| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 |
- /*
- * json rpc client
- *
- * @build make jsonrpc
- * @server bin/jsonrpc_server 1234
- * @client bin/jsonrpc_client 127.0.0.1 1234 add 1 2
- *
- */
- #include "hloop.h"
- #include "hatomic.h"
- #include "hbase.h"
- #include "hsocket.h"
- #include "jsonrpc.h"
- #include "cJSON.h"
- // hloop_create_tcp_client -> on_connect -> hio_write -> hio_read -> on_recv
- static int verbose = 0;
- static unpack_setting_t jsonrpc_unpack_setting;
- static void on_close(hio_t* io) {
- printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(io));
- cJSON* jreq = (cJSON*)(hevent_userdata(io));
- if (jreq) {
- cJSON_Delete(jreq);
- hevent_set_userdata(io, NULL);
- }
- hloop_stop(hevent_loop(io));
- }
- static void on_recv(hio_t* io, void* readbuf, int readbytes) {
- // printf("on_recv fd=%d readbytes=%d\n", hio_fd(io), readbytes);
- if (verbose) {
- char localaddrstr[SOCKADDR_STRLEN] = {0};
- char peeraddrstr[SOCKADDR_STRLEN] = {0};
- printf("[%s] <=> [%s]\n",
- SOCKADDR_STR(hio_localaddr(io), localaddrstr),
- SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
- }
- // unpack
- jsonrpc_message msg;
- memset(&msg, 0, sizeof(msg));
- int packlen = jsonrpc_unpack(&msg, readbuf, readbytes);
- if (packlen < 0) {
- printf("jsonrpc_unpack failed!\n");
- return;
- }
- assert(packlen == readbytes);
- printf("< %.*s\n", msg.head.length, msg.body);
- // cJSON_Parse
- cJSON* jres = cJSON_ParseWithLength(msg.body, msg.head.length);
- cJSON* jerror = cJSON_GetObjectItem(jres, "error");
- cJSON* jresult = cJSON_GetObjectItem(jres, "result");
- // ...
- cJSON_Delete(jres);
- hio_close(io);
- }
- static void on_connect(hio_t* io) {
- printf("on_connect fd=%d\n", hio_fd(io));
- if (verbose) {
- char localaddrstr[SOCKADDR_STRLEN] = {0};
- char peeraddrstr[SOCKADDR_STRLEN] = {0};
- printf("connect connfd=%d [%s] => [%s]\n", hio_fd(io),
- SOCKADDR_STR(hio_localaddr(io), localaddrstr),
- SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
- }
- hio_setcb_read(io, on_recv);
- hio_set_unpack(io, &jsonrpc_unpack_setting);
- hio_read(io);
- cJSON* jreq = (cJSON*)(hevent_userdata(io));
- hevent_set_userdata(io, NULL);
- assert(jreq != NULL);
- // cJSON_Print -> pack -> hio_write
- jsonrpc_message msg;
- memset(&msg, 0, sizeof(msg));
- msg.body = cJSON_PrintUnformatted(jreq);
- msg.head.length = strlen(msg.body);
- printf("> %.*s\n", msg.head.length, msg.body);
- // pack
- unsigned int packlen = jsonrpc_package_length(&msg.head);
- unsigned char* writebuf = NULL;
- HV_ALLOC(writebuf, packlen);
- packlen = jsonrpc_pack(&msg, writebuf, packlen);
- if (packlen > 0) {
- hio_write(io, writebuf, packlen);
- }
- cJSON_Delete(jreq);
- cJSON_free((void*)msg.body);
- HV_FREE(writebuf);
- }
- static int jsonrpc_call(hloop_t* loop, const char* host, int port, const char* method, const char* param1, const char* param2) {
- hio_t* connio = hio_create_socket(loop, host, port, HIO_TYPE_TCP, HIO_CLIENT_SIDE);
- if (connio == NULL) {
- return -1;
- }
- // printf("connfd=%d\n", hio_fd(connio));
- // construct request
- cJSON* jreq = cJSON_CreateObject();
- static hatomic_t s_id = HATOMIC_VAR_INIT(0);
- cJSON_AddItemToObject(jreq, "id", cJSON_CreateNumber(++s_id));
- cJSON_AddItemToObject(jreq, "method", cJSON_CreateString(method));
- cJSON* jparams = cJSON_CreateArray();
- cJSON_AddItemToArray(jparams, cJSON_CreateNumber(atoi(param1)));
- cJSON_AddItemToArray(jparams, cJSON_CreateNumber(atoi(param2)));
- cJSON_AddItemToObject(jreq, "params", jparams);
- hevent_set_userdata(connio, jreq);
- hio_setcb_connect(connio, on_connect);
- hio_setcb_close(connio, on_close);
- hio_connect(connio);
- return 0;
- }
- int main(int argc, char** argv) {
- if (argc < 6) {
- printf("Usage: %s host port method param1 param2\n", argv[0]);
- printf("method = [add, sub, mul, div]\n");
- printf("Examples:\n");
- printf(" %s 127.0.0.1 1234 add 1 2\n", argv[0]);
- printf(" %s 127.0.0.1 1234 div 1 0\n", argv[0]);
- return -10;
- }
- const char* host = argv[1];
- int port = atoi(argv[2]);
- const char* method = argv[3];
- const char* param1 = argv[4];
- const char* param2 = argv[5];
- // init jsonrpc_unpack_setting
- memset(&jsonrpc_unpack_setting, 0, sizeof(unpack_setting_t));
- jsonrpc_unpack_setting.mode = UNPACK_BY_LENGTH_FIELD;
- jsonrpc_unpack_setting.package_max_length = DEFAULT_PACKAGE_MAX_LENGTH;
- jsonrpc_unpack_setting.body_offset = JSONRPC_HEAD_LENGTH;
- jsonrpc_unpack_setting.length_field_offset = 1;
- jsonrpc_unpack_setting.length_field_bytes = 4;
- jsonrpc_unpack_setting.length_field_coding = ENCODE_BY_BIG_ENDIAN;
- hloop_t* loop = hloop_new(0);
- jsonrpc_call(loop, host, port, method, param1, param2);
- hloop_run(loop);
- hloop_free(&loop);
- return 0;
- }
|