mirror of
https://codeberg.org/grunfink/snac2.git
synced 2025-01-11 16:43:38 +00:00
ea9c030249
Most of xs.h seems to expect that buffers are rounded up to block size, so we should preserve that invariant here. (In particular, xs_expand will avoid calling xs_realloc if the new size fits in the same block, which means that if we don't pad out the data it will expand out of the memory we're allocated.)
191 lines
4.7 KiB
C
191 lines
4.7 KiB
C
/* copyright (c) 2022 - 2023 grunfink / MIT license */
|
|
|
|
#ifndef _XS_CURL_H
|
|
|
|
#define _XS_CURL_H
|
|
|
|
xs_dict *xs_http_request(const char *method, const char *url,
|
|
const xs_dict *headers,
|
|
const xs_str *body, int b_size, int *status,
|
|
xs_str **payload, int *p_size, int timeout);
|
|
|
|
#ifdef XS_IMPLEMENTATION
|
|
|
|
#include <curl/curl.h>
|
|
|
|
static size_t _header_callback(char *buffer, size_t size,
|
|
size_t nitems, xs_dict **userdata)
|
|
{
|
|
xs_dict *headers = *userdata;
|
|
xs *l;
|
|
|
|
/* get the line */
|
|
l = xs_str_new(NULL);
|
|
l = xs_append_m(l, buffer, size * nitems);
|
|
l = xs_strip_i(l);
|
|
|
|
/* only the HTTP/x 200 line and the last one doesn't have ': ' */
|
|
if (xs_str_in(l, ": ") != -1) {
|
|
xs *knv = xs_split_n(l, ": ", 1);
|
|
|
|
xs_tolower_i(xs_list_get(knv, 0));
|
|
|
|
headers = xs_dict_set(headers, xs_list_get(knv, 0), xs_list_get(knv, 1));
|
|
}
|
|
else
|
|
if (xs_startswith(l, "HTTP/"))
|
|
headers = xs_dict_set(headers, "_proto", l);
|
|
|
|
*userdata = headers;
|
|
|
|
return nitems * size;
|
|
}
|
|
|
|
|
|
struct _payload_data {
|
|
char *data;
|
|
int size;
|
|
int offset;
|
|
};
|
|
|
|
static int _data_callback(void *buffer, size_t size,
|
|
size_t nitems, struct _payload_data *pd)
|
|
{
|
|
int sz = size * nitems;
|
|
|
|
/* open space */
|
|
pd->size += sz;
|
|
pd->data = xs_realloc(pd->data, _xs_blk_size(pd->size + 1));
|
|
|
|
/* copy data */
|
|
memcpy(pd->data + pd->offset, buffer, sz);
|
|
pd->offset += sz;
|
|
|
|
return sz;
|
|
}
|
|
|
|
|
|
static int _post_callback(char *buffer, size_t size,
|
|
size_t nitems, struct _payload_data *pd)
|
|
{
|
|
/* size of data left */
|
|
int sz = pd->size - pd->offset;
|
|
|
|
/* if it's still bigger than the provided space, trim */
|
|
if (sz > (int) (size * nitems))
|
|
sz = size * nitems;
|
|
|
|
memcpy(buffer, pd->data + pd->offset, sz);
|
|
|
|
/* skip sent data */
|
|
pd->offset += sz;
|
|
|
|
return sz;
|
|
}
|
|
|
|
|
|
xs_dict *xs_http_request(const char *method, const char *url,
|
|
const xs_dict *headers,
|
|
const xs_str *body, int b_size, int *status,
|
|
xs_str **payload, int *p_size, int timeout)
|
|
/* does an HTTP request */
|
|
{
|
|
xs_dict *response;
|
|
CURL *curl;
|
|
struct curl_slist *list = NULL;
|
|
xs_dict *p;
|
|
xs_str *k;
|
|
xs_val *v;
|
|
long lstatus;
|
|
struct _payload_data pd;
|
|
|
|
response = xs_dict_new();
|
|
|
|
curl = curl_easy_init();
|
|
|
|
curl_easy_setopt(curl, CURLOPT_URL, url);
|
|
|
|
if (timeout <= 0)
|
|
timeout = 8;
|
|
|
|
curl_easy_setopt(curl, CURLOPT_TIMEOUT, (long) timeout);
|
|
|
|
#ifdef FORCE_HTTP_1_1
|
|
/* force HTTP/1.1 */
|
|
curl_easy_setopt(curl, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
|
|
#endif
|
|
|
|
/* obey redirections */
|
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
|
|
|
|
/* store response headers here */
|
|
curl_easy_setopt(curl, CURLOPT_HEADERDATA, &response);
|
|
curl_easy_setopt(curl, CURLOPT_HEADERFUNCTION, _header_callback);
|
|
|
|
struct _payload_data ipd = { NULL, 0, 0 };
|
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &ipd);
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, _data_callback);
|
|
|
|
if (strcmp(method, "POST") == 0 || strcmp(method, "PUT") == 0) {
|
|
curl_easy_setopt(curl, method[1] == 'O' ? CURLOPT_POST : CURLOPT_UPLOAD, 1L);
|
|
|
|
if (body != NULL) {
|
|
if (b_size <= 0)
|
|
b_size = xs_size(body);
|
|
|
|
/* add the content-length header */
|
|
curl_easy_setopt(curl, CURLOPT_INFILESIZE, b_size);
|
|
|
|
pd.data = (char *)body;
|
|
pd.size = b_size;
|
|
pd.offset = 0;
|
|
|
|
curl_easy_setopt(curl, CURLOPT_READDATA, &pd);
|
|
curl_easy_setopt(curl, CURLOPT_READFUNCTION, _post_callback);
|
|
}
|
|
}
|
|
|
|
/* fill the request headers */
|
|
p = (xs_dict *)headers;
|
|
while (xs_dict_iter(&p, &k, &v)) {
|
|
xs *h = xs_fmt("%s: %s", k, v);
|
|
|
|
list = curl_slist_append(list, h);
|
|
}
|
|
|
|
/* disable server support for 100-continue */
|
|
list = curl_slist_append(list, "Expect:");
|
|
|
|
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, list);
|
|
|
|
/* do it */
|
|
curl_easy_perform(curl);
|
|
|
|
curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &lstatus);
|
|
|
|
curl_easy_cleanup(curl);
|
|
|
|
curl_slist_free_all(list);
|
|
|
|
if (status != NULL)
|
|
*status = (int) lstatus;
|
|
|
|
if (p_size != NULL)
|
|
*p_size = ipd.size;
|
|
|
|
if (payload != NULL) {
|
|
*payload = ipd.data;
|
|
|
|
/* add an asciiz just in case (but not touching p_size) */
|
|
if (ipd.data != NULL)
|
|
ipd.data[ipd.size] = '\0';
|
|
}
|
|
else
|
|
xs_free(ipd.data);
|
|
|
|
return response;
|
|
}
|
|
|
|
#endif /* XS_IMPLEMENTATION */
|
|
|
|
#endif /* _XS_CURL_H */
|