Now closes unwanted client connections when a request is finished. Also added additional tests
This commit is contained in:
parent
20d3c44366
commit
362c1a2591
2 changed files with 84 additions and 13 deletions
|
|
@ -165,11 +165,26 @@ namespace SimpleWeb {
|
||||||
auto connection=session->connection;
|
auto connection=session->connection;
|
||||||
auto response=session->response;
|
auto response=session->response;
|
||||||
auto request_callback=std::make_shared<std::function<void(std::shared_ptr<Response>, const error_code&)>>(std::move(request_callback_));
|
auto request_callback=std::make_shared<std::function<void(std::shared_ptr<Response>, const error_code&)>>(std::move(request_callback_));
|
||||||
|
auto connections=this->connections;
|
||||||
auto connections_mutex=this->connections_mutex;
|
auto connections_mutex=this->connections_mutex;
|
||||||
session->callback=[connection, response, request_callback, connections_mutex](const error_code &ec) {
|
session->callback=[connection, response, request_callback, connections, connections_mutex](const error_code &ec) {
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(*connections_mutex);
|
std::lock_guard<std::mutex> lock(*connections_mutex);
|
||||||
connection->in_use=false;
|
connection->in_use=false;
|
||||||
|
|
||||||
|
// Remove unused connections, but keep one open for HTTP persistent connection:
|
||||||
|
size_t unused_connections=0;
|
||||||
|
for(auto it=connections->begin();it!=connections->end();) {
|
||||||
|
if((*it)->in_use)
|
||||||
|
++it;
|
||||||
|
else {
|
||||||
|
++unused_connections;
|
||||||
|
if(unused_connections>1)
|
||||||
|
it=connections->erase(it);
|
||||||
|
else
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(*request_callback)
|
if(*request_callback)
|
||||||
|
|
@ -208,11 +223,26 @@ namespace SimpleWeb {
|
||||||
auto connection=session->connection;
|
auto connection=session->connection;
|
||||||
auto response=session->response;
|
auto response=session->response;
|
||||||
auto request_callback=std::make_shared<std::function<void(std::shared_ptr<Response>, const error_code&)>>(std::move(request_callback_));
|
auto request_callback=std::make_shared<std::function<void(std::shared_ptr<Response>, const error_code&)>>(std::move(request_callback_));
|
||||||
|
auto connections=this->connections;
|
||||||
auto connections_mutex=this->connections_mutex;
|
auto connections_mutex=this->connections_mutex;
|
||||||
session->callback=[connection, response, request_callback, connections_mutex](const error_code &ec) {
|
session->callback=[connection, response, request_callback, connections, connections_mutex](const error_code &ec) {
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(*connections_mutex);
|
std::lock_guard<std::mutex> lock(*connections_mutex);
|
||||||
connection->in_use=false;
|
connection->in_use=false;
|
||||||
|
|
||||||
|
// Remove unused connections, but keep one open for HTTP persistent connection:
|
||||||
|
size_t unused_connections=0;
|
||||||
|
for(auto it=connections->begin();it!=connections->end();) {
|
||||||
|
if((*it)->in_use)
|
||||||
|
++it;
|
||||||
|
else {
|
||||||
|
++unused_connections;
|
||||||
|
if(unused_connections>1)
|
||||||
|
it=connections->erase(it);
|
||||||
|
else
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(*request_callback)
|
if(*request_callback)
|
||||||
|
|
@ -242,10 +272,10 @@ namespace SimpleWeb {
|
||||||
std::string host;
|
std::string host;
|
||||||
unsigned short port;
|
unsigned short port;
|
||||||
|
|
||||||
std::vector<std::shared_ptr<Connection>> connections;
|
std::shared_ptr<std::vector<std::shared_ptr<Connection>>> connections;
|
||||||
std::shared_ptr<std::mutex> connections_mutex;
|
std::shared_ptr<std::mutex> connections_mutex;
|
||||||
|
|
||||||
ClientBase(const std::string& host_port, unsigned short default_port) : io_service(new asio::io_service()), connections_mutex(new std::mutex()) {
|
ClientBase(const std::string& host_port, unsigned short default_port) : io_service(new asio::io_service()), connections(new std::vector<std::shared_ptr<Connection>>()), connections_mutex(new std::mutex()) {
|
||||||
auto parsed_host_port=parse_host_port(host_port, default_port);
|
auto parsed_host_port=parse_host_port(host_port, default_port);
|
||||||
host=parsed_host_port.first;
|
host=parsed_host_port.first;
|
||||||
port=parsed_host_port.second;
|
port=parsed_host_port.second;
|
||||||
|
|
@ -254,19 +284,15 @@ namespace SimpleWeb {
|
||||||
std::shared_ptr<Connection> get_connection() {
|
std::shared_ptr<Connection> get_connection() {
|
||||||
std::shared_ptr<Connection> connection;
|
std::shared_ptr<Connection> connection;
|
||||||
std::lock_guard<std::mutex> lock(*connections_mutex);
|
std::lock_guard<std::mutex> lock(*connections_mutex);
|
||||||
for(auto it=connections.begin();it!=connections.end();) {
|
for(auto it=connections->begin();it!=connections->end();++it) {
|
||||||
if((*it)->in_use)
|
if(!(*it)->in_use && !connection) {
|
||||||
++it;
|
|
||||||
else if(!connection) {
|
|
||||||
connection=*it;
|
connection=*it;
|
||||||
++it;
|
break;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
it=connections.erase(it);
|
|
||||||
}
|
}
|
||||||
if(!connection) {
|
if(!connection) {
|
||||||
connection=create_connection();
|
connection=create_connection();
|
||||||
connections.emplace_back(connection);
|
connections->emplace_back(connection);
|
||||||
}
|
}
|
||||||
connection->reconnecting=false;
|
connection->reconnecting=false;
|
||||||
connection->in_use=true;
|
connection->in_use=true;
|
||||||
|
|
|
||||||
|
|
@ -129,13 +129,16 @@ int main() {
|
||||||
{
|
{
|
||||||
HttpClient client("localhost:8080");
|
HttpClient client("localhost:8080");
|
||||||
|
|
||||||
// test performing the stream version of the request methods first
|
HttpClient::Connection *connection;
|
||||||
{
|
{
|
||||||
|
// test performing the stream version of the request methods first
|
||||||
stringstream output;
|
stringstream output;
|
||||||
stringstream content("A string");
|
stringstream content("A string");
|
||||||
auto r=client.request("POST", "/string", content);
|
auto r=client.request("POST", "/string", content);
|
||||||
output << r->content.rdbuf();
|
output << r->content.rdbuf();
|
||||||
assert(output.str()=="A string");
|
assert(output.str()=="A string");
|
||||||
|
assert(client.connections->size()==1);
|
||||||
|
connection=client.connections->front().get();
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
@ -143,6 +146,8 @@ int main() {
|
||||||
auto r=client.request("POST", "/string", "A string");
|
auto r=client.request("POST", "/string", "A string");
|
||||||
output << r->content.rdbuf();
|
output << r->content.rdbuf();
|
||||||
assert(output.str()=="A string");
|
assert(output.str()=="A string");
|
||||||
|
assert(client.connections->size()==1);
|
||||||
|
assert(connection==client.connections->front().get());
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
@ -150,6 +155,8 @@ int main() {
|
||||||
auto r=client.request("GET", "/header", "", {{"test1", "test"}, {"test2", "ing"}});
|
auto r=client.request("GET", "/header", "", {{"test1", "test"}, {"test2", "ing"}});
|
||||||
output << r->content.rdbuf();
|
output << r->content.rdbuf();
|
||||||
assert(output.str()=="testing");
|
assert(output.str()=="testing");
|
||||||
|
assert(client.connections->size()==1);
|
||||||
|
assert(connection==client.connections->front().get());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -183,13 +190,51 @@ int main() {
|
||||||
}
|
}
|
||||||
for(auto &thread: threads)
|
for(auto &thread: threads)
|
||||||
thread.join();
|
thread.join();
|
||||||
|
assert(client.connections->size()==100);
|
||||||
client.io_service->reset();
|
client.io_service->reset();
|
||||||
client.io_service->run();
|
client.io_service->run();
|
||||||
|
assert(client.connections->size()==1);
|
||||||
for(auto call: calls)
|
for(auto call: calls)
|
||||||
assert(call);
|
assert(call);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HttpClient client("localhost:8080");
|
||||||
|
assert(client.connections->size()==0);
|
||||||
|
for(size_t c=0;c<5000;++c) {
|
||||||
|
auto r1=client.request("POST", "/string", "A string");
|
||||||
|
assert(SimpleWeb::status_code(r1->status_code)==SimpleWeb::StatusCode::success_ok);
|
||||||
|
assert(r1->content.string()=="A string");
|
||||||
|
assert(client.connections->size()==1);
|
||||||
|
|
||||||
|
stringstream content("A string");
|
||||||
|
auto r2 = client.request("POST", "/string", content);
|
||||||
|
assert(SimpleWeb::status_code(r2->status_code) == SimpleWeb::StatusCode::success_ok);
|
||||||
|
assert(r2->content.string() == "A string");
|
||||||
|
assert(client.connections->size() == 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for(size_t c=0;c<500;++c) {
|
||||||
|
{
|
||||||
|
HttpClient client("localhost:8080");
|
||||||
|
auto r=client.request("POST", "/string", "A string");
|
||||||
|
assert(SimpleWeb::status_code(r->status_code)==SimpleWeb::StatusCode::success_ok);
|
||||||
|
assert(r->content.string()=="A string");
|
||||||
|
assert(client.connections->size()==1);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
HttpClient client("localhost:8080");
|
||||||
|
stringstream content("A string");
|
||||||
|
auto r = client.request("POST", "/string", content);
|
||||||
|
assert(SimpleWeb::status_code(r->status_code) == SimpleWeb::StatusCode::success_ok);
|
||||||
|
assert(r->content.string() == "A string");
|
||||||
|
assert(client.connections->size() == 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
server.stop();
|
server.stop();
|
||||||
server_thread.join();
|
server_thread.join();
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue