mirror of
https://github.com/VCMP-SqMod/SqMod.git
synced 2025-02-12 15:57: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.
234 lines
5.9 KiB
C++
234 lines
5.9 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/HTTPSClientSession.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/SecureServerSocket.h"
|
|
#include "Poco/Net/NetException.h"
|
|
#include "Poco/Thread.h"
|
|
#include <iostream>
|
|
|
|
using Poco::Net::HTTPSClientSession;
|
|
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);
|
|
std::unique_ptr<char[]> pBuffer(new char[_bufSize]);
|
|
int flags;
|
|
int n;
|
|
do
|
|
{
|
|
n = ws.receiveFrame(pBuffer.get(), static_cast<int>(_bufSize), flags);
|
|
if (n == 0)
|
|
break;
|
|
ws.sendFrame(pBuffer.get(), n, flags);
|
|
}
|
|
while ((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::SecureServerSocket ss(0);
|
|
Poco::Net::HTTPServer server(new WebSocketRequestHandlerFactory, ss, new Poco::Net::HTTPServerParams);
|
|
server.start();
|
|
|
|
Poco::Thread::sleep(200);
|
|
|
|
HTTPSClientSession cs("127.0.0.1", ss.address().port());
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws");
|
|
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, 0, 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, 0, 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, 0, 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, 0, 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, 0, 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::SecureServerSocket ss(0);
|
|
Poco::Net::HTTPServer server(new WebSocketRequestHandlerFactory(msgSize), ss, new Poco::Net::HTTPServerParams);
|
|
server.start();
|
|
|
|
Poco::Thread::sleep(200);
|
|
|
|
HTTPSClientSession cs("127.0.0.1", ss.address().port());
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws");
|
|
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, 0, n) == 0);
|
|
|
|
server.stop();
|
|
}
|
|
|
|
|
|
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);
|
|
|
|
return pSuite;
|
|
}
|