mirror of
https://github.com/pocoproject/poco.git
synced 2025-10-28 19:51:58 +01:00
* add PMTU discovery #2329 * add socket gather/scatter capabilities #2330 (win, udp) * enable WSAPoll * add FastMemoryPool * add receiveFrom() with native args * allow copying of StringTokenizer * add AtomicFlag and SpinlockMutex * update .gitignore * UDPServer and client #2343 (windows) * fix warnings * fix warnings * regenerate Net VS solutions * regenerate CppUnit projects/solutions * clang fixes * gcc fixes * try to fix travis * more travis fixes * more travis fixes * handle UDPClient exception * fix makefiles and init order warnings * add UNIX gather/scatter sendto/recvfrom implementations and tests * run travis tests as sudo * try to run tests as sudo, 2nd attempt * fix warning * use mutex in reactor * lock-order-inversion in SocketReactor #2346 * add PMTU discovery #2329 (linux) * ICMPSocket does not check reply address #1921 * remove some ignored tests * add PMTU discovery #2329 (reconcile logic with #1921) * fix native receiveFrome() * reinstate ignoring of proxy errors * add testMTU to ignore list * add include atomic * NTPClient not checking reply address #2348 * some ICMP/MTU fixes * UDPSocketReader cleanup * resolve some socket inheritance warnings * add NTP time sync to ignored tests * SocketNotifier not thread-safe #2345 * prevent x64 samples build attempt for win32 * build TestApp and Library * fix ICMP tests * regen VS projects * regen VS projects and add missing 2012 files * remove debug prints
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;
|
|
}
|