http_api: Implement http_api_connection class to serve RPC over HTTP
This commit is contained in:
parent
fe7eed6beb
commit
f461dee432
1 changed files with 119 additions and 0 deletions
119
include/fc/rpc/http_api.hpp
Normal file
119
include/fc/rpc/http_api.hpp
Normal file
|
|
@ -0,0 +1,119 @@
|
|||
#pragma once
|
||||
#include <fc/io/json.hpp>
|
||||
#include <fc/network/http/connection.hpp>
|
||||
#include <fc/network/http/server.hpp>
|
||||
#include <fc/reflect/variant.hpp>
|
||||
#include <fc/rpc/api_connection.hpp>
|
||||
#include <fc/rpc/state.hpp>
|
||||
|
||||
namespace fc { namespace rpc {
|
||||
|
||||
class http_api_connection : public api_connection
|
||||
{
|
||||
public:
|
||||
~http_api_connection()
|
||||
{
|
||||
}
|
||||
|
||||
http_api_connection()
|
||||
{
|
||||
_rpc_state.add_method( "call", [this]( const variants& args ) -> variant {
|
||||
FC_ASSERT( args.size() == 3 && args[2].is_array() );
|
||||
return this->receive_call( args[0].as_uint64(),
|
||||
args[1].as_string(),
|
||||
args[2].get_array() );
|
||||
});
|
||||
|
||||
_rpc_state.add_method( "notice", [this]( const variants& args ) -> variant {
|
||||
FC_ASSERT( args.size() == 2 && args[1].is_array() );
|
||||
this->receive_notice( args[0].as_uint64(), args[1].get_array() );
|
||||
return variant();
|
||||
});
|
||||
|
||||
_rpc_state.add_method( "callback", [this]( const variants& args ) -> variant {
|
||||
FC_ASSERT( args.size() == 2 && args[1].is_array() );
|
||||
this->receive_callback( args[0].as_uint64(), args[1].get_array() );
|
||||
return variant();
|
||||
});
|
||||
|
||||
_rpc_state.on_unhandled( [&]( const std::string& method_name, const variants& args ){
|
||||
return this->receive_call( 0, method_name, args );
|
||||
});
|
||||
}
|
||||
|
||||
virtual variant send_call( api_id_type api_id,
|
||||
string method_name,
|
||||
variants args = variants() ) override
|
||||
{
|
||||
// HTTP has no way to do this, so do nothing
|
||||
return variant();
|
||||
}
|
||||
|
||||
virtual variant send_callback( uint64_t callback_id, variants args = variants() ) override
|
||||
{
|
||||
// HTTP has no way to do this, so do nothing
|
||||
return variant();
|
||||
}
|
||||
|
||||
virtual void send_notice( uint64_t callback_id, variants args = variants() ) override
|
||||
{
|
||||
// HTTP has no way to do this, so do nothing
|
||||
return;
|
||||
}
|
||||
|
||||
void on_request( const fc::http::request& req, const fc::http::server::response& resp )
|
||||
{
|
||||
// this must be called by outside HTTP server's on_request method
|
||||
std::string resp_body;
|
||||
http::reply::status_code resp_status;
|
||||
|
||||
try
|
||||
{
|
||||
resp.add_header( "Content-Type", "application/json" );
|
||||
std::string req_body( req.body.begin(), req.body.end() );
|
||||
auto var = fc::json::from_string( req_body );
|
||||
const auto& var_obj = var.get_object();
|
||||
|
||||
if( var_obj.contains( "method" ) )
|
||||
{
|
||||
auto call = var.as<fc::rpc::request>();
|
||||
try
|
||||
{
|
||||
auto result = _rpc_state.local_call( call.method, call.params );
|
||||
resp_body = fc::json::to_string( fc::rpc::response( *call.id, result ) );
|
||||
resp_status = http::reply::OK;
|
||||
}
|
||||
catch ( const fc::exception& e )
|
||||
{
|
||||
resp_body = fc::json::to_string( fc::rpc::response( *call.id, error_object{ 1, e.to_detail_string(), fc::variant(e)} ) );
|
||||
resp_status = http::reply::InternalServerError;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
resp_status = http::reply::BadRequest;
|
||||
resp_body = "";
|
||||
}
|
||||
}
|
||||
catch ( const fc::exception& e )
|
||||
{
|
||||
resp_status = http::reply::InternalServerError;
|
||||
resp_body = "";
|
||||
wdump((e.to_detail_string()));
|
||||
}
|
||||
try
|
||||
{
|
||||
resp.set_status( resp_status );
|
||||
resp.set_length( resp_body.length() );
|
||||
resp.write( resp_body.c_str(), resp_body.length() );
|
||||
}
|
||||
catch( const fc::exception& e )
|
||||
{
|
||||
wdump((e.to_detail_string()));
|
||||
}
|
||||
return;
|
||||
}
|
||||
fc::rpc::state _rpc_state;
|
||||
};
|
||||
|
||||
} } // namespace fc::rpc
|
||||
Loading…
Reference in a new issue