Towards Dev

A publication for sharing projects, ideas, codes, and new theories.

Follow publication

evaluate() PostgreSQL Function for Evaluating Stored Expressions (Part 1)

Christoph Bussler
Towards Dev
Published in
5 min readDec 8, 2022

--

Query predicates as data

{
"make": "Koenigsegg",
"model": "CC850",
"color": "silver",
"horsepower": 1385,
"price": 3650000
}
| car_id (int) | new_car (jsonb)        |
+--------------+------------------------+
| 1 | { |
| | "make": "Koenigsegg", |
| | "model": "CC850", |
| | "color": "silver", |
| | "horsepower": 1385, |
| | "price": 3650000 |
| | } |
+--------------+------------------------+
| 2 | { |
| | "make": "Honda", |
| | "model": "Jazz", |
| | "color": "silver", |
| | "horsepower": 0, |
| | "price": 21394 |
| | } |
+--------------+------------------------+
| customer_id | name      | interest                       |
| (int) | (varchar) | (varchar) |
+-------------+-----------+--------------------------------+
| 100 | A | object -> 'horsepower' > 1000 |
+-------------+-----------+--------------------------------+
| 101 | B | object -> 'price' < 100000 and |
| | | object ->> 'color' = 'silver' |
+-------------+-----------+--------------------------------+
| customer_id | name      | interest                              |
| (int) | (varchar) | (varchar) |
+-------------+----------------+----------------------------------+
| 100 | A | (object -> 'horsepower')::int > 1000 |
+-------------+----------------+----------------------------------+
| 101 | B | (object -> 'price')::int < 100000 and |
| | | object ->> 'color' = 'silver' |
+-------------------+----------------+----------------------------+

Evaluate() function

select cust.customer_id, 
car.car_id
from customer cust,
car car
where evaluate(car.new_car, cust.interest);

Implementation

CREATE OR REPLACE FUNCTION evaluate(
p_object JSONB,
p_expression VARCHAR
)
RETURNS BOOLEAN
LANGUAGE plpgsql
AS
-- Expression evaluation on object
-- (a) execute the expression on object of type JSONB
-- (b) return TRUE if the expression evaluates to true
-- (c) return FALSE if the expression evaluates to false
$$
DECLARE
v_result BOOLEAN;
BEGIN
EXECUTE format('SELECT'
|| ' CASE'
|| ' WHEN (SELECT count(*)'
|| ' FROM (SELECT $1 AS object) temp'
|| ' WHERE ('
|| $2
|| ' )) = 1 THEN TRUE'
|| ' ELSE FALSE'
|| ' END')
USING p_object::JSONB, p_expression::TEXT
INTO v_result;
RETURN v_result;
END;
$$;

COMMENT ON FUNCTION evaluate(
p_object JSONB,
p_expression VARCHAR)
IS 'Function to evaluate an expression on an JSON object';
| customer_id | car_id |
+-------------+--------+
| 100 | 1 |
+-------------+--------+
| 101 | 2 |
+-------------+--------+

Improvements

Summary

Reference

--

--

No responses yet