Asio based Asynchronous Rpc
An implementation of the ping / pong service.
protobuf message definition:
syntax = "proto3";
package pb;
message request
{
optional bytes command = 1;
}
message response
{
optional bytes results = 1;
}
service service
{
rpc execute(request) returns (response);
}
option cc_generic_services = true;client side implementation:
#define BOOST_ASIO_HAS_IO_URING
#define BOOST_ASIO_DISABLE_EPOLL
#include <thread>
#include <iostream>
#include <arpc.hpp>
#include <ping.pb.h>
namespace net = boost::asio;
namespace gp = google::protobuf;
struct task
{
arpc::controller controller;
pb::request request;
pb::response response;
arpc::Closure* done;
};
class client
{
public:
client(net::io_context& ioc, const std::string& host, const std::string& port) : ioc(ioc), work(ioc), host(host), port(port)
{
channel = new arpc::channel(ioc);
service = new pb::service::Stub(channel, pb::service::STUB_OWNS_CHANNEL);
}
void ping()
{
auto t = std::make_shared<task>();
auto& controller = t->controller;
controller.host(host);
controller.port(port);
controller.timeout(80);
t->request.set_command("ping");
t->done = gp::NewCallback(this, &client::done, t);
service->execute(&t->controller, &t->request, &t->response, t->done);
}
void done(std::shared_ptr<task> t)
{
auto& controller = t->controller;
if (controller.Failed())
std::cerr << "ErrorCode: " << controller.ErrorCode() << " ErrorText: " << controller.ErrorText() << std::endl;
std::cout << t->response.DebugString();
}
~client()
{
delete service;
}
private:
net::io_context& ioc;
net::io_context::work work;
arpc::channel* channel;
pb::service* service;
std::string host;
std::string port;
};
int main(int argc, char* argv[])
{
if (argc != 3)
{
std::cout << "Usage: " << argv[0] << " <host> <port>" << std::endl;
return 1;
}
std::string host(argv[1]);
std::string port(argv[2]);
net::io_context ioc;
client c(ioc, host, port);
std::thread t([&]{ ioc.run(); });
constexpr size_t size = 100;
char buff[size];
while (fgets(buff, size, stdin))
c.ping();
t.join();
return 0;
}server side implementation:
#define BOOST_ASIO_HAS_IO_URING
#define BOOST_ASIO_DISABLE_EPOLL
#include <iostream>
#include <arpc.hpp>
#include <ping.pb.h>
namespace net = boost::asio;
namespace gp = google::protobuf;
void done()
{
std::cout << "got called" << std::endl;
}
class service : public pb::service
{
public:
service()
{
}
void execute(gp::RpcController* controller, const pb::request* request, pb::response* response, gp::Closure* done)
{
std::cout << request->DebugString();
if (request->command() != "ping")
controller->SetFailed("unknown command");
else
response->set_results("pong");
done->Run();
}
~service()
{
}
};
int main(int argc, char* argv[])
{
if (argc != 3)
{
std::cout << "Usage: " << argv[0] << " <host> <port>" << std::endl;
return 1;
}
std::string host(argv[1]);
std::string port(argv[2]);
net::io_context ioc;
service s;
arpc::server server(ioc, host, port);
server.register_service(&s, gp::NewPermanentCallback(&done));
server.run();
ioc.run();
return 0;
}arpc is a Remote Procedure Call (RPC) library, which is header-only, extensible and modern C++ oriented.
It's built on top off the boost and protobuf, it's based on the Proactor design pattern with performance in mind.
arpc enables you to do network programming with tcp protocol in a straightforward, asynchronous and OOP manner.
arpc provides the following features:
- timeout The upper limit of the total time for the call to time out between a RPC request and response
- callback The callable to be invoked immediately after a RPC request or response has been accepted and processed
- controller A way to manipulate settings specific to the RPC implementation and to find out about RPC-level errors
By default, the example is using the io_uring backend, it's disabled by default in Boost.Asio.
This backend may be used for all I/O objects, including sockets, timers, and posix descriptors.
To disable the io_uring backend, just comment out the following lines in code under the example directory:
#define BOOST_ASIO_HAS_IO_URING
#define BOOST_ASIO_DISABLE_EPOLLThe library relies on a C++20 compiler and standard library
More specifically, arpc requires a compiler/standard library supporting the following C++20 features (non-exhaustively):
- concepts
- lambda templates
- All the C++20 type traits from the <type_traits> header
arpc is header-only. To use it just add the necessary #include line to your source files, like this:
#include <arpc.hpp>To build the example with cmake, cd to the root of the project and setup the build directory:
mkdir build
cd build
cmake ..Make and install the executables:
make -j4
make install
The executables are now located at the bin directory of the root of the project.
The example can also be built with the script build.sh, just run it, the executables will be put at the /tmp directory.
Please see example.
arpc is licensed as Boost Software License 1.0.