1
0
mirror of https://github.com/anope/anope.git synced 2026-06-24 20:46:38 +02:00
Files
anope/modules/rpc/jsonrpc.cpp
T
2025-02-20 12:53:15 +00:00

245 lines
6.3 KiB
C++

/*
*
* (C) 2010-2025 Anope Team
* Contact us at team@anope.org
*
* Please read COPYING and README for further details.
*/
#include "module.h"
#include "modules/rpc.h"
#include "modules/httpd.h"
#include "yyjson/yyjson.c"
template<class... Ts> struct overloaded : Ts... { using Ts::operator()...; };
template<class... Ts> overloaded(Ts...) -> overloaded<Ts...>;
inline Anope::string yyjson_get_astr(yyjson_val *val, const char *key)
{
const auto *str = yyjson_get_str(yyjson_obj_get(val, key));
return str ? str : "";
}
class MyJSONRPCServiceInterface final
: public RPCServiceInterface
, public HTTPPage
{
private:
Anope::map<RPCEvent *> events;
static void SendError(HTTPReply &reply, int64_t code, const Anope::string &message, const Anope::string &id)
{
Log(LOG_DEBUG) << "JSON-RPC error " << code << ": " << message;
// {"jsonrpc": "2.0", "error": {"code": -32601, "message": "Method not found"}, "id": "1"}
auto* doc = yyjson_mut_doc_new(nullptr);
auto* root = yyjson_mut_obj(doc);
yyjson_mut_doc_set_root(doc, root);
auto *error = yyjson_mut_obj(doc);
yyjson_mut_obj_add_sint(doc, error, "code", code);
yyjson_mut_obj_add_strn(doc, error, "message", message.c_str(), message.length());
yyjson_mut_obj_add_val(doc, root, "error", error);
yyjson_mut_obj_add_str(doc, root, "jsonrpc", "2.0");
if (id.empty())
yyjson_mut_obj_add_null(doc, root, "id");
else
yyjson_mut_obj_add_strn(doc, root, "id", id.c_str(), id.length());
auto *json = yyjson_mut_write(doc, YYJSON_WRITE_ALLOW_INVALID_UNICODE | YYJSON_WRITE_NEWLINE_AT_END, nullptr);
if (json)
{
reply.Write(json);
free(json);
}
yyjson_mut_doc_free(doc);
}
static void SerializeObject(yyjson_mut_doc *doc, yyjson_mut_val *root, const char *key, const RPCBlock &block)
{
auto *result = yyjson_mut_obj(doc);
for (const auto &reply : block.GetReplies())
{
// Captured structured bindings are a C++20 extension.
const auto &k = reply.first;
std::visit(overloaded
{
[&doc, &result, &k](const RPCBlock &b)
{
SerializeObject(doc, result, k.c_str(), b);
},
[&doc, &result, &k](const Anope::string &s)
{
yyjson_mut_obj_add_strn(doc, result, k.c_str(), s.c_str(), s.length());
},
[&doc, &result, &k](std::nullptr_t)
{
yyjson_mut_obj_add_null(doc, result, k.c_str());
},
[&doc, &result, &k](bool b)
{
yyjson_mut_obj_add_bool(doc, result, k.c_str(), b);
},
[&doc, &result, &k](double d)
{
yyjson_mut_obj_add_real(doc, result, k.c_str(), d);
},
[&doc, &result, &k](int64_t i)
{
yyjson_mut_obj_add_int(doc, result, k.c_str(), i);
},
[&doc, &result, &k](uint64_t u)
{
yyjson_mut_obj_add_uint(doc, result, k.c_str(), u);
},
}, reply.second);
}
yyjson_mut_obj_add_val(doc, root, key, result);
}
public:
MyJSONRPCServiceInterface(Module *creator, const Anope::string &sname)
: RPCServiceInterface(creator, sname)
, HTTPPage("/jsonrpc", "application/json")
{
}
bool Register(RPCEvent *event) override
{
return this->events.emplace(event->GetEvent(), event).second;
}
bool Unregister(RPCEvent *event) override
{
return this->events.erase(event->GetEvent()) != 0;
}
bool OnRequest(HTTPProvider *provider, const Anope::string &page_name, HTTPClient *client, HTTPMessage &message, HTTPReply &reply) override
{
auto *doc = yyjson_read(message.content.c_str(), message.content.length(), YYJSON_READ_ALLOW_TRAILING_COMMAS | YYJSON_READ_ALLOW_INVALID_UNICODE);
if (!doc)
{
SendError(reply, -32700, "JSON parse error", "");
return true;
}
auto *root = yyjson_doc_get_root(doc);
if (!yyjson_is_obj(root))
{
// TODO: handle an array of JSON-RPC requests
yyjson_doc_free(doc);
SendError(reply, -32600, "Wrong JSON root element", "");
return true;
}
const auto id = yyjson_get_astr(root, "id");
const auto jsonrpc = yyjson_get_astr(root, "jsonrpc");
if (!jsonrpc.empty() && jsonrpc != "2.0")
{
yyjson_doc_free(doc);
SendError(reply, -32600, "Unsupported JSON-RPC version", id);
return true;
}
RPCRequest request(reply);
request.id = id;
request.name = yyjson_get_astr(root, "method");
auto *params = yyjson_obj_get(root, "params");
size_t idx, max;
yyjson_val *val;
yyjson_arr_foreach(params, idx, max, val)
{
const auto *str = yyjson_get_str(val);
request.data.push_back(str ? str : "");
}
yyjson_doc_free(doc);
auto event = this->events.find(request.name);
if (event == this->events.end())
{
SendError(reply, -32601, "Method not found", id);
return true;
}
if (!event->second->Run(this, client, request))
return false;
this->Reply(request);
return true;
}
void Reply(RPCRequest &request) override
{
if (request.GetError())
{
SendError(request.reply, request.GetError()->first, request.GetError()->second, request.id);
return;
}
// {"jsonrpc": "2.0", "method": "update", "params": [1,2,3,4,5]}
auto* doc = yyjson_mut_doc_new(nullptr);
auto* root = yyjson_mut_obj(doc);
yyjson_mut_doc_set_root(doc, root);
if (request.id.empty())
yyjson_mut_obj_add_null(doc, root, "id");
else
yyjson_mut_obj_add_strn(doc, root, "id", request.id.c_str(), request.id.length());
if (!request.GetReplies().empty())
SerializeObject(doc, root, "result", request);
yyjson_mut_obj_add_str(doc, root, "jsonrpc", "2.0");
auto *json = yyjson_mut_write(doc, YYJSON_WRITE_ALLOW_INVALID_UNICODE | YYJSON_WRITE_NEWLINE_AT_END, nullptr);
if (json)
{
request.reply.Write(json);
free(json);
}
yyjson_mut_doc_free(doc);
}
};
class ModuleJSONRPC final
: public Module
{
private:
ServiceReference<HTTPProvider> httpref;
MyJSONRPCServiceInterface jsonrpcinterface;
public:
ModuleJSONRPC(const Anope::string &modname, const Anope::string &creator)
: Module(modname, creator, EXTRA | VENDOR)
, jsonrpcinterface(this, "rpc")
{
}
~ModuleJSONRPC() override
{
if (httpref)
httpref->UnregisterPage(&jsonrpcinterface);
}
void OnReload(Configuration::Conf *conf) override
{
if (httpref)
httpref->UnregisterPage(&jsonrpcinterface);
this->httpref = ServiceReference<HTTPProvider>("HTTPProvider", conf->GetModule(this)->Get<const Anope::string>("server", "httpd/main"));
if (!httpref)
throw ConfigException("Unable to find http reference, is httpd loaded?");
httpref->RegisterPage(&jsonrpcinterface);
}
};
MODULE_INIT(ModuleJSONRPC)