Skip to content

Commit d312525

Browse files
committed
feat: support net.http_delete with request body
Addresses #77 (comment). Now it's possible to do: ```sql select net.http_delete( url :='http://localhost:8080/delete_w_body' , body := '{"key": "val"}' ); ``` Backwards compatibility is tested. No body will be sent if the body parameter is NULL (the default).
1 parent 3cbe173 commit d312525

File tree

5 files changed

+92
-3
lines changed

5 files changed

+92
-3
lines changed

nix/nginx/conf/custom.conf

+9
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,15 @@ location /delete {
4949
echo_duplicate 1 $echo_client_request_headers$is_args$query_string;
5050
}
5151

52+
location /delete_w_body {
53+
if ($request_method != 'DELETE'){
54+
return 405;
55+
}
56+
57+
echo_read_request_body;
58+
echo $request_body;
59+
}
60+
5261
location /redirect_me {
5362
return 301 /to_here;
5463
}

sql/pg_net--0.14.0--0.15.0.sql

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
drop function net.http_delete (text, jsonb, jsonb, integer);
2+
3+
create function net.http_delete(
4+
-- url for the request
5+
url text,
6+
-- key/value pairs to be url encoded and appended to the `url`
7+
params jsonb default '{}'::jsonb,
8+
-- key/values to be included in request headers
9+
headers jsonb default '{}'::jsonb,
10+
-- the maximum number of milliseconds the request may take before being cancelled
11+
timeout_milliseconds int default 5000,
12+
-- optional body of the request
13+
body jsonb default NULL
14+
)
15+
-- request_id reference
16+
returns bigint
17+
volatile
18+
parallel safe
19+
language plpgsql
20+
as $$
21+
declare
22+
request_id bigint;
23+
params_array text[];
24+
begin
25+
select coalesce(array_agg(net._urlencode_string(key) || '=' || net._urlencode_string(value)), '{}')
26+
into params_array
27+
from jsonb_each_text(params);
28+
29+
-- Add to the request queue
30+
insert into net.http_request_queue(method, url, headers, body, timeout_milliseconds)
31+
values (
32+
'DELETE',
33+
net._encode_url_with_params_array(url, params_array),
34+
headers,
35+
convert_to(body::text, 'UTF8'),
36+
timeout_milliseconds
37+
)
38+
returning id
39+
into request_id;
40+
41+
return request_id;
42+
end
43+
$$;

sql/pg_net.sql

+5-3
Original file line numberDiff line numberDiff line change
@@ -223,11 +223,12 @@ create or replace function net.http_delete(
223223
-- key/values to be included in request headers
224224
headers jsonb default '{}'::jsonb,
225225
-- the maximum number of milliseconds the request may take before being cancelled
226-
timeout_milliseconds int default 5000
226+
timeout_milliseconds int default 5000,
227+
-- optional body of the request
228+
body jsonb default NULL
227229
)
228230
-- request_id reference
229231
returns bigint
230-
strict
231232
volatile
232233
parallel safe
233234
language plpgsql
@@ -241,11 +242,12 @@ begin
241242
from jsonb_each_text(params);
242243

243244
-- Add to the request queue
244-
insert into net.http_request_queue(method, url, headers, timeout_milliseconds)
245+
insert into net.http_request_queue(method, url, headers, body, timeout_milliseconds)
245246
values (
246247
'DELETE',
247248
net._encode_url_with_params_array(url, params_array),
248249
headers,
250+
convert_to(body::text, 'UTF8'),
249251
timeout_milliseconds
250252
)
251253
returning id

src/core.c

+3
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,9 @@ static void init_curl_handle(CURLM *curl_mhandle, MemoryContext curl_memctx, int
124124

125125
if (strcasecmp(method, "DELETE") == 0) {
126126
EREPORT_CURL_SETOPT(curl_ez_handle, CURLOPT_CUSTOMREQUEST, "DELETE");
127+
if (reqBody) {
128+
EREPORT_CURL_SETOPT(curl_ez_handle, CURLOPT_POSTFIELDS, reqBody);
129+
}
127130
}
128131

129132
EREPORT_CURL_SETOPT(curl_ez_handle, CURLOPT_WRITEFUNCTION, body_cb);

test/test_http_delete.py

+32
Original file line numberDiff line numberDiff line change
@@ -160,3 +160,35 @@ def test_http_delete_positional_args(sess):
160160
assert response is not None
161161
assert response[0] == "SUCCESS"
162162
assert response[1] == "ok"
163+
164+
165+
def test_http_delete_with_body(sess):
166+
"""delete with request body works"""
167+
168+
# Create a request
169+
(request_id,) = sess.execute(text(
170+
"""
171+
select net.http_delete(
172+
url :='http://localhost:8080/delete_w_body'
173+
, body := '{"key": "val"}'
174+
);
175+
"""
176+
)).fetchone()
177+
178+
# Commit so background worker can start
179+
sess.commit()
180+
181+
# Collect the response, waiting as needed
182+
(response_json,) = sess.execute(
183+
text(
184+
"""
185+
select
186+
((x.response).body)::jsonb body_json
187+
from
188+
net._http_collect_response(:request_id, async:=false) x;
189+
"""
190+
),
191+
{"request_id": request_id},
192+
).fetchone()
193+
194+
assert response_json["key"] == "val"

0 commit comments

Comments
 (0)