Searpc is a simple C language RPC framework based on GObject system. Searpc handles the serialization/deserialization part of RPC, the transport part is left to users.
The serialization/deserialization uses JSON format via json-glib library. A serialized json object is returned from server to client after executing the RPC function. Each RPC function defined in the server side should take an extra GError argument to report error. The returned json object contains three fields:
Just
./autogen.sh; ./configure; make; make install
To enable profile, Use
CFLAGS="-DPROFILE" ./configure
When profile is enabled, the time spend in each rpc call will be printed.
In the client side, you need to:
The client needs to create a SearpcClient object and supply a transport function. For example:
/* create an rpc_client and supply the transport function. */
SearpcClient *rpc_client;
rpc_client = searpc_client_new();
rpc_client->transport = transport_callback;
rpc_client->arg = &sockfd;
Suppose we have a get_substring
function defined in server as follows:
gchar *get_substring (const gchar *orig_str, int sub_len, GError **error)
To call this function, we type:
gchar* result;
GError *error = NULL;
result = searpc_client_call__string (client, "get_substring", &error,
2, "string", "hello", "int", 2);
string
in searpc_client_call__string
specify the return type. "get_substring"
is the function name. The remain parameters specify the number of parameters the rpc
function took and the type of each parameter and its value. So
2, "string", "hello", "int", 2
means "get_substring" takes 2 parameters, the first is of type "string", the value is "hello", the second is of type "int", the value is 2.
When the client-side function is called, Searpc does the following work:
Your transport function is supposed to:
The prototype of the transport function is:
/*
* arg: rpc_client->arg. Normally a socket number.
* fcall_str: the JSON data stream generated by Searpc.
* fcall_len: the length of `fcall_str`.
* ret_len: place to get the length of the returned json data stream.
* Returns: A newly allocated string stores the JSON data stream.
*/
static char *transport_callback (void *arg, const char *fcall_str, size_t fcall_len, size_t *ret_len);
In the server side, you need to:
And Searpc handles the others for you.
Marshal: The process of unpacking the function arguments from JSON data, call the RPC function and packing the result into JSON data format is called marshalling. The function used to pack the result is called a marshal.
Signature: Every function has a signature determined by its return type and parameter types. Knowning a function's signature enable us to use a corresponding marshal to call it and convert the result into json string.
First write rpc_table.py to contain the rpc function signatures as follows:
# [ <ret-type>, [<arg_types>] ]
func_table = [
[ "int", ["string"] ],
[ "string", ["int", "string"] ],
]
Add makefile rule:
searpc-signature.h searpc-marshal.h: rpc_table.py
python searpc-codegen.py rpc_table.py
searpc-signature.h
and searpc-marshal.h
will be created containing the
function signatures and corresponding marshals. searpc-marshal.h
also contains
a function called register_marshals
.
Then we init the server as follows:
#include "searpc-signature.h"
#include "searpc-marshal.h"
static void
init_rpc_service(void)
{
/* register_marshals is defined in searpc-marshal.h */
searpc_server_init(register_marshals);
}
To register a function we first need to create a service. A service is a set of functions.
Suppose we want to make searpc_strlen
callable from some network
clients, we can do this by putting the following code somewhere:
static int
searpc_strlen(const char *str)
{
if (str == NULL)
return -1;
else
return strlen(str);
}
static void
register_functions()
{
searpc_create_service("searpc-demo");
/* The first parameter is the implementation function.
* The second parameter is the name of the rpc function the
* client would call.
* The third parameter is the signature.
*/
searpc_server_register_function("searpc-demo",
searpc_strlen,
"searpc_strlen",
searpc_signature_int__string());
}
The seaprc_server_register_function
routine registers a function as
a RPC function. The
prototype of this function is:
/*
* service: the name of the service
* func: pointer to the function you want to register
* fname: the name of the function. It would be the key of your
* function in the fucntion hash table.
* signature: the identifier used to get the corresponding marshal.
* Returns: a gboolean value indicating success or failure
*/
gboolean searpc_server_register_function (const char *service,
void* func,
const gchar *fname,
gchar *signature);
After the registration, you should listen to the socket and wait for the
incoming request data stream. Once you get a valid request, call the
searpc_server_call_function()
routine, which will automatically do the
following work for you:
The prototype of searpc_server_call_function
is:
/*
* service: Service name.
* data: The incoming JSON data stream.
* len: The length of **`data`**.
* ret_len: Place to hold the length of the JSON data stream to be returned
* Returns: The JSON data containing the result of the RPC
*/
gchar* searpc_server_call_function (const char *service,
gchar *data, gsize len, gsize *ret_len)
The value returned by searpc_server_call_function()
is the JSON data
ready to send back to the client.
Note, the JSON data stream from client does not contain the service name, it's left to the transport layer to solve the problem. There are several ways, for example:
Pysearpc is the Python binding of Searpc. Only the client side function is
supported. To use it, simply define a class which inherits SearpcClient, and
provide a call_remote_func_sync
method, which is equivalent to the
transport_callback
.
To define your RPC funtion, use the @searpc_func
decorator. It is
equivalent to SEARPC_CLIENT_DEFUN_XXX__YYY
macro. To define a RPC
function which accepts multiple params, here is an example:
class SampeSearpcClient(SearpcClient):
def call_remote_func_sync(self, fcall_str):
# your transport code here
...
@searpc_func("int", ["string", "string"])
def searpc_demo_func(self):
# this is enough for the client side
pass
See the demo program for a more detailed example.
There are well-commented demos in both C and Python.
To run the demo, run the server demo in a shell, and run the client demo in another. To run the python demo, you should first install the package and setup the PYTHONPATH appropriately.
The following packages are required to build libsearpc:
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。
1. 开源生态
2. 协作、人、软件
3. 评估模型