-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpgcurl.c
127 lines (106 loc) · 3.81 KB
/
pgcurl.c
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
127
/*
* PostgreSQL functions to interface with libcurl.
* Copyright (c) 2011 Hannu Valtonen <[email protected]>
* See the file COPYING for distribution terms.
*/
#include "pgcurl.h"
#ifdef PG_MODULE_MAGIC
PG_MODULE_MAGIC;
#endif
static struct pgcurl_global globals;
void _PG_init(void)
{
MemoryContext old_ctxt;
globals.pg_ctxt = AllocSetContextCreate(TopMemoryContext,
"pgcurl global context",
ALLOCSET_SMALL_MINSIZE,
ALLOCSET_SMALL_INITSIZE,
ALLOCSET_SMALL_MAXSIZE);
old_ctxt = MemoryContextSwitchTo(globals.pg_ctxt);
}
static size_t read_callback(void *ptr, size_t size, size_t nmemb, void *userp)
{
struct help_struct *pooh = (struct help_struct *)userp;
if(size*nmemb < 1)
return 0;
if(pooh->size) {
*(char *)ptr = pooh->string[0]; /* copy one single byte */
pooh->string++; /* advance pointer */
pooh->size--; /* less data left */
return 1; /* we return 1 byte at a time! */
}
return 0; /* no more data left to deliver */
}
static size_t
write_callback(void *ptr, size_t size, size_t nmemb, void *data)
{
size_t realsize = size * nmemb;
struct help_struct *mem = (struct help_struct *)data;
mem->string = repalloc(mem->string, mem->size + realsize + 1);
if (mem->string == NULL)
elog(ERROR, "not enough memory to realloc)\n");
memcpy(&(mem->string[mem->size]), ptr, realsize);
mem->size += realsize;
mem->string[mem->size] = 0;
return realsize;
}
Datum pgcurl_put(PG_FUNCTION_ARGS)
{
return curl_do_actual_work(PGCURL_PUT, fcinfo);
}
Datum pgcurl_get(PG_FUNCTION_ARGS)
{
return curl_do_actual_work(PGCURL_GET, fcinfo);
}
Datum pgcurl_delete(PG_FUNCTION_ARGS)
{
return curl_do_actual_work(PGCURL_DELETE, fcinfo);
}
Datum curl_do_actual_work(int method_type, PG_FUNCTION_ARGS)
{
CURL *curl;
CURLcode res;
StringInfoData buf;
text *url_text = PG_GETARG_TEXT_P(0), *payload_text;
char *url, *payload;
struct help_struct read_chunk, write_chunk;
read_chunk.string = palloc(1); /* will be grown as needed by the realloc above */
read_chunk.size = 0; /* no data at this point */
url = DatumGetCString(DirectFunctionCall1(textout, PointerGetDatum(url_text)));
curl = curl_easy_init();
if(curl) {
curl_easy_setopt(curl, CURLOPT_URL, url);
curl_easy_setopt(curl, CURLOPT_VERBOSE, 0);
curl_easy_setopt(curl, CURLOPT_USERAGENT, "pgcurl-agent/1.0");
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); /* Consider making this configurable */
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); /* Consider making this configurable */
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void *)&read_chunk);
if (method_type == PGCURL_PUT) {
curl_easy_setopt(curl, CURLOPT_UPLOAD, 1);
payload_text = PG_GETARG_TEXT_P(0);
payload = DatumGetCString(DirectFunctionCall1(textout, PointerGetDatum(payload_text)));
write_chunk.string = payload;
write_chunk.size = strlen(payload);
curl_easy_setopt(curl, CURLOPT_READFUNCTION, read_callback);
curl_easy_setopt(curl, CURLOPT_READDATA, &write_chunk);
curl_easy_setopt(curl, CURLOPT_INFILESIZE, write_chunk.size);
}
if (method_type == PGCURL_POST) {
curl_easy_setopt(curl, CURLOPT_POST, 1);
}
if (method_type == PGCURL_DELETE) {
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "DELETE");
}
res = curl_easy_perform(curl);
if (res != 0)
elog(WARNING, "PGCurl error: %s, URL: %s\n", curl_easy_strerror(res), url);
curl_easy_cleanup(curl);
}
initStringInfo(&buf);
if(read_chunk.string) {
appendStringInfo(&buf, "%s", read_chunk.string);
pfree(read_chunk.string);
}
PG_RETURN_DATUM(DirectFunctionCall1(textin, CStringGetDatum(buf.data)));
}