4
4
CREATE EXTENSION IF NOT EXISTS pg_net SCHEMA extensions;
5
5
6
6
-- Create supabase_functions schema
7
- CREATE SCHEMA supabase_functions AUTHORIZATION supabase_admin;
7
+ CREATE SCHEMA IF NOT EXISTS supabase_functions AUTHORIZATION supabase_admin;
8
8
9
9
GRANT USAGE ON SCHEMA supabase_functions TO postgres, anon, authenticated, service_role;
10
10
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON TABLES TO postgres, anon, authenticated, service_role;
11
11
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON FUNCTIONS TO postgres, anon, authenticated, service_role;
12
12
ALTER DEFAULT PRIVILEGES IN SCHEMA supabase_functions GRANT ALL ON SEQUENCES TO postgres, anon, authenticated, service_role;
13
13
14
14
-- supabase_functions.migrations definition
15
- CREATE TABLE supabase_functions .migrations (
15
+ CREATE TABLE IF NOT EXISTS supabase_functions .migrations (
16
16
version text PRIMARY KEY ,
17
17
inserted_at timestamptz NOT NULL DEFAULT NOW()
18
18
);
19
19
20
20
-- 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' );
22
25
23
26
-- supabase_functions.hooks definition
24
- CREATE TABLE supabase_functions .hooks (
27
+ CREATE TABLE IF NOT EXISTS supabase_functions .hooks (
25
28
id bigserial PRIMARY KEY ,
26
29
hook_table_id integer NOT NULL ,
27
30
hook_name text NOT NULL ,
@@ -32,78 +35,104 @@ CREATE INDEX supabase_functions_hooks_request_id_idx ON supabase_functions.hooks
32
35
CREATE INDEX supabase_functions_hooks_h_table_id_h_name_idx ON supabase_functions .hooks USING btree (hook_table_id, hook_name);
33
36
COMMENT ON TABLE supabase_functions.hooks IS ' Supabase Functions Hooks: Audit trail for triggered hooks.' ;
34
37
35
- CREATE FUNCTION supabase_functions .http_request()
38
+ CREATE OR REPLACE FUNCTION supabase_functions .http_request()
36
39
RETURNS trigger
37
40
LANGUAGE plpgsql
41
+ SECURITY DEFINER
42
+ SET search_path TO ' supabase_functions'
38
43
AS $function$
39
44
DECLARE
40
- request_id bigint ;
45
+ local_request_id bigint ;
41
46
payload jsonb;
42
47
url text := TG_ARGV[0 ]::text ;
43
48
method text := TG_ARGV[1 ]::text ;
44
49
headers jsonb DEFAULT ' {}' ::jsonb;
45
50
params jsonb DEFAULT ' {}' ::jsonb;
46
- timeout_ms integer DEFAULT 1000 ;
51
+ timeout_ms integer ;
52
+ retry_count integer DEFAULT 0 ;
53
+ max_retries integer := COALESCE(TG_ARGV[5 ]::integer , 0 );
54
+ succeeded boolean := FALSE;
55
+ retry_delays double precision [] := ARRAY[0 , 0 .250 , 0 .500 , 1 .000 , 2 .500 , 5 .000 ];
56
+ status_code integer := 0 ;
47
57
BEGIN
48
58
IF url IS NULL OR url = ' null' THEN
49
59
RAISE EXCEPTION ' url argument is missing' ;
50
60
END IF;
51
-
52
61
IF method IS NULL OR method = ' null' THEN
53
62
RAISE EXCEPTION ' method argument is missing' ;
54
63
END IF;
55
-
56
64
IF TG_ARGV[2 ] IS NULL OR TG_ARGV[2 ] = ' null' THEN
57
65
headers = ' {"Content-Type": "application/json"}' ::jsonb;
58
66
ELSE
59
67
headers = TG_ARGV[2 ]::jsonb;
60
68
END IF;
61
-
62
69
IF TG_ARGV[3 ] IS NULL OR TG_ARGV[3 ] = ' null' THEN
63
70
params = ' {}' ::jsonb;
64
71
ELSE
65
72
params = TG_ARGV[3 ]::jsonb;
66
73
END IF;
67
-
68
- IF TG_ARGV[4 ] IS NULL OR TG_ARGV[4 ] = ' null' THEN
69
- timeout_ms = 1000 ;
70
- ELSE
74
+ IF TG_ARGV[4 ] IS NOT NULL OR TG_ARGV[4 ] <> ' null' THEN
71
75
timeout_ms = TG_ARGV[4 ]::integer ;
72
76
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
-
77
+ -- Retry loop
78
+ WHILE NOT succeeded AND retry_count <= max_retries LOOP
79
+ PERFORM pg_sleep(retry_delays[retry_count + 1 ]);
80
+ IF retry_delays[retry_count + 1 ] > 0 THEN
81
+ RAISE WARNING ' Retrying HTTP request: {retry_attempt: %, url: "%", timeout_ms: %, retry_delay_ms: %}' ,
82
+ retry_count, url, timeout_ms, retry_delays[retry_count + 1 ] * 1000 ;
83
+ END IF;
84
+ retry_count := retry_count + 1 ;
85
+ BEGIN
86
+ CASE
87
+ WHEN method = ' GET' THEN
88
+ SELECT http_get INTO local_request_id FROM net .http_get (
89
+ url,
90
+ params,
91
+ headers,
92
+ timeout_ms
93
+ );
94
+ WHEN method = ' POST' THEN
95
+ payload = jsonb_build_object(
96
+ ' old_record' , OLD,
97
+ ' record' , NEW,
98
+ ' type' , TG_OP,
99
+ ' table' , TG_TABLE_NAME,
100
+ ' schema' , TG_TABLE_SCHEMA
101
+ );
102
+ SELECT http_post INTO local_request_id FROM net .http_post (
103
+ url,
104
+ payload,
105
+ params,
106
+ headers,
107
+ timeout_ms
108
+ );
109
+ ELSE
110
+ RAISE EXCEPTION ' method argument % is invalid' , method;
111
+ END CASE;
112
+ IF local_request_id IS NOT NULL THEN
113
+ SELECT (response).status_code::integer
114
+ INTO status_code
115
+ FROM net ._http_collect_response (local_request_id);
116
+ IF status_code < 500 THEN
117
+ succeeded := TRUE;
118
+ END IF;
119
+ END IF;
120
+ -- Exit loop on successful request
121
+ EXIT WHEN succeeded;
122
+ EXCEPTION
123
+ WHEN OTHERS THEN
124
+ IF retry_count > max_retries THEN
125
+ -- If retries exhausted, re-raise exception
126
+ RAISE EXCEPTION ' HTTP request failed after % retries. SQL Error: { %, % }' ,
127
+ max_retries, SQLERRM, SQLSTATE;
128
+ END IF;
129
+ END;
130
+ END LOOP;
131
+ -- Failed retries are not logged
102
132
INSERT INTO supabase_functions .hooks
103
133
(hook_table_id, hook_name, request_id)
104
134
VALUES
105
- (TG_RELID, TG_NAME, request_id);
106
-
135
+ (TG_RELID, TG_NAME, local_request_id);
107
136
RETURN NEW;
108
137
END
109
138
$function$;
@@ -231,8 +260,6 @@ BEGIN
231
260
END
232
261
$$;
233
262
234
- INSERT INTO supabase_functions .migrations (version) VALUES (' 20210809183423_update_grants' );
235
-
236
263
ALTER function supabase_functions .http_request () SECURITY DEFINER;
237
264
ALTER function supabase_functions .http_request () SET search_path = supabase_functions;
238
265
REVOKE ALL ON FUNCTION supabase_functions .http_request () FROM PUBLIC;
0 commit comments