/* * json rpc server * * @build make jsonrpc * @server bin/jsonrpc_server 1234 * @client bin/jsonrpc_client 127.0.0.1 1234 add 1 2 * */ #include "hloop.h" #include "hbase.h" #include "hsocket.h" #include "cJSON.h" #include "router.h" #include "handler.h" // hloop_create_tcp_server -> on_accept -> hio_read -> on_recv -> hio_write static int verbose = 0; static unpack_setting_t jsonrpc_unpack_setting; jsonrpc_router router[] = { {"add", calc_add}, {"sub", calc_sub}, {"mul", calc_mul}, {"div", calc_div}, }; #define JSONRPC_ROUTER_NUM (sizeof(router)/sizeof(router[0])) static void on_close(hio_t* io) { printf("on_close fd=%d error=%d\n", hio_fd(io), hio_error(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)); } // cJSON_Parse -> router -> cJSON_Print -> hio_write char* req_str = (char*)readbuf; printf("> %s\n", req_str); cJSON* jreq = cJSON_Parse(req_str); cJSON* jres = cJSON_CreateObject(); cJSON* jid = cJSON_GetObjectItem(jreq, "id"); cJSON* jmethod = cJSON_GetObjectItem(jreq, "method"); if (cJSON_IsNumber(jid)) { long id = cJSON_GetNumberValue(jid); cJSON_AddItemToObject(jres, "id", cJSON_CreateNumber(id)); } if (cJSON_IsString(jmethod)) { // router char* method = cJSON_GetStringValue(jmethod); bool found = false; for (int i = 0; i < JSONRPC_ROUTER_NUM; ++i) { if (strcmp(method, router[i].method) == 0) { found = true; router[i].handler(jreq, jres); break; } } if (!found) { not_found(jreq, jres); } } else { bad_request(jreq, jres); } char* resp_str = cJSON_PrintUnformatted(jres); printf("< %s\n", resp_str); // NOTE: +1 for \0 hio_write(io, resp_str, strlen(resp_str) + 1); cJSON_Delete(jreq); cJSON_Delete(jres); cJSON_free(resp_str); } static void on_accept(hio_t* io) { printf("on_accept connfd=%d\n", hio_fd(io)); if (verbose) { char localaddrstr[SOCKADDR_STRLEN] = {0}; char peeraddrstr[SOCKADDR_STRLEN] = {0}; printf("accept connfd=%d [%s] <= [%s]\n", hio_fd(io), SOCKADDR_STR(hio_localaddr(io), localaddrstr), SOCKADDR_STR(hio_peeraddr(io), peeraddrstr)); } hio_setcb_close(io, on_close); hio_setcb_read(io, on_recv); hio_set_unpack(io, &jsonrpc_unpack_setting); hio_read(io); } int main(int argc, char** argv) { if (argc < 2) { printf("Usage: %s port\n", argv[0]); return -10; } int port = atoi(argv[1]); // init jsonrpc_unpack_setting memset(&jsonrpc_unpack_setting, 0, sizeof(unpack_setting_t)); jsonrpc_unpack_setting.mode = UNPACK_BY_DELIMITER; jsonrpc_unpack_setting.package_max_length = DEFAULT_PACKAGE_MAX_LENGTH; jsonrpc_unpack_setting.delimiter[0] = '\0'; jsonrpc_unpack_setting.delimiter_bytes = 1; hloop_t* loop = hloop_new(0); hio_t* listenio = hloop_create_tcp_server(loop, "0.0.0.0", port, on_accept); if (listenio == NULL) { return -20; } printf("listenfd=%d\n", hio_fd(listenio)); hloop_run(loop); hloop_free(&loop); return 0; }