gambas-source-code/gb.net.curl/src/CCurl.c
Benoît Minisini dd3b512daf [CONFIGURATION]
* NEW: Update copyrights.


git-svn-id: svn://localhost/gambas/trunk@4384 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2011-12-31 02:39:20 +00:00

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
};