dd3b512daf
* NEW: Update copyrights. git-svn-id: svn://localhost/gambas/trunk@4384 867c0c6c-44f3-4631-809d-bfa615b0a4ec
558 lines
11 KiB
C
558 lines
11 KiB
C
/***************************************************************************
|
|
|
|
CCurl.c
|
|
|
|
(c) 2003-2008 Daniel Campos Fernández <dcamposf@gmail.com>
|
|
|
|
This program 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, or (at your option)
|
|
any later version.
|
|
|
|
This program 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.
|
|
|
|
***************************************************************************/
|
|
|
|
#define __CCURL_C
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
|
|
#include <curl/curl.h>
|
|
#include <curl/easy.h>
|
|
#include <curl/multi.h>
|
|
|
|
#include "main.h"
|
|
#include "gambas.h"
|
|
#include "CCurl.h"
|
|
#include "CProxy.h"
|
|
|
|
|
|
DECLARE_EVENT (EVENT_FINISHED);
|
|
DECLARE_EVENT (EVENT_ERROR);
|
|
DECLARE_EVENT (EVENT_CONNECT);
|
|
DECLARE_EVENT (EVENT_READ);
|
|
|
|
/*****************************************************
|
|
CURLM : a pointer to use curl_multi interface,
|
|
allowing asynchrnous work without using threads
|
|
in this class.
|
|
******************************************************/
|
|
CURLM *CCURL_multicurl;
|
|
int CCURL_pipe[2] = {-1, -1};
|
|
|
|
/******************************************************
|
|
Events from this class
|
|
******************************************************/
|
|
GB_STREAM_DESC CurlStream =
|
|
{
|
|
open: CCURL_stream_open,
|
|
close: CCURL_stream_close,
|
|
read: CCURL_stream_read,
|
|
write: CCURL_stream_write,
|
|
seek: CCURL_stream_seek,
|
|
tell: CCURL_stream_tell,
|
|
flush: CCURL_stream_flush,
|
|
eof: CCURL_stream_eof,
|
|
lof: CCURL_stream_lof,
|
|
handle: CCURL_stream_handle,
|
|
};
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
// STREAM //
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
/* not allowed stream methods */
|
|
|
|
int CCURL_stream_handle(GB_STREAM *stream) { return 0;}
|
|
int CCURL_stream_open(GB_STREAM *stream, const char *path, int mode, void *data){return -1;}
|
|
int CCURL_stream_seek(GB_STREAM *stream, int64_t pos, int whence){ return -1;}
|
|
int CCURL_stream_tell(GB_STREAM *stream, int64_t *pos){return -1; }
|
|
int CCURL_stream_flush(GB_STREAM *stream) { return 0;}
|
|
int CCURL_stream_close(GB_STREAM *stream) { return -1;}
|
|
int CCURL_stream_write(GB_STREAM *stream, char *buffer, int len){return -1;}
|
|
|
|
int CCURL_stream_lof(GB_STREAM *stream, int64_t *len)
|
|
{
|
|
void *_object = STREAM_TO_OBJECT(stream);
|
|
|
|
*len = 0;
|
|
|
|
if ((THIS_STATUS != NET_RECEIVING_DATA ) && (THIS_STATUS != NET_INACTIVE))
|
|
return -1;
|
|
|
|
*len = GB.StringLength(THIS->data);
|
|
return 0;
|
|
}
|
|
|
|
int CCURL_stream_eof(GB_STREAM *stream)
|
|
{
|
|
void *_object = STREAM_TO_OBJECT(stream);
|
|
|
|
if ((THIS_STATUS != NET_RECEIVING_DATA ) && (THIS_STATUS != NET_INACTIVE)) return -1;
|
|
if (!GB.StringLength(THIS->data)) return -1;
|
|
return 0;
|
|
}
|
|
|
|
int CCURL_stream_read(GB_STREAM *stream, char *buffer, int len)
|
|
{
|
|
void *_object = STREAM_TO_OBJECT(stream);
|
|
int len_data;
|
|
char *new_data;
|
|
|
|
if ((THIS_STATUS != NET_RECEIVING_DATA ) && (THIS_STATUS != NET_INACTIVE)) return -1;
|
|
|
|
len_data = GB.StringLength(THIS->data);
|
|
|
|
if (len_data < len) return -1;
|
|
|
|
memcpy(buffer, THIS->data, len);
|
|
|
|
len_data -= len;
|
|
|
|
if (len_data > 0)
|
|
new_data = GB.NewString(THIS->data + len, len_data);
|
|
else
|
|
new_data = NULL;
|
|
|
|
GB.FreeString(&THIS->data);
|
|
THIS->data = new_data;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void raise_event(void *_object, int event)
|
|
{
|
|
GB.Raise(THIS, event, 0);
|
|
GB.Unref(POINTER(&_object));
|
|
}
|
|
|
|
void CURL_raise_finished(void *_object)
|
|
{
|
|
raise_event(THIS, EVENT_FINISHED);
|
|
}
|
|
|
|
void CURL_raise_error(void *_object)
|
|
{
|
|
raise_event(THIS, EVENT_ERROR);
|
|
}
|
|
|
|
void CURL_raise_connect(void *_object)
|
|
{
|
|
raise_event(THIS, EVENT_CONNECT);
|
|
}
|
|
|
|
void CURL_raise_read(void *_object)
|
|
{
|
|
raise_event(THIS, EVENT_READ);
|
|
}
|
|
|
|
void CURL_manage_error(void *_object, int error)
|
|
{
|
|
if (THIS_FILE)
|
|
{
|
|
fclose(THIS_FILE);
|
|
THIS_FILE = NULL;
|
|
}
|
|
|
|
switch (error)
|
|
{
|
|
case CURLE_OK:
|
|
|
|
if (THIS->async)
|
|
{
|
|
#if DEBUG
|
|
fprintf(stderr, "-- [%p] curl_multi_remove_handle(%p)\n", THIS, THIS_CURL);
|
|
#endif
|
|
curl_multi_remove_handle(CCURL_multicurl,THIS_CURL);
|
|
}
|
|
GB.Ref(THIS);
|
|
GB.Post(CURL_raise_finished, (intptr_t)THIS);
|
|
CURL_stop(THIS);
|
|
THIS_STATUS = NET_INACTIVE;
|
|
break;
|
|
|
|
default:
|
|
|
|
if (THIS->async)
|
|
{
|
|
#if DEBUG
|
|
fprintf(stderr, "-- [%p] curl_multi_remove_handle(%p)\n", THIS, THIS_CURL);
|
|
#endif
|
|
curl_multi_remove_handle(CCURL_multicurl,THIS_CURL);
|
|
}
|
|
GB.Ref(THIS);
|
|
GB.Post(CURL_raise_error, (intptr_t)THIS);
|
|
CURL_stop(THIS);
|
|
THIS_STATUS = (- (1000 + error));
|
|
break;
|
|
}
|
|
|
|
if (THIS->async)
|
|
GB.Unref(POINTER(&_object));
|
|
}
|
|
|
|
void CURL_init_stream(void *_object)
|
|
{
|
|
THIS->stream.desc = &CurlStream;
|
|
THIS->stream.tag = THIS;
|
|
}
|
|
|
|
/***************************************************************
|
|
This CallBack is called each event loop by Gambas to test
|
|
the status of curl descriptors
|
|
***************************************************************/
|
|
|
|
static void stop_post()
|
|
{
|
|
if (CCURL_pipe[0] < 0) return;
|
|
|
|
GB.Watch (CCURL_pipe[0], GB_WATCH_NONE, NULL, 0);
|
|
close(CCURL_pipe[0]);
|
|
close(CCURL_pipe[1]);
|
|
CCURL_pipe[0]=-1;
|
|
}
|
|
|
|
static void CCURL_post_curl(intptr_t data)
|
|
{
|
|
CURLMsg *Msg;
|
|
int nread;
|
|
int post=1;
|
|
void *_object;
|
|
char *tmp;
|
|
struct timespec mywait;
|
|
|
|
do
|
|
{
|
|
mywait.tv_sec=0;
|
|
mywait.tv_nsec=1000000;
|
|
nanosleep(&mywait,NULL);
|
|
}
|
|
while(CURLM_CALL_MULTI_PERFORM == curl_multi_perform(CCURL_multicurl,&nread));
|
|
|
|
if (!nread) post=0;
|
|
|
|
do
|
|
{
|
|
Msg=curl_multi_info_read(CCURL_multicurl,&nread);
|
|
if (!Msg) nread=0;
|
|
if (Msg)
|
|
{
|
|
curl_easy_getinfo(Msg->easy_handle,CURLINFO_PRIVATE,&tmp);
|
|
_object=(void*)tmp;
|
|
CURL_manage_error(THIS,Msg->data.result);
|
|
}
|
|
}
|
|
while (nread);
|
|
|
|
if (!post)
|
|
stop_post();
|
|
}
|
|
|
|
void CURL_stop(void *_object)
|
|
{
|
|
if (THIS_FILE)
|
|
{
|
|
fclose(THIS_FILE);
|
|
THIS_FILE = NULL;
|
|
}
|
|
|
|
if (THIS_CURL)
|
|
{
|
|
#if DEBUG
|
|
fprintf(stderr, "-- [%p] curl_multi_remove_handle(%p)\n", THIS, THIS_CURL);
|
|
#endif
|
|
curl_multi_remove_handle(CCURL_multicurl,THIS_CURL);
|
|
#if DEBUG
|
|
fprintf(stderr, "-- [%p] curl_easycleanup(%p)\n", THIS, THIS_CURL);
|
|
#endif
|
|
curl_easy_cleanup(THIS_CURL);
|
|
THIS_CURL = NULL;
|
|
}
|
|
|
|
THIS_STATUS = NET_INACTIVE;
|
|
}
|
|
|
|
static void Curl_init_post(void)
|
|
{
|
|
if (CCURL_pipe[0]!=-1) return;
|
|
|
|
if (pipe(CCURL_pipe))
|
|
{
|
|
fprintf(stderr, "gb.net.curl: warning: unable to create the client watching pipe: %s\n", strerror(errno));
|
|
return;
|
|
}
|
|
|
|
GB.Watch (CCURL_pipe[0], GB_WATCH_READ, CCURL_post_curl, 0);
|
|
if (write(CCURL_pipe[1], "1", sizeof(char)) != 1)
|
|
fprintf(stderr, "gb.net.curl: warning: unable to write to the client watching pipe: %s\n", strerror(errno));
|
|
}
|
|
|
|
void CURL_start_post(void *_object)
|
|
{
|
|
Curl_init_post();
|
|
curl_multi_add_handle(CCURL_multicurl, THIS_CURL);
|
|
GB.Ref(THIS);
|
|
}
|
|
|
|
bool CURL_check_active(void *_object)
|
|
{
|
|
if (THIS_STATUS > 0)
|
|
{
|
|
GB.Error("Property is read-only while client is active");
|
|
return TRUE;
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
BEGIN_PROPERTY(Curl_User)
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnString(THIS->user.user);
|
|
else
|
|
{
|
|
if (CURL_check_active(THIS))
|
|
return;
|
|
|
|
GB.StoreString(PROP(GB_STRING), &(THIS->user.user));
|
|
}
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_Async)
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnBoolean(THIS->async);
|
|
else
|
|
{
|
|
if (CURL_check_active(THIS))
|
|
return;
|
|
|
|
THIS->async = VPROP(GB_BOOLEAN);
|
|
}
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_Timeout)
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnInteger(THIS->timeout);
|
|
else
|
|
{
|
|
int timeout;
|
|
|
|
if (CURL_check_active(THIS))
|
|
return;
|
|
|
|
timeout = VPROP(GB_INTEGER);
|
|
if (timeout < 0)
|
|
timeout = 0;
|
|
|
|
THIS->timeout = timeout;
|
|
}
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_Password)
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnString(THIS->user.pwd);
|
|
else
|
|
{
|
|
if (CURL_check_active(THIS))
|
|
return;
|
|
|
|
GB.StoreString(PROP(GB_STRING), &(THIS->user.pwd));
|
|
}
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_Status)
|
|
|
|
GB.ReturnInteger(THIS_STATUS);
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_ErrorText)
|
|
|
|
if (THIS_STATUS >= 0)
|
|
GB.ReturnVoidString();
|
|
else
|
|
GB.ReturnConstZeroString(curl_easy_strerror((-THIS_STATUS) - 1000));
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_PROPERTY(Curl_URL)
|
|
|
|
char *url, *tmp;
|
|
char *protocol;
|
|
|
|
if (READ_PROPERTY)
|
|
{
|
|
GB.ReturnString(THIS_URL);
|
|
return;
|
|
}
|
|
|
|
if (CURL_check_active(THIS))
|
|
return;
|
|
|
|
if (PLENGTH() == 0)
|
|
goto UNKNOWN_PROTOCOL;
|
|
|
|
url = GB.NewString(PSTRING(), PLENGTH());
|
|
|
|
if (GB.Is(THIS, GB.FindClass("FtpClient")))
|
|
{
|
|
protocol = CURL_get_protocol(url, "ftp://");
|
|
if (strcmp(protocol, "ftp://"))
|
|
goto UNKNOWN_PROTOCOL;
|
|
}
|
|
else if (GB.Is(THIS, GB.FindClass("HttpClient")))
|
|
{
|
|
protocol = CURL_get_protocol(url, "http://");
|
|
if (strcmp(protocol, "http://") && strcmp(protocol, "https://"))
|
|
goto UNKNOWN_PROTOCOL;
|
|
}
|
|
else
|
|
goto UNKNOWN_PROTOCOL;
|
|
|
|
if (strncmp(url, protocol, strlen(protocol)))
|
|
{
|
|
tmp = GB.NewZeroString(protocol);
|
|
tmp = GB.AddString(tmp, url, GB.StringLength(url));
|
|
GB.FreeString(&url);
|
|
url = tmp;
|
|
}
|
|
|
|
GB.FreeString(&THIS_URL);
|
|
THIS_URL = url;
|
|
return;
|
|
|
|
UNKNOWN_PROTOCOL:
|
|
|
|
GB.Error("Unknown protocol");
|
|
|
|
END_PROPERTY
|
|
|
|
BEGIN_METHOD_VOID(Curl_new)
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "Curl_new: %p\n", THIS);
|
|
#endif
|
|
|
|
CURL_user_init(&THIS->user);
|
|
CURL_proxy_init(&THIS->proxy.proxy);
|
|
|
|
THIS->proxy.parent_status = (int*)&THIS_STATUS;
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(Curl_free)
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "Curl_free: %p\n", THIS);
|
|
#endif
|
|
|
|
GB.FreeString(&THIS_URL);
|
|
|
|
if (THIS_FILE)
|
|
fclose(THIS_FILE);
|
|
|
|
if (THIS_CURL)
|
|
{
|
|
#if DEBUG
|
|
fprintf(stderr, "-- [%p] curl_easy_cleanup(%p)\n", THIS, THIS_CURL);
|
|
#endif
|
|
curl_easy_cleanup(THIS_CURL);
|
|
}
|
|
|
|
CURL_user_clear(&THIS->user);
|
|
CURL_proxy_clear(&THIS->proxy.proxy);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(Curl_init)
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "-- curl_multi_init()\n");
|
|
#endif
|
|
CCURL_multicurl = curl_multi_init();
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(Curl_exit)
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "-- curl_multi_cleanup()\n");
|
|
#endif
|
|
curl_multi_cleanup(CCURL_multicurl);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(Curl_Peek)
|
|
|
|
GB.ReturnString(THIS->data);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_PROPERTY(Curl_Debug)
|
|
|
|
if (READ_PROPERTY)
|
|
GB.ReturnBoolean(THIS->debug);
|
|
else
|
|
THIS->debug = VPROP(GB_BOOLEAN);
|
|
|
|
END_PROPERTY
|
|
|
|
//*************************************************************************
|
|
//#################### GAMBAS INTERFACE ###################################
|
|
//*************************************************************************
|
|
GB_DESC CCurlDesc[] =
|
|
{
|
|
GB_DECLARE("Curl", sizeof(CCURL)), GB_NOT_CREATABLE(),
|
|
|
|
GB_INHERITS("Stream"),
|
|
|
|
GB_STATIC_METHOD("_init", NULL, Curl_init, NULL),
|
|
GB_STATIC_METHOD("_exit", NULL, Curl_exit, NULL),
|
|
|
|
GB_METHOD("_new", NULL, Curl_new, NULL),
|
|
GB_METHOD("_free", NULL, Curl_free, NULL),
|
|
GB_METHOD("Peek","s", Curl_Peek, NULL),
|
|
|
|
GB_PROPERTY("URL", "s", Curl_URL),
|
|
GB_PROPERTY("User", "s", Curl_User),
|
|
GB_PROPERTY("Password", "s", Curl_Password),
|
|
GB_PROPERTY("Async", "b", Curl_Async),
|
|
GB_PROPERTY("Timeout", "i", Curl_Timeout),
|
|
GB_PROPERTY_SELF("Proxy", ".Curl.Proxy"),
|
|
GB_PROPERTY_READ("Status", "i", Curl_Status),
|
|
GB_PROPERTY_READ("ErrorText", "s", Curl_ErrorText),
|
|
GB_PROPERTY("Debug", "b", Curl_Debug),
|
|
|
|
GB_EVENT("Finished", NULL, NULL, &EVENT_FINISHED),
|
|
GB_EVENT("Connect", NULL, NULL, &EVENT_CONNECT),
|
|
GB_EVENT("Read", NULL, NULL, &EVENT_READ),
|
|
GB_EVENT("Error", NULL,NULL, &EVENT_ERROR),
|
|
|
|
GB_END_DECLARE
|
|
};
|
|
|