mongoose/examples/json-rpc-over-websocket/main.c

86 lines
3.2 KiB
C
Raw Normal View History

2021-11-23 00:45:14 +08:00
// Copyright (c) 2020 Cesanta Software Limited
// All rights reserved
//
// See https://mongoose.ws/tutorials/json-rpc-over-websocket/
#include "mongoose.h"
static const char *s_listen_on = "ws://localhost:8000";
static const char *s_web_root = "web_root";
2022-08-01 05:51:59 +08:00
static struct mg_rpc *s_rpc_head = NULL;
2021-11-23 00:45:14 +08:00
2022-07-27 07:46:05 +08:00
static void rpc_sum(struct mg_rpc_req *r) {
2021-11-23 00:45:14 +08:00
double a = 0.0, b = 0.0;
2022-07-27 07:46:05 +08:00
mg_json_get_num(r->frame, "$.params[0]", &a);
mg_json_get_num(r->frame, "$.params[1]", &b);
mg_rpc_ok(r, "%g", a + b);
2021-11-23 00:45:14 +08:00
}
2022-07-27 07:46:05 +08:00
static void rpc_mul(struct mg_rpc_req *r) {
2021-11-23 00:45:14 +08:00
double a = 0.0, b = 0.0;
2022-07-27 07:46:05 +08:00
mg_json_get_num(r->frame, "$.params[0]", &a);
mg_json_get_num(r->frame, "$.params[1]", &b);
mg_rpc_ok(r, "%g", a * b);
2021-11-23 00:45:14 +08:00
}
// This RESTful server implements the following endpoints:
// /websocket - upgrade to Websocket, and implement websocket echo server
// any other URI serves static files from s_web_root
static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
if (ev == MG_EV_OPEN) {
// c->is_hexdumping = 1;
} else if (ev == MG_EV_WS_OPEN) {
c->data[0] = 'W'; // Mark this connection as an established WS client
2021-11-23 00:45:14 +08:00
} else if (ev == MG_EV_HTTP_MSG) {
struct mg_http_message *hm = (struct mg_http_message *) ev_data;
if (mg_http_match_uri(hm, "/websocket")) {
// Upgrade to websocket. From now on, a connection is a full-duplex
// Websocket connection, which will receive MG_EV_WS_MSG events.
mg_ws_upgrade(c, hm, NULL);
} else {
// Serve static files
struct mg_http_serve_opts opts = {.root_dir = s_web_root};
mg_http_serve_dir(c, ev_data, &opts);
}
} else if (ev == MG_EV_WS_MSG) {
// Got websocket frame. Received data is wm->data
struct mg_ws_message *wm = (struct mg_ws_message *) ev_data;
struct mg_iobuf io = {0, 0, 0, 512};
struct mg_rpc_req r = {&s_rpc_head, 0, mg_pfn_iobuf, &io, 0, wm->data};
2022-07-30 14:55:26 +08:00
mg_rpc_process(&r);
if (io.buf) mg_ws_send(c, (char *) io.buf, io.len, WEBSOCKET_OP_TEXT);
mg_iobuf_free(&io);
2021-11-23 00:45:14 +08:00
}
(void) fn_data;
}
static void timer_fn(void *arg) {
struct mg_mgr *mgr = (struct mg_mgr *) arg;
2022-06-10 16:40:22 +08:00
// Broadcast message to all connected websocket clients.
2021-11-23 00:45:14 +08:00
for (struct mg_connection *c = mgr->conns; c != NULL; c = c->next) {
if (c->data[0] != 'W') continue;
2023-05-18 01:26:45 +08:00
mg_ws_printf(c, WEBSOCKET_OP_TEXT, "{%m:%m,%m:[%d,%d,%d]}",
MG_ESC("method"), MG_ESC("notification1"), MG_ESC("params"), 1,
2, 3);
2021-11-23 00:45:14 +08:00
}
}
int main(void) {
2022-08-01 18:19:32 +08:00
struct mg_mgr mgr; // Event manager
mg_mgr_init(&mgr); // Init event manager
mg_log_set(MG_LL_DEBUG); // Set log level
mg_timer_add(&mgr, 5000, MG_TIMER_REPEAT, timer_fn, &mgr); // Init timer
2022-07-27 07:46:05 +08:00
// Configure JSON-RPC functions we're going to handle
mg_rpc_add(&s_rpc_head, mg_str("sum"), rpc_sum, NULL);
mg_rpc_add(&s_rpc_head, mg_str("mul"), rpc_mul, NULL);
mg_rpc_add(&s_rpc_head, mg_str("rpc.list"), mg_rpc_list, &s_rpc_head);
2021-11-23 00:45:14 +08:00
printf("Starting WS listener on %s/websocket\n", s_listen_on);
mg_http_listen(&mgr, s_listen_on, fn, NULL); // Create HTTP listener
for (;;) mg_mgr_poll(&mgr, 1000); // Infinite event loop
mg_mgr_free(&mgr); // Deallocate event manager
2022-08-01 05:51:59 +08:00
mg_rpc_del(&s_rpc_head, NULL); // Deallocate RPC handlers
2021-11-23 00:45:14 +08:00
return 0;
}