mirror of
https://github.com/VCMP-SqMod/SqMod.git
synced 2025-02-15 01:07:12 +01:00
Switched to POCO library for unified platform/library interface. Deprecated the external module API. It was creating more problems than solving. Removed most built-in libraries in favor of system libraries for easier maintenance. Cleaned and secured code with help from static analyzers.
291 lines
7.8 KiB
C++
291 lines
7.8 KiB
C++
//
|
|
// WebSocketTest.cpp
|
|
//
|
|
// Copyright (c) 2012, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#include "WebSocketTest.h"
|
|
#include "CppUnit/TestCaller.h"
|
|
#include "CppUnit/TestSuite.h"
|
|
#include "Poco/Net/WebSocket.h"
|
|
#include "Poco/Net/SocketStream.h"
|
|
#include "Poco/Net/HTTPClientSession.h"
|
|
#include "Poco/Net/HTTPServer.h"
|
|
#include "Poco/Net/HTTPServerParams.h"
|
|
#include "Poco/Net/HTTPRequestHandler.h"
|
|
#include "Poco/Net/HTTPRequestHandlerFactory.h"
|
|
#include "Poco/Net/HTTPServerRequest.h"
|
|
#include "Poco/Net/HTTPServerResponse.h"
|
|
#include "Poco/Net/ServerSocket.h"
|
|
#include "Poco/Net/NetException.h"
|
|
#include "Poco/Thread.h"
|
|
#include "Poco/Buffer.h"
|
|
|
|
|
|
using Poco::Net::HTTPClientSession;
|
|
using Poco::Net::HTTPRequest;
|
|
using Poco::Net::HTTPResponse;
|
|
using Poco::Net::HTTPServerRequest;
|
|
using Poco::Net::HTTPServerResponse;
|
|
using Poco::Net::SocketStream;
|
|
using Poco::Net::WebSocket;
|
|
using Poco::Net::WebSocketException;
|
|
|
|
|
|
namespace
|
|
{
|
|
class WebSocketRequestHandler: public Poco::Net::HTTPRequestHandler
|
|
{
|
|
public:
|
|
WebSocketRequestHandler(std::size_t bufSize = 1024): _bufSize(bufSize)
|
|
{
|
|
}
|
|
|
|
void handleRequest(HTTPServerRequest& request, HTTPServerResponse& response)
|
|
{
|
|
try
|
|
{
|
|
WebSocket ws(request, response);
|
|
Poco::Buffer<char> buffer(_bufSize);
|
|
int flags;
|
|
int n;
|
|
do
|
|
{
|
|
n = ws.receiveFrame(buffer.begin(), static_cast<int>(buffer.size()), flags);
|
|
ws.sendFrame(buffer.begin(), n, flags);
|
|
}
|
|
while (n > 0 || (flags & WebSocket::FRAME_OP_BITMASK) != WebSocket::FRAME_OP_CLOSE);
|
|
}
|
|
catch (WebSocketException& exc)
|
|
{
|
|
switch (exc.code())
|
|
{
|
|
case WebSocket::WS_ERR_HANDSHAKE_UNSUPPORTED_VERSION:
|
|
response.set("Sec-WebSocket-Version", WebSocket::WEBSOCKET_VERSION);
|
|
// fallthrough
|
|
case WebSocket::WS_ERR_NO_HANDSHAKE:
|
|
case WebSocket::WS_ERR_HANDSHAKE_NO_VERSION:
|
|
case WebSocket::WS_ERR_HANDSHAKE_NO_KEY:
|
|
response.setStatusAndReason(HTTPResponse::HTTP_BAD_REQUEST);
|
|
response.setContentLength(0);
|
|
response.send();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
std::size_t _bufSize;
|
|
};
|
|
|
|
class WebSocketRequestHandlerFactory: public Poco::Net::HTTPRequestHandlerFactory
|
|
{
|
|
public:
|
|
WebSocketRequestHandlerFactory(std::size_t bufSize = 1024): _bufSize(bufSize)
|
|
{
|
|
}
|
|
|
|
Poco::Net::HTTPRequestHandler* createRequestHandler(const HTTPServerRequest& request)
|
|
{
|
|
return new WebSocketRequestHandler(_bufSize);
|
|
}
|
|
|
|
private:
|
|
std::size_t _bufSize;
|
|
};
|
|
}
|
|
|
|
|
|
WebSocketTest::WebSocketTest(const std::string& name): CppUnit::TestCase(name)
|
|
{
|
|
}
|
|
|
|
|
|
WebSocketTest::~WebSocketTest()
|
|
{
|
|
}
|
|
|
|
|
|
void WebSocketTest::testWebSocket()
|
|
{
|
|
Poco::Net::ServerSocket ss(0);
|
|
Poco::Net::HTTPServer server(new WebSocketRequestHandlerFactory, ss, new Poco::Net::HTTPServerParams);
|
|
server.start();
|
|
|
|
Poco::Thread::sleep(200);
|
|
|
|
HTTPClientSession cs("127.0.0.1", ss.address().port());
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws", HTTPRequest::HTTP_1_1);
|
|
HTTPResponse response;
|
|
WebSocket ws(cs, request, response);
|
|
|
|
std::string payload("x");
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
char buffer[1024] = {};
|
|
int flags;
|
|
int n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
|
|
for (int i = 2; i < 20; i++)
|
|
{
|
|
payload.assign(i, 'x');
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
Poco::Buffer<char> pocobuffer(0);
|
|
assertTrue(0 == pocobuffer.size());
|
|
n = ws.receiveFrame(pocobuffer, flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (n == pocobuffer.size());
|
|
assertTrue (payload.compare(0, payload.size(), pocobuffer.begin(), n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
}
|
|
|
|
for (int i = 125; i < 129; i++)
|
|
{
|
|
payload.assign(i, 'x');
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
Poco::Buffer<char> pocobuffer(0);
|
|
n = ws.receiveFrame(pocobuffer, flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), pocobuffer.begin(), n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
}
|
|
|
|
payload = "Hello, world!";
|
|
ws.sendFrame(payload.data(), (int) payload.size());
|
|
n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_TEXT);
|
|
|
|
payload = "Hello, universe!";
|
|
ws.sendFrame(payload.data(), (int) payload.size(), WebSocket::FRAME_BINARY);
|
|
n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
assertTrue (flags == WebSocket::FRAME_BINARY);
|
|
|
|
ws.shutdown();
|
|
n = ws.receiveFrame(buffer, sizeof(buffer), flags);
|
|
assertTrue (n == 2);
|
|
assertTrue ((flags & WebSocket::FRAME_OP_BITMASK) == WebSocket::FRAME_OP_CLOSE);
|
|
|
|
server.stop();
|
|
}
|
|
|
|
|
|
void WebSocketTest::testWebSocketLarge()
|
|
{
|
|
const int msgSize = 64000;
|
|
|
|
Poco::Net::ServerSocket ss(0);
|
|
Poco::Net::HTTPServer server(new WebSocketRequestHandlerFactory(msgSize), ss, new Poco::Net::HTTPServerParams);
|
|
server.start();
|
|
|
|
Poco::Thread::sleep(200);
|
|
|
|
HTTPClientSession cs("127.0.0.1", ss.address().port());
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws", HTTPRequest::HTTP_1_1);
|
|
HTTPResponse response;
|
|
WebSocket ws(cs, request, response);
|
|
ws.setSendBufferSize(msgSize);
|
|
ws.setReceiveBufferSize(msgSize);
|
|
std::string payload(msgSize, 'x');
|
|
SocketStream sstr(ws);
|
|
sstr << payload;
|
|
sstr.flush();
|
|
|
|
char buffer[msgSize + 1] = {};
|
|
int flags;
|
|
int n = 0;
|
|
do
|
|
{
|
|
n += ws.receiveFrame(buffer + n, sizeof(buffer) - n, flags);
|
|
} while (n > 0 && n < msgSize);
|
|
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer, n) == 0);
|
|
}
|
|
|
|
|
|
void WebSocketTest::testOneLargeFrame(int msgSize)
|
|
{
|
|
Poco::Net::ServerSocket ss(0);
|
|
Poco::Net::HTTPServer server(new WebSocketRequestHandlerFactory(msgSize), ss, new Poco::Net::HTTPServerParams);
|
|
server.start();
|
|
|
|
Poco::Thread::sleep(200);
|
|
|
|
HTTPClientSession cs("127.0.0.1", ss.address().port());
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws", HTTPRequest::HTTP_1_1);
|
|
HTTPResponse response;
|
|
WebSocket ws(cs, request, response);
|
|
ws.setSendBufferSize(msgSize);
|
|
ws.setReceiveBufferSize(msgSize);
|
|
std::string payload(msgSize, 'x');
|
|
|
|
ws.sendFrame(payload.data(), msgSize);
|
|
|
|
Poco::Buffer<char> buffer(msgSize);
|
|
int flags;
|
|
int n;
|
|
|
|
n = ws.receiveFrame(buffer.begin(), static_cast<int>(buffer.size()), flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), buffer.begin(), n) == 0);
|
|
|
|
ws.sendFrame(payload.data(), msgSize);
|
|
|
|
Poco::Buffer<char> pocobuffer(0);
|
|
|
|
n = ws.receiveFrame(pocobuffer, flags);
|
|
assertTrue (n == payload.size());
|
|
assertTrue (payload.compare(0, payload.size(), pocobuffer.begin(), n) == 0);
|
|
}
|
|
|
|
|
|
void WebSocketTest::testWebSocketLargeInOneFrame()
|
|
{
|
|
testOneLargeFrame(64000);
|
|
testOneLargeFrame(70000);
|
|
}
|
|
|
|
|
|
void WebSocketTest::setUp()
|
|
{
|
|
}
|
|
|
|
|
|
void WebSocketTest::tearDown()
|
|
{
|
|
}
|
|
|
|
|
|
CppUnit::Test* WebSocketTest::suite()
|
|
{
|
|
CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("WebSocketTest");
|
|
|
|
CppUnit_addTest(pSuite, WebSocketTest, testWebSocket);
|
|
CppUnit_addTest(pSuite, WebSocketTest, testWebSocketLarge);
|
|
CppUnit_addTest(pSuite, WebSocketTest, testWebSocketLargeInOneFrame);
|
|
|
|
return pSuite;
|
|
}
|