99a3bbc4f9
* use C++11 enums * small rename * update tests * update tests * update ws * update ws * update README.md
539 lines
17 KiB
C++
539 lines
17 KiB
C++
/*
|
|
* IXCobraConnection.cpp
|
|
* Author: Benjamin Sergeant
|
|
* Copyright (c) 2017-2018 Machine Zone. All rights reserved.
|
|
*/
|
|
|
|
#include "IXCobraConnection.h"
|
|
#include <ixcrypto/IXHMac.h>
|
|
#include <ixwebsocket/IXWebSocket.h>
|
|
|
|
#include <algorithm>
|
|
#include <stdexcept>
|
|
#include <cmath>
|
|
#include <cassert>
|
|
#include <cstring>
|
|
|
|
|
|
namespace ix
|
|
{
|
|
TrafficTrackerCallback CobraConnection::_trafficTrackerCallback = nullptr;
|
|
constexpr size_t CobraConnection::kQueueMaxSize;
|
|
|
|
CobraConnection::CobraConnection() :
|
|
_webSocket(new WebSocket()),
|
|
_publishMode(CobraConnection_PublishMode_Immediate),
|
|
_authenticated(false),
|
|
_eventCallback(nullptr)
|
|
{
|
|
_pdu["action"] = "rtm/publish";
|
|
|
|
initWebSocketOnMessageCallback();
|
|
}
|
|
|
|
CobraConnection::~CobraConnection()
|
|
{
|
|
disconnect();
|
|
setEventCallback(nullptr);
|
|
}
|
|
|
|
void CobraConnection::setTrafficTrackerCallback(const TrafficTrackerCallback& callback)
|
|
{
|
|
_trafficTrackerCallback = callback;
|
|
}
|
|
|
|
void CobraConnection::resetTrafficTrackerCallback()
|
|
{
|
|
setTrafficTrackerCallback(nullptr);
|
|
}
|
|
|
|
void CobraConnection::invokeTrafficTrackerCallback(size_t size, bool incoming)
|
|
{
|
|
if (_trafficTrackerCallback)
|
|
{
|
|
_trafficTrackerCallback(size, incoming);
|
|
}
|
|
}
|
|
|
|
void CobraConnection::setEventCallback(const EventCallback& eventCallback)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_eventCallbackMutex);
|
|
_eventCallback = eventCallback;
|
|
}
|
|
|
|
void CobraConnection::invokeEventCallback(ix::CobraConnectionEventType eventType,
|
|
const std::string& errorMsg,
|
|
const WebSocketHttpHeaders& headers,
|
|
const std::string& subscriptionId)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_eventCallbackMutex);
|
|
if (_eventCallback)
|
|
{
|
|
_eventCallback(eventType, errorMsg, headers, subscriptionId);
|
|
}
|
|
}
|
|
|
|
void CobraConnection::invokeErrorCallback(const std::string& errorMsg,
|
|
const std::string& serializedPdu)
|
|
{
|
|
std::stringstream ss;
|
|
ss << errorMsg << " : received pdu => " << serializedPdu;
|
|
invokeEventCallback(ix::CobraConnection_EventType_Error, ss.str());
|
|
}
|
|
|
|
void CobraConnection::disconnect()
|
|
{
|
|
_authenticated = false;
|
|
_webSocket->stop();
|
|
}
|
|
|
|
void CobraConnection::initWebSocketOnMessageCallback()
|
|
{
|
|
_webSocket->setOnMessageCallback(
|
|
[this](ix::WebSocketMessageType messageType,
|
|
const std::string& str,
|
|
size_t wireSize,
|
|
const ix::WebSocketErrorInfo& error,
|
|
const ix::WebSocketOpenInfo& openInfo,
|
|
const ix::WebSocketCloseInfo& closeInfo)
|
|
{
|
|
CobraConnection::invokeTrafficTrackerCallback(wireSize, true);
|
|
|
|
std::stringstream ss;
|
|
if (messageType == ix::WebSocketMessageType::Open)
|
|
{
|
|
invokeEventCallback(ix::CobraConnection_EventType_Open,
|
|
std::string(),
|
|
openInfo.headers);
|
|
sendHandshakeMessage();
|
|
}
|
|
else if (messageType == ix::WebSocketMessageType::Close)
|
|
{
|
|
_authenticated = false;
|
|
|
|
std::stringstream ss;
|
|
ss << "Close code " << closeInfo.code;
|
|
ss << " reason " << closeInfo.reason;
|
|
invokeEventCallback(ix::CobraConnection_EventType_Closed,
|
|
ss.str());
|
|
}
|
|
else if (messageType == ix::WebSocketMessageType::Message)
|
|
{
|
|
Json::Value data;
|
|
Json::Reader reader;
|
|
if (!reader.parse(str, data))
|
|
{
|
|
invokeErrorCallback("Invalid json", str);
|
|
return;
|
|
}
|
|
|
|
if (!data.isMember("action"))
|
|
{
|
|
invokeErrorCallback("Missing action", str);
|
|
return;
|
|
}
|
|
|
|
auto action = data["action"].asString();
|
|
|
|
if (action == "auth/handshake/ok")
|
|
{
|
|
if (!handleHandshakeResponse(data))
|
|
{
|
|
invokeErrorCallback("Error extracting nonce from handshake response", str);
|
|
}
|
|
}
|
|
else if (action == "auth/handshake/error")
|
|
{
|
|
invokeErrorCallback("Handshake error", str);
|
|
}
|
|
else if (action == "auth/authenticate/ok")
|
|
{
|
|
_authenticated = true;
|
|
invokeEventCallback(ix::CobraConnection_EventType_Authenticated);
|
|
flushQueue();
|
|
}
|
|
else if (action == "auth/authenticate/error")
|
|
{
|
|
invokeErrorCallback("Authentication error", str);
|
|
}
|
|
else if (action == "rtm/subscription/data")
|
|
{
|
|
handleSubscriptionData(data);
|
|
}
|
|
else if (action == "rtm/subscribe/ok")
|
|
{
|
|
if (!handleSubscriptionResponse(data))
|
|
{
|
|
invokeErrorCallback("Error processing subscribe response", str);
|
|
}
|
|
}
|
|
else if (action == "rtm/subscribe/error")
|
|
{
|
|
invokeErrorCallback("Subscription error", str);
|
|
}
|
|
else if (action == "rtm/unsubscribe/ok")
|
|
{
|
|
if (!handleUnsubscriptionResponse(data))
|
|
{
|
|
invokeErrorCallback("Error processing subscribe response", str);
|
|
}
|
|
}
|
|
else if (action == "rtm/unsubscribe/error")
|
|
{
|
|
invokeErrorCallback("Unsubscription error", str);
|
|
}
|
|
else
|
|
{
|
|
invokeErrorCallback("Un-handled message type", str);
|
|
}
|
|
}
|
|
else if (messageType == ix::WebSocketMessageType::Error)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Connection error: " << error.reason << std::endl;
|
|
ss << "#retries: " << error.retries << std::endl;
|
|
ss << "Wait time(ms): " << error.wait_time << std::endl;
|
|
ss << "HTTP Status: " << error.http_status << std::endl;
|
|
invokeErrorCallback(ss.str(), std::string());
|
|
}
|
|
});
|
|
}
|
|
|
|
void CobraConnection::setPublishMode(CobraConnectionPublishMode publishMode)
|
|
{
|
|
_publishMode = publishMode;
|
|
}
|
|
|
|
void CobraConnection::configure(const std::string& appkey,
|
|
const std::string& endpoint,
|
|
const std::string& rolename,
|
|
const std::string& rolesecret,
|
|
const WebSocketPerMessageDeflateOptions& webSocketPerMessageDeflateOptions)
|
|
{
|
|
_roleName = rolename;
|
|
_roleSecret = rolesecret;
|
|
|
|
std::stringstream ss;
|
|
ss << endpoint;
|
|
ss << "/v2?appkey=";
|
|
ss << appkey;
|
|
|
|
std::string url = ss.str();
|
|
_webSocket->setUrl(url);
|
|
_webSocket->setPerMessageDeflateOptions(webSocketPerMessageDeflateOptions);
|
|
}
|
|
|
|
//
|
|
// Handshake message schema.
|
|
//
|
|
// handshake = {
|
|
// "action": "auth/handshake",
|
|
// "body": {
|
|
// "data": {
|
|
// "role": role
|
|
// },
|
|
// "method": "role_secret"
|
|
// },
|
|
// }
|
|
//
|
|
//
|
|
bool CobraConnection::sendHandshakeMessage()
|
|
{
|
|
Json::Value data;
|
|
data["role"] = _roleName;
|
|
|
|
Json::Value body;
|
|
body["data"] = data;
|
|
body["method"] = "role_secret";
|
|
|
|
Json::Value pdu;
|
|
pdu["action"] = "auth/handshake";
|
|
pdu["body"] = body;
|
|
|
|
std::string serializedJson = serializeJson(pdu);
|
|
CobraConnection::invokeTrafficTrackerCallback(serializedJson.size(), false);
|
|
|
|
return _webSocket->send(serializedJson).success;
|
|
}
|
|
|
|
//
|
|
// Extract the nonce from the handshake response
|
|
// use it to compute a hash during authentication
|
|
//
|
|
// {
|
|
// "action": "auth/handshake/ok",
|
|
// "body": {
|
|
// "data": {
|
|
// "nonce": "MTI0Njg4NTAyMjYxMzgxMzgzMg==",
|
|
// "version": "0.0.24"
|
|
// }
|
|
// }
|
|
// }
|
|
//
|
|
bool CobraConnection::handleHandshakeResponse(const Json::Value& pdu)
|
|
{
|
|
if (!pdu.isMember("body")) return false;
|
|
Json::Value body = pdu["body"];
|
|
|
|
if (!body.isMember("data")) return false;
|
|
Json::Value data = body["data"];
|
|
|
|
if (!data.isMember("nonce")) return false;
|
|
Json::Value nonce = data["nonce"];
|
|
|
|
if (!nonce.isString()) return false;
|
|
|
|
return sendAuthMessage(nonce.asString());
|
|
}
|
|
|
|
//
|
|
// Authenticate message schema.
|
|
//
|
|
// challenge = {
|
|
// "action": "auth/authenticate",
|
|
// "body": {
|
|
// "method": "role_secret",
|
|
// "credentials": {
|
|
// "hash": computeHash(secret, nonce)
|
|
// }
|
|
// },
|
|
// }
|
|
//
|
|
bool CobraConnection::sendAuthMessage(const std::string& nonce)
|
|
{
|
|
Json::Value credentials;
|
|
credentials["hash"] = hmac(nonce, _roleSecret);
|
|
|
|
Json::Value body;
|
|
body["credentials"] = credentials;
|
|
body["method"] = "role_secret";
|
|
|
|
Json::Value pdu;
|
|
pdu["action"] = "auth/authenticate";
|
|
pdu["body"] = body;
|
|
|
|
std::string serializedJson = serializeJson(pdu);
|
|
CobraConnection::invokeTrafficTrackerCallback(serializedJson.size(), false);
|
|
|
|
return _webSocket->send(serializedJson).success;
|
|
}
|
|
|
|
bool CobraConnection::handleSubscriptionResponse(const Json::Value& pdu)
|
|
{
|
|
if (!pdu.isMember("body")) return false;
|
|
Json::Value body = pdu["body"];
|
|
|
|
if (!body.isMember("subscription_id")) return false;
|
|
Json::Value subscriptionId = body["subscription_id"];
|
|
|
|
if (!subscriptionId.isString()) return false;
|
|
|
|
invokeEventCallback(ix::CobraConnection_EventType_Subscribed,
|
|
std::string(), WebSocketHttpHeaders(),
|
|
subscriptionId.asString());
|
|
return true;
|
|
}
|
|
|
|
bool CobraConnection::handleUnsubscriptionResponse(const Json::Value& pdu)
|
|
{
|
|
if (!pdu.isMember("body")) return false;
|
|
Json::Value body = pdu["body"];
|
|
|
|
if (!body.isMember("subscription_id")) return false;
|
|
Json::Value subscriptionId = body["subscription_id"];
|
|
|
|
if (!subscriptionId.isString()) return false;
|
|
|
|
invokeEventCallback(ix::CobraConnection_EventType_UnSubscribed,
|
|
std::string(), WebSocketHttpHeaders(),
|
|
subscriptionId.asString());
|
|
return true;
|
|
}
|
|
|
|
bool CobraConnection::handleSubscriptionData(const Json::Value& pdu)
|
|
{
|
|
if (!pdu.isMember("body")) return false;
|
|
Json::Value body = pdu["body"];
|
|
|
|
// Identify subscription_id, so that we can find
|
|
// which callback to execute
|
|
if (!body.isMember("subscription_id")) return false;
|
|
Json::Value subscriptionId = body["subscription_id"];
|
|
|
|
std::lock_guard<std::mutex> lock(_cbsMutex);
|
|
auto cb = _cbs.find(subscriptionId.asString());
|
|
if (cb == _cbs.end()) return false; // cannot find callback
|
|
|
|
// Extract messages now
|
|
if (!body.isMember("messages")) return false;
|
|
Json::Value messages = body["messages"];
|
|
|
|
for (auto&& msg : messages)
|
|
{
|
|
cb->second(msg);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CobraConnection::connect()
|
|
{
|
|
_webSocket->start();
|
|
return true;
|
|
}
|
|
|
|
bool CobraConnection::isConnected() const
|
|
{
|
|
return _webSocket->getReadyState() == ix::ReadyState::Open;
|
|
}
|
|
|
|
bool CobraConnection::isAuthenticated() const
|
|
{
|
|
return isConnected() && _authenticated;
|
|
}
|
|
|
|
std::string CobraConnection::serializeJson(const Json::Value& value)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_jsonWriterMutex);
|
|
return _jsonWriter.write(value);
|
|
}
|
|
|
|
//
|
|
// publish is not thread safe as we are trying to reuse some Json objects.
|
|
//
|
|
bool CobraConnection::publish(const Json::Value& channels,
|
|
const Json::Value& msg)
|
|
{
|
|
_body["channels"] = channels;
|
|
_body["message"] = msg;
|
|
_pdu["body"] = _body;
|
|
|
|
std::string serializedJson = serializeJson(_pdu);
|
|
|
|
if (_publishMode == CobraConnection_PublishMode_Batch)
|
|
{
|
|
enqueue(serializedJson);
|
|
return true;
|
|
}
|
|
|
|
//
|
|
// Fast path. We are authenticated and the publishing succeed
|
|
// This should happen for 99% of the cases.
|
|
//
|
|
if (_authenticated && publishMessage(serializedJson))
|
|
{
|
|
return true;
|
|
}
|
|
else // Or else we enqueue
|
|
// Slow code path is when we haven't connected yet (startup),
|
|
// or when the connection drops for some reason.
|
|
{
|
|
enqueue(serializedJson);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void CobraConnection::subscribe(const std::string& channel,
|
|
SubscriptionCallback cb)
|
|
{
|
|
// Create and send a subscribe pdu
|
|
Json::Value body;
|
|
body["channel"] = channel;
|
|
|
|
Json::Value pdu;
|
|
pdu["action"] = "rtm/subscribe";
|
|
pdu["body"] = body;
|
|
|
|
_webSocket->send(pdu.toStyledString());
|
|
|
|
// Set the callback
|
|
std::lock_guard<std::mutex> lock(_cbsMutex);
|
|
_cbs[channel] = cb;
|
|
}
|
|
|
|
void CobraConnection::unsubscribe(const std::string& channel)
|
|
{
|
|
{
|
|
std::lock_guard<std::mutex> lock(_cbsMutex);
|
|
auto cb = _cbs.find(channel);
|
|
if (cb == _cbs.end()) return;
|
|
|
|
_cbs.erase(cb);
|
|
}
|
|
|
|
// Create and send an unsubscribe pdu
|
|
Json::Value body;
|
|
body["subscription_id"] = channel;
|
|
|
|
Json::Value pdu;
|
|
pdu["action"] = "rtm/unsubscribe";
|
|
pdu["body"] = body;
|
|
|
|
_webSocket->send(pdu.toStyledString());
|
|
}
|
|
|
|
//
|
|
// Enqueue strategy drops old messages when we are at full capacity
|
|
//
|
|
// If we want to keep only 3 items max in the queue:
|
|
//
|
|
// enqueue(A) -> [A]
|
|
// enqueue(B) -> [B, A]
|
|
// enqueue(C) -> [C, B, A]
|
|
// enqueue(D) -> [D, C, B] -- now we drop A, the oldest message,
|
|
// -- and keep the 'fresh ones'
|
|
//
|
|
void CobraConnection::enqueue(const std::string& msg)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_queueMutex);
|
|
|
|
if (_messageQueue.size() == CobraConnection::kQueueMaxSize)
|
|
{
|
|
_messageQueue.pop_back();
|
|
}
|
|
_messageQueue.push_front(msg);
|
|
}
|
|
|
|
//
|
|
// We process messages back (oldest) to front (newest) to respect ordering
|
|
// when sending them. If we fail to send something, we put it back in the queue
|
|
// at the end we picked it up originally (at the end).
|
|
//
|
|
bool CobraConnection::flushQueue()
|
|
{
|
|
std::lock_guard<std::mutex> lock(_queueMutex);
|
|
|
|
while (!_messageQueue.empty())
|
|
{
|
|
auto&& msg = _messageQueue.back();
|
|
if (!publishMessage(msg))
|
|
{
|
|
_messageQueue.push_back(msg);
|
|
return false;
|
|
}
|
|
_messageQueue.pop_back();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CobraConnection::publishMessage(const std::string& serializedJson)
|
|
{
|
|
auto webSocketSendInfo = _webSocket->send(serializedJson);
|
|
CobraConnection::invokeTrafficTrackerCallback(webSocketSendInfo.wireSize,
|
|
false);
|
|
return webSocketSendInfo.success;
|
|
}
|
|
|
|
void CobraConnection::suspend()
|
|
{
|
|
disconnect();
|
|
}
|
|
|
|
void CobraConnection::resume()
|
|
{
|
|
connect();
|
|
}
|
|
|
|
} // namespace ix
|