mirror of
https://github.com/EQEmu/Server.git
synced 2025-12-11 21:01:29 +00:00
217 lines
6.0 KiB
C++
217 lines
6.0 KiB
C++
#include "websocket_server.h"
|
|
#include "../event/event_loop.h"
|
|
#include "../event/timer.h"
|
|
#include <fmt/format.h>
|
|
#include <map>
|
|
#include <list>
|
|
|
|
struct MethodHandlerEntry
|
|
{
|
|
MethodHandlerEntry(EQ::Net::WebsocketServer::MethodHandler h, int s) {
|
|
handler = h;
|
|
status = s;
|
|
}
|
|
|
|
EQ::Net::WebsocketServer::MethodHandler handler;
|
|
int status;
|
|
};
|
|
|
|
struct EQ::Net::WebsocketServer::Impl
|
|
{
|
|
std::unique_ptr<TCPServer> server;
|
|
std::unique_ptr<EQ::Timer> ping_timer;
|
|
std::map<std::shared_ptr<websocket_connection>, std::unique_ptr<WebsocketServerConnection>> connections;
|
|
std::map<std::string, MethodHandlerEntry> methods;
|
|
websocket_server websocket_server;
|
|
LoginHandler login_handler;
|
|
};
|
|
|
|
EQ::Net::WebsocketServer::WebsocketServer(const std::string &addr, int port)
|
|
{
|
|
_impl.reset(new Impl());
|
|
_impl->server.reset(new EQ::Net::TCPServer());
|
|
_impl->server->Listen(addr, port, false, [this](std::shared_ptr<EQ::Net::TCPConnection> connection) {
|
|
auto wsc = _impl->websocket_server.get_connection();
|
|
WebsocketServerConnection *c = new WebsocketServerConnection(this, connection, wsc);
|
|
_impl->connections.insert(std::make_pair(wsc, std::unique_ptr<WebsocketServerConnection>(c)));
|
|
});
|
|
|
|
_impl->websocket_server.set_write_handler(
|
|
[this](websocketpp::connection_hdl hdl, char const *data, size_t size) -> websocketpp::lib::error_code {
|
|
auto c = _impl->websocket_server.get_con_from_hdl(hdl);
|
|
auto iter = _impl->connections.find(c);
|
|
if (iter != _impl->connections.end()) {
|
|
iter->second->GetTCPConnection()->Write(data, size);
|
|
}
|
|
|
|
return websocketpp::lib::error_code();
|
|
});
|
|
|
|
_impl->ping_timer.reset(new EQ::Timer(5000, true, [this](EQ::Timer *t) {
|
|
auto iter = _impl->connections.begin();
|
|
|
|
while (iter != _impl->connections.end()) {
|
|
try {
|
|
auto &connection = iter->second;
|
|
connection->GetWebsocketConnection()->ping("keepalive");
|
|
}
|
|
catch (std::exception) {
|
|
iter->second->GetTCPConnection()->Disconnect();
|
|
}
|
|
|
|
iter++;
|
|
}
|
|
}));
|
|
|
|
_impl->methods.insert(std::make_pair("login", MethodHandlerEntry(std::bind(&WebsocketServer::Login, this, std::placeholders::_1, std::placeholders::_2), 0)));
|
|
_impl->methods.insert(std::make_pair("subscribe", MethodHandlerEntry(std::bind(&WebsocketServer::Subscribe, this, std::placeholders::_1, std::placeholders::_2), 0)));
|
|
_impl->methods.insert(std::make_pair("unsubscribe", MethodHandlerEntry(std::bind(&WebsocketServer::Unsubscribe, this, std::placeholders::_1, std::placeholders::_2), 0)));
|
|
_impl->login_handler = [](const WebsocketServerConnection* connection, const std::string& user, const std::string& pass) {
|
|
WebsocketLoginStatus ret;
|
|
ret.account_name = "admin";
|
|
|
|
if (connection->RemoteIP() == "127.0.0.1" || connection->RemoteIP() == "::") {
|
|
ret.logged_in = true;
|
|
return ret;
|
|
}
|
|
|
|
ret.logged_in = false;
|
|
return ret;
|
|
};
|
|
|
|
_impl->websocket_server.clear_access_channels(websocketpp::log::alevel::all);
|
|
}
|
|
|
|
EQ::Net::WebsocketServer::~WebsocketServer()
|
|
{
|
|
}
|
|
|
|
void EQ::Net::WebsocketServer::ReleaseConnection(WebsocketServerConnection *connection)
|
|
{
|
|
//Clear any subscriptions here
|
|
|
|
_impl->connections.erase(connection->GetWebsocketConnection());
|
|
}
|
|
|
|
Json::Value EQ::Net::WebsocketServer::HandleRequest(WebsocketServerConnection *connection, const std::string &method, const Json::Value ¶ms)
|
|
{
|
|
Json::Value err;
|
|
if (method != "login") {
|
|
if (!connection->IsAuthorized()) {
|
|
throw WebsocketException("Not logged in");
|
|
}
|
|
}
|
|
|
|
auto iter = _impl->methods.find(method);
|
|
if (iter != _impl->methods.end()) {
|
|
auto &s = iter->second;
|
|
if (s.status > connection->GetStatus()) {
|
|
throw WebsocketException("Status too low");
|
|
}
|
|
|
|
return s.handler(connection, params);
|
|
}
|
|
|
|
throw WebsocketException("Unknown Method");
|
|
}
|
|
|
|
void EQ::Net::WebsocketServer::SetMethodHandler(const std::string &method, MethodHandler handler, int required_status)
|
|
{
|
|
//Reserved method names
|
|
if (method == "subscribe" ||
|
|
method == "unsubscribe" ||
|
|
method == "login") {
|
|
return;
|
|
}
|
|
|
|
_impl->methods.insert_or_assign(method, MethodHandlerEntry(handler, required_status));
|
|
}
|
|
|
|
void EQ::Net::WebsocketServer::SetLoginHandler(LoginHandler handler)
|
|
{
|
|
_impl->login_handler = handler;
|
|
}
|
|
|
|
void EQ::Net::WebsocketServer::DispatchEvent(const std::string &evt, Json::Value data, int required_status)
|
|
{
|
|
try {
|
|
Json::Value event_obj;
|
|
event_obj["type"] = "event";
|
|
event_obj["event"] = evt;
|
|
event_obj["data"] = data;
|
|
|
|
std::stringstream payload;
|
|
payload << event_obj;
|
|
|
|
for (auto &iter : _impl->connections) {
|
|
auto &c = iter.second;
|
|
|
|
//Might be better to get rid of subscriptions and just send everything and
|
|
//let the client sort out what they want idk
|
|
if (c->GetStatus() >= required_status && c->IsSubscribed(evt)) {
|
|
c->GetWebsocketConnection()->send(payload.str());
|
|
}
|
|
}
|
|
}
|
|
catch (std::exception) {
|
|
}
|
|
}
|
|
|
|
Json::Value EQ::Net::WebsocketServer::Login(WebsocketServerConnection *connection, const Json::Value ¶ms)
|
|
{
|
|
Json::Value ret;
|
|
|
|
try {
|
|
Json::Value ret;
|
|
|
|
auto user = params[0].asString();
|
|
auto pass = params[1].asString();
|
|
|
|
auto r = _impl->login_handler(connection, user, pass);
|
|
|
|
if (r.logged_in) {
|
|
connection->SetAuthorized(true, r.account_name, r.account_id, 255);
|
|
ret["status"] = "Ok";
|
|
}
|
|
else {
|
|
connection->SetAuthorized(false, "", 0, 0);
|
|
ret["status"] = "Not Authorized";
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
catch (std::exception) {
|
|
throw WebsocketException("Unable to process login request");
|
|
}
|
|
}
|
|
|
|
Json::Value EQ::Net::WebsocketServer::Subscribe(WebsocketServerConnection *connection, const Json::Value ¶ms)
|
|
{
|
|
Json::Value ret;
|
|
|
|
try {
|
|
auto evt = params[0].asString();
|
|
connection->Subscribe(evt);
|
|
ret["status"] = "Ok";
|
|
return ret;
|
|
}
|
|
catch (std::exception) {
|
|
throw WebsocketException("Unable to process subscribe request");
|
|
}
|
|
}
|
|
|
|
Json::Value EQ::Net::WebsocketServer::Unsubscribe(WebsocketServerConnection *connection, const Json::Value ¶ms)
|
|
{
|
|
Json::Value ret;
|
|
|
|
try {
|
|
auto evt = params[0].asString();
|
|
connection->Unsubscribe(evt);
|
|
ret["status"] = "Ok";
|
|
return ret;
|
|
}
|
|
catch (std::exception) {
|
|
throw WebsocketException("Unable to process unsubscribe request");
|
|
}
|
|
}
|