2006-12-21 03:57:49 +00:00
|
|
|
/*=============================================================================
|
|
|
|
client.cpp
|
|
|
|
===============================================================================
|
|
|
|
This is the C++ XML-RPC client library for Xmlrpc-c.
|
|
|
|
|
|
|
|
Note that unlike most of Xmlprc-c's C++ API, this is _not_ based on the
|
|
|
|
C client library. This code is independent of the C client library, and
|
|
|
|
is based directly on the client XML transport libraries (with a little
|
|
|
|
help from internal C utility libraries).
|
|
|
|
=============================================================================*/
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
#include <stdlib.h>
|
2006-12-21 03:57:49 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <string>
|
2008-05-23 20:56:24 +00:00
|
|
|
#include <vector>
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
#include "xmlrpc-c/girerr.hpp"
|
|
|
|
using girerr::error;
|
2008-05-23 20:56:24 +00:00
|
|
|
using girerr::throwf;
|
2006-12-21 03:57:49 +00:00
|
|
|
#include "xmlrpc-c/girmem.hpp"
|
|
|
|
using girmem::autoObjectPtr;
|
|
|
|
using girmem::autoObject;
|
2008-05-23 20:56:24 +00:00
|
|
|
#include "env_wrap.hpp"
|
2006-12-21 03:57:49 +00:00
|
|
|
#include "xmlrpc-c/base.h"
|
|
|
|
#include "xmlrpc-c/client.h"
|
|
|
|
#include "xmlrpc-c/transport.h"
|
|
|
|
#include "xmlrpc-c/base.hpp"
|
|
|
|
#include "xmlrpc-c/xml.hpp"
|
|
|
|
#include "xmlrpc-c/client.hpp"
|
2008-05-23 20:56:24 +00:00
|
|
|
#include "transport_config.h"
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace xmlrpc_c;
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
throwIfError(env_wrap const& env) {
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
class memblockStringWrapper {
|
|
|
|
|
|
|
|
public:
|
|
|
|
memblockStringWrapper(string const value) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->memblockP = XMLRPC_MEMBLOCK_NEW(char, &env.env_c, 0);
|
|
|
|
throwIfError(env);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
XMLRPC_MEMBLOCK_APPEND(char, &env.env_c, this->memblockP,
|
2006-12-21 03:57:49 +00:00
|
|
|
value.c_str(), value.size());
|
2008-05-23 20:56:24 +00:00
|
|
|
throwIfError(env);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
memblockStringWrapper(xmlrpc_mem_block * const memblockP) :
|
|
|
|
memblockP(memblockP) {};
|
|
|
|
|
|
|
|
~memblockStringWrapper() {
|
|
|
|
XMLRPC_MEMBLOCK_FREE(char, this->memblockP);
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlrpc_mem_block * memblockP;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace xmlrpc_c {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
struct client_xml_impl {
|
|
|
|
/* We have both kinds of pointers to give the user flexibility -- we
|
|
|
|
have constructors that take both. But the simple pointer
|
|
|
|
'transportP' is valid in both cases.
|
|
|
|
*/
|
|
|
|
clientXmlTransport * transportP;
|
|
|
|
clientXmlTransportPtr transportPtr;
|
|
|
|
xmlrpc_dialect dialect;
|
|
|
|
|
|
|
|
client_xml_impl(clientXmlTransport * const transportP,
|
|
|
|
xmlrpc_dialect const dialect = xmlrpc_dialect_i8) :
|
|
|
|
transportP(transportP),
|
|
|
|
dialect(dialect) {}
|
|
|
|
|
|
|
|
client_xml_impl(clientXmlTransportPtr const transportPtr,
|
|
|
|
clientXmlTransport * const transportP,
|
|
|
|
xmlrpc_dialect const dialect = xmlrpc_dialect_i8) :
|
|
|
|
transportP(transportP),
|
|
|
|
transportPtr(transportPtr),
|
|
|
|
dialect(dialect) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
carriageParm::carriageParm() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
carriageParm::~carriageParm() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
carriageParmPtr::carriageParmPtr() {
|
|
|
|
// Base class constructor will construct pointer that points to nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
carriageParmPtr::carriageParmPtr(
|
|
|
|
carriageParm * const carriageParmP) : autoObjectPtr(carriageParmP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
carriageParm *
|
|
|
|
carriageParmPtr::operator->() const {
|
|
|
|
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<carriageParm *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
carriageParm *
|
|
|
|
carriageParmPtr::get() const {
|
|
|
|
return dynamic_cast<carriageParm *>(objectP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
carriageParm_http0::carriageParm_http0() :
|
|
|
|
c_serverInfoP(NULL) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
carriageParm_http0::carriageParm_http0(string const serverUrl) {
|
|
|
|
this->c_serverInfoP = NULL;
|
|
|
|
|
|
|
|
this->instantiate(serverUrl);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
carriageParm_http0::~carriageParm_http0() {
|
|
|
|
|
|
|
|
if (this->c_serverInfoP)
|
|
|
|
xmlrpc_server_info_free(this->c_serverInfoP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::instantiate(string const serverUrl) {
|
|
|
|
|
|
|
|
if (c_serverInfoP)
|
|
|
|
throw(error("object already instantiated"));
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
this->c_serverInfoP =
|
|
|
|
xmlrpc_server_info_new(&env.env_c, serverUrl.c_str());
|
|
|
|
throwIfError(env);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
carriageParm_http0::setUser(string const userid,
|
|
|
|
string const password) {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_set_user(
|
|
|
|
&env.env_c, this->c_serverInfoP, userid.c_str(), password.c_str());
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::allowAuthBasic() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_allow_auth_basic(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::disallowAuthBasic() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_disallow_auth_basic(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::allowAuthDigest() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_allow_auth_digest(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::disallowAuthDigest() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_disallow_auth_digest(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::allowAuthNegotiate() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_allow_auth_negotiate(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::disallowAuthNegotiate() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_disallow_auth_negotiate(
|
|
|
|
&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::allowAuthNtlm() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_allow_auth_ntlm(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::disallowAuthNtlm() {
|
|
|
|
|
|
|
|
if (!this->c_serverInfoP)
|
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
xmlrpc_server_info_disallow_auth_ntlm(&env.env_c, this->c_serverInfoP);
|
|
|
|
|
|
|
|
if (env.env_c.fault_occurred)
|
|
|
|
throw(error(env.env_c.fault_string));
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
carriageParm_http0::setBasicAuth(string const username,
|
|
|
|
string const password) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (!this->c_serverInfoP)
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("object not instantiated"));
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
xmlrpc_server_info_set_basic_auth(
|
2008-05-23 20:56:24 +00:00
|
|
|
&env.env_c, this->c_serverInfoP, username.c_str(), password.c_str());
|
|
|
|
throwIfError(env);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
carriageParm_http0Ptr::carriageParm_http0Ptr() {
|
|
|
|
// Base class constructor will construct pointer that points to nothing
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
carriageParm_http0Ptr::carriageParm_http0Ptr(
|
|
|
|
carriageParm_http0 * const carriageParmP) :
|
|
|
|
carriageParmPtr(carriageParmP) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
carriageParm_http0 *
|
|
|
|
carriageParm_http0Ptr::operator->() const {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<carriageParm_http0 *>(p);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransaction::xmlTransaction() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
xmlTransaction::finish(string const& responseXml) const {
|
|
|
|
|
|
|
|
xml::trace("XML-RPC RESPONSE", responseXml);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
xmlTransaction::finishErr(error const&) const {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransactionPtr::xmlTransactionPtr() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTransactionPtr::xmlTransactionPtr(xmlTransaction * xmlTransP) :
|
|
|
|
autoObjectPtr(xmlTransP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
xmlTransaction *
|
|
|
|
xmlTransactionPtr::operator->() const {
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<xmlTransaction *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
struct xmlTranCtl {
|
|
|
|
/*----------------------------------------------------------------------------
|
|
|
|
This contains information needed to conduct a transaction. You
|
|
|
|
construct it as you start the transaction and destroy it after the
|
|
|
|
work is done. You need this only for an asynchronous one, because
|
|
|
|
where the user starts and finishes the RPC in the same
|
|
|
|
libxmlrpc_client call, you can just keep this information in
|
|
|
|
various stack variables, and it's faster and easier to understand
|
|
|
|
that way.
|
|
|
|
|
|
|
|
The C transport is designed to take a xmlrpc_call_info argument for
|
|
|
|
similar stuff needed by the the C client object. But it's really
|
|
|
|
opaque to the transport, so we just let xmlTranCtl masquerade as
|
|
|
|
xmlprc_call_info in our call to the C transport.
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
xmlTranCtl(xmlTransactionPtr const& xmlTranP,
|
|
|
|
string const& callXml) :
|
|
|
|
|
|
|
|
xmlTranP(xmlTranP) {
|
|
|
|
|
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
this->callXmlP = XMLRPC_MEMBLOCK_NEW(char, &env.env_c, 0);
|
|
|
|
throwIfError(env);
|
|
|
|
|
|
|
|
XMLRPC_MEMBLOCK_APPEND(char, &env.env_c, this->callXmlP,
|
|
|
|
callXml.c_str(), callXml.size());
|
|
|
|
throwIfError(env);
|
|
|
|
}
|
|
|
|
|
|
|
|
~xmlTranCtl() {
|
|
|
|
XMLRPC_MEMBLOCK_FREE(char, this->callXmlP);
|
|
|
|
}
|
|
|
|
|
|
|
|
xmlTransactionPtr const xmlTranP;
|
|
|
|
// The transaction we're controlling. Most notable use of this is
|
|
|
|
// that this object we inform when the transaction is done. This
|
|
|
|
// is where the response XML and other transaction results go.
|
|
|
|
|
|
|
|
xmlrpc_mem_block * callXmlP;
|
|
|
|
// The XML of the call. This is what the transport transports.
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
clientXmlTransport::~clientXmlTransport() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
clientXmlTransport::start(carriageParm * const carriageParmP,
|
|
|
|
string const& callXml,
|
|
|
|
xmlTransactionPtr const& xmlTranP) {
|
2008-05-23 20:56:24 +00:00
|
|
|
|
|
|
|
// Note: derived class clientXmlTransport_http overrides this,
|
|
|
|
// so it doesn't normally get used.
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
string responseXml;
|
|
|
|
|
|
|
|
this->call(carriageParmP, callXml, &responseXml);
|
|
|
|
|
|
|
|
xmlTranP->finish(responseXml);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2008-05-23 20:56:24 +00:00
|
|
|
clientXmlTransport::finishAsync(xmlrpc_c::timeout) {
|
|
|
|
|
|
|
|
// Since our start() does the whole thing, there's nothing for
|
|
|
|
// us to do.
|
|
|
|
|
|
|
|
// A derived class that overrides start() with something properly
|
|
|
|
// asynchronous had better also override finishAsync() with something
|
|
|
|
// substantial.
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
clientXmlTransport::asyncComplete(
|
|
|
|
struct xmlrpc_call_info * const callInfoP,
|
|
|
|
xmlrpc_mem_block * const responseXmlMP,
|
|
|
|
xmlrpc_env const transportEnv) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTranCtl * const xmlTranCtlP = reinterpret_cast<xmlTranCtl *>(callInfoP);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
if (transportEnv.fault_occurred) {
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTranCtlP->xmlTranP->finishErr(error(transportEnv.fault_string));
|
2006-12-21 03:57:49 +00:00
|
|
|
} else {
|
|
|
|
string const responseXml(
|
|
|
|
XMLRPC_MEMBLOCK_CONTENTS(char, responseXmlMP),
|
|
|
|
XMLRPC_MEMBLOCK_SIZE(char, responseXmlMP));
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTranCtlP->xmlTranP->finish(responseXml);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
} catch(error) {
|
|
|
|
/* We can't throw an error back to C code, and the async_complete
|
|
|
|
interface does not provide for failure, so we define ->finish()
|
|
|
|
as not being capable of throwing an error.
|
|
|
|
*/
|
|
|
|
assert(false);
|
|
|
|
}
|
2008-05-23 20:56:24 +00:00
|
|
|
delete(xmlTranCtlP);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
/* Ordinarily, *xmlTranCtlP is the last reference to
|
|
|
|
xmlTranCtlP->xmlTranP, so that will get destroyed too. But
|
2006-12-21 03:57:49 +00:00
|
|
|
->finish() could conceivably create a new reference to
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTranCtlP->xmlTranP, and then it would keep living.
|
2006-12-21 03:57:49 +00:00
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
clientXmlTransport::setInterrupt(int *) {
|
|
|
|
|
|
|
|
throwf("The client XML transport is not interruptible");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
clientXmlTransportPtr::clientXmlTransportPtr() {
|
|
|
|
// Base class constructor will construct pointer that points to nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
clientXmlTransportPtr::clientXmlTransportPtr(
|
|
|
|
clientXmlTransport * const transportP) : autoObjectPtr(transportP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
clientXmlTransport *
|
|
|
|
clientXmlTransportPtr::get() const {
|
|
|
|
return dynamic_cast<clientXmlTransport *>(objectP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
clientXmlTransport *
|
|
|
|
clientXmlTransportPtr::operator->() const {
|
|
|
|
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<clientXmlTransport *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
clientXmlTransport_http::~clientXmlTransport_http() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
clientXmlTransport_http::call(
|
|
|
|
carriageParm * const carriageParmP,
|
|
|
|
string const& callXml,
|
|
|
|
string * const responseXmlP) {
|
|
|
|
|
|
|
|
carriageParm_http0 * const carriageParmHttpP =
|
|
|
|
dynamic_cast<carriageParm_http0 *>(carriageParmP);
|
|
|
|
|
|
|
|
if (carriageParmHttpP == NULL)
|
|
|
|
throw(error("HTTP client XML transport called with carriage "
|
|
|
|
"parameter object not of class carriageParm_http"));
|
|
|
|
|
|
|
|
memblockStringWrapper callXmlM(callXml);
|
|
|
|
|
|
|
|
xmlrpc_mem_block * responseXmlMP;
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
|
|
|
|
|
|
|
this->c_transportOpsP->call(&env.env_c,
|
2006-12-21 03:57:49 +00:00
|
|
|
this->c_transportP,
|
|
|
|
carriageParmHttpP->c_serverInfoP,
|
|
|
|
callXmlM.memblockP,
|
|
|
|
&responseXmlMP);
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
throwIfError(env);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
memblockStringWrapper responseHolder(responseXmlMP);
|
|
|
|
// Makes responseXmlMP get freed at end of scope
|
|
|
|
|
|
|
|
*responseXmlP = string(XMLRPC_MEMBLOCK_CONTENTS(char, responseXmlMP),
|
|
|
|
XMLRPC_MEMBLOCK_SIZE(char, responseXmlMP));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
clientXmlTransport_http::start(
|
|
|
|
carriageParm * const carriageParmP,
|
|
|
|
string const& callXml,
|
|
|
|
xmlTransactionPtr const& xmlTranP) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
env_wrap env;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
carriageParm_http0 * const carriageParmHttpP =
|
|
|
|
dynamic_cast<carriageParm_http0 *>(carriageParmP);
|
|
|
|
|
|
|
|
if (carriageParmHttpP == NULL)
|
|
|
|
throw(error("HTTP client XML transport called with carriage "
|
|
|
|
"parameter object not of type carriageParm_http"));
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlTranCtl * const tranCtlP(new xmlTranCtl(xmlTranP, callXml));
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
this->c_transportOpsP->send_request(
|
2008-05-23 20:56:24 +00:00
|
|
|
&env.env_c,
|
2006-12-21 03:57:49 +00:00
|
|
|
this->c_transportP,
|
|
|
|
carriageParmHttpP->c_serverInfoP,
|
2008-05-23 20:56:24 +00:00
|
|
|
tranCtlP->callXmlP,
|
2006-12-21 03:57:49 +00:00
|
|
|
&this->asyncComplete,
|
2008-05-23 20:56:24 +00:00
|
|
|
reinterpret_cast<xmlrpc_call_info *>(tranCtlP));
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
throwIfError(env);
|
2006-12-21 03:57:49 +00:00
|
|
|
} catch (...) {
|
2008-05-23 20:56:24 +00:00
|
|
|
delete tranCtlP;
|
2006-12-21 03:57:49 +00:00
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
clientXmlTransport_http::finishAsync(xmlrpc_c::timeout const timeout) {
|
|
|
|
|
|
|
|
xmlrpc_timeoutType const c_timeoutType(
|
|
|
|
timeout.finite ? timeout_yes : timeout_no);
|
|
|
|
xmlrpc_timeout const c_timeout(timeout.duration);
|
|
|
|
|
|
|
|
this->c_transportOpsP->finish_asynch(
|
|
|
|
this->c_transportP, c_timeoutType, c_timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
clientXmlTransport_http::setInterrupt(int * const interruptP) {
|
|
|
|
|
|
|
|
if (this->c_transportOpsP->set_interrupt)
|
|
|
|
this->c_transportOpsP->set_interrupt(this->c_transportP, interruptP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool const haveCurl(
|
2006-12-21 03:57:49 +00:00
|
|
|
#if MUST_BUILD_CURL_CLIENT
|
2008-05-23 20:56:24 +00:00
|
|
|
true
|
|
|
|
#else
|
|
|
|
false
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
|
|
|
|
bool const haveLibwww(
|
|
|
|
#if MUST_BUILD_LIBWWW_CLIENT
|
|
|
|
true
|
|
|
|
#else
|
|
|
|
false
|
|
|
|
#endif
|
|
|
|
);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
bool const haveWininet(
|
|
|
|
#if MUST_BUILD_WININET_CLIENT
|
|
|
|
true
|
|
|
|
#else
|
|
|
|
false
|
|
|
|
#endif
|
|
|
|
);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
vector<string>
|
|
|
|
clientXmlTransport_http::availableTypes() {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
vector<string> retval;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (haveCurl)
|
|
|
|
retval.push_back("curl");
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (haveLibwww)
|
|
|
|
retval.push_back("libwww");
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (haveWininet)
|
|
|
|
retval.push_back("wininet");
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
return retval;
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientXmlTransportPtr
|
|
|
|
clientXmlTransport_http::create() {
|
|
|
|
/*----------------------------------------------------------------------------
|
|
|
|
Make an HTTP Client XML transport of any kind (Caller doesn't care).
|
|
|
|
|
|
|
|
Caller can find out what kind he got by trying dynamic casts.
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
Caller can use a carriageParm_http0 with the transport.
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
if (haveCurl)
|
|
|
|
return clientXmlTransportPtr(new clientXmlTransport_curl());
|
|
|
|
else if (haveLibwww)
|
|
|
|
return clientXmlTransportPtr(new clientXmlTransport_libwww());
|
|
|
|
else if (haveWininet)
|
|
|
|
return clientXmlTransportPtr(new clientXmlTransport_wininet());
|
|
|
|
else
|
|
|
|
throwf("This XML-RPC client library contains no HTTP XML transports");
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransaction::clientTransaction() {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransactionPtr::clientTransactionPtr() {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransactionPtr::clientTransactionPtr(
|
|
|
|
clientTransaction * const transP) : autoObjectPtr(transP) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransactionPtr::~clientTransactionPtr() {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransaction *
|
|
|
|
clientTransactionPtr::operator->() const {
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<clientTransaction *>(p);
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
client::~client() {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
client::start(carriageParm * const carriageParmP,
|
|
|
|
string const& methodName,
|
|
|
|
paramList const& paramList,
|
|
|
|
clientTransactionPtr const& tranP) {
|
|
|
|
/*----------------------------------------------------------------------------
|
|
|
|
Start an RPC, wait for it to complete, and finish it.
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
Usually, a derived class overrides this with something that does
|
|
|
|
not wait for the RPC to complete, but rather arranges for something
|
|
|
|
to finish the RPC later when the RPC does complete.
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
rpcOutcome outcome;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->call(carriageParmP, methodName, paramList, &outcome);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
tranP->finish(outcome);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
client::finishAsync(xmlrpc_c::timeout) {
|
|
|
|
|
|
|
|
// Since our start() does the whole thing, there's nothing for
|
|
|
|
// us to do.
|
|
|
|
|
|
|
|
// A derived class that overrides start() with something properly
|
|
|
|
// asynchronous had better also override finishAsync() with something
|
|
|
|
// substantial.
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
client::setInterrupt(int *) {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
throwf("Clients of this type are not interruptible");
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientPtr::clientPtr() {
|
|
|
|
// Base class constructor will construct pointer that points to nothing
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
clientPtr::clientPtr(
|
|
|
|
client * const clientP) : autoObjectPtr(clientP) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
client *
|
|
|
|
clientPtr::operator->() const {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<client *>(p);
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
|
|
|
|
client *
|
|
|
|
clientPtr::get() const {
|
|
|
|
return dynamic_cast<client *>(objectP);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
client_xml::client_xml(clientXmlTransport * const transportP) {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP = new client_xml_impl(transportP);
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
client_xml::client_xml(clientXmlTransportPtr const transportPtr) {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP = new client_xml_impl(transportPtr, transportPtr.get());
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
client_xml::client_xml(clientXmlTransport * const transportP,
|
|
|
|
xmlrpc_dialect const dialect) {
|
|
|
|
|
|
|
|
this->implP = new client_xml_impl(transportP, dialect);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
client_xml::client_xml(clientXmlTransportPtr const transportPtr,
|
|
|
|
xmlrpc_dialect const dialect) {
|
|
|
|
|
|
|
|
this->implP = new client_xml_impl(transportPtr, transportPtr.get(),
|
|
|
|
dialect);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
client_xml::~client_xml() {
|
|
|
|
|
|
|
|
delete(this->implP);
|
|
|
|
}
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
client_xml::call(carriageParm * const carriageParmP,
|
2008-05-23 20:56:24 +00:00
|
|
|
string const& methodName,
|
2006-12-21 03:57:49 +00:00
|
|
|
paramList const& paramList,
|
|
|
|
rpcOutcome * const outcomeP) {
|
|
|
|
|
|
|
|
string callXml;
|
|
|
|
string responseXml;
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
xml::generateCall(methodName, paramList, this->implP->dialect, &callXml);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
xml::trace("XML-RPC CALL", callXml);
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
try {
|
|
|
|
this->implP->transportP->call(carriageParmP, callXml, &responseXml);
|
|
|
|
} catch (error const& error) {
|
|
|
|
throwf("Unable to transport XML to server and "
|
|
|
|
"get XML response back. %s", error.what());
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
xml::trace("XML-RPC RESPONSE", responseXml);
|
2008-05-23 20:56:24 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
xml::parseResponse(responseXml, outcomeP);
|
|
|
|
} catch (error const& error) {
|
|
|
|
throwf("Response XML from server is not valid XML-RPC response. %s",
|
|
|
|
error.what());
|
|
|
|
}
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
client_xml::start(carriageParm * const carriageParmP,
|
2008-05-23 20:56:24 +00:00
|
|
|
string const& methodName,
|
2006-12-21 03:57:49 +00:00
|
|
|
paramList const& paramList,
|
|
|
|
clientTransactionPtr const& tranP) {
|
|
|
|
|
|
|
|
string callXml;
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
xml::generateCall(methodName, paramList, this->implP->dialect, &callXml);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
xml::trace("XML-RPC CALL", callXml);
|
|
|
|
|
|
|
|
xmlTransaction_clientPtr const xmlTranP(tranP);
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->transportP->start(carriageParmP, callXml, xmlTranP);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
client_xml::finishAsync(xmlrpc_c::timeout const timeout) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->transportP->finishAsync(timeout);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
void
|
|
|
|
client_xml::setInterrupt(int * const interruptP) {
|
|
|
|
|
|
|
|
this->implP->transportP->setInterrupt(interruptP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serverAccessor::serverAccessor(clientPtr const clientP,
|
|
|
|
carriageParmPtr const carriageParmP) :
|
|
|
|
|
|
|
|
clientP(clientP), carriageParmP(carriageParmP) {};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
serverAccessor::call(std::string const& methodName,
|
|
|
|
xmlrpc_c::paramList const& paramList,
|
|
|
|
xmlrpc_c::rpcOutcome * const outcomeP) const {
|
|
|
|
|
|
|
|
this->clientP->call(this->carriageParmP.get(),
|
|
|
|
methodName,
|
|
|
|
paramList,
|
|
|
|
outcomeP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serverAccessorPtr::serverAccessorPtr() {
|
|
|
|
// Base class constructor will construct pointer that points to nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serverAccessorPtr::serverAccessorPtr(
|
|
|
|
serverAccessor * const serverAccessorParmP) :
|
|
|
|
autoObjectPtr(serverAccessorParmP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serverAccessor *
|
|
|
|
serverAccessorPtr::operator->() const {
|
|
|
|
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<serverAccessor *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serverAccessor *
|
|
|
|
serverAccessorPtr::get() const {
|
|
|
|
return dynamic_cast<serverAccessor *>(objectP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
connection::connection(client * const clientP,
|
|
|
|
carriageParm * const carriageParmP) :
|
|
|
|
clientP(clientP), carriageParmP(carriageParmP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
connection::~connection() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
struct rpc_impl {
|
|
|
|
enum state {
|
|
|
|
STATE_UNFINISHED, // RPC is running or not started yet
|
|
|
|
STATE_ERROR, // We couldn't execute the RPC
|
|
|
|
STATE_FAILED, // RPC executed successfully, but failed per XML-RPC
|
|
|
|
STATE_SUCCEEDED // RPC is done, no exception
|
|
|
|
};
|
|
|
|
enum state state;
|
|
|
|
girerr::error * errorP; // Defined only in STATE_ERROR
|
|
|
|
rpcOutcome outcome;
|
|
|
|
// Defined only in STATE_FAILED and STATE_SUCCEEDED
|
|
|
|
string methodName;
|
|
|
|
xmlrpc_c::paramList paramList;
|
|
|
|
|
|
|
|
rpc_impl(string const& methodName,
|
|
|
|
xmlrpc_c::paramList const& paramList) :
|
|
|
|
state(STATE_UNFINISHED),
|
|
|
|
methodName(methodName),
|
|
|
|
paramList(paramList) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rpc::rpc(string const methodName,
|
|
|
|
paramList const& paramList) {
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP = new rpc_impl(methodName, paramList);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rpc::~rpc() {
|
2008-05-23 20:56:24 +00:00
|
|
|
|
|
|
|
if (this->implP->state == rpc_impl::STATE_ERROR)
|
|
|
|
delete(this->implP->errorP);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
delete(this->implP);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::call(client * const clientP,
|
|
|
|
carriageParm * const carriageParmP) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (this->implP->state != rpc_impl::STATE_UNFINISHED)
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("Attempt to execute an RPC that has already been "
|
|
|
|
"executed"));
|
|
|
|
|
|
|
|
clientP->call(carriageParmP,
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->methodName,
|
|
|
|
this->implP->paramList,
|
|
|
|
&this->implP->outcome);
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->state = this->implP->outcome.succeeded() ?
|
|
|
|
rpc_impl::STATE_SUCCEEDED : rpc_impl::STATE_FAILED;
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::call(connection const& connection) {
|
|
|
|
|
|
|
|
this->call(connection.clientP, connection.carriageParmP);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::start(client * const clientP,
|
|
|
|
carriageParm * const carriageParmP) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
if (this->implP->state != rpc_impl::STATE_UNFINISHED)
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("Attempt to execute an RPC that has already been "
|
|
|
|
"executed"));
|
|
|
|
|
|
|
|
clientP->start(carriageParmP,
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->methodName,
|
|
|
|
this->implP->paramList,
|
2006-12-21 03:57:49 +00:00
|
|
|
rpcPtr(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::start(xmlrpc_c::connection const& connection) {
|
|
|
|
|
|
|
|
this->start(connection.clientP, connection.carriageParmP);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::finish(rpcOutcome const& outcome) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->state =
|
|
|
|
outcome.succeeded() ?
|
|
|
|
rpc_impl::STATE_SUCCEEDED : rpc_impl::STATE_FAILED;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->outcome = outcome;
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
this->notifyComplete();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::finishErr(error const& error) {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
this->implP->state = rpc_impl::STATE_ERROR;
|
|
|
|
this->implP->errorP = new girerr::error(error);
|
2006-12-21 03:57:49 +00:00
|
|
|
this->notifyComplete();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
rpc::notifyComplete() {
|
|
|
|
/*----------------------------------------------------------------------------
|
|
|
|
Anyone who does RPCs asynchronously and doesn't use polling will
|
|
|
|
want to make his own class derived from 'rpc' and override this
|
|
|
|
with a notifyFinish() that does something.
|
|
|
|
|
|
|
|
Typically, notifyFinish() will queue the RPC so some other thread
|
|
|
|
will deal with the fact that the RPC is finished.
|
|
|
|
|
|
|
|
|
|
|
|
In the absence of the aforementioned queueing, the RPC becomes
|
|
|
|
unreferenced as soon as our Caller releases his reference, so the
|
|
|
|
RPC gets destroyed when we return.
|
|
|
|
-----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
value
|
|
|
|
rpc::getResult() const {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
switch (this->implP->state) {
|
|
|
|
case rpc_impl::STATE_UNFINISHED:
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("Attempt to get result of RPC that is not finished."));
|
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_ERROR:
|
|
|
|
throw(*this->implP->errorP);
|
2006-12-21 03:57:49 +00:00
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_FAILED:
|
|
|
|
throw(error("RPC response indicates failure. " +
|
|
|
|
this->implP->outcome.getFault().getDescription()));
|
2006-12-21 03:57:49 +00:00
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_SUCCEEDED: {
|
2006-12-21 03:57:49 +00:00
|
|
|
// All normal
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
return this->implP->outcome.getResult();
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fault
|
|
|
|
rpc::getFault() const {
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
switch (this->implP->state) {
|
|
|
|
case rpc_impl::STATE_UNFINISHED:
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("Attempt to get fault from RPC that is not finished"));
|
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_ERROR:
|
|
|
|
throw(*this->implP->errorP);
|
2006-12-21 03:57:49 +00:00
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_SUCCEEDED:
|
2006-12-21 03:57:49 +00:00
|
|
|
throw(error("Attempt to get fault from an RPC that succeeded"));
|
|
|
|
break;
|
2008-05-23 20:56:24 +00:00
|
|
|
case rpc_impl::STATE_FAILED: {
|
2006-12-21 03:57:49 +00:00
|
|
|
// All normal
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
return this->implP->outcome.getFault();
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
rpc::isFinished() const {
|
2008-05-23 20:56:24 +00:00
|
|
|
return (this->implP->state != rpc_impl::STATE_UNFINISHED);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
rpc::isSuccessful() const {
|
2008-05-23 20:56:24 +00:00
|
|
|
return (this->implP->state == rpc_impl::STATE_SUCCEEDED);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rpcPtr::rpcPtr() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-05-23 20:56:24 +00:00
|
|
|
rpcPtr::rpcPtr(rpc * const rpcP) : clientTransactionPtr(rpcP) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rpcPtr::rpcPtr(string const methodName,
|
2008-05-23 20:56:24 +00:00
|
|
|
xmlrpc_c::paramList const& paramList) :
|
|
|
|
clientTransactionPtr(new rpc(methodName, paramList)) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rpc *
|
|
|
|
rpcPtr::operator->() const {
|
|
|
|
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<rpc *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransaction_client::xmlTransaction_client(
|
|
|
|
clientTransactionPtr const& tranP) :
|
|
|
|
tranP(tranP) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
xmlTransaction_client::finish(string const& responseXml) const {
|
|
|
|
|
|
|
|
xml::trace("XML-RPC RESPONSE", responseXml);
|
|
|
|
|
|
|
|
try {
|
|
|
|
rpcOutcome outcome;
|
|
|
|
|
|
|
|
xml::parseResponse(responseXml, &outcome);
|
|
|
|
|
|
|
|
this->tranP->finish(outcome);
|
2008-05-23 20:56:24 +00:00
|
|
|
} catch (error const& error) {
|
|
|
|
this->tranP->finishErr(error);
|
2006-12-21 03:57:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
xmlTransaction_client::finishErr(error const& error) const {
|
|
|
|
|
|
|
|
this->tranP->finishErr(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransaction_clientPtr::xmlTransaction_clientPtr() {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransaction_clientPtr::xmlTransaction_clientPtr(
|
2008-05-23 20:56:24 +00:00
|
|
|
clientTransactionPtr const& tranP) :
|
|
|
|
xmlTransactionPtr(new xmlTransaction_client(tranP)) {}
|
2006-12-21 03:57:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
xmlTransaction_client *
|
|
|
|
xmlTransaction_clientPtr::operator->() const {
|
|
|
|
autoObject * const p(this->objectP);
|
|
|
|
return dynamic_cast<xmlTransaction_client *>(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace
|