WebSocketServer: Only encode event once and when it's required

This commit is contained in:
tt2468 2021-04-29 08:52:40 -07:00
parent d8e84321b1
commit 4028d42931
2 changed files with 36 additions and 5 deletions

View File

@ -196,16 +196,16 @@ std::string WebSocketServer::GetConnectUrl()
void WebSocketServer::BroadcastEvent(uint64_t requiredIntent, std::string eventType, json eventData) void WebSocketServer::BroadcastEvent(uint64_t requiredIntent, std::string eventType, json eventData)
{ {
QtConcurrent::run(&_threadPool, [=]() { QtConcurrent::run(&_threadPool, [=]() {
// Populate message object
json eventMessage; json eventMessage;
eventMessage["messageType"] = "Event"; eventMessage["messageType"] = "Event";
eventMessage["eventType"] = eventType; eventMessage["eventType"] = eventType;
if (eventData.is_object()) if (eventData.is_object())
eventMessage["eventData"] = eventData; eventMessage["eventData"] = eventData;
// I hate to have to encode all supported types, but it's more efficient at scale than doing it per-session. // Initialize objects. The broadcast process only dumps the data when its needed.
std::string messageJson = eventMessage.dump(); std::string messageJson;
auto messageMsgPack = json::to_msgpack(eventMessage); std::string messageMsgPack;
std::string messageMsgPackString(messageMsgPack.begin(), messageMsgPack.end());
std::unique_lock<std::mutex> lock(_sessionMutex); std::unique_lock<std::mutex> lock(_sessionMutex);
for (auto & it : _sessions) { for (auto & it : _sessions) {
@ -214,10 +214,17 @@ void WebSocketServer::BroadcastEvent(uint64_t requiredIntent, std::string eventT
if ((it.second.EventSubscriptions() & requiredIntent) != 0) { if ((it.second.EventSubscriptions() & requiredIntent) != 0) {
switch (it.second.Encoding()) { switch (it.second.Encoding()) {
case WebSocketEncoding::Json: case WebSocketEncoding::Json:
if (messageJson.empty()) {
messageJson = eventMessage.dump();
}
_server.send((websocketpp::connection_hdl)it.first, messageJson, websocketpp::frame::opcode::text); _server.send((websocketpp::connection_hdl)it.first, messageJson, websocketpp::frame::opcode::text);
break; break;
case WebSocketEncoding::MsgPack: case WebSocketEncoding::MsgPack:
_server.send((websocketpp::connection_hdl)it.first, messageMsgPackString, websocketpp::frame::opcode::binary); if (messageMsgPack.empty()) {
auto msgPackData = json::to_msgpack(eventMessage);
messageMsgPack = std::string(msgPackData.begin(), msgPackData.end());
}
_server.send((websocketpp::connection_hdl)it.first, messageMsgPack, websocketpp::frame::opcode::binary);
break; break;
} }
} }
@ -278,9 +285,28 @@ void WebSocketServer::onClose(websocketpp::connection_hdl hdl)
{ {
auto conn = _server.get_con_from_hdl(hdl); auto conn = _server.get_con_from_hdl(hdl);
// Get info from the session and then delete it
std::unique_lock<std::mutex> lock(_sessionMutex); std::unique_lock<std::mutex> lock(_sessionMutex);
auto &session = _sessions[hdl];
bool isIdentified = session.IsIdentified();
uint64_t connectedAt = session.ConnectedAt();
uint64_t incomingMessages = session.IncomingMessages();
uint64_t outgoingMessages = session.OutgoingMessages();
std::string remoteAddress = session.RemoteAddress();
_sessions.erase(hdl); _sessions.erase(hdl);
lock.unlock(); lock.unlock();
// Build SessionState object for signal
WebSocketSessionState state;
state.remoteAddress = remoteAddress;
state.connectedAt = connectedAt;
state.incomingMessages = incomingMessages;
state.outgoingMessages = outgoingMessages;
// Emit signals
emit ClientDisconnected(state, conn->get_local_close_code());
if (isIdentified)
emit IdentifiedClientDisconnected(state, conn->get_local_close_code());
} }
void WebSocketServer::onMessage(websocketpp::connection_hdl hdl, websocketpp::server<websocketpp::config::asio>::message_ptr message) void WebSocketServer::onMessage(websocketpp::connection_hdl hdl, websocketpp::server<websocketpp::config::asio>::message_ptr message)

View File

@ -76,6 +76,11 @@ class WebSocketServer : QObject
public Q_SLOTS: public Q_SLOTS:
void BroadcastEvent(uint64_t requiredIntent, std::string eventType, json eventData = nullptr); void BroadcastEvent(uint64_t requiredIntent, std::string eventType, json eventData = nullptr);
signals:
void ClientIdentified(const WebSocketSessionState state);
void ClientDisconnected(const WebSocketSessionState state, const uint16_t closeCode);
void IdentifiedClientDisconnected(const WebSocketSessionState state, const uint16_t closeCode);
private: private:
void ServerRunner(); void ServerRunner();