2019-09-06 05:48:38 +02:00
|
|
|
/*
|
|
|
|
* cmd_satori_chat.cpp
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2017 Machine Zone. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
#include "IXTest.h"
|
|
|
|
#include "catch.hpp"
|
2019-09-06 05:48:38 +02:00
|
|
|
#include <chrono>
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <iostream>
|
2019-09-06 05:48:38 +02:00
|
|
|
#include <ixcobra/IXCobraConnection.h>
|
|
|
|
#include <ixcrypto/IXUuid.h>
|
2019-09-24 06:04:01 +02:00
|
|
|
#include <ixsnake/IXRedisServer.h>
|
|
|
|
#include <ixsnake/IXSnakeServer.h>
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
using namespace ix;
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
std::atomic<size_t> incomingBytes(0);
|
|
|
|
std::atomic<size_t> outgoingBytes(0);
|
|
|
|
|
|
|
|
void setupTrafficTrackerCallback()
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ix::CobraConnection::setTrafficTrackerCallback([](size_t size, bool incoming) {
|
|
|
|
if (incoming)
|
2019-09-06 05:48:38 +02:00
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
incomingBytes += size;
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
outgoingBytes += size;
|
|
|
|
}
|
|
|
|
});
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class SatoriChat
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
public:
|
|
|
|
SatoriChat(const std::string& user,
|
|
|
|
const std::string& session,
|
|
|
|
const std::string& endpoint);
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
void subscribe(const std::string& channel);
|
|
|
|
void start();
|
|
|
|
void stop();
|
|
|
|
void run();
|
|
|
|
bool isReady() const;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
void sendMessage(const std::string& text);
|
|
|
|
size_t getReceivedMessagesCount() const;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
bool hasPendingMessages() const;
|
|
|
|
Json::Value popMessage();
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
private:
|
|
|
|
std::string _user;
|
|
|
|
std::string _session;
|
|
|
|
std::string _endpoint;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
std::queue<Json::Value> _publish_queue;
|
|
|
|
mutable std::mutex _queue_mutex;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
std::thread _thread;
|
|
|
|
std::atomic<bool> _stop;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
ix::CobraConnection _conn;
|
|
|
|
std::atomic<bool> _connectedAndSubscribed;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
std::queue<Json::Value> _receivedQueue;
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
std::mutex _logMutex;
|
2019-09-06 05:48:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
SatoriChat::SatoriChat(const std::string& user,
|
2019-09-06 07:02:10 +02:00
|
|
|
const std::string& session,
|
2019-09-23 19:25:23 +02:00
|
|
|
const std::string& endpoint)
|
|
|
|
: _user(user)
|
|
|
|
, _session(session)
|
|
|
|
, _endpoint(endpoint)
|
|
|
|
, _stop(false)
|
|
|
|
, _connectedAndSubscribed(false)
|
2019-09-06 05:48:38 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void SatoriChat::start()
|
|
|
|
{
|
|
|
|
_thread = std::thread(&SatoriChat::run, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SatoriChat::stop()
|
|
|
|
{
|
|
|
|
_stop = true;
|
|
|
|
_thread.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SatoriChat::isReady() const
|
|
|
|
{
|
|
|
|
return _connectedAndSubscribed;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t SatoriChat::getReceivedMessagesCount() const
|
|
|
|
{
|
|
|
|
return _receivedQueue.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SatoriChat::hasPendingMessages() const
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(_queue_mutex);
|
|
|
|
return !_publish_queue.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
Json::Value SatoriChat::popMessage()
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(_queue_mutex);
|
|
|
|
auto msg = _publish_queue.front();
|
|
|
|
_publish_queue.pop();
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Callback to handle received messages, that are printed on the console
|
|
|
|
//
|
|
|
|
void SatoriChat::subscribe(const std::string& channel)
|
|
|
|
{
|
|
|
|
std::string filter;
|
2019-09-23 19:25:23 +02:00
|
|
|
_conn.subscribe(channel, filter, [this](const Json::Value& msg) {
|
2019-09-24 06:51:55 +02:00
|
|
|
spdlog::info("receive {}", msg.toStyledString());
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
if (!msg.isObject()) return;
|
|
|
|
if (!msg.isMember("user")) return;
|
|
|
|
if (!msg.isMember("text")) return;
|
|
|
|
if (!msg.isMember("session")) return;
|
|
|
|
|
|
|
|
std::string msg_user = msg["user"].asString();
|
|
|
|
std::string msg_text = msg["text"].asString();
|
|
|
|
std::string msg_session = msg["session"].asString();
|
|
|
|
|
|
|
|
// We are not interested in messages
|
|
|
|
// from a different session.
|
|
|
|
if (msg_session != _session) return;
|
|
|
|
|
|
|
|
// We are not interested in our own messages
|
|
|
|
if (msg_user == _user) return;
|
|
|
|
|
|
|
|
_receivedQueue.push(msg);
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << std::endl << msg_user << " > " << msg_text << std::endl << _user << " > ";
|
|
|
|
log(ss.str());
|
|
|
|
});
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SatoriChat::sendMessage(const std::string& text)
|
|
|
|
{
|
|
|
|
Json::Value msg;
|
|
|
|
msg["user"] = _user;
|
|
|
|
msg["session"] = _session;
|
|
|
|
msg["text"] = text;
|
|
|
|
|
|
|
|
std::unique_lock<std::mutex> lock(_queue_mutex);
|
|
|
|
_publish_queue.push(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Do satori communication on a background thread, where we can have
|
|
|
|
// something like an event loop that publish, poll and receive data
|
|
|
|
//
|
|
|
|
void SatoriChat::run()
|
|
|
|
{
|
|
|
|
// "chat" conf
|
|
|
|
std::string appkey("FC2F10139A2BAc53BB72D9db967b024f");
|
|
|
|
std::string channel = _session;
|
|
|
|
std::string role = "_sub";
|
|
|
|
std::string secret = "66B1dA3ED5fA074EB5AE84Dd8CE3b5ba";
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
_conn.configure(
|
|
|
|
appkey, _endpoint, role, secret, ix::WebSocketPerMessageDeflateOptions(true));
|
2019-09-06 05:48:38 +02:00
|
|
|
_conn.connect();
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
_conn.setEventCallback([this, channel](ix::CobraConnectionEventType eventType,
|
|
|
|
const std::string& errMsg,
|
2019-09-24 06:04:01 +02:00
|
|
|
const ix::WebSocketHttpHeaders& headers,
|
2019-09-23 19:25:23 +02:00
|
|
|
const std::string& subscriptionId,
|
|
|
|
CobraConnection::MsgId msgId) {
|
|
|
|
if (eventType == ix::CobraConnection_EventType_Open)
|
2019-09-06 05:48:38 +02:00
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
log("Subscriber connected: " + _user);
|
2019-09-24 06:04:01 +02:00
|
|
|
for (auto&& it : headers)
|
|
|
|
{
|
|
|
|
log("Headers " + it.first + " " + it.second);
|
|
|
|
}
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
else if (eventType == ix::CobraConnection_EventType_Authenticated)
|
|
|
|
{
|
|
|
|
log("Subscriber authenticated: " + _user);
|
|
|
|
subscribe(channel);
|
|
|
|
}
|
|
|
|
else if (eventType == ix::CobraConnection_EventType_Error)
|
|
|
|
{
|
|
|
|
log(errMsg + _user);
|
|
|
|
}
|
|
|
|
else if (eventType == ix::CobraConnection_EventType_Closed)
|
|
|
|
{
|
|
|
|
log("Connection closed: " + _user);
|
|
|
|
}
|
|
|
|
else if (eventType == ix::CobraConnection_EventType_Subscribed)
|
|
|
|
{
|
|
|
|
log("Subscription ok: " + _user + " subscription_id " + subscriptionId);
|
|
|
|
_connectedAndSubscribed = true;
|
|
|
|
}
|
|
|
|
else if (eventType == ix::CobraConnection_EventType_UnSubscribed)
|
|
|
|
{
|
|
|
|
log("Unsubscription ok: " + _user + " subscription_id " + subscriptionId);
|
|
|
|
}
|
|
|
|
else if (eventType == ix::CobraConnection_EventType_Published)
|
|
|
|
{
|
|
|
|
Logger() << "Subscriber: published message acked: " << msgId;
|
|
|
|
}
|
|
|
|
});
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
while (!_stop)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
while (hasPendingMessages())
|
|
|
|
{
|
|
|
|
auto msg = popMessage();
|
|
|
|
|
|
|
|
std::string text = msg["text"].asString();
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "Sending msg [" << text << "]";
|
|
|
|
log(ss.str());
|
2019-09-12 20:43:52 +02:00
|
|
|
|
2019-09-06 05:48:38 +02:00
|
|
|
Json::Value channels;
|
|
|
|
channels.append(channel);
|
|
|
|
_conn.publish(channels, msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ix::msleep(50);
|
|
|
|
}
|
|
|
|
|
|
|
|
_conn.unsubscribe(channel);
|
|
|
|
|
|
|
|
ix::msleep(50);
|
|
|
|
_conn.disconnect();
|
2019-09-12 20:43:52 +02:00
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
_conn.setEventCallback([](ix::CobraConnectionEventType /*eventType*/,
|
|
|
|
const std::string& /*errMsg*/,
|
|
|
|
const ix::WebSocketHttpHeaders& /*headers*/,
|
|
|
|
const std::string& /*subscriptionId*/,
|
|
|
|
CobraConnection::MsgId /*msgId*/) { ; });
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
} // namespace
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
TEST_CASE("Cobra_chat", "[cobra_chat]")
|
|
|
|
{
|
|
|
|
SECTION("Exchange and count sent/received messages.")
|
|
|
|
{
|
2019-09-06 07:02:10 +02:00
|
|
|
int port = getFreePort();
|
|
|
|
snake::AppConfig appConfig = makeSnakeServerConfig(port);
|
2019-09-24 06:04:01 +02:00
|
|
|
|
|
|
|
// Start a redis server
|
|
|
|
ix::RedisServer redisServer(appConfig.redisPort);
|
|
|
|
auto res = redisServer.listen();
|
|
|
|
REQUIRE(res.first);
|
|
|
|
redisServer.start();
|
|
|
|
|
|
|
|
// Start a snake server
|
2019-09-06 05:48:38 +02:00
|
|
|
snake::SnakeServer snakeServer(appConfig);
|
|
|
|
snakeServer.run();
|
|
|
|
|
|
|
|
int timeout;
|
|
|
|
setupTrafficTrackerCallback();
|
|
|
|
|
|
|
|
std::string session = ix::generateSessionId();
|
2019-09-06 07:02:10 +02:00
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "ws://localhost:" << port;
|
|
|
|
std::string endpoint = ss.str();
|
|
|
|
|
|
|
|
SatoriChat chatA("jean", session, endpoint);
|
|
|
|
SatoriChat chatB("paul", session, endpoint);
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
chatA.start();
|
|
|
|
chatB.start();
|
|
|
|
|
|
|
|
// Wait for all chat instance to be ready
|
|
|
|
timeout = 10 * 1000; // 10s
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
if (chatA.isReady() && chatB.isReady()) break;
|
|
|
|
ix::msleep(10);
|
2019-09-12 20:43:52 +02:00
|
|
|
|
2019-09-06 05:48:38 +02:00
|
|
|
timeout -= 10;
|
|
|
|
if (timeout <= 0)
|
|
|
|
{
|
2019-09-10 21:19:22 +02:00
|
|
|
snakeServer.stop();
|
2019-09-24 06:04:01 +02:00
|
|
|
redisServer.stop();
|
2019-09-06 05:48:38 +02:00
|
|
|
REQUIRE(false); // timeout
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a bit of extra time, for the subscription to be active
|
|
|
|
ix::msleep(1000);
|
|
|
|
|
|
|
|
chatA.sendMessage("from A1");
|
|
|
|
chatA.sendMessage("from A2");
|
|
|
|
chatA.sendMessage("from A3");
|
|
|
|
|
|
|
|
chatB.sendMessage("from B1");
|
|
|
|
chatB.sendMessage("from B2");
|
|
|
|
|
|
|
|
// 1. Wait for all messages to be sent
|
|
|
|
timeout = 10 * 1000; // 10s
|
|
|
|
while (chatA.hasPendingMessages() || chatB.hasPendingMessages())
|
|
|
|
{
|
|
|
|
ix::msleep(10);
|
2019-09-12 20:43:52 +02:00
|
|
|
|
2019-09-06 05:48:38 +02:00
|
|
|
timeout -= 10;
|
|
|
|
if (timeout <= 0)
|
|
|
|
{
|
2019-09-10 21:19:22 +02:00
|
|
|
snakeServer.stop();
|
2019-09-24 06:04:01 +02:00
|
|
|
redisServer.stop();
|
2019-09-06 05:48:38 +02:00
|
|
|
REQUIRE(false); // timeout
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Give us 1s for all messages to be received
|
|
|
|
ix::msleep(1000);
|
|
|
|
|
|
|
|
chatA.stop();
|
|
|
|
chatB.stop();
|
|
|
|
|
|
|
|
REQUIRE(chatA.getReceivedMessagesCount() == 2);
|
|
|
|
REQUIRE(chatB.getReceivedMessagesCount() == 3);
|
|
|
|
|
2019-09-24 06:51:55 +02:00
|
|
|
spdlog::info("Incoming bytes {}", incomingBytes);
|
|
|
|
spdlog::info("Outgoing bytes {}", outgoingBytes);
|
2019-09-06 05:48:38 +02:00
|
|
|
|
2019-09-24 06:51:55 +02:00
|
|
|
spdlog::info("Stopping snake server...");
|
2019-09-06 05:48:38 +02:00
|
|
|
snakeServer.stop();
|
2019-09-24 06:04:01 +02:00
|
|
|
|
2019-09-24 06:51:55 +02:00
|
|
|
spdlog::info("Stopping redis server...");
|
2019-09-24 06:04:01 +02:00
|
|
|
redisServer.stop();
|
2019-09-06 05:48:38 +02:00
|
|
|
}
|
|
|
|
}
|