299 lines
8.8 KiB
C++
299 lines
8.8 KiB
C++
/*
|
|
* ws_autobahn.cpp
|
|
* Author: Benjamin Sergeant
|
|
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
*/
|
|
|
|
//
|
|
// 1. First you need to generate a config file in a config folder,
|
|
// which can use a white list of test to execute (with globbing),
|
|
// or a black list of tests to ignore
|
|
//
|
|
// config/fuzzingserver.json
|
|
// {
|
|
// "url": "ws://127.0.0.1:9001",
|
|
// "outdir": "./reports/clients",
|
|
// "cases": ["2.*"],
|
|
// "exclude-cases": [
|
|
// ],
|
|
// "exclude-agent-cases": {}
|
|
// }
|
|
//
|
|
//
|
|
// 2 Run the test server (using docker)
|
|
// docker run -it --rm -v "${PWD}/config:/config" -v "${PWD}/reports:/reports" -p 9001:9001 --name
|
|
// fuzzingserver crossbario/autobahn-testsuite
|
|
//
|
|
// 3. Run this command
|
|
// ws autobahn -q --url ws://localhost:9001
|
|
//
|
|
// 4. A HTML report will be generated, you can inspect it to see if you are compliant or not
|
|
//
|
|
|
|
#include <atomic>
|
|
#include <condition_variable>
|
|
#include <ixwebsocket/IXSocket.h>
|
|
#include <ixwebsocket/IXWebSocket.h>
|
|
#include <mutex>
|
|
#include <spdlog/spdlog.h>
|
|
#include <sstream>
|
|
|
|
|
|
namespace
|
|
{
|
|
std::string truncate(const std::string& str, size_t n)
|
|
{
|
|
if (str.size() < n)
|
|
{
|
|
return str;
|
|
}
|
|
else
|
|
{
|
|
return str.substr(0, n) + "...";
|
|
}
|
|
}
|
|
} // namespace
|
|
|
|
namespace ix
|
|
{
|
|
class AutobahnTestCase
|
|
{
|
|
public:
|
|
AutobahnTestCase(const std::string& _url, bool quiet);
|
|
void run();
|
|
|
|
private:
|
|
void log(const std::string& msg);
|
|
|
|
std::string _url;
|
|
ix::WebSocket _webSocket;
|
|
|
|
bool _quiet;
|
|
|
|
std::mutex _mutex;
|
|
std::condition_variable _condition;
|
|
};
|
|
|
|
AutobahnTestCase::AutobahnTestCase(const std::string& url, bool quiet)
|
|
: _url(url)
|
|
, _quiet(quiet)
|
|
{
|
|
_webSocket.disableAutomaticReconnection();
|
|
|
|
// FIXME: this should be on by default
|
|
ix::WebSocketPerMessageDeflateOptions webSocketPerMessageDeflateOptions(
|
|
true, false, false, 15, 15);
|
|
_webSocket.setPerMessageDeflateOptions(webSocketPerMessageDeflateOptions);
|
|
}
|
|
|
|
void AutobahnTestCase::log(const std::string& msg)
|
|
{
|
|
if (!_quiet)
|
|
{
|
|
spdlog::info(msg);
|
|
}
|
|
}
|
|
|
|
void AutobahnTestCase::run()
|
|
{
|
|
_webSocket.setUrl(_url);
|
|
|
|
std::stringstream ss;
|
|
log(std::string("Connecting to url: ") + _url);
|
|
|
|
_webSocket.setOnMessageCallback([this](const ix::WebSocketMessagePtr& msg) {
|
|
std::stringstream ss;
|
|
if (msg->type == ix::WebSocketMessageType::Open)
|
|
{
|
|
log("autobahn: connected");
|
|
ss << "Uri: " << msg->openInfo.uri << std::endl;
|
|
ss << "Handshake Headers:" << std::endl;
|
|
for (auto it : msg->openInfo.headers)
|
|
{
|
|
ss << it.first << ": " << it.second << std::endl;
|
|
}
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Close)
|
|
{
|
|
ss << "autobahn: connection closed:";
|
|
ss << " code " << msg->closeInfo.code;
|
|
ss << " reason " << msg->closeInfo.reason << std::endl;
|
|
|
|
_condition.notify_one();
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Message)
|
|
{
|
|
ss << "Received " << msg->wireSize << " bytes" << std::endl;
|
|
|
|
ss << "autobahn: received message: " << truncate(msg->str, 40) << std::endl;
|
|
|
|
_webSocket.send(msg->str, msg->binary);
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Error)
|
|
{
|
|
ss << "Connection error: " << msg->errorInfo.reason << std::endl;
|
|
ss << "#retries: " << msg->errorInfo.retries << std::endl;
|
|
ss << "Wait time(ms): " << msg->errorInfo.wait_time << std::endl;
|
|
ss << "HTTP Status: " << msg->errorInfo.http_status << std::endl;
|
|
|
|
// And error can happen, in which case the test-case is marked done
|
|
_condition.notify_one();
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Fragment)
|
|
{
|
|
ss << "Received message fragment" << std::endl;
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Ping)
|
|
{
|
|
ss << "Received ping" << std::endl;
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Pong)
|
|
{
|
|
ss << "Received pong" << std::endl;
|
|
}
|
|
else
|
|
{
|
|
ss << "Invalid ix::WebSocketMessageType" << std::endl;
|
|
}
|
|
|
|
log(ss.str());
|
|
});
|
|
|
|
_webSocket.start();
|
|
|
|
log("Waiting for test completion ...");
|
|
std::unique_lock<std::mutex> lock(_mutex);
|
|
_condition.wait(lock);
|
|
|
|
_webSocket.stop();
|
|
}
|
|
|
|
bool generateReport(const std::string& url)
|
|
{
|
|
ix::WebSocket webSocket;
|
|
std::string reportUrl(url);
|
|
reportUrl += "/updateReports?agent=ixwebsocket";
|
|
webSocket.setUrl(reportUrl);
|
|
webSocket.disableAutomaticReconnection();
|
|
|
|
std::atomic<bool> success(true);
|
|
std::condition_variable condition;
|
|
|
|
webSocket.setOnMessageCallback([&condition, &success](const ix::WebSocketMessagePtr& msg) {
|
|
if (msg->type == ix::WebSocketMessageType::Close)
|
|
{
|
|
spdlog::info("Report generated");
|
|
condition.notify_one();
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Error)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Connection error: " << msg->errorInfo.reason << std::endl;
|
|
ss << "#retries: " << msg->errorInfo.retries << std::endl;
|
|
ss << "Wait time(ms): " << msg->errorInfo.wait_time << std::endl;
|
|
ss << "HTTP Status: " << msg->errorInfo.http_status << std::endl;
|
|
spdlog::info(ss.str());
|
|
|
|
success = false;
|
|
}
|
|
});
|
|
|
|
webSocket.start();
|
|
std::mutex mutex;
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
condition.wait(lock);
|
|
webSocket.stop();
|
|
|
|
if (!success)
|
|
{
|
|
spdlog::error("Cannot generate report at url {}", reportUrl);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
int getTestCaseCount(const std::string& url)
|
|
{
|
|
ix::WebSocket webSocket;
|
|
std::string caseCountUrl(url);
|
|
caseCountUrl += "/getCaseCount";
|
|
webSocket.setUrl(caseCountUrl);
|
|
webSocket.disableAutomaticReconnection();
|
|
|
|
int count = -1;
|
|
std::condition_variable condition;
|
|
|
|
webSocket.setOnMessageCallback([&condition, &count](const ix::WebSocketMessagePtr& msg) {
|
|
if (msg->type == ix::WebSocketMessageType::Close)
|
|
{
|
|
condition.notify_one();
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Error)
|
|
{
|
|
std::stringstream ss;
|
|
ss << "Connection error: " << msg->errorInfo.reason << std::endl;
|
|
ss << "#retries: " << msg->errorInfo.retries << std::endl;
|
|
ss << "Wait time(ms): " << msg->errorInfo.wait_time << std::endl;
|
|
ss << "HTTP Status: " << msg->errorInfo.http_status << std::endl;
|
|
spdlog::info(ss.str());
|
|
|
|
condition.notify_one();
|
|
}
|
|
else if (msg->type == ix::WebSocketMessageType::Message)
|
|
{
|
|
// response is a string
|
|
std::stringstream ss;
|
|
ss << msg->str;
|
|
ss >> count;
|
|
}
|
|
});
|
|
|
|
webSocket.start();
|
|
std::mutex mutex;
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
condition.wait(lock);
|
|
webSocket.stop();
|
|
|
|
if (count == -1)
|
|
{
|
|
spdlog::error("Cannot retrieve test case count at url {}", caseCountUrl);
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
//
|
|
// make && bench ws autobahn --url ws://localhost:9001
|
|
//
|
|
int ws_autobahn_main(const std::string& url, bool quiet)
|
|
{
|
|
int testCasesCount = getTestCaseCount(url);
|
|
spdlog::info("Test cases count: {}", testCasesCount);
|
|
|
|
if (testCasesCount == -1)
|
|
{
|
|
spdlog::error("Cannot retrieve test case count at url {}", url);
|
|
return 1;
|
|
}
|
|
|
|
testCasesCount++;
|
|
|
|
for (int i = 1; i < testCasesCount; ++i)
|
|
{
|
|
spdlog::info("Execute test case {}", i);
|
|
|
|
int caseNumber = i;
|
|
|
|
std::stringstream ss;
|
|
ss << url << "/runCase?case=" << caseNumber << "&agent=ixwebsocket";
|
|
|
|
std::string url(ss.str());
|
|
|
|
AutobahnTestCase testCase(url, quiet);
|
|
testCase.run();
|
|
}
|
|
|
|
return generateReport(url) ? 0 : 1;
|
|
}
|
|
} // namespace ix
|