Skip to content

Commit e6c0474

Browse files
committed
feat: support webhook retries
1 parent 1fd3dc3 commit e6c0474

File tree

2 files changed

+141
-89
lines changed

2 files changed

+141
-89
lines changed

migrations/db/migrations/20240822021428_enable_webhooks_by_default.sql

+77-49
Original file line numberDiff line numberDiff line change
@@ -4,106 +4,136 @@
44
CREATE EXTENSION IF NOT EXISTS pg_net SCHEMA extensions;
55

66
-- Create supabase_functions schema
7-
CREATE SCHEMA supabase_functions AUTHORIZATION supabase_admin;
7+
CREATE SCHEMA IF NOT EXISTS supabase_functions AUTHORIZATION supabase_admin;
88

99
GRANT USAGE ON SCHEMA supabase_functions TO postgres, anon, authenticated, service_role;
1010
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON TABLES TO postgres, anon, authenticated, service_role;
1111
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON FUNCTIONS TO postgres, anon, authenticated, service_role;
1212
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON SEQUENCES TO postgres, anon, authenticated, service_role;
1313

1414
-- supabase_functions.migrations definition
15-
CREATE TABLE supabase_functions.migrations (
15+
CREATE TABLE IF NOT EXISTS supabase_functions.migrations (
1616
version text PRIMARY KEY,
1717
inserted_at timestamptz NOT NULL DEFAULT NOW()
1818
);
1919

2020
-- Initial supabase_functions migration
21-
INSERT INTO supabase_functions.migrations (version) VALUES ('initial');
21+
INSERT INTO supabase_functions.migrations (version) VALUES
22+
('initial'),
23+
('20210809183423_update_grants'),
24+
('20240125163000_add_retry_to_http_request')
25+
ON CONFLICT DO NOTHING;
2226

2327
-- supabase_functions.hooks definition
24-
CREATE TABLE supabase_functions.hooks (
28+
CREATE TABLE IF NOT EXISTS supabase_functions.hooks (
2529
id bigserial PRIMARY KEY,
2630
hook_table_id integer NOT NULL,
2731
hook_name text NOT NULL,
2832
created_at timestamptz NOT NULL DEFAULT NOW(),
2933
request_id bigint
3034
);
31-
CREATE INDEX supabase_functions_hooks_request_id_idx ON supabase_functions.hooks USING btree (request_id);
32-
CREATE INDEX supabase_functions_hooks_h_table_id_h_name_idx ON supabase_functions.hooks USING btree (hook_table_id, hook_name);
35+
CREATE INDEX IF NOT EXISTS supabase_functions_hooks_request_id_idx ON supabase_functions.hooks USING btree (request_id);
36+
CREATE INDEX IF NOT EXISTS supabase_functions_hooks_h_table_id_h_name_idx ON supabase_functions.hooks USING btree (hook_table_id, hook_name);
3337
COMMENT ON TABLE supabase_functions.hooks IS 'Supabase Functions Hooks: Audit trail for triggered hooks.';
3438

35-
CREATE FUNCTION supabase_functions.http_request()
39+
CREATE OR REPLACE FUNCTION supabase_functions.http_request()
3640
RETURNS trigger
3741
LANGUAGE plpgsql
42+
SECURITY DEFINER
43+
SET search_path TO 'supabase_functions'
3844
AS $function$
3945
DECLARE
40-
request_id bigint;
46+
local_request_id bigint;
4147
payload jsonb;
4248
url text := TG_ARGV[0]::text;
4349
method text := TG_ARGV[1]::text;
4450
headers jsonb DEFAULT '{}'::jsonb;
4551
params jsonb DEFAULT '{}'::jsonb;
46-
timeout_ms integer DEFAULT 1000;
52+
timeout_ms integer;
53+
retry_count integer DEFAULT 0;
54+
max_retries integer := COALESCE(TG_ARGV[5]::integer, 0);
55+
succeeded boolean := FALSE;
56+
retry_delays double precision[] := ARRAY[0, 0.250, 0.500, 1.000, 2.500, 5.000];
57+
status_code integer := 0;
4758
BEGIN
4859
IF url IS NULL OR url = 'null' THEN
4960
RAISE EXCEPTION 'url argument is missing';
5061
END IF;
51-
5262
IF method IS NULL OR method = 'null' THEN
5363
RAISE EXCEPTION 'method argument is missing';
5464
END IF;
55-
5665
IF TG_ARGV[2] IS NULL OR TG_ARGV[2] = 'null' THEN
5766
headers = '{"Content-Type": "application/json"}'::jsonb;
5867
ELSE
5968
headers = TG_ARGV[2]::jsonb;
6069
END IF;
61-
6270
IF TG_ARGV[3] IS NULL OR TG_ARGV[3] = 'null' THEN
6371
params = '{}'::jsonb;
6472
ELSE
6573
params = TG_ARGV[3]::jsonb;
6674
END IF;
67-
68-
IF TG_ARGV[4] IS NULL OR TG_ARGV[4] = 'null' THEN
69-
timeout_ms = 1000;
70-
ELSE
75+
IF TG_ARGV[4] IS NOT NULL OR TG_ARGV[4] <> 'null' THEN
7176
timeout_ms = TG_ARGV[4]::integer;
7277
END IF;
73-
74-
CASE
75-
WHEN method = 'GET' THEN
76-
SELECT http_get INTO request_id FROM net.http_get(
77-
url,
78-
params,
79-
headers,
80-
timeout_ms
81-
);
82-
WHEN method = 'POST' THEN
83-
payload = jsonb_build_object(
84-
'old_record', OLD,
85-
'record', NEW,
86-
'type', TG_OP,
87-
'table', TG_TABLE_NAME,
88-
'schema', TG_TABLE_SCHEMA
89-
);
90-
91-
SELECT http_post INTO request_id FROM net.http_post(
92-
url,
93-
payload,
94-
params,
95-
headers,
96-
timeout_ms
97-
);
98-
ELSE
99-
RAISE EXCEPTION 'method argument % is invalid', method;
100-
END CASE;
101-
78+
-- Retry loop
79+
WHILE NOT succeeded AND retry_count <= max_retries LOOP
80+
PERFORM pg_sleep(retry_delays[retry_count + 1]);
81+
IF retry_delays[retry_count + 1] > 0 THEN
82+
RAISE WARNING 'Retrying HTTP request: {retry_attempt: %, url: "%", timeout_ms: %, retry_delay_ms: %}',
83+
retry_count, url, timeout_ms, retry_delays[retry_count + 1] * 1000;
84+
END IF;
85+
retry_count := retry_count + 1;
86+
BEGIN
87+
CASE
88+
WHEN method = 'GET' THEN
89+
SELECT http_get INTO local_request_id FROM net.http_get(
90+
url,
91+
params,
92+
headers,
93+
timeout_ms
94+
);
95+
WHEN method = 'POST' THEN
96+
payload = jsonb_build_object(
97+
'old_record', OLD,
98+
'record', NEW,
99+
'type', TG_OP,
100+
'table', TG_TABLE_NAME,
101+
'schema', TG_TABLE_SCHEMA
102+
);
103+
SELECT http_post INTO local_request_id FROM net.http_post(
104+
url,
105+
payload,
106+
params,
107+
headers,
108+
timeout_ms
109+
);
110+
ELSE
111+
RAISE EXCEPTION 'method argument % is invalid', method;
112+
END CASE;
113+
IF local_request_id IS NOT NULL THEN
114+
SELECT (response).status_code::integer
115+
INTO status_code
116+
FROM net._http_collect_response(local_request_id);
117+
IF status_code < 500 THEN
118+
succeeded := TRUE;
119+
END IF;
120+
END IF;
121+
-- Exit loop on successful request
122+
EXIT WHEN succeeded;
123+
EXCEPTION
124+
WHEN OTHERS THEN
125+
IF retry_count > max_retries THEN
126+
-- If retries exhausted, re-raise exception
127+
RAISE EXCEPTION 'HTTP request failed after % retries. SQL Error: { %, % }',
128+
max_retries, SQLERRM, SQLSTATE;
129+
END IF;
130+
END;
131+
END LOOP;
132+
-- Failed retries are not logged
102133
INSERT INTO supabase_functions.hooks
103134
(hook_table_id, hook_name, request_id)
104135
VALUES
105-
(TG_RELID, TG_NAME, request_id);
106-
136+
(TG_RELID, TG_NAME, local_request_id);
107137
RETURN NEW;
108138
END
109139
$function$;
@@ -231,8 +261,6 @@ BEGIN
231261
END
232262
$$;
233263

234-
INSERT INTO supabase_functions.migrations (version) VALUES ('20210809183423_update_grants');
235-
236264
ALTER function supabase_functions.http_request() SECURITY DEFINER;
237265
ALTER function supabase_functions.http_request() SET search_path = supabase_functions;
238266
REVOKE ALL ON FUNCTION supabase_functions.http_request() FROM PUBLIC;

migrations/schema.sql

+64-40
Original file line numberDiff line numberDiff line change
@@ -604,73 +604,97 @@ CREATE FUNCTION supabase_functions.http_request() RETURNS trigger
604604
SET search_path TO 'supabase_functions'
605605
AS $$
606606
DECLARE
607-
request_id bigint;
607+
local_request_id bigint;
608608
payload jsonb;
609609
url text := TG_ARGV[0]::text;
610610
method text := TG_ARGV[1]::text;
611611
headers jsonb DEFAULT '{}'::jsonb;
612612
params jsonb DEFAULT '{}'::jsonb;
613-
timeout_ms integer DEFAULT 1000;
613+
timeout_ms integer;
614+
retry_count integer DEFAULT 0;
615+
max_retries integer := COALESCE(TG_ARGV[5]::integer, 0);
616+
succeeded boolean := FALSE;
617+
retry_delays double precision[] := ARRAY[0, 0.250, 0.500, 1.000, 2.500, 5.000];
618+
status_code integer := 0;
614619
BEGIN
615620
IF url IS NULL OR url = 'null' THEN
616621
RAISE EXCEPTION 'url argument is missing';
617622
END IF;
618-
619623
IF method IS NULL OR method = 'null' THEN
620624
RAISE EXCEPTION 'method argument is missing';
621625
END IF;
622-
623626
IF TG_ARGV[2] IS NULL OR TG_ARGV[2] = 'null' THEN
624627
headers = '{"Content-Type": "application/json"}'::jsonb;
625628
ELSE
626629
headers = TG_ARGV[2]::jsonb;
627630
END IF;
628-
629631
IF TG_ARGV[3] IS NULL OR TG_ARGV[3] = 'null' THEN
630632
params = '{}'::jsonb;
631633
ELSE
632634
params = TG_ARGV[3]::jsonb;
633635
END IF;
634-
635-
IF TG_ARGV[4] IS NULL OR TG_ARGV[4] = 'null' THEN
636-
timeout_ms = 1000;
637-
ELSE
636+
IF TG_ARGV[4] IS NOT NULL OR TG_ARGV[4] <> 'null' THEN
638637
timeout_ms = TG_ARGV[4]::integer;
639638
END IF;
640-
641-
CASE
642-
WHEN method = 'GET' THEN
643-
SELECT http_get INTO request_id FROM net.http_get(
644-
url,
645-
params,
646-
headers,
647-
timeout_ms
648-
);
649-
WHEN method = 'POST' THEN
650-
payload = jsonb_build_object(
651-
'old_record', OLD,
652-
'record', NEW,
653-
'type', TG_OP,
654-
'table', TG_TABLE_NAME,
655-
'schema', TG_TABLE_SCHEMA
656-
);
657-
658-
SELECT http_post INTO request_id FROM net.http_post(
659-
url,
660-
payload,
661-
params,
662-
headers,
663-
timeout_ms
664-
);
665-
ELSE
666-
RAISE EXCEPTION 'method argument % is invalid', method;
667-
END CASE;
668-
639+
-- Retry loop
640+
WHILE NOT succeeded AND retry_count <= max_retries LOOP
641+
PERFORM pg_sleep(retry_delays[retry_count + 1]);
642+
IF retry_delays[retry_count + 1] > 0 THEN
643+
RAISE WARNING 'Retrying HTTP request: {retry_attempt: %, url: "%", timeout_ms: %, retry_delay_ms: %}',
644+
retry_count, url, timeout_ms, retry_delays[retry_count + 1] * 1000;
645+
END IF;
646+
retry_count := retry_count + 1;
647+
BEGIN
648+
CASE
649+
WHEN method = 'GET' THEN
650+
SELECT http_get INTO local_request_id FROM net.http_get(
651+
url,
652+
params,
653+
headers,
654+
timeout_ms
655+
);
656+
WHEN method = 'POST' THEN
657+
payload = jsonb_build_object(
658+
'old_record', OLD,
659+
'record', NEW,
660+
'type', TG_OP,
661+
'table', TG_TABLE_NAME,
662+
'schema', TG_TABLE_SCHEMA
663+
);
664+
SELECT http_post INTO local_request_id FROM net.http_post(
665+
url,
666+
payload,
667+
params,
668+
headers,
669+
timeout_ms
670+
);
671+
ELSE
672+
RAISE EXCEPTION 'method argument % is invalid', method;
673+
END CASE;
674+
IF local_request_id IS NOT NULL THEN
675+
SELECT (response).status_code::integer
676+
INTO status_code
677+
FROM net._http_collect_response(local_request_id);
678+
IF status_code < 500 THEN
679+
succeeded := TRUE;
680+
END IF;
681+
END IF;
682+
-- Exit loop on successful request
683+
EXIT WHEN succeeded;
684+
EXCEPTION
685+
WHEN OTHERS THEN
686+
IF retry_count > max_retries THEN
687+
-- If retries exhausted, re-raise exception
688+
RAISE EXCEPTION 'HTTP request failed after % retries. SQL Error: { %, % }',
689+
max_retries, SQLERRM, SQLSTATE;
690+
END IF;
691+
END;
692+
END LOOP;
693+
-- Failed retries are not logged
669694
INSERT INTO supabase_functions.hooks
670695
(hook_table_id, hook_name, request_id)
671696
VALUES
672-
(TG_RELID, TG_NAME, request_id);
673-
697+
(TG_RELID, TG_NAME, local_request_id);
674698
RETURN NEW;
675699
END
676700
$$;

0 commit comments

Comments
 (0)