mirror of
https://github.com/oatpp/oatpp.git
synced 2025-03-31 18:30:22 +08:00
better test framework. + ClientServerTestRunner
This commit is contained in:
parent
1c4fb11296
commit
f7ce8ea873
@ -33,6 +33,8 @@
|
||||
|
||||
#include "oatpp/core/macro/component.hpp"
|
||||
|
||||
#include <list>
|
||||
|
||||
namespace oatpp { namespace test { namespace web {
|
||||
|
||||
class ClientServerTestRunner {
|
||||
@ -40,6 +42,7 @@ public:
|
||||
typedef oatpp::web::server::HttpRouter HttpRouter;
|
||||
typedef oatpp::web::server::api::ApiController ApiController;
|
||||
private:
|
||||
std::list<std::shared_ptr<ApiController>> m_controllers;
|
||||
OATPP_COMPONENT(std::shared_ptr<HttpRouter>, m_router);
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::ServerConnectionProvider>, m_connectionProvider);
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::server::ConnectionHandler>, m_connectionHandler);
|
||||
@ -51,6 +54,7 @@ public:
|
||||
|
||||
void addController(const std::shared_ptr<ApiController>& controller) {
|
||||
controller->addEndpointsToRouter(m_router);
|
||||
m_controllers.push_back(controller);
|
||||
}
|
||||
|
||||
template<typename Lambda>
|
||||
|
@ -156,7 +156,7 @@ public:
|
||||
|
||||
#define OATPP_ASSERT(EXP) \
|
||||
if(!(EXP)) { \
|
||||
OATPP_LOGE("ASSERT[FAILED]", #EXP); \
|
||||
OATPP_LOGE("\033[1mASSERT\033[0m[\033[1;31mFAILED\033[0m]", #EXP); \
|
||||
exit(EXIT_FAILURE); \
|
||||
}
|
||||
|
||||
|
@ -50,7 +50,7 @@ std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getC
|
||||
struct sockaddr_in client;
|
||||
|
||||
if ((host == NULL) || (host->h_addr == NULL)) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Error retrieving DNS information.");
|
||||
OATPP_LOGD("[oatpp::network::client::SimpleTCPConnectionProvider::getConnection()]", "Error. Can't retrieve DNS information.");
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
@ -62,7 +62,7 @@ std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getC
|
||||
oatpp::data::v_io_handle clientHandle = socket(AF_INET, SOCK_STREAM, 0);
|
||||
|
||||
if (clientHandle < 0) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Error creating socket.");
|
||||
OATPP_LOGD("[oatpp::network::client::SimpleTCPConnectionProvider::getConnection()]", "Error. Can't create socket.");
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
@ -70,13 +70,13 @@ std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getC
|
||||
int yes = 1;
|
||||
v_int32 ret = setsockopt(clientHandle, SOL_SOCKET, SO_NOSIGPIPE, &yes, sizeof(int));
|
||||
if(ret < 0) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Warning failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
OATPP_LOGD("[oatpp::network::client::SimpleTCPConnectionProvider::getConnection()]", "Warning. Failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
}
|
||||
#endif
|
||||
|
||||
if (connect(clientHandle, (struct sockaddr *)&client, sizeof(client)) != 0 ) {
|
||||
::close(clientHandle);
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Could not connect");
|
||||
OATPP_LOGD("[oatpp::network::client::SimpleTCPConnectionProvider::getConnection()]", "Error. Could not connect.");
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
@ -105,7 +105,7 @@ oatpp::async::Action SimpleTCPConnectionProvider::getConnectionAsync(oatpp::asyn
|
||||
struct hostent* host = gethostbyname((const char*) m_host->getData());
|
||||
|
||||
if ((host == NULL) || (host->h_addr == NULL)) {
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Error retrieving DNS information.");
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Error. Can't retrieve DNS information.");
|
||||
}
|
||||
|
||||
bzero(&m_client, sizeof(m_client));
|
||||
@ -116,7 +116,7 @@ oatpp::async::Action SimpleTCPConnectionProvider::getConnectionAsync(oatpp::asyn
|
||||
m_clientHandle = socket(AF_INET, SOCK_STREAM, 0);
|
||||
|
||||
if (m_clientHandle < 0) {
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Error creating socket.");
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Error. Can't create socket.");
|
||||
}
|
||||
|
||||
fcntl(m_clientHandle, F_SETFL, O_NONBLOCK);
|
||||
@ -125,7 +125,7 @@ oatpp::async::Action SimpleTCPConnectionProvider::getConnectionAsync(oatpp::asyn
|
||||
int yes = 1;
|
||||
v_int32 ret = setsockopt(m_clientHandle, SOL_SOCKET, SO_NOSIGPIPE, &yes, sizeof(int));
|
||||
if(ret < 0) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Warning failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
OATPP_LOGD("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]", "Warning. Failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
}
|
||||
#endif
|
||||
|
||||
@ -145,7 +145,7 @@ oatpp::async::Action SimpleTCPConnectionProvider::getConnectionAsync(oatpp::asyn
|
||||
return repeat();
|
||||
}
|
||||
::close(m_clientHandle);
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Can't connect");
|
||||
return error("[oatpp::network::client::SimpleTCPConnectionProvider::getConnectionAsync()]: Error. Can't connect.");
|
||||
}
|
||||
|
||||
};
|
||||
|
@ -76,14 +76,14 @@ oatpp::data::v_io_handle SimpleTCPConnectionProvider::instantiateServer(){
|
||||
|
||||
ret = setsockopt(serverHandle, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int));
|
||||
if(ret < 0) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Warning failed to set %s for accepting socket", "SO_REUSEADDR");
|
||||
OATPP_LOGD("[oatpp::network::server::SimpleTCPConnectionProvider::instantiateServer()]", "Warning. Failed to set %s for accepting socket", "SO_REUSEADDR");
|
||||
}
|
||||
|
||||
ret = bind(serverHandle, (struct sockaddr *)&addr, sizeof(addr));
|
||||
|
||||
if(ret != 0) {
|
||||
::close(serverHandle);
|
||||
throw std::runtime_error("Can't bind to address");
|
||||
throw std::runtime_error("[oatpp::network::server::SimpleTCPConnectionProvider::instantiateServer()]: Error. Can't bind to address.");
|
||||
return -1 ;
|
||||
}
|
||||
|
||||
@ -100,9 +100,7 @@ oatpp::data::v_io_handle SimpleTCPConnectionProvider::instantiateServer(){
|
||||
}
|
||||
|
||||
std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getConnection(){
|
||||
|
||||
//oatpp::test::PerformanceChecker checker("Accept Checker");
|
||||
|
||||
|
||||
oatpp::data::v_io_handle handle = accept(m_serverHandle, nullptr, nullptr);
|
||||
|
||||
if (handle < 0) {
|
||||
@ -110,7 +108,7 @@ std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getC
|
||||
if(error == EAGAIN || error == EWOULDBLOCK){
|
||||
return nullptr;
|
||||
} else {
|
||||
OATPP_LOGD("Server", "Error: %d", error);
|
||||
OATPP_LOGD("[oatpp::network::server::SimpleTCPConnectionProvider::getConnection()]", "Error. %d", error);
|
||||
return nullptr;
|
||||
}
|
||||
}
|
||||
@ -119,7 +117,7 @@ std::shared_ptr<oatpp::data::stream::IOStream> SimpleTCPConnectionProvider::getC
|
||||
int yes = 1;
|
||||
v_int32 ret = setsockopt(handle, SOL_SOCKET, SO_NOSIGPIPE, &yes, sizeof(int));
|
||||
if(ret < 0) {
|
||||
OATPP_LOGD("SimpleTCPConnectionProvider", "Warning failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
OATPP_LOGD("[oatpp::network::server::SimpleTCPConnectionProvider::getConnection()]", "Warning. Failed to set %s for socket", "SO_NOSIGPIPE");
|
||||
}
|
||||
#endif
|
||||
|
||||
|
@ -63,7 +63,7 @@ public:
|
||||
*
|
||||
* It may be implemented later
|
||||
*/
|
||||
throw std::runtime_error("[oatpp::network::server::SimpleTCPConnectionProvider::getConnectionAsync()] not implemented.");
|
||||
throw std::runtime_error("[oatpp::network::server::SimpleTCPConnectionProvider::getConnectionAsync()]: Error. Not implemented.");
|
||||
}
|
||||
|
||||
v_word16 getPort(){
|
||||
|
@ -89,11 +89,14 @@ std::shared_ptr<Interface::ConnectionSubmission> Interface::connectNonBlocking()
|
||||
return submission;
|
||||
}
|
||||
|
||||
std::shared_ptr<Socket> Interface::accept() {
|
||||
std::shared_ptr<Socket> Interface::accept(const bool& waitingHandle) {
|
||||
std::unique_lock<std::mutex> lock(m_mutex);
|
||||
while (m_submissions.getFirstNode() == nullptr) {
|
||||
while (waitingHandle && m_submissions.getFirstNode() == nullptr) {
|
||||
m_condition.wait(lock);
|
||||
}
|
||||
if(!waitingHandle) {
|
||||
return nullptr;
|
||||
}
|
||||
return acceptSubmission(m_submissions.popFront());
|
||||
}
|
||||
|
||||
@ -104,5 +107,9 @@ std::shared_ptr<Socket> Interface::acceptNonBlocking() {
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
void Interface::notifyAcceptors() {
|
||||
m_condition.notify_all();
|
||||
}
|
||||
|
||||
}}}
|
||||
|
@ -70,9 +70,21 @@ public:
|
||||
|
||||
std::shared_ptr<ConnectionSubmission> connect();
|
||||
std::shared_ptr<ConnectionSubmission> connectNonBlocking();
|
||||
|
||||
std::shared_ptr<Socket> accept();
|
||||
|
||||
/**
|
||||
*
|
||||
* @param waitingHandle
|
||||
* @return
|
||||
*/
|
||||
std::shared_ptr<Socket> accept(const bool& waitingHandle = true);
|
||||
|
||||
std::shared_ptr<Socket> acceptNonBlocking();
|
||||
|
||||
/**
|
||||
* Notify all threads that are waiting on accept().
|
||||
* Those threads that have waitingHandle changed to false will be unblocked.
|
||||
*/
|
||||
void notifyAcceptors();
|
||||
|
||||
oatpp::String getName() {
|
||||
return m_name;
|
||||
|
@ -27,13 +27,16 @@
|
||||
namespace oatpp { namespace network { namespace virtual_ { namespace server {
|
||||
|
||||
void ConnectionProvider::close() {
|
||||
//
|
||||
m_open = false;
|
||||
m_interface->notifyAcceptors();
|
||||
}
|
||||
|
||||
std::shared_ptr<ConnectionProvider::IOStream> ConnectionProvider::getConnection() {
|
||||
auto socket = m_interface->accept();
|
||||
socket->setNonBlocking(m_nonBlocking);
|
||||
socket->setMaxAvailableToReadWrtie(m_maxAvailableToRead, m_maxAvailableToWrite);
|
||||
auto socket = m_interface->accept(m_open);
|
||||
if(socket) {
|
||||
socket->setNonBlocking(m_nonBlocking);
|
||||
socket->setMaxAvailableToReadWrtie(m_maxAvailableToRead, m_maxAvailableToWrite);
|
||||
}
|
||||
return socket;
|
||||
}
|
||||
|
||||
|
@ -34,6 +34,7 @@ class ConnectionProvider : public oatpp::network::ServerConnectionProvider {
|
||||
private:
|
||||
std::shared_ptr<virtual_::Interface> m_interface;
|
||||
bool m_nonBlocking;
|
||||
bool m_open;
|
||||
data::v_io_size m_maxAvailableToRead;
|
||||
data::v_io_size m_maxAvailableToWrite;
|
||||
public:
|
||||
@ -41,6 +42,7 @@ public:
|
||||
ConnectionProvider(const std::shared_ptr<virtual_::Interface>& interface, bool nonBlocking = false)
|
||||
: m_interface(interface)
|
||||
, m_nonBlocking(nonBlocking)
|
||||
, m_open(true)
|
||||
, m_maxAvailableToRead(-1)
|
||||
, m_maxAvailableToWrite(-1)
|
||||
{
|
||||
@ -75,7 +77,7 @@ public:
|
||||
*
|
||||
* It may be implemented later
|
||||
*/
|
||||
throw std::runtime_error("[oatpp::network::virtual_::server::ConnectionProvider::getConnectionAsync()] not implemented.");
|
||||
throw std::runtime_error("[oatpp::network::virtual_::server::ConnectionProvider::getConnectionAsync()]: Error. Not implemented.");
|
||||
}
|
||||
|
||||
};
|
||||
|
@ -65,6 +65,7 @@ void runTests() {
|
||||
|
||||
OATPP_RUN_TEST(oatpp::test::network::virtual_::PipeTest);
|
||||
OATPP_RUN_TEST(oatpp::test::network::virtual_::InterfaceTest);
|
||||
|
||||
OATPP_RUN_TEST(oatpp::test::web::FullTest);
|
||||
OATPP_RUN_TEST(oatpp::test::web::FullAsyncTest);
|
||||
|
||||
|
@ -35,59 +35,108 @@
|
||||
|
||||
#include "oatpp/parser/json/mapping/ObjectMapper.hpp"
|
||||
|
||||
#include "oatpp/network/server/SimpleTCPConnectionProvider.hpp"
|
||||
#include "oatpp/network/client/SimpleTCPConnectionProvider.hpp"
|
||||
|
||||
#include "oatpp/network/virtual_/client/ConnectionProvider.hpp"
|
||||
#include "oatpp/network/virtual_/server/ConnectionProvider.hpp"
|
||||
#include "oatpp/network/virtual_/Interface.hpp"
|
||||
#include "oatpp/network/server/Server.hpp"
|
||||
|
||||
#include "oatpp/core/macro/component.hpp"
|
||||
|
||||
#include "oatpp-test/web/ClientServerTestRunner.hpp"
|
||||
|
||||
namespace oatpp { namespace test { namespace web {
|
||||
|
||||
namespace {
|
||||
|
||||
//#define OATPP_TEST_USE_REAL_PORT
|
||||
|
||||
class TestComponent {
|
||||
public:
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, virtualInterface)([] {
|
||||
return oatpp::network::virtual_::Interface::createShared("virtualhost");
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::ServerConnectionProvider>, serverConnectionProvider)([this] {
|
||||
#ifdef OATPP_TEST_USE_REAL_PORT
|
||||
return oatpp::network::server::SimpleTCPConnectionProvider::createShared(8000, true /* nonBlocking */);
|
||||
#else
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, interface);
|
||||
auto provider = oatpp::network::virtual_::server::ConnectionProvider::createShared(interface, true /* nonBlocking */);
|
||||
//provider->setSocketMaxAvailableToReadWrtie(123, 11);
|
||||
provider->setSocketMaxAvailableToReadWrtie(-1, -1);
|
||||
return provider;
|
||||
#endif
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::web::server::HttpRouter>, httpRouter)([] {
|
||||
return oatpp::web::server::HttpRouter::createShared();
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::server::ConnectionHandler>, serverConnectionHandler)([] {
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::web::server::HttpRouter>, router);
|
||||
return oatpp::web::server::AsyncHttpConnectionHandler::createShared(router);
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::data::mapping::ObjectMapper>, objectMapper)([] {
|
||||
return oatpp::parser::json::mapping::ObjectMapper::createShared();
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::ClientConnectionProvider>, clientConnectionProvider)([this] {
|
||||
#ifdef OATPP_TEST_USE_REAL_PORT
|
||||
return oatpp::network::client::SimpleTCPConnectionProvider::createShared("127.0.0.1", 8000);
|
||||
#else
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, interface);
|
||||
auto provider = oatpp::network::virtual_::client::ConnectionProvider::createShared(interface);
|
||||
//provider->setSocketMaxAvailableToReadWrtie(12421, 21312);
|
||||
provider->setSocketMaxAvailableToReadWrtie(-1, -1);
|
||||
return provider;
|
||||
#endif
|
||||
}());
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
void FullAsyncTest::onRun() {
|
||||
|
||||
auto interface = oatpp::network::virtual_::Interface::createShared("virtualhost");
|
||||
|
||||
auto serverConnectionProvider = oatpp::network::virtual_::server::ConnectionProvider::createShared(interface, true);
|
||||
auto clientConnectionProvider = oatpp::network::virtual_::client::ConnectionProvider::createShared(interface);
|
||||
|
||||
serverConnectionProvider->setSocketMaxAvailableToReadWrtie(123, 11);
|
||||
clientConnectionProvider->setSocketMaxAvailableToReadWrtie(12421, 21312);
|
||||
|
||||
//serverConnectionProvider->setSocketMaxAvailableToReadWrtie(1, 1);
|
||||
//clientConnectionProvider->setSocketMaxAvailableToReadWrtie(1, 1);
|
||||
|
||||
auto objectMapper = oatpp::parser::json::mapping::ObjectMapper::createShared();
|
||||
|
||||
auto router = oatpp::web::server::HttpRouter::createShared();
|
||||
auto connectionHandler = oatpp::web::server::AsyncHttpConnectionHandler::createShared(router);
|
||||
|
||||
auto controller = app::ControllerAsync::createShared(objectMapper);
|
||||
controller->addEndpointsToRouter(router);
|
||||
|
||||
auto requestExecutor = oatpp::web::client::HttpRequestExecutor::createShared(clientConnectionProvider);
|
||||
|
||||
auto client = app::Client::createShared(requestExecutor, objectMapper);
|
||||
|
||||
auto server = oatpp::network::server::Server::createShared(serverConnectionProvider, connectionHandler);
|
||||
|
||||
std::thread clientThread([client, server, connectionHandler, objectMapper]{
|
||||
|
||||
for(v_int32 i = 0; i < 1000; i ++) {
|
||||
TestComponent component;
|
||||
|
||||
oatpp::test::web::ClientServerTestRunner runner;
|
||||
|
||||
runner.addController(app::ControllerAsync::createShared());
|
||||
|
||||
runner.run([] {
|
||||
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::ClientConnectionProvider>, clientConnectionProvider);
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::data::mapping::ObjectMapper>, objectMapper);
|
||||
|
||||
auto requestExecutor = oatpp::web::client::HttpRequestExecutor::createShared(clientConnectionProvider);
|
||||
auto client = app::Client::createShared(requestExecutor, objectMapper);
|
||||
|
||||
auto connection = client->getConnection();
|
||||
|
||||
v_int32 iterationsStep = 1000;
|
||||
|
||||
for(v_int32 i = 0; i < iterationsStep * 10; i ++) {
|
||||
|
||||
{ // test simple GET
|
||||
auto response = client->getRoot();
|
||||
auto response = client->getRoot(connection);
|
||||
auto value = response->readBodyToString();
|
||||
OATPP_ASSERT(value == "Hello World Async!!!");
|
||||
}
|
||||
|
||||
{ // test GET with path parameter
|
||||
auto response = client->getWithParams("my_test_param-Async");
|
||||
auto response = client->getWithParams("my_test_param-Async", connection);
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
OATPP_ASSERT(dto);
|
||||
OATPP_ASSERT(dto->testValue == "my_test_param-Async");
|
||||
}
|
||||
|
||||
{ // test GET with header parameter
|
||||
auto response = client->getWithHeaders("my_test_header-Async");
|
||||
auto response = client->getWithHeaders("my_test_header-Async", connection);
|
||||
//auto str = response->readBodyToString();
|
||||
//OATPP_LOGE("AAA", "code=%d, str='%s'", response->statusCode, str->c_str());
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
@ -96,7 +145,7 @@ void FullAsyncTest::onRun() {
|
||||
}
|
||||
|
||||
{ // test POST with body
|
||||
auto response = client->postBody("my_test_body-Async");
|
||||
auto response = client->postBody("my_test_body-Async", connection);
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
OATPP_ASSERT(dto);
|
||||
OATPP_ASSERT(dto->testValue == "my_test_body-Async");
|
||||
@ -108,34 +157,21 @@ void FullAsyncTest::onRun() {
|
||||
stream.write("0123456789", 10);
|
||||
}
|
||||
auto data = stream.toString();
|
||||
auto response = client->echoBody(data);
|
||||
auto response = client->echoBody(data, connection);
|
||||
|
||||
auto returnedData = response->readBodyToString();
|
||||
|
||||
OATPP_ASSERT(returnedData);
|
||||
OATPP_ASSERT(returnedData == data);
|
||||
}
|
||||
|
||||
if((i + 1) % iterationsStep == 0) {
|
||||
OATPP_LOGD("i", "%d", i + 1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
try {
|
||||
connectionHandler->stop();
|
||||
server->stop();
|
||||
client->getConnection(); // wake blocking server accept
|
||||
} catch(std::runtime_error e) {
|
||||
// DO NOTHING
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
std::thread serverThread([server]{
|
||||
server->run();
|
||||
});
|
||||
|
||||
clientThread.join();
|
||||
serverThread.join();
|
||||
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(1000));
|
||||
|
||||
}
|
||||
|
||||
|
@ -26,7 +26,6 @@
|
||||
|
||||
#include "oatpp/web/app/Client.hpp"
|
||||
|
||||
#include "oatpp/web/app/ControllerAsync.hpp"
|
||||
#include "oatpp/web/app/Controller.hpp"
|
||||
|
||||
#include "oatpp/web/client/HttpRequestExecutor.hpp"
|
||||
@ -36,67 +35,115 @@
|
||||
|
||||
#include "oatpp/parser/json/mapping/ObjectMapper.hpp"
|
||||
|
||||
#include "oatpp/network/server/SimpleTCPConnectionProvider.hpp"
|
||||
#include "oatpp/network/client/SimpleTCPConnectionProvider.hpp"
|
||||
|
||||
#include "oatpp/network/virtual_/client/ConnectionProvider.hpp"
|
||||
#include "oatpp/network/virtual_/server/ConnectionProvider.hpp"
|
||||
#include "oatpp/network/virtual_/Interface.hpp"
|
||||
|
||||
#include "oatpp/network/server/Server.hpp"
|
||||
#include "oatpp/core/macro/component.hpp"
|
||||
|
||||
#include "oatpp-test/web/ClientServerTestRunner.hpp"
|
||||
|
||||
namespace oatpp { namespace test { namespace web {
|
||||
|
||||
namespace {
|
||||
|
||||
//#define OATPP_TEST_USE_REAL_PORT
|
||||
|
||||
class TestComponent {
|
||||
public:
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, virtualInterface)([] {
|
||||
return oatpp::network::virtual_::Interface::createShared("virtualhost");
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::ServerConnectionProvider>, serverConnectionProvider)([this] {
|
||||
#ifdef OATPP_TEST_USE_REAL_PORT
|
||||
return oatpp::network::server::SimpleTCPConnectionProvider::createShared(8000);
|
||||
#else
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, interface);
|
||||
auto provider = oatpp::network::virtual_::server::ConnectionProvider::createShared(interface);
|
||||
//provider->setSocketMaxAvailableToReadWrtie(123, 11);
|
||||
provider->setSocketMaxAvailableToReadWrtie(-1, -1);
|
||||
return provider;
|
||||
#endif
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::web::server::HttpRouter>, httpRouter)([] {
|
||||
return oatpp::web::server::HttpRouter::createShared();
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::server::ConnectionHandler>, serverConnectionHandler)([] {
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::web::server::HttpRouter>, router);
|
||||
return oatpp::web::server::HttpConnectionHandler::createShared(router);
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::data::mapping::ObjectMapper>, objectMapper)([] {
|
||||
return oatpp::parser::json::mapping::ObjectMapper::createShared();
|
||||
}());
|
||||
|
||||
OATPP_CREATE_COMPONENT(std::shared_ptr<oatpp::network::ClientConnectionProvider>, clientConnectionProvider)([this] {
|
||||
#ifdef OATPP_TEST_USE_REAL_PORT
|
||||
return oatpp::network::client::SimpleTCPConnectionProvider::createShared("127.0.0.1", 8000);
|
||||
#else
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::virtual_::Interface>, interface);
|
||||
auto provider = oatpp::network::virtual_::client::ConnectionProvider::createShared(interface);
|
||||
//provider->setSocketMaxAvailableToReadWrtie(12421, 21312);
|
||||
provider->setSocketMaxAvailableToReadWrtie(-1, -1);
|
||||
return provider;
|
||||
#endif
|
||||
}());
|
||||
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
void FullTest::onRun() {
|
||||
|
||||
auto interface = oatpp::network::virtual_::Interface::createShared("virtualhost");
|
||||
|
||||
auto serverConnectionProvider = oatpp::network::virtual_::server::ConnectionProvider::createShared(interface);
|
||||
auto clientConnectionProvider = oatpp::network::virtual_::client::ConnectionProvider::createShared(interface);
|
||||
|
||||
serverConnectionProvider->setSocketMaxAvailableToReadWrtie(123, 11);
|
||||
clientConnectionProvider->setSocketMaxAvailableToReadWrtie(12421, 21312);
|
||||
TestComponent component;
|
||||
|
||||
//serverConnectionProvider->setSocketMaxAvailableToReadWrtie(1, 1);
|
||||
//clientConnectionProvider->setSocketMaxAvailableToReadWrtie(1, 1);
|
||||
|
||||
auto objectMapper = oatpp::parser::json::mapping::ObjectMapper::createShared();
|
||||
|
||||
auto router = oatpp::web::server::HttpRouter::createShared();
|
||||
auto connectionHandler = oatpp::web::server::HttpConnectionHandler::createShared(router);
|
||||
|
||||
auto controller = app::Controller::createShared(objectMapper);
|
||||
controller->addEndpointsToRouter(router);
|
||||
|
||||
auto requestExecutor = oatpp::web::client::HttpRequestExecutor::createShared(clientConnectionProvider);
|
||||
|
||||
auto client = app::Client::createShared(requestExecutor, objectMapper);
|
||||
|
||||
auto server = oatpp::network::server::Server::createShared(serverConnectionProvider, connectionHandler);
|
||||
|
||||
std::thread clientThread([client, server, objectMapper]{
|
||||
|
||||
for(v_int32 i = 0; i < 1000; i ++) {
|
||||
oatpp::test::web::ClientServerTestRunner runner;
|
||||
|
||||
runner.addController(app::Controller::createShared());
|
||||
|
||||
runner.run([] {
|
||||
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::network::ClientConnectionProvider>, clientConnectionProvider);
|
||||
OATPP_COMPONENT(std::shared_ptr<oatpp::data::mapping::ObjectMapper>, objectMapper);
|
||||
|
||||
auto requestExecutor = oatpp::web::client::HttpRequestExecutor::createShared(clientConnectionProvider);
|
||||
auto client = app::Client::createShared(requestExecutor, objectMapper);
|
||||
|
||||
auto connection = client->getConnection();
|
||||
|
||||
v_int32 iterationsStep = 1000;
|
||||
|
||||
for(v_int32 i = 0; i < iterationsStep * 10; i ++) {
|
||||
|
||||
{ // test simple GET
|
||||
auto response = client->getRoot();
|
||||
auto response = client->getRoot(connection);
|
||||
auto value = response->readBodyToString();
|
||||
OATPP_ASSERT(value == "Hello World!!!");
|
||||
}
|
||||
|
||||
|
||||
{ // test GET with path parameter
|
||||
auto response = client->getWithParams("my_test_param");
|
||||
auto response = client->getWithParams("my_test_param", connection);
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
OATPP_ASSERT(dto);
|
||||
OATPP_ASSERT(dto->testValue == "my_test_param");
|
||||
}
|
||||
|
||||
|
||||
{ // test GET with header parameter
|
||||
auto response = client->getWithHeaders("my_test_header");
|
||||
auto response = client->getWithHeaders("my_test_header", connection);
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
OATPP_ASSERT(dto);
|
||||
OATPP_ASSERT(dto->testValue == "my_test_header");
|
||||
}
|
||||
|
||||
|
||||
{ // test POST with body
|
||||
auto response = client->postBody("my_test_body");
|
||||
auto response = client->postBody("my_test_body", connection);
|
||||
auto dto = response->readBodyToDto<app::TestDto>(objectMapper);
|
||||
OATPP_ASSERT(dto);
|
||||
OATPP_ASSERT(dto->testValue == "my_test_body");
|
||||
@ -108,29 +155,19 @@ void FullTest::onRun() {
|
||||
stream.write("0123456789", 10);
|
||||
}
|
||||
auto data = stream.toString();
|
||||
auto response = client->echoBody(data);
|
||||
auto response = client->echoBody(data, connection);
|
||||
auto returnedData = response->readBodyToString();
|
||||
OATPP_ASSERT(returnedData);
|
||||
OATPP_ASSERT(returnedData == data);
|
||||
}
|
||||
|
||||
|
||||
if((i + 1) % iterationsStep == 0) {
|
||||
OATPP_LOGD("i", "%d", i + 1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
try {
|
||||
server->stop();
|
||||
client->getRoot(); // wake blocking server accept
|
||||
} catch(std::runtime_error e) {
|
||||
// DO NOTHING
|
||||
}
|
||||
|
||||
|
||||
});
|
||||
|
||||
std::thread serverThread([server]{
|
||||
server->run();
|
||||
});
|
||||
|
||||
clientThread.join();
|
||||
serverThread.join();
|
||||
|
||||
}
|
||||
|
||||
|
@ -42,20 +42,20 @@ public:
|
||||
{}
|
||||
public:
|
||||
|
||||
static std::shared_ptr<Controller> createShared(const std::shared_ptr<ObjectMapper>& objectMapper){
|
||||
static std::shared_ptr<Controller> createShared(const std::shared_ptr<ObjectMapper>& objectMapper = OATPP_GET_COMPONENT(std::shared_ptr<ObjectMapper>)){
|
||||
return std::make_shared<Controller>(objectMapper);
|
||||
}
|
||||
|
||||
#include OATPP_CODEGEN_BEGIN(ApiController)
|
||||
|
||||
ENDPOINT("GET", "/", root) {
|
||||
OATPP_LOGD(TAG, "GET '/'");
|
||||
//OATPP_LOGD(TAG, "GET '/'");
|
||||
return createResponse(Status::CODE_200, "Hello World!!!");
|
||||
}
|
||||
|
||||
ENDPOINT("GET", "params/{param}", getWithParams,
|
||||
PATH(String, param)) {
|
||||
OATPP_LOGD(TAG, "GET params/%s", param->c_str());
|
||||
//OATPP_LOGD(TAG, "GET params/%s", param->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = param;
|
||||
return createDtoResponse(Status::CODE_200, dto);
|
||||
@ -63,7 +63,7 @@ public:
|
||||
|
||||
ENDPOINT("GET", "headers", getWithHeaders,
|
||||
HEADER(String, param, "X-TEST-HEADER")) {
|
||||
OATPP_LOGD(TAG, "GET headers {X-TEST-HEADER: %s}", param->c_str());
|
||||
//OATPP_LOGD(TAG, "GET headers {X-TEST-HEADER: %s}", param->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = param;
|
||||
return createDtoResponse(Status::CODE_200, dto);
|
||||
@ -71,7 +71,7 @@ public:
|
||||
|
||||
ENDPOINT("POST", "body", postBody,
|
||||
BODY_STRING(String, body)) {
|
||||
OATPP_LOGD(TAG, "POST body %s", body->c_str());
|
||||
//OATPP_LOGD(TAG, "POST body %s", body->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = body;
|
||||
return createDtoResponse(Status::CODE_200, dto);
|
||||
@ -79,7 +79,7 @@ public:
|
||||
|
||||
ENDPOINT("POST", "echo", echo,
|
||||
BODY_STRING(String, body)) {
|
||||
OATPP_LOGD(TAG, "POST body(echo) size=%d", body->getSize());
|
||||
//OATPP_LOGD(TAG, "POST body(echo) size=%d", body->getSize());
|
||||
return createResponse(Status::CODE_200, body);
|
||||
}
|
||||
|
||||
|
@ -43,7 +43,7 @@ public:
|
||||
{}
|
||||
public:
|
||||
|
||||
static std::shared_ptr<ControllerAsync> createShared(const std::shared_ptr<ObjectMapper>& objectMapper){
|
||||
static std::shared_ptr<ControllerAsync> createShared(const std::shared_ptr<ObjectMapper>& objectMapper = OATPP_GET_COMPONENT(std::shared_ptr<ObjectMapper>)){
|
||||
return std::make_shared<ControllerAsync>(objectMapper);
|
||||
}
|
||||
|
||||
@ -54,56 +54,56 @@ public:
|
||||
ENDPOINT_ASYNC_INIT(Root)
|
||||
|
||||
Action act() {
|
||||
OATPP_LOGD(TAG, "GET '/'");
|
||||
//OATPP_LOGD(TAG, "GET '/'");
|
||||
return _return(controller->createResponse(Status::CODE_200, "Hello World Async!!!"));
|
||||
}
|
||||
|
||||
|
||||
};
|
||||
|
||||
|
||||
ENDPOINT_ASYNC("GET", "params/{param}", GetWithParams) {
|
||||
|
||||
|
||||
ENDPOINT_ASYNC_INIT(GetWithParams)
|
||||
|
||||
|
||||
Action act() {
|
||||
auto param = request->getPathVariable("param");
|
||||
OATPP_LOGD(TAG, "GET params/%s", param->c_str());
|
||||
//OATPP_LOGD(TAG, "GET params/%s", param->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = param;
|
||||
return _return(controller->createDtoResponse(Status::CODE_200, dto));
|
||||
}
|
||||
|
||||
|
||||
};
|
||||
|
||||
|
||||
ENDPOINT_ASYNC("GET", "headers", GetWithHeaders) {
|
||||
|
||||
|
||||
ENDPOINT_ASYNC_INIT(GetWithHeaders)
|
||||
|
||||
|
||||
Action act() {
|
||||
auto param = request->getHeader("X-TEST-HEADER");
|
||||
OATPP_LOGD(TAG, "GET headers {X-TEST-HEADER: %s}", param->c_str());
|
||||
//OATPP_LOGD(TAG, "GET headers {X-TEST-HEADER: %s}", param->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = param;
|
||||
return _return(controller->createDtoResponse(Status::CODE_200, dto));
|
||||
}
|
||||
|
||||
|
||||
};
|
||||
|
||||
|
||||
ENDPOINT_ASYNC("POST", "body", PostBody) {
|
||||
|
||||
|
||||
ENDPOINT_ASYNC_INIT(PostBody)
|
||||
|
||||
|
||||
Action act() {
|
||||
OATPP_LOGD(TAG, "POST body. Reading body...");
|
||||
//OATPP_LOGD(TAG, "POST body. Reading body...");
|
||||
return request->readBodyToStringAsync(this, &PostBody::onBodyRead);
|
||||
}
|
||||
|
||||
|
||||
Action onBodyRead(const String& body) {
|
||||
OATPP_LOGD(TAG, "POST body %s", body->c_str());
|
||||
//OATPP_LOGD(TAG, "POST body %s", body->c_str());
|
||||
auto dto = TestDto::createShared();
|
||||
dto->testValue = body;
|
||||
return _return(controller->createDtoResponse(Status::CODE_200, dto));
|
||||
}
|
||||
|
||||
|
||||
};
|
||||
|
||||
ENDPOINT_ASYNC("POST", "echo", Echo) {
|
||||
@ -111,12 +111,12 @@ public:
|
||||
ENDPOINT_ASYNC_INIT(Echo)
|
||||
|
||||
Action act() {
|
||||
OATPP_LOGD(TAG, "POST body(echo). Reading body...");
|
||||
//OATPP_LOGD(TAG, "POST body(echo). Reading body...");
|
||||
return request->readBodyToStringAsync(this, &Echo::onBodyRead);
|
||||
}
|
||||
|
||||
Action onBodyRead(const String& body) {
|
||||
OATPP_LOGD(TAG, "POST echo size=%d", body->getSize());
|
||||
//OATPP_LOGD(TAG, "POST echo size=%d", body->getSize());
|
||||
return _return(controller->createResponse(Status::CODE_200, body));
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user