Sindbad~EG File Manager
/*
curlapi.c
cURL plugin.
gSOAP XML Web services tools
Copyright (C) 2000-2017, Robert van Engelen, Genivia Inc., All Rights Reserved.
This part of the software is released under one of the following licenses:
GPL or the gSOAP public license.
--------------------------------------------------------------------------------
gSOAP public license.
The contents of this file are subject to the gSOAP Public License Version 1.3
(the "License"); you may not use this file except in compliance with the
License. You may obtain a copy of the License at
http://www.cs.fsu.edu/~engelen/soaplicense.html
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
for the specific language governing rights and limitations under the License.
The Initial Developer of the Original Code is Robert A. van Engelen.
Copyright (C) 2000-2017, Robert van Engelen, Genivia Inc., All Rights Reserved.
--------------------------------------------------------------------------------
GPL license.
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 of the License, 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., 59 Temple
Place, Suite 330, Boston, MA 02111-1307 USA
Author contact information:
engelen@genivia.com / engelen@acm.org
This program is released under the GPL with the additional exemption that
compiling, linking, and/or using OpenSSL is allowed.
--------------------------------------------------------------------------------
A commercial use license is available from Genivia, Inc., contact@genivia.com
--------------------------------------------------------------------------------
*/
/**
@mainpage The CURL plugin
[TOC]
@section curl_0 Overview
The CURL plugin for gSOAP provides a bridge for the gSOAP engine to use libcurl
for internet communications. While gSOAP provides a full HTTP stack, libcurl
can be used to support additional protocols and features by replacing gSOAP's
HTTP stack.
@section curl_1 CURL plugin setup
To use the CURL plugin:
-# Add `#include "plugin/curlapi.h"` to your client-side code and compile your
code together with `plugin/curlapi.c`. Link your code with libcurl.
-# Add `curl_global_init(CURL_GLOBAL_ALL)` at the start of your program to
initialize CURL. Add `curl_global_cleanup()` at the end of your program.
-# In your source code where you create a `soap` context, register the plugin
with this `soap` context, Or use the `soap` member of a soapcpp2-generated
C++ proxy class. Use `soap_register_plugin(soap, soap_curl)` to register.
-# Alternatively, if you have a `CURL *curl` handle already set up, then
register the plugin with `soap_register_plugin_arg(soap, soap_curl, curl)`.
The benefit of this is that you can set CURL options of the handle. Do not
delete this handle until the `soap` context is deleted.
-# If you register multiple other plugins with the context, you should register
the CURL plugin always first.
The plugin is not limited to SOAP calls, you can use it with XML REST and JSON
in gSOAP. The plugin registry steps are the same for any client-side API
service calls.
The CURL plugin supports SOAP with MTOM attachments, including streaming MTOM.
Other plugins can be combined with this plugin, such as WSSE for WS-Security.
@note The CURL plugin increases the overhead of HTTP calls compared to the
gSOAP HTTP stack. The overhead is due to buffering of the entire outbound
message before sending and buffering of the entire message received. By
contrast, gSOAP uses a streaming approach and only buffers the socket
communications to (de)serialize XML directly into C/C++ data.
@section curl_2 Configuration and settings
To use the CURL plugin, register the plugin with the current `soap` context
using `soap_register_plugin(soap, soap_curl)`. This also creates a new CURL
handle that is internally used by the plugin until the `soap` context is
deleted. For C++ proxy classes generated with soapcpp2, register the plugin
with the `soap` member of the proxy class.
The gSOAP HTTP chunked transfer mode `SOAP_IO_CHUNK` and timeout settings are
also used by the CURL plugin, when set, as follows:
@code
#include "plugin/curlapi.h"
...
struct soap *soap;
curl_global_init(CURL_GLOBAL_ALL);
soap = soap_new1(SOAP_IO_CHUNK | SOAP_XML_INDENT);
soap_register_plugin(soap, soap_curl);
soap->connect_timeout = 60; // 1 minute
soap->send_timeout = 10; // 10 seconds
soap->recv_timeout = 10; // 10 seconds
soap->transfer_timeout = 20; // 20 seconds
...
// client program runs
...
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
curl_global_cleanup();
@endcode
It is strongly recommended to set timeouts. The timeout values specified here
are just examples. Actual values depend on the application's performance
characteristics.
HTTP proxy settings are used by the CURL plugin. You can specify the HTTP proxy
settings `soap->proxy_host` and `soap->proxy_port` with the HTTP proxy host
and port, respectively, and specify the HTTP proxy access credentials
`soap->proxy_userid` and `soap->proxy_passwd`.
Also compression is used by the CURL plugin when enabled with `SOAP_ENC_ZLIB`:
@code
#include "plugin/curlapi.h"
...
struct soap *soap;
curl_global_init(CURL_GLOBAL_ALL);
soap = soap_new1(SOAP_IO_CHUNK | SOAP_ENC_ZLIB | SOAP_XML_INDENT);
soap_register_plugin(soap, soap_curl);
...
// client program runs
...
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
curl_global_cleanup();
@endcode
When an transmission error occurs, use `soap_curl_reset(soap)` to reset the
plugin. This ensures that the gSOAP IO operations are reset and will behave
again normally.
Alternatively, you can create your own `CURL *curl` handle, configure it, and
pass it to the plugin as follows:
@code
#include "plugin/curlapi.h"
...
struct soap *soap;
CURL *curl;
curl_global_init(CURL_GLOBAL_ALL);
curl = curl_easy_init();
curl_easy_setopt(data->curl, CURLOPT_CONNECTTIMEOUT, 60L);
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT, 10L);
soap = soap_new1(SOAP_XML_INDENT);
soap_register_plugin_arg(soap, soap_curl, curl);
...
// client program runs
...
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
curl_easy_cleanup(curl);
...
curl_global_cleanup();
@endcode
Note that C++ proxy classes generated by soapcpp2 with option `-j` have a
`soap` member that should be used to register the plugin with:
@code
#include "plugin/curlapi.h"
...
Proxy proxy(SOAP_XML_INDENT);
CURL *curl;
curl_global_init(CURL_GLOBAL_ALL);
curl = curl_easy_init();
curl_easy_setopt(data->curl, CURLOPT_CONNECTTIMEOUT, 60L);
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT, 10L);
soap_register_plugin_arg(proxy.soap, soap_curl, curl);
...
// make calls with the proxy object
...
proxy.destroy();
curl_easy_cleanup(curl);
...
curl_global_cleanup();
@endcode
@section curl_3 SOAP client example
This example shows a calculator client application with CURL and gSOAP.
The soapcpp2 command is applied to `calc.h` with `soapcpp2 -c -CL calc.h`, where
`calc.h` is:
@code
//gsoap ns service name: calc Simple calculator service described at https://www.genivia.com/dev.html
//gsoap ns service protocol: SOAP
//gsoap ns service style: rpc
//gsoap ns service encoding: encoded
//gsoap ns service namespace: http://websrv.cs.fsu.edu/~engelen/calc.wsdl
//gsoap ns service location: http://websrv.cs.fsu.edu/~engelen/calcserver.cgi
//gsoap ns schema namespace: urn:calc
//gsoap ns service method: add Sums two values
int ns__add(double a, double b, double *result);
//gsoap ns service method: sub Subtracts two values
int ns__sub(double a, double b, double *result);
//gsoap ns service method: mul Multiplies two values
int ns__mul(double a, double b, double *result);
//gsoap ns service method: div Divides two values
int ns__div(double a, double b, double *result);
//gsoap ns service method: pow Raises a to b
int ns__pow(double a, double b, double *result);
@endcode
This generates `soapStub.h`, `soapH.h`, `soapC.c`, `soapClient.c`, and
`calc.nsmap`.
To keep this example small, the main program uses the calculator service to add
two values:
@code
#include "soapH.h"
#include "calc.nsmap"
#include "plugin/curlapi.h"
const char server[] = "http://websrv.cs.fsu.edu/~engelen/calcserver.cgi";
int main(int argc, char **argv)
{
struct soap *soap = soap_new1(SOAP_XML_INDENT);
double result;
curl_global_init(CURL_GLOBAL_ALL);
soap_register_plugin(soap, soap_curl);
if (soap_call_ns__add(soap, server, "", 2.0, 3.0, &result))
{
soap_print_fault(soap, stderr);
soap_curl_reset(soap);
}
else
printf("2 +3 = %g\n", result);
soap_destroy(soap);
soap_end(soap);
soap_free(soap);
curl_global_cleanup();
return 0;
}
@endcode
We compile this example program together with `stdsoap2.c`, `soapC.c`,
`soapClient.c`, `plugin/curlapi.c` and we link it with libcurl.
As stated previously, to use a current `CURL *curl` handle that you have
created, use `soap_register_plugin_arg(soap, soap_curl, curl)` to register the
plugin.
@section curl_4 JSON REST example
See the gSOAP [JSON documentation](https://www.genivia.com/doc/xml-rpc-json/html/index.html)
for details about using JSON with gSOAP in C and in C++.
A JSON client in C with CURL has the following outline:
@code
#include "plugin/curlapi.h"
#include "json.h"
struct Namespace namespaces[] = {{NULL,NULL,NULL,NULL}};
...
struct soap *ctx = soap_new1(SOAP_C_UTFSTRING | SOAP_XML_INDENT);
struct value *request = new_value(ctx);
struct value response;
curl_global_init(CURL_GLOBAL_ALL);
soap_register_plugin(ctx, soap_curl);
... // here we populate the request data to send
if (json_call(ctx, "endpoint URL", request, &response))
{
soap_print_fault(ctx, stderr);
soap_curl_reset(ctx);
}
else
{
... // use the response data here
}
soap_destroy(ctx); // delete objects
soap_end(ctx); // delete data
... // here we can make other calls etc.
soap_free(ctx); // delete the context
curl_global_cleanup();
@endcode
As stated previously, to use a current `CURL *curl` handle that you have
created, use `soap_register_plugin_arg(soap, soap_curl, curl)` to register the
plugin.
JSON in C++ is similar to the C example shown with the benefit of the
easy-to-use [JSON C++ API](https://www.genivia.com/doc/xml-rpc-json/html/index.html#cpp).
*/
#include "curlapi.h"
#ifdef __cplusplus
extern "C" {
#endif
/** Plugin identification for plugin registry */
const char soap_curl_id[] = SOAP_CURL_ID;
/******************************************************************************\
*
* Static protos
*
\******************************************************************************/
static int soap_curl_init(struct soap *soap, struct soap_curl_data *data, CURL *curl);
static void soap_curl_delete(struct soap *soap, struct soap_plugin *p);
static int soap_curl_connect_callback(struct soap *soap, const char *endpoint, const char *host, int port);
static int soap_curl_send_callback(struct soap *soap, const char *buf, size_t len);
static int soap_curl_prepare_init_recv_callback(struct soap *soap);
static int soap_curl_prepare_final_recv_callback(struct soap *soap);
static size_t soap_curl_recv_callback(struct soap *soap, char *buf, size_t size);
static size_t soap_curl_read_callback(void *buffer, size_t size, size_t nitems, void *ptr);
static size_t soap_curl_write_callback(void *buffer, size_t size, size_t nitems, void *ptr);
/******************************************************************************\
*
* Plugin registry functions
*
\******************************************************************************/
/**
@fn int soap_curl(struct soap *soap, struct soap_plugin *p, void *arg)
@brief Plugin registry function, used with soap_register_plugin and soap_register_plugin_arg.
@param soap context
@param[in,out] p plugin created in registry
@param[in] arg passed from soap_register_plugin_arg
@return SOAP_OK
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_curl(struct soap *soap, struct soap_plugin *p, void *arg)
{
DBGFUN("soap_curl");
p->id = soap_curl_id;
p->data = (void*)SOAP_MALLOC(soap, sizeof(struct soap_curl_data));
p->fcopy = NULL;
p->fdelete = soap_curl_delete;
if (!p->data)
return SOAP_EOM;
if (soap_curl_init(soap, (struct soap_curl_data*)p->data, (CURL*)arg))
{
SOAP_FREE(soap, p->data);
return SOAP_EOM;
}
return SOAP_OK;
}
/**
@fn int soap_curl_init(struct soap *soap, struct soap_wsa_data *data)
@brief Initializes plugin data.
@param soap context
@param[in,out] data plugin data
@return SOAP_OK
*/
static int soap_curl_init(struct soap *soap, struct soap_curl_data *data, CURL *curl)
{
DBGFUN("soap_curl_init");
data->soap = soap;
data->curl = curl;
data->own = (curl == NULL);
data->active = 0;
data->hdr = NULL;
data->blk = NULL;
data->ptr = NULL;
data->lst = NULL;
data->mode = soap->omode;
*data->buf = '\0';
soap->omode &= ~SOAP_IO;
soap->omode |= SOAP_IO_BUFFER;
soap->omode |= SOAP_ENC_PLAIN;
data->fconnect = soap->fconnect;
soap->fconnect = soap_curl_connect_callback;
data->fsend = soap->fsend;
soap->fsend = soap_curl_send_callback;
data->frecv = soap->frecv;
soap->frecv = soap_curl_recv_callback;
data->fprepareinitrecv = soap->fprepareinitrecv;
soap->fprepareinitrecv = soap_curl_prepare_init_recv_callback;
data->fpreparefinalrecv = soap->fpreparefinalrecv;
soap->fpreparefinalrecv = soap_curl_prepare_final_recv_callback;
return SOAP_OK;
}
/**
@fn void soap_curl_delete(struct soap *soap, struct soap_plugin *p)
@brief Deletes plugin data.
@param soap context
@param[in,out] p plugin
*/
static void soap_curl_delete(struct soap *soap, struct soap_plugin *p)
{
(void)soap;
struct soap_curl_data *data = (struct soap_curl_data*)p->data;
DBGFUN("soap_curl_delete");
if (data->lst)
soap_end_block(soap, data->lst);
if (data->curl && data->own)
curl_easy_cleanup(data->curl);
soap->fsend = data->fsend;
soap->frecv = data->frecv;
soap->fprepareinitrecv = data->fprepareinitrecv;
soap->fpreparefinalrecv = data->fpreparefinalrecv;
SOAP_FREE(soap, data);
}
/******************************************************************************\
*
* Plugin API calls
*
\******************************************************************************/
/**
@fn int soap_curl_reset(struct soap *soap)
@brief Reset the plugin so gSOAP IO behaves normally. This is an optional API call, not required except when serializing data after an error.
@param soap context
*/
SOAP_FMAC1
void
SOAP_FMAC2
soap_curl_reset(struct soap *soap)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
DBGFUN("soap_curl_reset");
if (data)
{
if (data->lst)
soap_end_block(soap, data->lst);
data->lst = NULL;
data->active = 0;
}
}
/******************************************************************************\
*
* Callbacks registered by plugin
*
\******************************************************************************/
/**
@fn int soap_curl_connect_callback(struct soap *soap, const char *endpoint, const char *host, int port)
@brief The fconnect callback invokes this function to override connecting to an endpoint.
@param soap context
@param endpoint URL to connect to, use "" if the CURL handle has a URL assigned with CURLOPT_URL
@param host not used
@param port not used
@return SOAP_OK or error code
*/
static int soap_curl_connect_callback(struct soap *soap, const char *endpoint, const char *host, int port)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
(void)host; (void)port;
DBGFUN1("soap_curl_connect_callback", "endpoint=%s", endpoint);
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
if (!data->curl) /* no CURL handle passed to soap_register_plugin() */
{
data->curl = curl_easy_init(); /* so set up our own */
if (!data->curl)
return soap->error = SOAP_EOM;
data->own = 1;
}
if (data->hdr)
curl_slist_free_all(data->hdr);
data->hdr = NULL;
data->blk = NULL;
data->ptr = NULL;
if (data->lst)
soap_end_block(soap, data->lst);
data->lst = NULL;
if (endpoint && *endpoint) /* if endpoint != "" then use it, otherwise use CURL's */
curl_easy_setopt(data->curl, CURLOPT_URL, endpoint);
curl_easy_setopt(data->curl, CURLOPT_USERAGENT, SOAP_CURL_ID);
if (soap->status == SOAP_POST || soap->status == SOAP_POST_FILE)
curl_easy_setopt(data->curl, CURLOPT_POST, 1L);
else if (soap->status == SOAP_GET)
curl_easy_setopt(data->curl, CURLOPT_HTTPGET, 1L);
else if (soap->status == SOAP_PUT)
curl_easy_setopt(data->curl, CURLOPT_PUT, 1L);
else if (soap->status == SOAP_DEL)
curl_easy_setopt(data->curl, CURLOPT_CUSTOMREQUEST, "DELETE");
if (soap->status == SOAP_POST || soap->status == SOAP_POST_FILE || soap->status == SOAP_PUT)
{
if (soap_http_content_type(soap, SOAP_OK))
{
(void)soap_memmove(soap->tmpbuf+14, sizeof(soap->tmpbuf), soap->tmpbuf, sizeof(soap->tmpbuf)-14);
soap->tmpbuf[sizeof(soap->tmpbuf)-1] = '\0';
(void)soap_memcpy(soap->tmpbuf, sizeof(soap->tmpbuf), "Content-Type: ", 14);
data->hdr = curl_slist_append(data->hdr, soap->tmpbuf);
if (!data->hdr)
return soap->error = SOAP_EOM;
}
if (soap->action)
{
(SOAP_SNPRINTF(soap->tmpbuf, sizeof(soap->tmpbuf), strlen(soap->action) + 14), "SOAPAction: \"%s\"", soap->action);
data->hdr = curl_slist_append(data->hdr, soap->tmpbuf);
if (!data->hdr)
return soap->error = SOAP_EOM;
}
if (soap->http_extra_header)
{
const char *header = soap->http_extra_header;
const char *next;
soap->http_extra_header = NULL; /* use http_extra_header once (assign new value before each call) */
while ((next = strstr(header, "\r\n")) != NULL)
{
if (next > header && next < header + sizeof(soap->tmpbuf))
{
soap_strncpy(soap->tmpbuf, sizeof(soap->tmpbuf), header, next - header);
if (*soap->tmpbuf)
{
data->hdr = curl_slist_append(data->hdr, soap->tmpbuf);
if (!data->hdr)
return soap->error = SOAP_EOM;
}
}
header = next + 2;
}
if (*header)
{
data->hdr = curl_slist_append(data->hdr, header);
if (!data->hdr)
return soap->error = SOAP_EOM;
}
}
if (data->hdr)
curl_easy_setopt(data->curl, CURLOPT_HTTPHEADER, (void*)data->hdr);
curl_easy_setopt(data->curl, CURLOPT_READFUNCTION, soap_curl_read_callback);
curl_easy_setopt(data->curl, CURLOPT_READDATA, (void*)data);
}
curl_easy_setopt(data->curl, CURLOPT_WRITEFUNCTION, soap_curl_write_callback);
curl_easy_setopt(data->curl, CURLOPT_WRITEDATA, (void*)data);
curl_easy_setopt(data->curl, CURLOPT_ERRORBUFFER, data->buf);
curl_easy_setopt(data->curl, CURLOPT_NOSIGNAL, 1L);
if (soap->connect_timeout > 0)
curl_easy_setopt(data->curl, CURLOPT_CONNECTTIMEOUT, (long)soap->connect_timeout);
else if (soap->connect_timeout < 0)
curl_easy_setopt(data->curl, CURLOPT_CONNECTTIMEOUT_MS, -(long)soap->connect_timeout/1000);
if (soap->transfer_timeout > 0)
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT, (long)soap->transfer_timeout);
else if (soap->send_timeout > 0)
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT, (long)soap->send_timeout);
else if (soap->send_timeout < 0)
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT_MS, -(long)soap->send_timeout/1000);
else if (soap->recv_timeout > 0)
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT, (long)soap->recv_timeout);
else if (soap->recv_timeout < 0)
curl_easy_setopt(data->curl, CURLOPT_TIMEOUT_MS, -(long)soap->recv_timeout/1000);
if (soap->proxy_host)
{
(SOAP_SNPRINTF(soap->tmpbuf, sizeof(soap->tmpbuf), strlen(soap->proxy_host) + 8 + 8), "http://%s:%d", soap->proxy_host,soap->proxy_port);
curl_easy_setopt(data->curl, CURLOPT_PROXY, (void*)soap->tmpbuf);
if (soap->proxy_userid)
{
size_t len_proxy_userid = strlen(soap->proxy_userid);
size_t len_proxy_passwd = soap->proxy_passwd ? strlen(soap->proxy_passwd) : 0;
if ((len_proxy_userid + len_proxy_passwd) * 3 + 2 < sizeof(soap->tmpbuf))
{
size_t pos = 0;
size_t i;
for (i = 0; i < len_proxy_userid; i++)
{
(SOAP_SNPRINTF(soap->tmpbuf + pos, sizeof(soap->tmpbuf) - pos, 4), "%%%02x", (int)((unsigned char)soap->proxy_userid[i]));
pos += 3;
}
soap->tmpbuf[pos] = ':';
pos++;
for (i = 0; i < len_proxy_passwd; i++)
{
(SOAP_SNPRINTF(soap->tmpbuf + pos, sizeof(soap->tmpbuf) - pos, 4), "%%%02x", (int)((unsigned char)soap->proxy_passwd[i]));
pos += 3;
}
soap->tmpbuf[pos] = '\0';
curl_easy_setopt(data->curl, CURLOPT_PROXYUSERPWD, (void*)soap->tmpbuf);
}
}
}
soap->omode &= ~SOAP_IO; /* reset IO modes */
soap->omode |= SOAP_IO_BUFFER; /* buffer the output */
soap->omode |= SOAP_ENC_PLAIN; /* no HTTP headers */
soap->omode &= ~SOAP_ENC_ZLIB;
/* store data sent by engine in an isolated blist */
if (!(data->lst = soap_alloc_block(soap)))
return soap->error;
soap->blist = soap->blist->next;
/* activate callbacks */
data->active = 1;
return SOAP_OK;
}
/**
@fn int soap_curl_send_callback(struct soap *soap, const char *buf, size_t len)
@brief The fsend callback invokes this function to override sending data by saving in a blist.
@param soap context
@param buf data to send
@param len number of bytes to send
@return SOAP_OK or error code
*/
static int soap_curl_send_callback(struct soap *soap, const char *buf, size_t len)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
char *blk;
DBGFUN1("soap_curl_send_callback", "len=%zu", len);
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
if (!data->active)
return data->fsend(soap, buf, len);
if (!data->curl || !data->lst)
return soap->error = SOAP_PLUGIN_ERROR;
if (len > 0)
{
blk = (char*)soap_push_block(soap, data->lst, len);
if (!blk)
return soap->error;
(void)soap_memcpy((void*)blk, len, (const void*)buf, len);
}
return SOAP_OK;
}
/**
@fn int soap_curl_prepare_init_recv_callback(struct soap *soap)
@brief The fprepareinitrecv callback invokes this function to override the start of receiving data and ending of sending.
@param soap context
@return SOAP_OK or error code
*/
static int soap_curl_prepare_init_recv_callback(struct soap *soap)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
long status;
const char *s = NULL;
CURLcode res;
DBGFUN("soap_curl_prepare_init_recv_callback");
if (!data || !data->curl)
return soap->error = SOAP_PLUGIN_ERROR;
if (!data->active || !data->lst)
{
if (data->fprepareinitrecv)
return data->fprepareinitrecv(soap);
return SOAP_OK;
}
if ((data->mode & SOAP_IO) == SOAP_IO_CHUNK)
{
/* HTTP chunking mode was set with soap_init1() */
data->hdr = curl_slist_append(data->hdr, "Transfer-Encoding: chunked");
if (!data->hdr)
{
soap->error = SOAP_EOM;
return 0;
}
curl_easy_setopt(data->curl, CURLOPT_HTTPHEADER, data->hdr);
}
else
{
/* content length is the size of the message saved */
curl_easy_setopt(data->curl, CURLOPT_POSTFIELDSIZE, data->lst->size);
}
if ((data->mode & SOAP_ENC_ZLIB))
{
/* enable all supported built-in compressions */
curl_easy_setopt(data->curl, CURLOPT_ACCEPT_ENCODING, "");
}
data->hdr = curl_slist_append(data->hdr, "Expect:"); /* remove Expect: 100 */
if (data->hdr)
{
curl_easy_setopt(data->curl, CURLOPT_HTTPHEADER, (void*)data->hdr);
res = curl_easy_perform(data->curl);
curl_slist_free_all(data->hdr);
data->hdr = NULL;
if (res != CURLE_OK)
return soap_sender_fault(soap, curl_easy_strerror(res), "origin: soap_curl plugin");
}
curl_easy_getinfo(data->curl, CURLINFO_RESPONSE_CODE, &status);
if (!curl_easy_getinfo(data->curl, CURLINFO_CONTENT_TYPE, &s) && s)
soap->http_content = soap_strdup(soap, s);
DBGLOG(TEST, SOAP_MESSAGE(fdebug, "cURL HTTP response code %ld content type %s\n", status, s ? s : ""));
soap->status = (int)status;
if ((soap->status >= 200 && soap->status <= 299) /* OK, Accepted, etc */
|| soap->status == 400 /* Bad Request */
|| soap->status == 500) /* Internal Server Error */
{
if (data->fprepareinitrecv)
return data->fprepareinitrecv(soap);
return SOAP_OK;
}
/* read HTTP body for error details and return HTTP error */
return soap_set_receiver_error(soap, "HTTP Error", soap_http_get_body(soap, NULL), soap->status);
}
/**
@fn int soap_curl_prepare_final_recv_callback(struct soap *soap)
@brief The fpreparefinalrecv callback resets the recv callback.
@param soap context
@return SOAP_OK or error code
*/
static int soap_curl_prepare_final_recv_callback(struct soap *soap)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
DBGFUN("soap_curl_prepare_final_recv_callback");
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
/* deactivate callbacks */
data->active = 0;
if (data->fpreparefinalrecv)
return data->fpreparefinalrecv(soap);
return SOAP_OK;
}
/**
@fn size_t soap_curl_recv_callback(struct soap *soap, char *buf, size_t size)
@brief The frecv callback invokes this function to override receiving data that is stored in a blist.
@param soap context
@param buf receive in this buffer
@param size buffer size
@return number of bytes read, 0 for end or error
*/
static size_t soap_curl_recv_callback(struct soap *soap, char *buf, size_t size)
{
struct soap_curl_data *data = (struct soap_curl_data*)soap_lookup_plugin(soap, soap_curl_id);
size_t len;
DBGFUN1("soap_curl_recv_callback", "size=%zu", size);
if (!data)
{
soap->error = SOAP_PLUGIN_ERROR;
return 0;
}
if (!data->active)
return data->frecv(soap, buf, size);
if (!data->lst)
{
long status;
curl_easy_getinfo(data->curl, CURLINFO_RESPONSE_CODE, &status);
DBGLOG(TEST, SOAP_MESSAGE(fdebug, "cURL HTTP response code %ld\n", status));
soap->error = (int)status;
return 0;
}
if (!data->blk)
{
data->ptr = data->blk = soap_first_block(data->soap, data->lst);
if (!data->blk)
{
soap_end_block(soap, data->lst);
data->lst = NULL;
return 0;
}
}
len = soap_block_size(soap, data->lst) - (data->ptr - data->blk);
if (len > size)
len = size;
(void)soap_memcpy((void*)buf, size, (const void*)data->ptr, len);
data->ptr += len;
if (data->ptr >= data->blk + soap_block_size(data->soap, data->lst))
{
data->ptr = data->blk = soap_next_block(data->soap, data->lst);
if (!data->blk)
{
soap_end_block(soap, data->lst);
data->lst = NULL;
}
}
soap->length += len;
return len;
}
/******************************************************************************\
*
* CURL callbacks
*
\******************************************************************************/
/**
@fn size_t soap_curl_read_callback(void *buffer, size_t size, size_t nitems, void *ptr)
@brief The CURL read callback invokes this function to read data to send.
@param buffer read data into this buffer, buffer is size*nitems large.
@param size
@param nitems
@param ptr points to soap_curl_data plugin data
@return number of bytes read, 0 for end or error
*/
static size_t soap_curl_read_callback(void *buffer, size_t size, size_t nitems, void *ptr)
{
struct soap_curl_data *data = (struct soap_curl_data*)ptr;
struct soap *soap = data->soap;
size_t len;
DBGFUN2("soap_curl_read_callback", "size=%zu", size, "nitems=%zu", nitems);
if (!data->lst)
return 0;
if (!data->blk)
{
data->ptr = data->blk = soap_first_block(soap, data->lst);
if (!data->blk)
{
soap_end_block(soap, data->lst);
data->lst = NULL;
return 0;
}
}
len = soap_block_size(soap, data->lst) - (data->ptr - data->blk);
if (len > size * nitems)
len = size * nitems;
(void)soap_memcpy((void*)buffer, size * nitems, (const void*)data->ptr, len);
data->ptr += len;
if (data->ptr >= data->blk + soap_block_size(soap, data->lst))
{
data->ptr = data->blk = soap_next_block(soap, data->lst);
if (!data->blk)
{
soap_end_block(soap, data->lst);
data->lst = NULL;
}
}
return len;
}
/**
@fn size_t soap_curl_write_callback(void *buffer, size_t size, size_t nitems, void *ptr)
@brief The CURL write callback invokes this function to write data that was received.
@param buffer data to write of size*nitems bytes total.
@param size
@param nitems
@param ptr points to soap_curl_data plugin data
@return number of bytes written, 0 for error
*/
static size_t soap_curl_write_callback(void *buffer, size_t size, size_t nitems, void *ptr)
{
struct soap_curl_data *data = (struct soap_curl_data*)ptr;
struct soap *soap = data->soap;
size_t len = size * nitems;
char *s;
DBGFUN2("soap_curl_write_callback", "size=%zu", size, "nitems=%zu", nitems);
if (!data->lst)
{
/* store data received in an isolated blist */
if (!(data->lst = soap_alloc_block(soap)))
return 0;
soap->blist = soap->blist->next;
}
s = (char*)soap_push_block(soap, data->lst, len);
if (!s)
return 0;
(void)soap_memcpy((void*)s, len, buffer, len);
return len;
}
#ifdef __cplusplus
}
#endif
Sindbad File Manager Version 1.0, Coded By Sindbad EG ~ The Terrorists