177 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <nano/boost/asio/bind_executor.hpp>
 | |
| #include <nano/lib/json_error_response.hpp>
 | |
| #include <nano/lib/rpc_handler_interface.hpp>
 | |
| #include <nano/lib/rpcconfig.hpp>
 | |
| #include <nano/lib/utility.hpp>
 | |
| #include <nano/rpc/rpc_connection.hpp>
 | |
| #include <nano/rpc/rpc_handler.hpp>
 | |
| 
 | |
| #include <boost/algorithm/string.hpp>
 | |
| #include <boost/algorithm/string/predicate.hpp>
 | |
| #ifdef NANO_SECURE_RPC
 | |
| #include <boost/asio/ssl/stream.hpp>
 | |
| #endif
 | |
| #include <boost/format.hpp>
 | |
| 
 | |
| nano::rpc_connection::rpc_connection (nano::rpc_config const & rpc_config, boost::asio::io_context & io_ctx, nano::logger & logger, nano::rpc_handler_interface & rpc_handler_interface) :
 | |
| 	socket (io_ctx),
 | |
| 	strand (io_ctx.get_executor ()),
 | |
| 	io_ctx (io_ctx),
 | |
| 	logger (logger),
 | |
| 	rpc_config (rpc_config),
 | |
| 	rpc_handler_interface (rpc_handler_interface)
 | |
| {
 | |
| 	responded.clear ();
 | |
| }
 | |
| 
 | |
| void nano::rpc_connection::parse_connection ()
 | |
| {
 | |
| 	read (socket);
 | |
| }
 | |
| 
 | |
| void nano::rpc_connection::prepare_head (unsigned version, boost::beast::http::status status)
 | |
| {
 | |
| 	res.version (version);
 | |
| 	res.result (status);
 | |
| 	res.set (boost::beast::http::field::allow, "POST, OPTIONS");
 | |
| 	res.set (boost::beast::http::field::content_type, "application/json");
 | |
| 	res.set (boost::beast::http::field::access_control_allow_origin, "*");
 | |
| 	res.set (boost::beast::http::field::access_control_allow_methods, "POST, OPTIONS");
 | |
| 	res.set (boost::beast::http::field::access_control_allow_headers, "Accept, Accept-Language, Content-Language, Content-Type");
 | |
| 	res.set (boost::beast::http::field::connection, "close");
 | |
| }
 | |
| 
 | |
| void nano::rpc_connection::write_result (std::string body, unsigned version, boost::beast::http::status status)
 | |
| {
 | |
| 	if (!responded.test_and_set ())
 | |
| 	{
 | |
| 		prepare_head (version, status);
 | |
| 		res.body () = body;
 | |
| 		res.prepare_payload ();
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		debug_assert (false && "RPC already responded and should only respond once");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void nano::rpc_connection::write_completion_handler (std::shared_ptr<nano::rpc_connection> const & rpc_connection)
 | |
| {
 | |
| 	// Intentional no-op
 | |
| }
 | |
| 
 | |
| template <typename STREAM_TYPE>
 | |
| void nano::rpc_connection::read (STREAM_TYPE & stream)
 | |
| {
 | |
| 	auto this_l (shared_from_this ());
 | |
| 	auto header_parser (std::make_shared<boost::beast::http::request_parser<boost::beast::http::empty_body>> ());
 | |
| 	header_parser->body_limit (rpc_config.max_request_size);
 | |
| 
 | |
| 	boost::beast::http::async_read_header (stream, buffer, *header_parser, boost::asio::bind_executor (strand, [this_l, &stream, header_parser] (boost::system::error_code const & ec, size_t bytes_transferred) {
 | |
| 		if (!ec)
 | |
| 		{
 | |
| 			if (boost::iequals (header_parser->get ()[boost::beast::http::field::expect], "100-continue"))
 | |
| 			{
 | |
| 				auto continue_response (std::make_shared<boost::beast::http::response<boost::beast::http::empty_body>> ());
 | |
| 				continue_response->version (11);
 | |
| 				continue_response->result (boost::beast::http::status::continue_);
 | |
| 				continue_response->set (boost::beast::http::field::server, "nano");
 | |
| 				boost::beast::http::async_write (stream, *continue_response, boost::asio::bind_executor (this_l->strand, [this_l, continue_response] (boost::system::error_code const & ec, size_t bytes_transferred) {}));
 | |
| 			}
 | |
| 
 | |
| 			this_l->parse_request (stream, header_parser);
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			this_l->logger.error (nano::log::type::rpc_connection, "RPC header error: ", ec.message ());
 | |
| 
 | |
| 			// Respond with the reason for the invalid header
 | |
| 			auto response_handler ([this_l, &stream] (std::string const & tree_a) {
 | |
| 				this_l->write_result (tree_a, 11);
 | |
| 				boost::beast::http::async_write (stream, this_l->res, boost::asio::bind_executor (this_l->strand, [this_l] (boost::system::error_code const & ec, size_t bytes_transferred) {
 | |
| 					this_l->write_completion_handler (this_l);
 | |
| 				}));
 | |
| 			});
 | |
| 			nano::json_error_response (response_handler, std::string ("Invalid header: ") + ec.message ());
 | |
| 		}
 | |
| 	}));
 | |
| }
 | |
| 
 | |
| template <typename STREAM_TYPE>
 | |
| void nano::rpc_connection::parse_request (STREAM_TYPE & stream, std::shared_ptr<boost::beast::http::request_parser<boost::beast::http::empty_body>> const & header_parser)
 | |
| {
 | |
| 	auto this_l (shared_from_this ());
 | |
| 	auto header_field_credentials_l (header_parser->get ()["nano-api-key"]);
 | |
| 	auto header_corr_id_l (header_parser->get ()["nano-correlation-id"]);
 | |
| 	auto body_parser (std::make_shared<boost::beast::http::request_parser<boost::beast::http::string_body>> (std::move (*header_parser)));
 | |
| 	std::string path_l = body_parser->get ().target ();
 | |
| 	boost::beast::http::async_read (stream, buffer, *body_parser, boost::asio::bind_executor (strand, [this_l, body_parser, header_field_credentials_l, header_corr_id_l, path_l, &stream] (boost::system::error_code const & ec, size_t bytes_transferred) {
 | |
| 		if (!ec)
 | |
| 		{
 | |
| 			this_l->io_ctx.post ([this_l, body_parser, header_field_credentials_l, header_corr_id_l, path_l, &stream] () {
 | |
| 				auto & req (body_parser->get ());
 | |
| 				auto start (std::chrono::steady_clock::now ());
 | |
| 				auto version (req.version ());
 | |
| 				std::stringstream ss;
 | |
| 				ss << std::hex << std::showbase << reinterpret_cast<uintptr_t> (this_l.get ());
 | |
| 				auto request_id = ss.str ();
 | |
| 				auto response_handler ([this_l, version, start, request_id, &stream] (std::string const & tree_a) {
 | |
| 					auto body = tree_a;
 | |
| 					this_l->write_result (body, version);
 | |
| 					boost::beast::http::async_write (stream, this_l->res, boost::asio::bind_executor (this_l->strand, [this_l] (boost::system::error_code const & ec, size_t bytes_transferred) {
 | |
| 						this_l->write_completion_handler (this_l);
 | |
| 					}));
 | |
| 
 | |
| 					// Bump logging level if RPC request logging is enabled
 | |
| 					this_l->logger.log (this_l->rpc_config.rpc_logging.log_rpc ? nano::log::level::info : nano::log::level::debug,
 | |
| 					nano::log::type::rpc_request, "RPC request {} completed in {} microseconds", request_id, std::chrono::duration_cast<std::chrono::microseconds> (std::chrono::steady_clock::now () - start).count ());
 | |
| 				});
 | |
| 
 | |
| 				std::string api_path_l = "/api/v2";
 | |
| 				int rpc_version_l = boost::starts_with (path_l, api_path_l) ? 2 : 1;
 | |
| 
 | |
| 				auto method = req.method ();
 | |
| 				switch (method)
 | |
| 				{
 | |
| 					case boost::beast::http::verb::post:
 | |
| 					{
 | |
| 						auto handler (std::make_shared<nano::rpc_handler> (this_l->rpc_config, req.body (), request_id, response_handler, this_l->rpc_handler_interface, this_l->logger));
 | |
| 						nano::rpc_handler_request_params request_params;
 | |
| 						request_params.rpc_version = rpc_version_l;
 | |
| 						request_params.credentials = header_field_credentials_l;
 | |
| 						request_params.correlation_id = header_corr_id_l;
 | |
| 						request_params.path = boost::algorithm::erase_first_copy (path_l, api_path_l);
 | |
| 						request_params.path = boost::algorithm::erase_first_copy (request_params.path, "/");
 | |
| 						handler->process_request (request_params);
 | |
| 						break;
 | |
| 					}
 | |
| 					case boost::beast::http::verb::options:
 | |
| 					{
 | |
| 						this_l->prepare_head (version);
 | |
| 						this_l->res.prepare_payload ();
 | |
| 						boost::beast::http::async_write (stream, this_l->res, boost::asio::bind_executor (this_l->strand, [this_l] (boost::system::error_code const & ec, size_t bytes_transferred) {
 | |
| 							this_l->write_completion_handler (this_l);
 | |
| 						}));
 | |
| 						break;
 | |
| 					}
 | |
| 					default:
 | |
| 					{
 | |
| 						nano::json_error_response (response_handler, "Can only POST requests");
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 			});
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			this_l->logger.error (nano::log::type::rpc_connection, "RPC read error: ", ec.message ());
 | |
| 		}
 | |
| 	}));
 | |
| }
 | |
| 
 | |
| template void nano::rpc_connection::read (socket_type &);
 | |
| template void nano::rpc_connection::parse_request (socket_type &, std::shared_ptr<boost::beast::http::request_parser<boost::beast::http::empty_body>> const &);
 | |
| #ifdef NANO_SECURE_RPC
 | |
| template void nano::rpc_connection::read (boost::asio::ssl::stream<socket_type &> &);
 | |
| template void nano::rpc_connection::parse_request (boost::asio::ssl::stream<socket_type &> &, std::shared_ptr<boost::beast::http::request_parser<boost::beast::http::empty_body>> const &);
 | |
| #endif
 | 
