/*============================================================================= 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). =============================================================================*/ #include #include #include #include #include "xmlrpc-c/girerr.hpp" using girerr::error; using girerr::throwf; #include "xmlrpc-c/girmem.hpp" using girmem::autoObjectPtr; using girmem::autoObject; #include "env_wrap.hpp" #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" #include "transport_config.h" using namespace std; using namespace xmlrpc_c; namespace { void throwIfError(env_wrap const& env) { if (env.env_c.fault_occurred) throw(error(env.env_c.fault_string)); } class memblockStringWrapper { public: memblockStringWrapper(string const value) { env_wrap env; this->memblockP = XMLRPC_MEMBLOCK_NEW(char, &env.env_c, 0); throwIfError(env); XMLRPC_MEMBLOCK_APPEND(char, &env.env_c, this->memblockP, value.c_str(), value.size()); throwIfError(env); } memblockStringWrapper(xmlrpc_mem_block * const memblockP) : memblockP(memblockP) {}; ~memblockStringWrapper() { XMLRPC_MEMBLOCK_FREE(char, this->memblockP); } xmlrpc_mem_block * memblockP; }; } // namespace namespace xmlrpc_c { 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) {} }; carriageParm::carriageParm() {} carriageParm::~carriageParm() {} 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(p); } carriageParm * carriageParmPtr::get() const { return dynamic_cast(objectP); } carriageParm_http0::carriageParm_http0() : c_serverInfoP(NULL) {} carriageParm_http0::carriageParm_http0(string const serverUrl) { this->c_serverInfoP = NULL; this->instantiate(serverUrl); } 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")); env_wrap env; this->c_serverInfoP = xmlrpc_server_info_new(&env.env_c, serverUrl.c_str()); throwIfError(env); } void carriageParm_http0::setUser(string const userid, string const password) { if (!this->c_serverInfoP) throw(error("object not instantiated")); 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)); } void carriageParm_http0::setBasicAuth(string const username, string const password) { if (!this->c_serverInfoP) throw(error("object not instantiated")); env_wrap env; xmlrpc_server_info_set_basic_auth( &env.env_c, this->c_serverInfoP, username.c_str(), password.c_str()); throwIfError(env); } carriageParm_http0Ptr::carriageParm_http0Ptr() { // Base class constructor will construct pointer that points to nothing } carriageParm_http0Ptr::carriageParm_http0Ptr( carriageParm_http0 * const carriageParmP) : carriageParmPtr(carriageParmP) {} carriageParm_http0 * carriageParm_http0Ptr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } xmlTransaction::xmlTransaction() {} void xmlTransaction::finish(string const& responseXml) const { xml::trace("XML-RPC RESPONSE", responseXml); } void xmlTransaction::finishErr(error const&) const { } xmlTransactionPtr::xmlTransactionPtr() {} xmlTransactionPtr::xmlTransactionPtr(xmlTransaction * xmlTransP) : autoObjectPtr(xmlTransP) {} xmlTransaction * xmlTransactionPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } 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. }; clientXmlTransport::~clientXmlTransport() {} void clientXmlTransport::start(carriageParm * const carriageParmP, string const& callXml, xmlTransactionPtr const& xmlTranP) { // Note: derived class clientXmlTransport_http overrides this, // so it doesn't normally get used. string responseXml; this->call(carriageParmP, callXml, &responseXml); xmlTranP->finish(responseXml); } void 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. } void clientXmlTransport::asyncComplete( struct xmlrpc_call_info * const callInfoP, xmlrpc_mem_block * const responseXmlMP, xmlrpc_env const transportEnv) { xmlTranCtl * const xmlTranCtlP = reinterpret_cast(callInfoP); try { if (transportEnv.fault_occurred) { xmlTranCtlP->xmlTranP->finishErr(error(transportEnv.fault_string)); } else { string const responseXml( XMLRPC_MEMBLOCK_CONTENTS(char, responseXmlMP), XMLRPC_MEMBLOCK_SIZE(char, responseXmlMP)); xmlTranCtlP->xmlTranP->finish(responseXml); } } 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); } delete(xmlTranCtlP); /* Ordinarily, *xmlTranCtlP is the last reference to xmlTranCtlP->xmlTranP, so that will get destroyed too. But ->finish() could conceivably create a new reference to xmlTranCtlP->xmlTranP, and then it would keep living. */ } 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(objectP); } clientXmlTransport * clientXmlTransportPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } clientXmlTransport_http::~clientXmlTransport_http() {} void clientXmlTransport_http::call( carriageParm * const carriageParmP, string const& callXml, string * const responseXmlP) { carriageParm_http0 * const carriageParmHttpP = dynamic_cast(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; env_wrap env; this->c_transportOpsP->call(&env.env_c, this->c_transportP, carriageParmHttpP->c_serverInfoP, callXmlM.memblockP, &responseXmlMP); throwIfError(env); 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) { env_wrap env; carriageParm_http0 * const carriageParmHttpP = dynamic_cast(carriageParmP); if (carriageParmHttpP == NULL) throw(error("HTTP client XML transport called with carriage " "parameter object not of type carriageParm_http")); xmlTranCtl * const tranCtlP(new xmlTranCtl(xmlTranP, callXml)); try { this->c_transportOpsP->send_request( &env.env_c, this->c_transportP, carriageParmHttpP->c_serverInfoP, tranCtlP->callXmlP, &this->asyncComplete, reinterpret_cast(tranCtlP)); throwIfError(env); } catch (...) { delete tranCtlP; 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); } 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( #if MUST_BUILD_CURL_CLIENT true #else false #endif ); bool const haveLibwww( #if MUST_BUILD_LIBWWW_CLIENT true #else false #endif ); bool const haveWininet( #if MUST_BUILD_WININET_CLIENT true #else false #endif ); vector clientXmlTransport_http::availableTypes() { vector retval; if (haveCurl) retval.push_back("curl"); if (haveLibwww) retval.push_back("libwww"); if (haveWininet) retval.push_back("wininet"); return retval; } 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. 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"); } clientTransaction::clientTransaction() {} clientTransactionPtr::clientTransactionPtr() {} clientTransactionPtr::clientTransactionPtr( clientTransaction * const transP) : autoObjectPtr(transP) {} clientTransactionPtr::~clientTransactionPtr() {} clientTransaction * clientTransactionPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } client::~client() {} 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. 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; this->call(carriageParmP, methodName, paramList, &outcome); tranP->finish(outcome); } 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. } void client::setInterrupt(int *) { throwf("Clients of this type are not interruptible"); } clientPtr::clientPtr() { // Base class constructor will construct pointer that points to nothing } clientPtr::clientPtr( client * const clientP) : autoObjectPtr(clientP) {} client * clientPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } client * clientPtr::get() const { return dynamic_cast(objectP); } client_xml::client_xml(clientXmlTransport * const transportP) { this->implP = new client_xml_impl(transportP); } client_xml::client_xml(clientXmlTransportPtr const transportPtr) { this->implP = new client_xml_impl(transportPtr, transportPtr.get()); } 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); } void client_xml::call(carriageParm * const carriageParmP, string const& methodName, paramList const& paramList, rpcOutcome * const outcomeP) { string callXml; string responseXml; xml::generateCall(methodName, paramList, this->implP->dialect, &callXml); xml::trace("XML-RPC CALL", callXml); 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()); } xml::trace("XML-RPC RESPONSE", responseXml); try { xml::parseResponse(responseXml, outcomeP); } catch (error const& error) { throwf("Response XML from server is not valid XML-RPC response. %s", error.what()); } } void client_xml::start(carriageParm * const carriageParmP, string const& methodName, paramList const& paramList, clientTransactionPtr const& tranP) { string callXml; xml::generateCall(methodName, paramList, this->implP->dialect, &callXml); xml::trace("XML-RPC CALL", callXml); xmlTransaction_clientPtr const xmlTranP(tranP); this->implP->transportP->start(carriageParmP, callXml, xmlTranP); } void client_xml::finishAsync(xmlrpc_c::timeout const timeout) { this->implP->transportP->finishAsync(timeout); } 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(p); } serverAccessor * serverAccessorPtr::get() const { return dynamic_cast(objectP); } connection::connection(client * const clientP, carriageParm * const carriageParmP) : clientP(clientP), carriageParmP(carriageParmP) {} connection::~connection() {} 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) { this->implP = new rpc_impl(methodName, paramList); } rpc::~rpc() { if (this->implP->state == rpc_impl::STATE_ERROR) delete(this->implP->errorP); delete(this->implP); } void rpc::call(client * const clientP, carriageParm * const carriageParmP) { if (this->implP->state != rpc_impl::STATE_UNFINISHED) throw(error("Attempt to execute an RPC that has already been " "executed")); clientP->call(carriageParmP, this->implP->methodName, this->implP->paramList, &this->implP->outcome); this->implP->state = this->implP->outcome.succeeded() ? rpc_impl::STATE_SUCCEEDED : rpc_impl::STATE_FAILED; } void rpc::call(connection const& connection) { this->call(connection.clientP, connection.carriageParmP); } void rpc::start(client * const clientP, carriageParm * const carriageParmP) { if (this->implP->state != rpc_impl::STATE_UNFINISHED) throw(error("Attempt to execute an RPC that has already been " "executed")); clientP->start(carriageParmP, this->implP->methodName, this->implP->paramList, rpcPtr(this)); } void rpc::start(xmlrpc_c::connection const& connection) { this->start(connection.clientP, connection.carriageParmP); } void rpc::finish(rpcOutcome const& outcome) { this->implP->state = outcome.succeeded() ? rpc_impl::STATE_SUCCEEDED : rpc_impl::STATE_FAILED; this->implP->outcome = outcome; this->notifyComplete(); } void rpc::finishErr(error const& error) { this->implP->state = rpc_impl::STATE_ERROR; this->implP->errorP = new girerr::error(error); 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 { switch (this->implP->state) { case rpc_impl::STATE_UNFINISHED: throw(error("Attempt to get result of RPC that is not finished.")); break; case rpc_impl::STATE_ERROR: throw(*this->implP->errorP); break; case rpc_impl::STATE_FAILED: throw(error("RPC response indicates failure. " + this->implP->outcome.getFault().getDescription())); break; case rpc_impl::STATE_SUCCEEDED: { // All normal } } return this->implP->outcome.getResult(); } fault rpc::getFault() const { switch (this->implP->state) { case rpc_impl::STATE_UNFINISHED: throw(error("Attempt to get fault from RPC that is not finished")); break; case rpc_impl::STATE_ERROR: throw(*this->implP->errorP); break; case rpc_impl::STATE_SUCCEEDED: throw(error("Attempt to get fault from an RPC that succeeded")); break; case rpc_impl::STATE_FAILED: { // All normal } } return this->implP->outcome.getFault(); } bool rpc::isFinished() const { return (this->implP->state != rpc_impl::STATE_UNFINISHED); } bool rpc::isSuccessful() const { return (this->implP->state == rpc_impl::STATE_SUCCEEDED); } rpcPtr::rpcPtr() {} rpcPtr::rpcPtr(rpc * const rpcP) : clientTransactionPtr(rpcP) {} rpcPtr::rpcPtr(string const methodName, xmlrpc_c::paramList const& paramList) : clientTransactionPtr(new rpc(methodName, paramList)) {} rpc * rpcPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(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); } catch (error const& error) { this->tranP->finishErr(error); } } void xmlTransaction_client::finishErr(error const& error) const { this->tranP->finishErr(error); } xmlTransaction_clientPtr::xmlTransaction_clientPtr() {} xmlTransaction_clientPtr::xmlTransaction_clientPtr( clientTransactionPtr const& tranP) : xmlTransactionPtr(new xmlTransaction_client(tranP)) {} xmlTransaction_client * xmlTransaction_clientPtr::operator->() const { autoObject * const p(this->objectP); return dynamic_cast(p); } } // namespace