7183a3d3de
Change-Id: Ice934380029bf27054e830fffc07a5d037d1430f Signed-off-by: Michael Meeks <michael.meeks@collabora.com>
290 lines
8 KiB
C++
290 lines
8 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
|
|
/*
|
|
* Copyright the Collabora Online contributors.
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <atomic>
|
|
#include <cerrno>
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <iostream>
|
|
#include <mutex>
|
|
#include <thread>
|
|
#include <assert.h>
|
|
|
|
#include <Poco/Net/HTMLForm.h>
|
|
#include <Poco/Net/HTTPClientSession.h>
|
|
#include <Poco/Net/HTTPSClientSession.h>
|
|
#include <Poco/Net/HTTPRequest.h>
|
|
#include <Poco/Net/HTTPResponse.h>
|
|
#include <Poco/Net/FilePartSource.h>
|
|
#include <Poco/Net/SSLManager.h>
|
|
#include <Poco/Net/WebSocket.h>
|
|
#include <Poco/Net/AcceptCertificateHandler.h>
|
|
#include <Poco/StreamCopier.h>
|
|
#include <Poco/URI.h>
|
|
#include <Poco/Util/Application.h>
|
|
#include <Poco/Util/HelpFormatter.h>
|
|
#include <Poco/Util/Option.h>
|
|
#include <Poco/Util/OptionSet.h>
|
|
#include <Poco/Runnable.h>
|
|
|
|
#include <Util.hpp>
|
|
|
|
using Poco::Net::HTTPClientSession;
|
|
using Poco::Net::HTTPRequest;
|
|
using Poco::Net::HTTPResponse;
|
|
using Poco::Net::WebSocket;
|
|
using Poco::Runnable;
|
|
using Poco::Util::Application;
|
|
|
|
const char *HostName = "127.0.0.1";
|
|
constexpr int HttpPortNumber = 9191;
|
|
constexpr int SslPortNumber = 9193;
|
|
|
|
static bool EnableHttps = false;
|
|
bool EnableExperimental = false;
|
|
|
|
struct Session
|
|
{
|
|
private:
|
|
std::string _session_name;
|
|
Poco::Net::HTTPClientSession *_session;
|
|
|
|
public:
|
|
Session(const char *session_name, bool https = false)
|
|
: _session_name(session_name)
|
|
{
|
|
if (https)
|
|
_session = new Poco::Net::HTTPSClientSession(HostName, SslPortNumber);
|
|
else
|
|
_session = new Poco::Net::HTTPClientSession(HostName, HttpPortNumber);
|
|
}
|
|
~Session()
|
|
{
|
|
delete _session;
|
|
}
|
|
|
|
void sendPing(int i)
|
|
{
|
|
Poco::Net::HTTPRequest request(
|
|
Poco::Net::HTTPRequest::HTTP_POST,
|
|
"/ping/" + _session_name + '/' + std::to_string(i));
|
|
try {
|
|
Poco::Net::HTMLForm form;
|
|
form.setEncoding(Poco::Net::HTMLForm::ENCODING_MULTIPART);
|
|
form.prepareSubmit(request);
|
|
form.write(_session->sendRequest(request));
|
|
}
|
|
catch (const Poco::Exception &e)
|
|
{
|
|
std::cerr << "Failed to write data: " << e.name() <<
|
|
' ' << e.message() << '\n';
|
|
throw;
|
|
}
|
|
}
|
|
|
|
std::string getResponseString()
|
|
{
|
|
Poco::Net::HTTPResponse response;
|
|
std::istream& responseStream = _session->receiveResponse(response);
|
|
const std::string result(std::istreambuf_iterator<char>(responseStream), {});
|
|
// std::cerr << "Got response '" << result << "'\n";
|
|
return result;
|
|
}
|
|
|
|
int getResponseInt()
|
|
{
|
|
int number = 42;
|
|
|
|
try {
|
|
// std::cerr << "try to get response\n";
|
|
const std::string result = getResponseString();
|
|
number = std::stoi(result);
|
|
}
|
|
catch (const Poco::Exception &e)
|
|
{
|
|
std::cerr << "Exception converting: " << e.name() <<
|
|
' ' << e.message() << '\n';
|
|
throw;
|
|
}
|
|
return number;
|
|
}
|
|
|
|
std::shared_ptr<WebSocket> getWebSocket()
|
|
{
|
|
_session->setTimeout(Poco::Timespan(10, 0));
|
|
HTTPRequest request(HTTPRequest::HTTP_GET, "/ws");
|
|
HTTPResponse response;
|
|
return std::make_shared<WebSocket>(*_session, request, response);
|
|
}
|
|
};
|
|
|
|
struct ThreadWorker : public Runnable
|
|
{
|
|
private:
|
|
const char *_domain;
|
|
public:
|
|
ThreadWorker(const char *domain = nullptr)
|
|
: _domain(domain)
|
|
{
|
|
}
|
|
virtual void run()
|
|
{
|
|
for (int i = 0; i < 100; ++i)
|
|
{
|
|
Session ping(_domain ? _domain : "init", EnableHttps);
|
|
ping.sendPing(i);
|
|
#ifndef NDEBUG
|
|
int back =
|
|
#endif
|
|
ping.getResponseInt();
|
|
assert(back == i + 1);
|
|
}
|
|
}
|
|
};
|
|
|
|
struct Client : public Poco::Util::Application
|
|
{
|
|
void testPing()
|
|
{
|
|
std::cerr << "testPing\n";
|
|
Session first("init", EnableHttps);
|
|
Session second("init", EnableHttps);
|
|
|
|
int count = 42, back;
|
|
first.sendPing(count);
|
|
second.sendPing(count + 1);
|
|
|
|
back = first.getResponseInt();
|
|
std::cerr << "testPing: " << back << '\n';
|
|
assert (back == count + 1);
|
|
|
|
back = second.getResponseInt();
|
|
std::cerr << "testPing: " << back << '\n';
|
|
assert (back == count + 2);
|
|
}
|
|
|
|
void testLadder()
|
|
{
|
|
std::cerr << "testLadder\n";
|
|
ThreadWorker ladder;
|
|
std::thread thread([&ladder]{ladder.run();});
|
|
thread.join();
|
|
}
|
|
|
|
void testParallel()
|
|
{
|
|
std::cerr << "testParallel\n";
|
|
const int num = 10;
|
|
std::thread snakes[num];
|
|
ThreadWorker ladders[num];
|
|
|
|
for (size_t i = 0; i < num; i++)
|
|
snakes[i] = std::thread([&ladders, i]{ladders[i].run();});
|
|
|
|
for (int i = 0; i < num; i++)
|
|
snakes[i].join();
|
|
}
|
|
|
|
void testWebsocketPingPong()
|
|
{
|
|
std::cerr << "testWebsocketPingPong\n";
|
|
Session session("ws", EnableHttps);
|
|
std::shared_ptr<WebSocket> ws = session.getWebSocket();
|
|
|
|
std::string send = "hello there";
|
|
ws->sendFrame(&send[0], send.length(),
|
|
WebSocket::SendFlags::FRAME_TEXT);
|
|
|
|
for (size_t i = 0; i < 10; i++)
|
|
{
|
|
ws->sendFrame(&i, sizeof(i), WebSocket::SendFlags::FRAME_BINARY);
|
|
size_t back[5];
|
|
int flags = 0;
|
|
#ifndef NDEBUG
|
|
int recvd =
|
|
#endif
|
|
ws->receiveFrame((void *)back, sizeof(back), flags);
|
|
assert(recvd == sizeof(size_t));
|
|
assert(back[0] == i + 1);
|
|
}
|
|
}
|
|
|
|
void testWebsocketEcho()
|
|
{
|
|
std::cerr << "testwebsocketEcho\n";
|
|
Session session("ws", EnableHttps);
|
|
std::shared_ptr<WebSocket> ws = session.getWebSocket();
|
|
|
|
std::vector<char> res;
|
|
|
|
std::srand(std::time(nullptr));
|
|
|
|
std::vector<char> data;
|
|
for (size_t i = 1; i < (1 << 14); ++i)
|
|
{
|
|
data.push_back((char)(Util::rng::getNext() / (UINT_MAX/256)));
|
|
ws->sendFrame(data.data(), data.size(), WebSocket::SendFlags::FRAME_BINARY);
|
|
|
|
res.resize(i);
|
|
int flags;
|
|
#ifndef NDEBUG
|
|
int recvd =
|
|
#endif
|
|
ws->receiveFrame(res.data(), res.size(), flags);
|
|
assert(recvd == static_cast<int>(i));
|
|
|
|
if (i == sizeof(size_t))
|
|
{
|
|
assert(*reinterpret_cast<const size_t*>(res.data()) ==
|
|
*reinterpret_cast<const size_t*>(data.data()) + 1);
|
|
}
|
|
else
|
|
{
|
|
assert(res == data);
|
|
}
|
|
}
|
|
}
|
|
|
|
public:
|
|
// coverity[root_function] : don't warn about uncaught exceptions
|
|
int main(const std::vector<std::string>& args) override
|
|
{
|
|
EnableHttps = (args.size() > 0 && args[0] == "ssl");
|
|
std::cerr << "Starting " << (EnableHttps ? "HTTPS" : "HTTP") << " client." << std::endl;
|
|
|
|
if (EnableHttps)
|
|
{
|
|
Poco::Net::initializeSSL();
|
|
// Just accept the certificate anyway for testing purposes
|
|
Poco::SharedPtr<Poco::Net::InvalidCertificateHandler> invalidCertHandler = new Poco::Net::AcceptCertificateHandler(false);
|
|
|
|
Poco::Net::Context::Params sslParams;
|
|
Poco::Net::Context::Ptr sslContext = new Poco::Net::Context(Poco::Net::Context::CLIENT_USE, sslParams);
|
|
Poco::Net::SSLManager::instance().initializeClient(nullptr, std::move(invalidCertHandler), std::move(sslContext));
|
|
}
|
|
|
|
testWebsocketPingPong();
|
|
testWebsocketEcho();
|
|
|
|
testPing();
|
|
testLadder();
|
|
testParallel();
|
|
|
|
return 0;
|
|
}
|
|
};
|
|
|
|
// coverity[root_function] : don't warn about uncaught exceptions
|
|
POCO_APP_MAIN(Client)
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|