2018-12-30 03:33:15 +01:00
|
|
|
/*
|
|
|
|
* IXTest.cpp
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2018 Machine Zone. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "IXTest.h"
|
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
#include <fstream>
|
2019-03-22 23:33:04 +01:00
|
|
|
#include <iomanip>
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <iostream>
|
2020-10-27 03:18:55 +01:00
|
|
|
#include <ixcrypto/IXUuid.h>
|
2020-11-07 18:34:47 +01:00
|
|
|
#include <ixwebsocket/IXNetSystem.h>
|
|
|
|
#include <ixwebsocket/IXWebSocket.h>
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <mutex>
|
2019-04-20 01:50:04 +02:00
|
|
|
#include <random>
|
2019-09-23 19:25:23 +02:00
|
|
|
#include <stack>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string>
|
|
|
|
#include <thread>
|
2019-04-20 01:50:04 +02:00
|
|
|
|
2018-12-30 03:33:15 +01:00
|
|
|
|
|
|
|
namespace ix
|
|
|
|
{
|
|
|
|
std::atomic<size_t> incomingBytes(0);
|
|
|
|
std::atomic<size_t> outgoingBytes(0);
|
2020-03-12 17:07:01 +01:00
|
|
|
std::mutex TLogger::_mutex;
|
2019-01-29 00:14:49 +01:00
|
|
|
std::stack<int> freePorts;
|
2018-12-30 03:33:15 +01:00
|
|
|
|
|
|
|
void setupWebSocketTrafficTrackerCallback()
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ix::WebSocket::setTrafficTrackerCallback([](size_t size, bool incoming) {
|
|
|
|
if (incoming)
|
|
|
|
{
|
|
|
|
incomingBytes += size;
|
|
|
|
}
|
|
|
|
else
|
2018-12-30 03:33:15 +01:00
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
outgoingBytes += size;
|
2018-12-30 03:33:15 +01:00
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
});
|
2018-12-30 03:33:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void reportWebSocketTraffic()
|
|
|
|
{
|
2020-03-12 17:07:01 +01:00
|
|
|
TLogger() << incomingBytes;
|
|
|
|
TLogger() << "Incoming bytes: " << incomingBytes;
|
|
|
|
TLogger() << "Outgoing bytes: " << outgoingBytes;
|
2018-12-30 03:33:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void msleep(int ms)
|
|
|
|
{
|
|
|
|
std::chrono::duration<double, std::milli> duration(ms);
|
|
|
|
std::this_thread::sleep_for(duration);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string generateSessionId()
|
|
|
|
{
|
|
|
|
auto now = std::chrono::system_clock::now();
|
2019-02-21 03:59:07 +01:00
|
|
|
auto seconds =
|
2019-09-23 19:25:23 +02:00
|
|
|
std::chrono::duration_cast<std::chrono::seconds>(now.time_since_epoch()).count();
|
2018-12-30 03:33:15 +01:00
|
|
|
|
|
|
|
return std::to_string(seconds);
|
|
|
|
}
|
2019-01-08 03:04:28 +01:00
|
|
|
|
|
|
|
void log(const std::string& msg)
|
|
|
|
{
|
2020-03-12 17:07:01 +01:00
|
|
|
TLogger() << msg;
|
2019-01-08 03:04:28 +01:00
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
void hexDump(const std::string& prefix, const std::string& s)
|
2019-03-22 23:33:04 +01:00
|
|
|
{
|
|
|
|
std::ostringstream ss;
|
|
|
|
bool upper_case = false;
|
|
|
|
|
|
|
|
for (std::string::size_type i = 0; i < s.length(); ++i)
|
|
|
|
{
|
2019-09-23 19:25:23 +02:00
|
|
|
ss << std::hex << std::setfill('0') << std::setw(2)
|
|
|
|
<< (upper_case ? std::uppercase : std::nouppercase) << (int) s[i];
|
2019-03-22 23:33:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << prefix << ": " << s << " => " << ss.str() << std::endl;
|
|
|
|
}
|
2019-05-16 23:25:31 +02:00
|
|
|
|
|
|
|
bool startWebSocketEchoServer(ix::WebSocketServer& server)
|
|
|
|
{
|
2020-07-24 21:49:36 +02:00
|
|
|
server.setOnClientMessageCallback(
|
2020-08-28 23:55:40 +02:00
|
|
|
[&server](std::shared_ptr<ConnectionState> connectionState,
|
2020-07-24 21:49:36 +02:00
|
|
|
WebSocket& webSocket,
|
|
|
|
const ix::WebSocketMessagePtr& msg) {
|
2020-08-28 23:55:40 +02:00
|
|
|
auto remoteIp = connectionState->getRemoteIp();
|
2020-07-24 04:29:41 +02:00
|
|
|
if (msg->type == ix::WebSocketMessageType::Open)
|
|
|
|
{
|
|
|
|
TLogger() << "New connection";
|
|
|
|
TLogger() << "Remote ip: " << remoteIp;
|
|
|
|
TLogger() << "Uri: " << msg->openInfo.uri;
|
|
|
|
TLogger() << "Headers:";
|
|
|
|
for (auto it : msg->openInfo.headers)
|
2019-05-16 23:25:31 +02:00
|
|
|
{
|
2020-07-24 04:29:41 +02:00
|
|
|
TLogger() << it.first << ": " << it.second;
|
2019-09-23 19:25:23 +02:00
|
|
|
}
|
2020-07-24 04:29:41 +02:00
|
|
|
}
|
|
|
|
else if (msg->type == ix::WebSocketMessageType::Close)
|
|
|
|
{
|
|
|
|
TLogger() << "Closed connection";
|
|
|
|
}
|
|
|
|
else if (msg->type == ix::WebSocketMessageType::Message)
|
|
|
|
{
|
|
|
|
for (auto&& client : server.getClients())
|
2019-09-23 19:25:23 +02:00
|
|
|
{
|
2020-07-24 21:49:36 +02:00
|
|
|
if (client.get() != &webSocket)
|
2019-05-16 23:25:31 +02:00
|
|
|
{
|
2020-07-24 04:29:41 +02:00
|
|
|
client->send(msg->str, msg->binary);
|
2019-05-16 23:25:31 +02:00
|
|
|
}
|
|
|
|
}
|
2020-07-24 04:29:41 +02:00
|
|
|
}
|
|
|
|
});
|
2019-05-16 23:25:31 +02:00
|
|
|
|
|
|
|
auto res = server.listen();
|
|
|
|
if (!res.first)
|
|
|
|
{
|
2020-03-12 17:07:01 +01:00
|
|
|
TLogger() << res.second;
|
2019-05-16 23:25:31 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
server.start();
|
|
|
|
return true;
|
|
|
|
}
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
std::vector<uint8_t> load(const std::string& path)
|
|
|
|
{
|
|
|
|
std::vector<uint8_t> memblock;
|
|
|
|
|
|
|
|
std::ifstream file(path);
|
|
|
|
if (!file.is_open()) return memblock;
|
|
|
|
|
|
|
|
file.seekg(0, file.end);
|
|
|
|
std::streamoff size = file.tellg();
|
|
|
|
file.seekg(0, file.beg);
|
|
|
|
|
|
|
|
memblock.resize((size_t) size);
|
2019-09-23 19:25:23 +02:00
|
|
|
file.read((char*) &memblock.front(), static_cast<std::streamsize>(size));
|
2019-09-06 05:48:38 +02:00
|
|
|
|
|
|
|
return memblock;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string readAsString(const std::string& path)
|
|
|
|
{
|
|
|
|
auto vec = load(path);
|
|
|
|
return std::string(vec.begin(), vec.end());
|
|
|
|
}
|
2020-03-21 01:00:18 +01:00
|
|
|
|
2020-03-20 20:21:45 +01:00
|
|
|
SocketTLSOptions makeClientTLSOptions()
|
|
|
|
{
|
|
|
|
SocketTLSOptions tlsOptionsClient;
|
|
|
|
tlsOptionsClient.certFile = ".certs/trusted-client-crt.pem";
|
|
|
|
tlsOptionsClient.keyFile = ".certs/trusted-client-key.pem";
|
|
|
|
tlsOptionsClient.caFile = ".certs/trusted-ca-crt.pem";
|
|
|
|
|
|
|
|
return tlsOptionsClient;
|
|
|
|
}
|
|
|
|
|
|
|
|
SocketTLSOptions makeServerTLSOptions(bool preferTLS)
|
|
|
|
{
|
|
|
|
// Start a fake sentry http server
|
|
|
|
SocketTLSOptions tlsOptionsServer;
|
|
|
|
tlsOptionsServer.certFile = ".certs/trusted-server-crt.pem";
|
|
|
|
tlsOptionsServer.keyFile = ".certs/trusted-server-key.pem";
|
|
|
|
tlsOptionsServer.caFile = ".certs/trusted-ca-crt.pem";
|
|
|
|
|
|
|
|
#if defined(IXWEBSOCKET_USE_MBED_TLS) || defined(IXWEBSOCKET_USE_OPEN_SSL)
|
|
|
|
tlsOptionsServer.tls = preferTLS;
|
|
|
|
#else
|
|
|
|
tlsOptionsServer.tls = false;
|
|
|
|
#endif
|
|
|
|
return tlsOptionsServer;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string getHttpScheme()
|
|
|
|
{
|
|
|
|
#if defined(IXWEBSOCKET_USE_MBED_TLS) || defined(IXWEBSOCKET_USE_OPEN_SSL)
|
|
|
|
std::string scheme("https://");
|
|
|
|
#else
|
|
|
|
std::string scheme("http://");
|
|
|
|
#endif
|
|
|
|
return scheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string getWsScheme(bool preferTLS)
|
|
|
|
{
|
|
|
|
std::string scheme;
|
|
|
|
#if defined(IXWEBSOCKET_USE_MBED_TLS) || defined(IXWEBSOCKET_USE_OPEN_SSL)
|
|
|
|
if (preferTLS)
|
|
|
|
{
|
|
|
|
scheme = "wss://";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
scheme = "ws://";
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
scheme = "ws://";
|
|
|
|
#endif
|
|
|
|
return scheme;
|
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
} // namespace ix
|