mirror of
git://git.gnupg.org/gnupg.git
synced 2024-11-04 20:38:50 +01:00
309 lines
6.2 KiB
C
309 lines
6.2 KiB
C
/* curl-shim.c - Implement a small subset of the curl API in terms of
|
|
* the iobuf HTTP API
|
|
*
|
|
* Copyright (C) 2005 Free Software Foundation, Inc.
|
|
*
|
|
* This file is part of GnuPG.
|
|
*
|
|
* GnuPG is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* GnuPG is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
|
|
* USA.
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include "http.h"
|
|
#include "util.h"
|
|
#include "ksutil.h"
|
|
#include "curl-shim.h"
|
|
|
|
static CURLcode
|
|
handle_error(CURL *curl,CURLcode err,const char *str)
|
|
{
|
|
if(curl->errorbuffer)
|
|
{
|
|
/* Make sure you never exceed CURL_ERROR_SIZE, currently set to
|
|
256 in curl-shim.h */
|
|
switch(err)
|
|
{
|
|
case CURLE_OK:
|
|
strcpy(curl->errorbuffer,"okay");
|
|
break;
|
|
|
|
case CURLE_UNSUPPORTED_PROTOCOL:
|
|
strcpy(curl->errorbuffer,"unsupported protocol");
|
|
break;
|
|
|
|
case CURLE_COULDNT_CONNECT:
|
|
strcpy(curl->errorbuffer,"couldn't connect");
|
|
break;
|
|
|
|
case CURLE_WRITE_ERROR:
|
|
strcpy(curl->errorbuffer,"write error");
|
|
break;
|
|
|
|
case CURLE_HTTP_RETURNED_ERROR:
|
|
sprintf(curl->errorbuffer,"url returned error %u",curl->status);
|
|
break;
|
|
|
|
default:
|
|
strcpy(curl->errorbuffer,"generic error");
|
|
break;
|
|
}
|
|
|
|
if(str && (strlen(curl->errorbuffer)+2+strlen(str)+1)<=CURL_ERROR_SIZE)
|
|
{
|
|
strcat(curl->errorbuffer,": ");
|
|
strcat(curl->errorbuffer,str);
|
|
}
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
CURLcode
|
|
curl_global_init(long flags)
|
|
{
|
|
return CURLE_OK;
|
|
}
|
|
|
|
void
|
|
curl_global_cleanup(void) {}
|
|
|
|
CURL *
|
|
curl_easy_init(void)
|
|
{
|
|
return calloc(1,sizeof(CURL));
|
|
}
|
|
|
|
void
|
|
curl_easy_cleanup(CURL *curl)
|
|
{
|
|
free(curl);
|
|
}
|
|
|
|
CURLcode
|
|
curl_easy_setopt(CURL *curl,CURLoption option,...)
|
|
{
|
|
va_list ap;
|
|
|
|
va_start(ap,option);
|
|
|
|
switch(option)
|
|
{
|
|
case CURLOPT_URL:
|
|
curl->url=va_arg(ap,char *);
|
|
break;
|
|
case CURLOPT_USERPWD:
|
|
curl->auth=va_arg(ap,char *);
|
|
break;
|
|
case CURLOPT_WRITEFUNCTION:
|
|
curl->writer=va_arg(ap,write_func);
|
|
break;
|
|
case CURLOPT_FILE:
|
|
curl->file=va_arg(ap,void *);
|
|
break;
|
|
case CURLOPT_ERRORBUFFER:
|
|
curl->errorbuffer=va_arg(ap,char *);
|
|
break;
|
|
case CURLOPT_PROXY:
|
|
curl->proxy=va_arg(ap,char *);
|
|
break;
|
|
case CURLOPT_POST:
|
|
curl->flags.post=va_arg(ap,unsigned int);
|
|
break;
|
|
case CURLOPT_POSTFIELDS:
|
|
curl->postfields=va_arg(ap,char *);
|
|
break;
|
|
case CURLOPT_FAILONERROR:
|
|
curl->flags.failonerror=va_arg(ap,unsigned int);
|
|
break;
|
|
default:
|
|
/* We ignore the huge majority of curl options */
|
|
break;
|
|
}
|
|
|
|
return handle_error(curl,CURLE_OK,NULL);
|
|
}
|
|
|
|
CURLcode
|
|
curl_easy_perform(CURL *curl)
|
|
{
|
|
int rc;
|
|
CURLcode err=CURLE_OK;
|
|
const char *errstr=NULL;
|
|
char *proxy=NULL;
|
|
|
|
/* Emulate the libcurl proxy behavior. If the calling program set a
|
|
proxy, use it. If it didn't set a proxy or set it to NULL, check
|
|
for one in the environment. If the calling program explicitly
|
|
set a null-string proxy, don't set a proxy at all. */
|
|
|
|
if(curl->proxy)
|
|
{
|
|
if(*curl->proxy)
|
|
proxy=curl->proxy;
|
|
}
|
|
else
|
|
proxy=getenv(HTTP_PROXY_ENV);
|
|
|
|
if(curl->flags.post)
|
|
{
|
|
rc=http_open(&curl->hd,HTTP_REQ_POST,curl->url,curl->auth,0,proxy);
|
|
if(rc==0)
|
|
{
|
|
char content_len[50];
|
|
unsigned int post_len=strlen(curl->postfields);
|
|
|
|
iobuf_writestr(curl->hd.fp_write,
|
|
"Content-Type: application/x-www-form-urlencoded\r\n");
|
|
sprintf(content_len,"Content-Length: %u\r\n",post_len);
|
|
|
|
iobuf_writestr(curl->hd.fp_write,content_len);
|
|
|
|
http_start_data(&curl->hd);
|
|
iobuf_write(curl->hd.fp_write,curl->postfields,post_len);
|
|
rc=http_wait_response(&curl->hd,&curl->status);
|
|
if(rc==0 && curl->flags.failonerror && curl->status>=300)
|
|
err=CURLE_HTTP_RETURNED_ERROR;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rc=http_open(&curl->hd,HTTP_REQ_GET,curl->url,curl->auth,0,proxy);
|
|
if(rc==0)
|
|
{
|
|
rc=http_wait_response(&curl->hd,&curl->status);
|
|
if(rc==0)
|
|
{
|
|
if(curl->flags.failonerror && curl->status>=300)
|
|
err=CURLE_HTTP_RETURNED_ERROR;
|
|
else
|
|
{
|
|
unsigned int maxlen=1024,buflen,len;
|
|
byte *line=NULL;
|
|
|
|
while((len=iobuf_read_line(curl->hd.fp_read,
|
|
&line,&buflen,&maxlen)))
|
|
{
|
|
maxlen=1024;
|
|
size_t ret;
|
|
|
|
ret=(curl->writer)(line,len,1,curl->file);
|
|
if(ret!=len)
|
|
{
|
|
err=CURLE_WRITE_ERROR;
|
|
break;
|
|
}
|
|
}
|
|
|
|
xfree(line);
|
|
http_close(&curl->hd);
|
|
}
|
|
}
|
|
else
|
|
http_close(&curl->hd);
|
|
}
|
|
}
|
|
|
|
switch(rc)
|
|
{
|
|
case 0:
|
|
break;
|
|
|
|
case G10ERR_INVALID_URI:
|
|
err=CURLE_UNSUPPORTED_PROTOCOL;
|
|
break;
|
|
|
|
case G10ERR_NETWORK:
|
|
errstr=strerror(errno);
|
|
err=CURLE_COULDNT_CONNECT;
|
|
break;
|
|
|
|
default:
|
|
errstr=g10_errstr(rc);
|
|
err=CURLE_COULDNT_CONNECT;
|
|
break;
|
|
}
|
|
|
|
return handle_error(curl,err,errstr);
|
|
}
|
|
|
|
/* This is not the same exact set that is allowed according to
|
|
RFC-2396, but it is what the real curl uses. */
|
|
#define VALID_URI_CHARS "abcdefghijklmnopqrstuvwxyz" \
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ" \
|
|
"0123456789"
|
|
|
|
char *
|
|
curl_escape(char *str,int length)
|
|
{
|
|
int len,max,idx,enc_idx=0;
|
|
char *enc;
|
|
|
|
if(length)
|
|
len=length;
|
|
else
|
|
len=strlen(str);
|
|
|
|
enc=malloc(len+1);
|
|
if(!enc)
|
|
return enc;
|
|
|
|
max=len;
|
|
|
|
for(idx=0;idx<len;idx++)
|
|
{
|
|
if(enc_idx+3>max)
|
|
{
|
|
char *tmp;
|
|
|
|
max+=100;
|
|
|
|
tmp=realloc(enc,max+1);
|
|
if(!tmp)
|
|
{
|
|
free(enc);
|
|
return NULL;
|
|
}
|
|
|
|
enc=tmp;
|
|
}
|
|
|
|
if(strchr(VALID_URI_CHARS,str[idx]))
|
|
enc[enc_idx++]=str[idx];
|
|
else
|
|
{
|
|
char numbuf[5];
|
|
sprintf(numbuf,"%%%02X",str[idx]);
|
|
strcpy(&enc[enc_idx],numbuf);
|
|
enc_idx+=3;
|
|
}
|
|
}
|
|
|
|
enc[enc_idx]='\0';
|
|
|
|
return enc;
|
|
}
|
|
|
|
void
|
|
curl_free(char *ptr)
|
|
{
|
|
free(ptr);
|
|
}
|