2015-01-05 21:40:24 +01:00
// Copyright (c) 2015 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
# include "httprpc.h"
# include "base58.h"
# include "chainparams.h"
# include "httpserver.h"
# include "rpcprotocol.h"
# include "rpcserver.h"
# include "random.h"
# include "sync.h"
# include "util.h"
# include "utilstrencodings.h"
# include "ui_interface.h"
2015-11-11 10:49:32 -05:00
# include "crypto/hmac_sha256.h"
# include <stdio.h>
# include "utilstrencodings.h"
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
# include <boost/algorithm/string.hpp> // boost::trim
2015-11-11 10:49:32 -05:00
# include <boost/foreach.hpp> //BOOST_FOREACH
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
/** Simple one-shot callback timer to be used by the RPC mechanism to e.g.
* re - lock the wellet .
*/
class HTTPRPCTimer : public RPCTimerBase
{
public :
2015-08-28 16:46:20 +02:00
HTTPRPCTimer ( struct event_base * eventBase , boost : : function < void ( void ) > & func , int64_t millis ) :
ev ( eventBase , false , func )
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
{
2015-08-28 16:46:20 +02:00
struct timeval tv ;
tv . tv_sec = millis / 1000 ;
tv . tv_usec = ( millis % 1000 ) * 1000 ;
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
ev . trigger ( & tv ) ;
}
private :
HTTPEvent ev ;
} ;
class HTTPRPCTimerInterface : public RPCTimerInterface
{
public :
HTTPRPCTimerInterface ( struct event_base * base ) : base ( base )
{
}
const char * Name ( )
{
return " HTTP " ;
}
2015-08-28 16:46:20 +02:00
RPCTimerBase * NewTimer ( boost : : function < void ( void ) > & func , int64_t millis )
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
{
2015-08-28 16:46:20 +02:00
return new HTTPRPCTimer ( base , func , millis ) ;
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
}
private :
struct event_base * base ;
} ;
/* Pre-base64-encoded authentication token */
static std : : string strRPCUserColonPass ;
/* Stored RPC timer interface (for unregistration) */
static HTTPRPCTimerInterface * httpRPCTimerInterface = 0 ;
static void JSONErrorReply ( HTTPRequest * req , const UniValue & objError , const UniValue & id )
{
// Send error reply from json-rpc error object
int nStatus = HTTP_INTERNAL_SERVER_ERROR ;
int code = find_value ( objError , " code " ) . get_int ( ) ;
if ( code = = RPC_INVALID_REQUEST )
nStatus = HTTP_BAD_REQUEST ;
else if ( code = = RPC_METHOD_NOT_FOUND )
nStatus = HTTP_NOT_FOUND ;
std : : string strReply = JSONRPCReply ( NullUniValue , objError , id ) ;
req - > WriteHeader ( " Content-Type " , " application/json " ) ;
req - > WriteReply ( nStatus , strReply ) ;
}
2015-11-11 10:49:32 -05:00
//This function checks username and password against -rpcauth
//entries from config file.
static bool multiUserAuthorized ( std : : string strUserPass )
{
if ( strUserPass . find ( " : " ) = = std : : string : : npos ) {
return false ;
}
std : : string strUser = strUserPass . substr ( 0 , strUserPass . find ( " : " ) ) ;
std : : string strPass = strUserPass . substr ( strUserPass . find ( " : " ) + 1 ) ;
if ( mapMultiArgs . count ( " -rpcauth " ) > 0 ) {
//Search for multi-user login/pass "rpcauth" from config
BOOST_FOREACH ( std : : string strRPCAuth , mapMultiArgs [ " -rpcauth " ] )
{
std : : vector < std : : string > vFields ;
boost : : split ( vFields , strRPCAuth , boost : : is_any_of ( " :$ " ) ) ;
if ( vFields . size ( ) ! = 3 ) {
//Incorrect formatting in config file
continue ;
}
std : : string strName = vFields [ 0 ] ;
if ( ! TimingResistantEqual ( strName , strUser ) ) {
continue ;
}
std : : string strSalt = vFields [ 1 ] ;
std : : string strHash = vFields [ 2 ] ;
unsigned int KEY_SIZE = 32 ;
unsigned char * out = new unsigned char [ KEY_SIZE ] ;
CHMAC_SHA256 ( reinterpret_cast < const unsigned char * > ( strSalt . c_str ( ) ) , strSalt . size ( ) ) . Write ( reinterpret_cast < const unsigned char * > ( strPass . c_str ( ) ) , strPass . size ( ) ) . Finalize ( out ) ;
std : : vector < unsigned char > hexvec ( out , out + KEY_SIZE ) ;
std : : string strHashFromPass = HexStr ( hexvec ) ;
if ( TimingResistantEqual ( strHashFromPass , strHash ) ) {
return true ;
}
}
}
return false ;
}
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
static bool RPCAuthorized ( const std : : string & strAuth )
{
if ( strRPCUserColonPass . empty ( ) ) // Belt-and-suspenders measure if InitRPCAuthentication was not called
return false ;
if ( strAuth . substr ( 0 , 6 ) ! = " Basic " )
return false ;
std : : string strUserPass64 = strAuth . substr ( 6 ) ;
boost : : trim ( strUserPass64 ) ;
std : : string strUserPass = DecodeBase64 ( strUserPass64 ) ;
2015-11-11 10:49:32 -05:00
//Check if authorized under single-user field
if ( TimingResistantEqual ( strUserPass , strRPCUserColonPass ) ) {
return true ;
}
return multiUserAuthorized ( strUserPass ) ;
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
}
static bool HTTPReq_JSONRPC ( HTTPRequest * req , const std : : string & )
{
// JSONRPC handles only POST
if ( req - > GetRequestMethod ( ) ! = HTTPRequest : : POST ) {
req - > WriteReply ( HTTP_BAD_METHOD , " JSONRPC server handles only POST requests " ) ;
return false ;
}
// Check authorization
std : : pair < bool , std : : string > authHeader = req - > GetHeader ( " authorization " ) ;
if ( ! authHeader . first ) {
req - > WriteReply ( HTTP_UNAUTHORIZED ) ;
return false ;
}
if ( ! RPCAuthorized ( authHeader . second ) ) {
LogPrintf ( " ThreadRPCServer incorrect password attempt from %s \n " , req - > GetPeer ( ) . ToString ( ) ) ;
/* Deter brute-forcing
If this results in a DoS the user really
shouldn ' t have their RPC port exposed . */
MilliSleep ( 250 ) ;
req - > WriteReply ( HTTP_UNAUTHORIZED ) ;
return false ;
}
JSONRequest jreq ;
try {
// Parse request
UniValue valRequest ;
if ( ! valRequest . read ( req - > ReadBody ( ) ) )
throw JSONRPCError ( RPC_PARSE_ERROR , " Parse error " ) ;
std : : string strReply ;
// singleton request
if ( valRequest . isObject ( ) ) {
jreq . parse ( valRequest ) ;
UniValue result = tableRPC . execute ( jreq . strMethod , jreq . params ) ;
// Send reply
strReply = JSONRPCReply ( result , NullUniValue , jreq . id ) ;
// array of requests
} else if ( valRequest . isArray ( ) )
strReply = JSONRPCExecBatch ( valRequest . get_array ( ) ) ;
else
throw JSONRPCError ( RPC_PARSE_ERROR , " Top-level object parse error " ) ;
req - > WriteHeader ( " Content-Type " , " application/json " ) ;
req - > WriteReply ( HTTP_OK , strReply ) ;
} catch ( const UniValue & objError ) {
JSONErrorReply ( req , objError , jreq . id ) ;
return false ;
} catch ( const std : : exception & e ) {
JSONErrorReply ( req , JSONRPCError ( RPC_PARSE_ERROR , e . what ( ) ) , jreq . id ) ;
return false ;
}
return true ;
}
static bool InitRPCAuthentication ( )
{
if ( mapArgs [ " -rpcpassword " ] = = " " )
{
LogPrintf ( " No rpcpassword set - using random cookie authentication \n " ) ;
if ( ! GenerateAuthCookie ( & strRPCUserColonPass ) ) {
uiInterface . ThreadSafeMessageBox (
_ ( " Error: A fatal internal error occurred, see debug.log for details " ) , // Same message as AbortNode
" " , CClientUIInterface : : MSG_ERROR ) ;
return false ;
}
} else {
2015-11-11 10:49:32 -05:00
LogPrintf ( " Config options rpcuser and rpcpassword will soon be deprecated. Locally-run instances may remove rpcuser to use cookie-based auth, or may be replaced with rpcauth. Please see share/rpcuser for rpcauth auth generation. " ) ;
evhttpd implementation
- *Replace usage of boost::asio with [libevent2](http://libevent.org/)*.
boost::asio is not part of C++11, so unlike other boost there is no
forwards-compatibility reason to stick with it. Together with #4738 (convert
json_spirit to UniValue), this rids Bitcoin Core of the worst offenders with
regard to compile-time slowness.
- *Replace spit-and-duct-tape http server with evhttp*. Front-end http handling
is handled by libevent, a work queue (with configurable depth and parallelism)
is used to handle application requests.
- *Wrap HTTP request in C++ class*; this makes the application code mostly
HTTP-server-neutral
- *Refactor RPC to move all http-specific code to a separate file*.
Theoreticaly this can allow building without HTTP server but with another RPC
backend, e.g. Qt's debug console (currently not implemented) or future RPC
mechanisms people may want to use.
- *HTTP dispatch mechanism*; services (e.g., RPC, REST) register which URL
paths they want to handle.
By using a proven, high-performance asynchronous networking library (also used
by Tor) and HTTP server, problems such as #5674, #5655, #344 should be avoided.
What works? bitcoind, bitcoin-cli, bitcoin-qt. Unit tests and RPC/REST tests
pass. The aim for now is everything but SSL support.
Configuration options:
- `-rpcthreads`: repurposed as "number of work handler threads". Still
defaults to 4.
- `-rpcworkqueue`: maximum depth of work queue. When this is reached, new
requests will return a 500 Internal Error.
- `-rpctimeout`: inactivity time, in seconds, after which to disconnect a
client.
- `-debug=http`: low-level http activity logging
2015-01-23 07:53:17 +01:00
strRPCUserColonPass = mapArgs [ " -rpcuser " ] + " : " + mapArgs [ " -rpcpassword " ] ;
}
return true ;
}
bool StartHTTPRPC ( )
{
LogPrint ( " rpc " , " Starting HTTP RPC server \n " ) ;
if ( ! InitRPCAuthentication ( ) )
return false ;
RegisterHTTPHandler ( " / " , true , HTTPReq_JSONRPC ) ;
assert ( EventBase ( ) ) ;
httpRPCTimerInterface = new HTTPRPCTimerInterface ( EventBase ( ) ) ;
RPCRegisterTimerInterface ( httpRPCTimerInterface ) ;
return true ;
}
void InterruptHTTPRPC ( )
{
LogPrint ( " rpc " , " Interrupting HTTP RPC server \n " ) ;
}
void StopHTTPRPC ( )
{
LogPrint ( " rpc " , " Stopping HTTP RPC server \n " ) ;
UnregisterHTTPHandler ( " / " , true ) ;
if ( httpRPCTimerInterface ) {
RPCUnregisterTimerInterface ( httpRPCTimerInterface ) ;
delete httpRPCTimerInterface ;
httpRPCTimerInterface = 0 ;
}
}