Uring 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:
#include <thread>
#include <iostream>
#include <urpc.hpp>
#include <ping.pb.h>
namespace net = unp;
namespace gp = google::protobuf;
struct task
{
urpc::controller controller;
pb::request request;
pb::response response;
urpc::Closure* done;
};
class client
{
public:
client(net::io_uring_context& ioc, const std::string& host, const std::string& port) : ioc(ioc), host(host), port(port)
{
channel = new urpc::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_uring_context& ioc;
urpc::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_uring_context ioc;
net::inplace_stop_source source;
client c(ioc, host, port);
std::thread t([&]{ ioc.run(source.get_token()); });
constexpr size_t size = 100;
char buff[size];
while (fgets(buff, size, stdin))
c.ping();
t.join();
return 0;
}server side implementation:
#include <iostream>
#include <urpc.hpp>
#include <ping.pb.h>
namespace net = unp;
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_uring_context ioc;
net::inplace_stop_source source;
service s;
urpc::server server(ioc, host, port);
server.register_service(&s, gp::NewPermanentCallback(&done));
server.run();
ioc.run(source.get_token());
return 0;
}urpc is a Remote Procedure Call (RPC) library, which is header-only, extensible and modern C++ oriented.
It's built on top off the unp and protobuf, it's based on the Proactor design pattern with performance in mind.
urpc enables you to do network programming with tcp protocol in a straightforward, asynchronous and OOP manner.
urpc 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
The library relies on a C++20 compiler and standard library
More specifically, urpc 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
urpc is header-only. To use it just add the necessary #include line to your source files, like this:
#include <urpc.hpp>git clone https://github.com/deepgrace/unp.git and place it with urpc under the same directory.
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.
urpc is licensed as Boost Software License 1.0.