summaryrefslogtreecommitdiff
path: root/modules/curl/curl.c
blob: b8b2bb63a4497bce2da4a9197da6f27cd00f40fc (plain)
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
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <curl/curl.h>

#include <config.h>
#include <lisp.h>

int plugin_is_GPL_compatible;
static Lisp_Object Qcurl;

struct buffer
{
  char *p;
  size_t size, capacity;
};

struct Lisp_CURL
{
  struct buffer buf;
  CURL *curl;
};

#define XCURL(x) ((struct Lisp_CURL*)XSAVE_POINTER (x, 0))

/* curl write callback */
static size_t
write_cb (void *src, size_t size, size_t nb, void *userp)
{
  struct buffer *buf = userp;
  size_t total = size*nb;

  if (buf->size + total > buf->capacity)
    {
      buf->capacity = 2 * (buf->size + total);
      buf->p = realloc (buf->p, buf->capacity);
    }

  memcpy (buf->p + buf->size, src, total);
  buf->size += total;
  buf->p[buf->size] = 0;

  return total;
}


EXFUN (Fcurl_make, 0);
DEFUN ("curl-make", Fcurl_make, Scurl_make, 0, 0, 0,
       doc: "Return a new CURL handle.")
  (void)
{
  struct Lisp_CURL *p = calloc (sizeof (*p), 1);
  p->buf.p = calloc (1, 1); /* so that realloc always work */
  p->buf.capacity = 0;
  p->curl = curl_easy_init ();
  return make_save_ptr ((void*)p);
}


EXFUN (Fcurl_fetch_url, 2);
DEFUN ("curl-fetch-url", Fcurl_fetch_url, Scurl_fetch_url, 2, 2, 0,
       doc: "Fetch and store the content of URL using HANDLE.\n"
       "Return t if successful otherwise return an error string.")
  (Lisp_Object handle, Lisp_Object url)
{
  CURLcode res;
  struct Lisp_CURL *c = XCURL (handle);

  curl_easy_setopt (c->curl, CURLOPT_URL, SSDATA (url));
  curl_easy_setopt (c->curl, CURLOPT_WRITEFUNCTION, write_cb);
  curl_easy_setopt (c->curl, CURLOPT_WRITEDATA, (void*)&c->buf);
  curl_easy_setopt (c->curl, CURLOPT_USERAGENT, "curl-in-emacs/1.0");
  res = curl_easy_perform (c->curl);

  if (res != CURLE_OK)
    {
      const char* error = curl_easy_strerror (res);
      return make_string (error, strlen (error));
    }

  return Qt;
}

EXFUN (Fcurl_content, 1);
DEFUN ("curl-content", Fcurl_content, Scurl_content, 1, 1, 0,
       doc: "Return the content of a successful fetch made in HANDLE.")
  (Lisp_Object handle)
{
  struct Lisp_CURL *c = XCURL (handle);
  return make_string (c->buf.p, c->buf.size);
}

EXFUN (Fcurl_free, 1);
DEFUN ("curl-free", Fcurl_free, Scurl_free, 1, 1, 0,
       doc: "Free curl HANDLE.")
  (Lisp_Object handle)
{
  struct Lisp_CURL *c = XCURL (handle);
  free (c->buf.p);
  curl_easy_cleanup (c->curl);

  return Qt;
}

void init ()
{
  curl_global_init (CURL_GLOBAL_ALL);
  /* when unloading: curl_global_cleanup(); */

  DEFSYM (Qcurl, "curl");

  defsubr (&Scurl_make);
  defsubr (&Scurl_fetch_url);
  defsubr (&Scurl_content);
  defsubr (&Scurl_free);

  Fprovide (Qcurl, Qnil);
}