forked from fraktalio/fstore-sql
-
Notifications
You must be signed in to change notification settings - Fork 0
/
extensions.sql
126 lines (109 loc) · 5.86 KB
/
extensions.sql
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
-- With this two extensions we enabled the event streaming to the edge-functions/event-handlers, automatically.
-- https://github.com/citusdata/pg_cron
-- https://github.com/supabase/pg_net
-- Your application does not have ot call `stream_events('view1')` function any more, `cron.job` will run `SELECT * from stream_events('view1');` for you, and publish event(s) to your edge-functions/http endpoints automatically. So, the database is doing all the job.
-- The `cron` job is managed(created/deleted) by triggers on the `view` table. So, whenever you register a new View, the cron job will be created automatically.
-- #######################################################################################
-- ###### pg_NET extension ######
-- #######################################################################################
create extension if not exists "pg_net" schema extensions;
-- #######################################################################################
-- ###### pg_CRON extension ######
-- #######################################################################################
-- enable 'pg_cron' extension
create extension if not exists "pg_cron";
-- #######################################################################################
-- ###### Stream events to the Edge ######
-- #######################################################################################
-- API: Create a function to stream events to a view/event handler/edge function(s).
-- The view/event handler is an HTTP endpoint/edge function that receives the events.
-- example: SELECT schedule_events('view1', '2 seconds', 'https://localhost:3000/functions/v1/event-handler');
-- unschedule: SELECT cron.unschedule('view1');
CREATE OR REPLACE FUNCTION schedule_events(v_view TEXT, v_schedule TEXT DEFAULT '1 seconds',
v_edge_function_url TEXT DEFAULT 'https://localhost:3000/functions/v1/event-handler')
RETURNS bigint AS
$$
DECLARE
sql_statement TEXT;
BEGIN
-- Construct the dynamic SQL statement
sql_statement := '
WITH event_result AS (
SELECT *
FROM stream_events(''' || v_view || ''', 1)
LIMIT 1
)
SELECT
net.http_post(
url:=''' || v_edge_function_url || ''',
body:=jsonb_build_object(''view'', ''' || v_view || ''', ''decider_id'', event_result.decider_id, ''offset'', event_result.offset, ''data'', event_result.data)
) AS request_id
FROM event_result
';
-- Execute the dynamic SQL statement
EXECUTE sql_statement;
-- Schedule the dynamic SQL statement to run at the specified interval
RETURN cron.schedule(v_view, v_schedule, sql_statement);
END;
$$ LANGUAGE plpgsql;
-- Trigger: Create a trigger function that will activate streaming events to a view/event handler automatically.
CREATE OR REPLACE FUNCTION on_insert_on_views() RETURNS trigger AS
'
BEGIN
IF NEW.pooling_delay_s IS NOT NULL THEN
-- Create the cron job. Stream events to the view/event handler every `NEW.pooling_delay` seconds
PERFORM schedule_events(NEW.view, NEW.pooling_delay_s || '' seconds'', NEW.edge_function_url);
-- If you do not want to use cron.job_run_details at all, then you can add cron.log_run = off to postgresql.conf.
-- or, delete old cron.job_run_details records of the current view, every day at noon
PERFORM cron.schedule(''delete_'' || NEW.view, ''0 12 * * *'',
$$DELETE FROM cron.job_run_details USING cron.job WHERE jobid = cron.job.jobid AND cron.job.jobname = NEW.view AND end_time < now() - interval ''1 days''$$);
END IF;
RETURN NEW;
END;
' LANGUAGE plpgsql;
DROP TRIGGER IF EXISTS t_on_insert_on_views ON "views";
CREATE TRIGGER t_on_insert_on_views
AFTER INSERT
ON "views"
FOR EACH ROW
EXECUTE FUNCTION on_insert_on_views();
-- Trigger: Create a trigger function that will activate streaming events to a view/event handler automatically on View update.
CREATE OR REPLACE FUNCTION on_update_on_views() RETURNS trigger AS
'
BEGIN
IF NEW.pooling_delay_s IS NOT NULL THEN
-- Create the cron job. Stream events to the view/event handler every `NEW.pooling_delay` seconds
PERFORM schedule_events(NEW.view, NEW.pooling_delay_s || '' seconds'', NEW.edge_function_url);
-- If you do not want to use cron.job_run_details at all, then you can add cron.log_run = off to postgresql.conf.
-- or, delete old cron.job_run_details records of the current view, every day at noon
PERFORM cron.schedule(''delete_'' || NEW.view, ''0 12 * * *'',
$$DELETE FROM cron.job_run_details USING cron.job WHERE jobid = cron.job.jobid AND cron.job.jobname = NEW.view AND end_time < now() - interval ''1 days''$$);
ELSE
IF OLD.pooling_delay_s IS NOT NULL THEN
-- Delete the cron job
PERFORM cron.unschedule(OLD.view);
END IF;
END IF;
RETURN NEW;
END;
' LANGUAGE plpgsql;
DROP TRIGGER IF EXISTS t_on_update_on_views ON "views";
CREATE TRIGGER t_on_update_on_views
AFTER UPDATE
ON "views"
FOR EACH ROW
EXECUTE FUNCTION on_update_on_views();
-- Create a trigger function to stop the cron job, and stop streaming events to a view/event handler.
CREATE OR REPLACE FUNCTION on_delete_on_views() RETURNS trigger AS
'
BEGIN
PERFORM cron.unschedule(OLD.view);
RETURN NEW;
END;
' LANGUAGE plpgsql;
DROP TRIGGER IF EXISTS t_on_delete_on_views ON "views";
CREATE TRIGGER t_on_delete_on_views
AFTER DELETE
ON "views"
FOR EACH ROW
EXECUTE FUNCTION on_delete_on_views();