a44a8cef99
need to be re-set over persistant requests
753 lines
20 KiB
Plaintext
753 lines
20 KiB
Plaintext
/* Perl interface for libcurl. Check out the file README for more info. */
|
|
|
|
#include "EXTERN.h"
|
|
#include "perl.h"
|
|
#include "XSUB.h"
|
|
|
|
#include <curl/curl.h>
|
|
#include <curl/easy.h>
|
|
|
|
#if (LIBCURL_VERSION_NUM<0x070702)
|
|
#define CURLOPT_HEADERFUNCTION 79
|
|
#define header_callback_func write_callback_func
|
|
#else
|
|
#define header_callback_func writeheader_callback_func
|
|
#endif
|
|
|
|
/* Lists that can be set via curl_easy_setopt() */
|
|
|
|
static struct curl_slist *httpheader = NULL, *quote = NULL, *postquote = NULL;
|
|
|
|
|
|
/* Buffer and varname for option CURLOPT_ERRORBUFFER */
|
|
|
|
static char errbuf[CURL_ERROR_SIZE];
|
|
static char *errbufvarname = NULL;
|
|
|
|
|
|
/* Callback functions */
|
|
|
|
static SV *read_callback = NULL, *write_callback = NULL,
|
|
*progress_callback = NULL, *passwd_callback = NULL,
|
|
*header_callback = NULL;
|
|
/* *closepolicy_callback = NULL; */
|
|
|
|
|
|
/* For storing the content */
|
|
|
|
static char *contbuf = NULL, *bufptr = NULL;
|
|
static int bufsize = 32768, contlen = 0;
|
|
|
|
|
|
/* Internal options for this perl module */
|
|
|
|
#define USE_INTERNAL_VARS 0x01
|
|
|
|
static int internal_options = 0;
|
|
|
|
|
|
/* Setup these global vars */
|
|
|
|
static void init_globals(void)
|
|
{
|
|
if (httpheader) curl_slist_free_all(httpheader);
|
|
if (quote) curl_slist_free_all(quote);
|
|
if (postquote) curl_slist_free_all(postquote);
|
|
httpheader = quote = postquote = NULL;
|
|
if (errbufvarname) free(errbufvarname);
|
|
errbufvarname = NULL;
|
|
if (contbuf == NULL) {
|
|
contbuf = malloc(bufsize + 1);
|
|
}
|
|
bufptr = contbuf;
|
|
*bufptr = '\0';
|
|
contlen = 0;
|
|
internal_options = 0;
|
|
}
|
|
|
|
|
|
/* Register a callback function */
|
|
|
|
static void register_callback(SV **callback, SV *function)
|
|
{
|
|
if (*callback == NULL) {
|
|
/* First time, create new SV */
|
|
*callback = newSVsv(function);
|
|
} else {
|
|
/* Been there, done that. Just overwrite the SV */
|
|
SvSetSV(*callback, function);
|
|
}
|
|
}
|
|
|
|
/* generic fwrite callback, which decides which callback to call */
|
|
static size_t
|
|
fwrite_wrapper (const void *ptr,
|
|
size_t size,
|
|
size_t nmemb,
|
|
void *stream,
|
|
void *call_function)
|
|
{
|
|
dSP;
|
|
int count, status;
|
|
SV *sv;
|
|
|
|
if (call_function) {
|
|
/* then we are doing a callback to perl */
|
|
|
|
ENTER;
|
|
SAVETMPS;
|
|
|
|
PUSHMARK(SP);
|
|
|
|
if (stream == stdout) {
|
|
sv = newSViv(0); /* FIXME: should cast stdout to GLOB somehow? */
|
|
} else { /* its already an SV */
|
|
sv = stream;
|
|
}
|
|
|
|
if (ptr != NULL) {
|
|
XPUSHs(sv_2mortal(newSVpvn(ptr, size * nmemb)));
|
|
} else {
|
|
XPUSHs(sv_2mortal(newSVpv("", 0)));
|
|
}
|
|
XPUSHs(sv_2mortal(newSVsv(sv))); /* CURLOPT_FILE SV* */
|
|
PUTBACK;
|
|
|
|
count = perl_call_sv((SV *) call_function, G_SCALAR);
|
|
|
|
SPAGAIN;
|
|
if (count != 1)
|
|
croak("Big trouble, perl_call_sv(write_callback) didn't return status\n");
|
|
|
|
status = POPi;
|
|
|
|
PUTBACK;
|
|
|
|
FREETMPS;
|
|
LEAVE;
|
|
return status;
|
|
|
|
} else {
|
|
/* default to a normal 'fwrite' */
|
|
/* stream could be a FILE * or an SV * */
|
|
FILE *f;
|
|
|
|
if (stream == stdout) { /* the only possible FILE ? Think so */
|
|
f = stream;
|
|
} else { /* its a GLOB */
|
|
f = IoIFP(sv_2io(stream)); /* may barf if not a GLOB */
|
|
}
|
|
|
|
return fwrite(ptr, size, nmemb, f);
|
|
}
|
|
}
|
|
|
|
/* Write callback for calling a perl callback */
|
|
size_t
|
|
write_callback_func( const void *ptr, size_t size,
|
|
size_t nmemb, void *stream)
|
|
{
|
|
return fwrite_wrapper(ptr,size,nmemb,stream,
|
|
write_callback);
|
|
}
|
|
|
|
/* header callback for calling a perl callback */
|
|
size_t
|
|
writeheader_callback_func( const void *ptr, size_t size,
|
|
size_t nmemb, void *stream)
|
|
{
|
|
return fwrite_wrapper(ptr,size,nmemb,stream,
|
|
header_callback);
|
|
}
|
|
|
|
size_t
|
|
read_callback_func( void *ptr, size_t size,
|
|
size_t nmemb, void *stream)
|
|
{
|
|
dSP ;
|
|
|
|
int count;
|
|
SV *sv;
|
|
STRLEN len;
|
|
size_t maxlen,mylen;
|
|
char *p;
|
|
|
|
maxlen = size*nmemb;
|
|
|
|
if (read_callback) {
|
|
/* we are doing a callback to perl */
|
|
|
|
ENTER ;
|
|
SAVETMPS ;
|
|
|
|
PUSHMARK(SP) ;
|
|
|
|
if (stream == stdin) {
|
|
sv = newSViv(0); /* should cast stdin to GLOB somehow? */
|
|
} else { /* its an SV */
|
|
sv = stream;
|
|
}
|
|
|
|
XPUSHs(sv_2mortal(newSViv(maxlen))); /* send how many bytes please */
|
|
XPUSHs(sv_2mortal(newSVsv(sv))); /* CURLOPT_INFILE SV* */
|
|
PUTBACK ;
|
|
|
|
count = perl_call_sv(read_callback, G_SCALAR);
|
|
|
|
SPAGAIN;
|
|
if (count != 1)
|
|
croak("Big trouble, perl_call_sv(read_callback) didn't return data\n");
|
|
|
|
sv = POPs;
|
|
p = SvPV(sv,len);
|
|
|
|
/* only allowed to return the number of bytes asked for */
|
|
mylen = len<maxlen ? len : maxlen;
|
|
memcpy(ptr,p,(size_t)mylen);
|
|
PUTBACK ;
|
|
|
|
FREETMPS ;
|
|
LEAVE ;
|
|
return (size_t) (mylen/size);
|
|
} else {
|
|
/* default to a normal 'fread' */
|
|
/* stream could be a FILE * or an SV * */
|
|
FILE *f;
|
|
|
|
if (stream == stdin) { /* the only possible FILE ? Think so*/
|
|
f = stream;
|
|
} else { /* its a GLOB */
|
|
f = IoIFP(sv_2io(stream)); /* may barf if not a GLOB */
|
|
}
|
|
|
|
return fread(ptr,size,nmemb,f);
|
|
}
|
|
}
|
|
|
|
/* Porgress callback for calling a perl callback */
|
|
|
|
static int progress_callback_func(void *clientp, size_t dltotal, size_t dlnow,
|
|
size_t ultotal, size_t ulnow)
|
|
{
|
|
dSP;
|
|
int count;
|
|
|
|
ENTER;
|
|
SAVETMPS;
|
|
PUSHMARK(sp);
|
|
if (clientp != NULL) {
|
|
XPUSHs(sv_2mortal(newSVpv(clientp, 0)));
|
|
} else {
|
|
XPUSHs(sv_2mortal(newSVpv("", 0)));
|
|
}
|
|
XPUSHs(sv_2mortal(newSViv(dltotal)));
|
|
XPUSHs(sv_2mortal(newSViv(dlnow)));
|
|
XPUSHs(sv_2mortal(newSViv(ultotal)));
|
|
XPUSHs(sv_2mortal(newSViv(ulnow)));
|
|
PUTBACK;
|
|
count = perl_call_sv(progress_callback, G_SCALAR);
|
|
SPAGAIN;
|
|
if (count != 1)
|
|
croak("Big trouble, perl_call_sv(progress_callback) didn't return 1\n");
|
|
count = POPi;
|
|
PUTBACK;
|
|
FREETMPS;
|
|
LEAVE;
|
|
return count;
|
|
}
|
|
|
|
|
|
/* Password callback for calling a perl callback */
|
|
|
|
static int passwd_callback_func(void *clientp, char *prompt, char *buffer,
|
|
int buflen)
|
|
{
|
|
dSP;
|
|
int count;
|
|
SV *sv;
|
|
STRLEN len;
|
|
size_t mylen;
|
|
char *p;
|
|
|
|
ENTER;
|
|
SAVETMPS;
|
|
PUSHMARK(sp);
|
|
if (clientp != NULL) {
|
|
XPUSHs(sv_2mortal(newSVsv(clientp)));
|
|
} else {
|
|
XPUSHs(sv_2mortal(newSVpv("", 0)));
|
|
}
|
|
XPUSHs(sv_2mortal(newSVpv(prompt, 0)));
|
|
XPUSHs(sv_2mortal(newSViv(buflen)));
|
|
PUTBACK;
|
|
count = perl_call_sv(passwd_callback, G_ARRAY);
|
|
SPAGAIN;
|
|
if (count != 2)
|
|
croak("Big trouble, perl_call_sv(passwd_callback) didn't return status + data\n");
|
|
|
|
sv = POPs;
|
|
count = POPi;
|
|
|
|
p = SvPV(sv,len);
|
|
|
|
/* only allowed to return the number of bytes asked for */
|
|
mylen = len<(buflen-1) ? len : (buflen-1);
|
|
memcpy(buffer,p,mylen);
|
|
buffer[buflen]=0; /* ensure C string terminates */
|
|
|
|
PUTBACK;
|
|
FREETMPS;
|
|
LEAVE;
|
|
return count;
|
|
}
|
|
|
|
|
|
#if 0
|
|
/* awaiting closepolicy prototype */
|
|
int
|
|
closepolicy_callback_func(void *clientp)
|
|
{
|
|
dSP;
|
|
int argc, status;
|
|
SV *pl_status;
|
|
|
|
ENTER;
|
|
SAVETMPS;
|
|
|
|
PUSHMARK(SP);
|
|
PUTBACK;
|
|
|
|
argc = perl_call_sv(closepolicy_callback, G_SCALAR);
|
|
SPAGAIN;
|
|
|
|
if (argc != 1) {
|
|
croak("Unexpected number of arguments returned from closefunction callback\n");
|
|
}
|
|
pl_status = POPs;
|
|
status = SvTRUE(pl_status) ? 0 : 1;
|
|
|
|
PUTBACK;
|
|
FREETMPS;
|
|
LEAVE;
|
|
|
|
return status;
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
/* Internal write callback. Only used if USE_INTERNAL_VARS was specified */
|
|
|
|
static size_t internal_write_callback(char *data, size_t size, size_t num,
|
|
FILE *fp)
|
|
{
|
|
int i;
|
|
|
|
size *= num;
|
|
if ((contlen + size) >= bufsize) {
|
|
bufsize *= 2;
|
|
contbuf = realloc(contbuf, bufsize + 1);
|
|
bufptr = contbuf + contlen;
|
|
}
|
|
contlen += size;
|
|
for (i = 0; i < size; i++) {
|
|
*bufptr++ = *data++;
|
|
}
|
|
*bufptr = '\0';
|
|
return size;
|
|
}
|
|
|
|
|
|
static int
|
|
constant(char *name, int arg)
|
|
{
|
|
errno = 0;
|
|
if (strncmp(name, "CURLINFO_", 9) == 0) {
|
|
name += 9;
|
|
switch (*name) {
|
|
case 'A':
|
|
case 'B':
|
|
case 'C':
|
|
case 'D':
|
|
if (strEQ(name, "CONNECT_TIME")) return CURLINFO_CONNECT_TIME;
|
|
break;
|
|
case 'E':
|
|
case 'F':
|
|
if (strEQ(name, "EFFECTIVE_URL")) return CURLINFO_EFFECTIVE_URL;
|
|
break;
|
|
case 'G':
|
|
case 'H':
|
|
if (strEQ(name, "HEADER_SIZE")) return CURLINFO_HEADER_SIZE;
|
|
if (strEQ(name, "HTTP_CODE")) return CURLINFO_HTTP_CODE;
|
|
break;
|
|
case 'I':
|
|
case 'J':
|
|
case 'K':
|
|
case 'L':
|
|
case 'M':
|
|
case 'N':
|
|
if (strEQ(name, "NAMELOOKUP_TIME")) return CURLINFO_NAMELOOKUP_TIME;
|
|
break;
|
|
case 'O':
|
|
case 'P':
|
|
if (strEQ(name, "PRETRANSFER_TIME")) return CURLINFO_PRETRANSFER_TIME;
|
|
break;
|
|
case 'Q':
|
|
case 'R':
|
|
if (strEQ(name, "REQUEST_SIZE")) return CURLINFO_REQUEST_SIZE;
|
|
break;
|
|
case 'S':
|
|
case 'T':
|
|
if (strEQ(name, "SIZE_DOWNLOAD")) return CURLINFO_SIZE_DOWNLOAD;
|
|
if (strEQ(name, "SIZE_UPLOAD")) return CURLINFO_SIZE_UPLOAD;
|
|
if (strEQ(name, "SPEED_DOWNLOAD")) return CURLINFO_SPEED_DOWNLOAD;
|
|
if (strEQ(name, "SPEED_UPLOAD")) return CURLINFO_SPEED_UPLOAD;
|
|
if (strEQ(name, "TOTAL_TIME")) return CURLINFO_TOTAL_TIME;
|
|
break;
|
|
case 'U':
|
|
case 'V':
|
|
case 'W':
|
|
case 'X':
|
|
case 'Y':
|
|
case 'Z':
|
|
break;
|
|
}
|
|
}
|
|
if (strncmp(name, "CURLOPT_", 8) == 0) {
|
|
name += 8;
|
|
switch (*name) {
|
|
case 'A':
|
|
case 'B':
|
|
if (strEQ(name, "AUTOREFERER")) return CURLOPT_AUTOREFERER;
|
|
break;
|
|
case 'C':
|
|
case 'D':
|
|
if (strEQ(name, "COOKIE")) return CURLOPT_COOKIE;
|
|
if (strEQ(name, "COOKIEFILE")) return CURLOPT_COOKIEFILE;
|
|
if (strEQ(name, "CRLF")) return CURLOPT_CRLF;
|
|
if (strEQ(name, "CUSTOMREQUEST")) return CURLOPT_CUSTOMREQUEST;
|
|
break;
|
|
case 'E':
|
|
case 'F':
|
|
if (strEQ(name, "ERRORBUFFER")) return CURLOPT_ERRORBUFFER;
|
|
if (strEQ(name, "FAILONERROR")) return CURLOPT_FAILONERROR;
|
|
if (strEQ(name, "FILE")) return CURLOPT_FILE;
|
|
if (strEQ(name, "FOLLOWLOCATION")) return CURLOPT_FOLLOWLOCATION;
|
|
if (strEQ(name, "FTPAPPEND")) return CURLOPT_FTPAPPEND;
|
|
if (strEQ(name, "FTPASCII")) return CURLOPT_FTPASCII;
|
|
if (strEQ(name, "FTPLISTONLY")) return CURLOPT_FTPLISTONLY;
|
|
if (strEQ(name, "FTPPORT")) return CURLOPT_FTPPORT;
|
|
break;
|
|
case 'G':
|
|
case 'H':
|
|
if (strEQ(name, "HEADER")) return CURLOPT_HEADER;
|
|
if (strEQ(name, "HEADERFUNCTION")) return CURLOPT_HEADERFUNCTION;
|
|
if (strEQ(name, "HTTPHEADER")) return CURLOPT_HTTPHEADER;
|
|
if (strEQ(name, "HTTPPOST")) return CURLOPT_HTTPPOST;
|
|
if (strEQ(name, "HTTPPROXYTUNNEL")) return CURLOPT_HTTPPROXYTUNNEL;
|
|
if (strEQ(name, "HTTPREQUEST")) return CURLOPT_HTTPREQUEST;
|
|
break;
|
|
case 'I':
|
|
case 'J':
|
|
if (strEQ(name, "INFILE")) return CURLOPT_INFILE;
|
|
if (strEQ(name, "INFILESIZE")) return CURLOPT_INFILESIZE;
|
|
if (strEQ(name, "INTERFACE")) return CURLOPT_INTERFACE;
|
|
break;
|
|
case 'K':
|
|
case 'L':
|
|
if (strEQ(name, "KRB4LEVEL")) return CURLOPT_KRB4LEVEL;
|
|
if (strEQ(name, "LOW_SPEED_LIMIT")) return CURLOPT_LOW_SPEED_LIMIT;
|
|
if (strEQ(name, "LOW_SPEED_TIME")) return CURLOPT_LOW_SPEED_TIME;
|
|
break;
|
|
case 'M':
|
|
case 'N':
|
|
if (strEQ(name, "MUTE")) return CURLOPT_MUTE;
|
|
if (strEQ(name, "NETRC")) return CURLOPT_NETRC;
|
|
if (strEQ(name, "NOBODY")) return CURLOPT_NOBODY;
|
|
if (strEQ(name, "NOPROGRESS")) return CURLOPT_NOPROGRESS;
|
|
if (strEQ(name, "NOTHING")) return CURLOPT_NOTHING;
|
|
break;
|
|
case 'O':
|
|
case 'P':
|
|
if (strEQ(name, "PASSWDDATA")) return CURLOPT_PASSWDDATA;
|
|
if (strEQ(name, "PASSWDFUNCTION")) return CURLOPT_PASSWDFUNCTION;
|
|
if (strEQ(name, "PORT")) return CURLOPT_PORT;
|
|
if (strEQ(name, "POST")) return CURLOPT_POST;
|
|
if (strEQ(name, "POSTFIELDS")) return CURLOPT_POSTFIELDS;
|
|
if (strEQ(name, "POSTFIELDSIZE")) return CURLOPT_POSTFIELDSIZE;
|
|
if (strEQ(name, "POSTQUOTE")) return CURLOPT_POSTQUOTE;
|
|
if (strEQ(name, "PROGRESSDATA")) return CURLOPT_PROGRESSDATA;
|
|
if (strEQ(name, "PROGRESSFUNCTION")) return CURLOPT_PROGRESSFUNCTION;
|
|
if (strEQ(name, "PROXY")) return CURLOPT_PROXY;
|
|
if (strEQ(name, "PROXYPORT")) return CURLOPT_PROXYPORT;
|
|
if (strEQ(name, "PROXYUSERPWD")) return CURLOPT_PROXYUSERPWD;
|
|
if (strEQ(name, "PUT")) return CURLOPT_PUT;
|
|
break;
|
|
case 'Q':
|
|
case 'R':
|
|
if (strEQ(name, "QUOTE")) return CURLOPT_QUOTE;
|
|
if (strEQ(name, "RANGE")) return CURLOPT_RANGE;
|
|
if (strEQ(name, "READFUNCTION")) return CURLOPT_READFUNCTION;
|
|
if (strEQ(name, "REFERER")) return CURLOPT_REFERER;
|
|
if (strEQ(name, "RESUME_FROM")) return CURLOPT_RESUME_FROM;
|
|
break;
|
|
case 'S':
|
|
case 'T':
|
|
if (strEQ(name, "SSLCERT")) return CURLOPT_SSLCERT;
|
|
if (strEQ(name, "SSLCERTPASSWD")) return CURLOPT_SSLCERTPASSWD;
|
|
if (strEQ(name, "SSLVERSION")) return CURLOPT_SSLVERSION;
|
|
if (strEQ(name, "STDERR")) return CURLOPT_STDERR;
|
|
if (strEQ(name, "TIMECONDITION")) return CURLOPT_TIMECONDITION;
|
|
if (strEQ(name, "TIMEOUT")) return CURLOPT_TIMEOUT;
|
|
if (strEQ(name, "TIMEVALUE")) return CURLOPT_TIMEVALUE;
|
|
if (strEQ(name, "TRANSFERTEXT")) return CURLOPT_TRANSFERTEXT;
|
|
break;
|
|
case 'U':
|
|
case 'V':
|
|
if (strEQ(name, "UPLOAD")) return CURLOPT_UPLOAD;
|
|
if (strEQ(name, "URL")) return CURLOPT_URL;
|
|
if (strEQ(name, "USERAGENT")) return CURLOPT_USERAGENT;
|
|
if (strEQ(name, "USERPWD")) return CURLOPT_USERPWD;
|
|
if (strEQ(name, "VERBOSE")) return CURLOPT_VERBOSE;
|
|
break;
|
|
case 'W':
|
|
case 'X':
|
|
case 'Y':
|
|
case 'Z':
|
|
if (strEQ(name, "WRITEFUNCTION")) return CURLOPT_WRITEFUNCTION;
|
|
if (strEQ(name, "WRITEHEADER")) return CURLOPT_WRITEHEADER;
|
|
if (strEQ(name, "WRITEINFO")) return CURLOPT_WRITEINFO;
|
|
break;
|
|
}
|
|
}
|
|
if (strEQ(name, "USE_INTERNAL_VARS")) return USE_INTERNAL_VARS;
|
|
errno = EINVAL;
|
|
return 0;
|
|
}
|
|
|
|
|
|
MODULE = Curl::easy PACKAGE = Curl::easy PREFIX = curl_easy_
|
|
|
|
int
|
|
constant(name,arg)
|
|
char * name
|
|
int arg
|
|
|
|
|
|
void *
|
|
curl_easy_init()
|
|
CODE:
|
|
init_globals();
|
|
RETVAL = curl_easy_init();
|
|
curl_easy_setopt(RETVAL, CURLOPT_HEADERFUNCTION, header_callback_func);
|
|
curl_easy_setopt(RETVAL, CURLOPT_WRITEFUNCTION, write_callback_func);
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
char *
|
|
curl_easy_version()
|
|
CODE:
|
|
RETVAL=curl_version();
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
int
|
|
curl_easy_setopt(curl, option, value)
|
|
void * curl
|
|
int option
|
|
SV * value
|
|
CODE:
|
|
if (option < CURLOPTTYPE_OBJECTPOINT) {
|
|
/* This is an option specifying an integer value: */
|
|
RETVAL = curl_easy_setopt(curl, option, (long)SvIV(value));
|
|
} else if (option == CURLOPT_FILE || option == CURLOPT_INFILE ||
|
|
option == CURLOPT_WRITEHEADER || option == CURLOPT_PROGRESSDATA ||
|
|
option == CURLOPT_PASSWDDATA) {
|
|
/* This is an option specifying an SV * value: */
|
|
RETVAL = curl_easy_setopt(curl, option, newSVsv(ST(2)));
|
|
} else if (option == CURLOPT_ERRORBUFFER) {
|
|
/* Pass in variable name for storing error messages... */
|
|
RETVAL = curl_easy_setopt(curl, option, errbuf);
|
|
if (errbufvarname) free(errbufvarname);
|
|
errbufvarname = strdup((char *)SvPV(value, PL_na));
|
|
} else if (option == CURLOPT_WRITEFUNCTION || option ==
|
|
CURLOPT_READFUNCTION || option == CURLOPT_PROGRESSFUNCTION ||
|
|
option == CURLOPT_PASSWDFUNCTION || option == CURLOPT_HEADERFUNCTION) {
|
|
/* This is an option specifying a callback function */
|
|
switch (option) {
|
|
case CURLOPT_WRITEFUNCTION:
|
|
register_callback(&write_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback_func);
|
|
break;
|
|
case CURLOPT_READFUNCTION:
|
|
register_callback(&read_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_READFUNCTION, read_callback_func);
|
|
break;
|
|
case CURLOPT_HEADERFUNCTION:
|
|
register_callback(&header_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_HEADERFUNCTION, header_callback_func);
|
|
case CURLOPT_PROGRESSFUNCTION:
|
|
register_callback(&progress_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, progress_callback_func);
|
|
break;
|
|
case CURLOPT_PASSWDFUNCTION:
|
|
register_callback(&passwd_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_PASSWDFUNCTION, passwd_callback_func);
|
|
break;
|
|
/* awaiting a prototype for the closepolicy function callback
|
|
case CURLOPT_CLOSEFUNCTION:
|
|
register_callback(&closepolicy_callback, value);
|
|
curl_easy_setopt(curl, CURLOPT_CLOSEFUNCTION, closepolicy_callback_func);
|
|
break;
|
|
*/
|
|
}
|
|
RETVAL = -1;
|
|
} else if (option == CURLOPT_HTTPHEADER || option == CURLOPT_QUOTE ||
|
|
option == CURLOPT_POSTQUOTE) {
|
|
/* This is an option specifying a list of curl_slist structs: */
|
|
AV *array = (AV *)SvRV(value);
|
|
struct curl_slist **slist = NULL;
|
|
/* We have to find out which list to use... */
|
|
switch (option) {
|
|
case CURLOPT_HTTPHEADER:
|
|
slist = &httpheader; break;
|
|
case CURLOPT_QUOTE:
|
|
slist = "e; break;
|
|
case CURLOPT_POSTQUOTE:
|
|
slist = &postquote; break;
|
|
}
|
|
/* free any previous list */
|
|
if (*slist) {
|
|
curl_slist_free_all(*slist);
|
|
*slist=NULL;
|
|
}
|
|
/* ...store the values into it... */
|
|
for (;;) {
|
|
SV *sv = av_shift(array);
|
|
int len = 0;
|
|
char *str = SvPV(sv, len);
|
|
if (len == 0) break;
|
|
*slist = curl_slist_append(*slist, str);
|
|
}
|
|
/* ...and pass the list into curl_easy_setopt() */
|
|
RETVAL = curl_easy_setopt(curl, option, *slist);
|
|
} else {
|
|
/* This is an option specifying a char * value: */
|
|
RETVAL = curl_easy_setopt(curl, option, SvPV(value, PL_na));
|
|
}
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
|
|
int
|
|
internal_setopt(option, value)
|
|
int option
|
|
int value
|
|
CODE:
|
|
if (value == 1) {
|
|
internal_options |= option;
|
|
} else {
|
|
internal_options &= !option;
|
|
}
|
|
RETVAL = 0;
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
|
|
int
|
|
curl_easy_perform(curl)
|
|
void * curl
|
|
CODE:
|
|
if (internal_options & USE_INTERNAL_VARS) {
|
|
/* Use internal callback which just stores the content into a buffer. */
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, internal_write_callback);
|
|
curl_easy_setopt(curl, CURLOPT_HEADER, 1);
|
|
}
|
|
RETVAL = curl_easy_perform(curl);
|
|
if (RETVAL && errbufvarname) {
|
|
/* If an error occurred and a varname for error messages has been
|
|
specified, store the error message. */
|
|
SV *sv = perl_get_sv(errbufvarname, TRUE | GV_ADDMULTI);
|
|
sv_setpv(sv, errbuf);
|
|
}
|
|
if (!RETVAL && (internal_options & USE_INTERNAL_VARS)) {
|
|
/* No error and internal variable for the content are to be used:
|
|
Split the data into headers and content and store them into
|
|
perl variables. */
|
|
SV *head_sv = perl_get_sv("Curl::easy::headers", TRUE | GV_ADDMULTI);
|
|
SV *cont_sv = perl_get_sv("Curl::easy::content", TRUE | GV_ADDMULTI);
|
|
char *p = contbuf;
|
|
int nl = 0, found = 0;
|
|
while (p < bufptr) {
|
|
if (nl && (*p == '\n' || *p == '\r')) {
|
|
/* found empty line, end of headers */
|
|
*p++ = '\0';
|
|
sv_setpv(head_sv, contbuf);
|
|
while (*p == '\n' || *p == '\r') {
|
|
p++;
|
|
}
|
|
sv_setpv(cont_sv, p);
|
|
found = 1;
|
|
break;
|
|
}
|
|
nl = (*p == '\n');
|
|
p++;
|
|
}
|
|
if (!found) {
|
|
sv_setpv(head_sv, "");
|
|
sv_setpv(cont_sv, contbuf);
|
|
}
|
|
}
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
|
|
int
|
|
curl_easy_getinfo(curl, option, value)
|
|
void * curl
|
|
int option
|
|
double value
|
|
CODE:
|
|
#ifdef __GNUC__
|
|
/* a(void) warnig about unnused variable */
|
|
(void) value;
|
|
#endif
|
|
switch (option & CURLINFO_TYPEMASK) {
|
|
case CURLINFO_STRING: {
|
|
char * value = (char *)SvPV(ST(2), PL_na);
|
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
|
sv_setpv(ST(2), value);
|
|
break;
|
|
}
|
|
case CURLINFO_LONG: {
|
|
long value = (long)SvIV(ST(2));
|
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
|
sv_setiv(ST(2), value);
|
|
break;
|
|
}
|
|
case CURLINFO_DOUBLE: {
|
|
double value = (double)SvNV(ST(2));
|
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
|
sv_setnv(ST(2), value);
|
|
break;
|
|
}
|
|
default: {
|
|
RETVAL = CURLE_BAD_FUNCTION_ARGUMENT;
|
|
break;
|
|
}
|
|
}
|
|
OUTPUT:
|
|
RETVAL
|
|
|
|
|
|
int
|
|
curl_easy_cleanup(curl)
|
|
void * curl
|
|
CODE:
|
|
curl_easy_cleanup(curl);
|
|
init_globals();
|
|
RETVAL = 0;
|
|
OUTPUT:
|
|
RETVAL
|
|
|