2019-06-14 01:04:59 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* press.cc
|
|
|
|
* An Tao
|
|
|
|
*
|
|
|
|
* Copyright 2018, An Tao. All rights reserved.
|
|
|
|
* https://github.com/an-tao/drogon
|
|
|
|
* Use of this source code is governed by the MIT license
|
|
|
|
* that can be found in the License file.
|
|
|
|
*
|
|
|
|
* Drogon
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "press.h"
|
|
|
|
#include "cmd.h"
|
|
|
|
#include <drogon/DrClassMap.h>
|
|
|
|
#include <iostream>
|
|
|
|
#include <memory>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <stdlib.h>
|
2020-01-25 03:58:20 +00:00
|
|
|
#ifndef _WIN32
|
2019-06-14 01:04:59 +00:00
|
|
|
#include <unistd.h>
|
2020-01-25 03:58:20 +00:00
|
|
|
#endif
|
2019-06-14 01:04:59 +00:00
|
|
|
|
|
|
|
using namespace drogon_ctl;
|
|
|
|
std::string press::detail()
|
|
|
|
{
|
|
|
|
return "Use press command to do stress testing\n"
|
|
|
|
"Usage:drogon_ctl press <options> <url>\n"
|
|
|
|
" -n num number of requests(default : 1)\n"
|
|
|
|
" -t num number of threads(default : 1)\n"
|
|
|
|
" -c num concurrent connections(default : 1)\n"
|
|
|
|
// " -k keep alive(default: no)\n"
|
|
|
|
" -q no progress indication(default: no)\n\n"
|
|
|
|
"example: drogon_ctl press -n 10000 -c 100 -t 4 -q "
|
|
|
|
"http://localhost:8080/index.html\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void outputErrorAndExit(const string_view &err)
|
|
|
|
{
|
|
|
|
std::cout << err << std::endl;
|
2019-07-26 14:22:12 +00:00
|
|
|
exit(1);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
void press::handleCommand(std::vector<std::string> ¶meters)
|
|
|
|
{
|
|
|
|
for (auto iter = parameters.begin(); iter != parameters.end(); iter++)
|
|
|
|
{
|
|
|
|
auto ¶m = *iter;
|
|
|
|
if (param.find("-n") == 0)
|
|
|
|
{
|
|
|
|
if (param == "-n")
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
++iter;
|
2019-06-14 01:04:59 +00:00
|
|
|
if (iter == parameters.end())
|
|
|
|
{
|
|
|
|
outputErrorAndExit("No number of requests!");
|
|
|
|
}
|
|
|
|
auto &num = *iter;
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfRequests_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of requests!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto num = param.substr(2);
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfRequests_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of requests!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (param.find("-t") == 0)
|
|
|
|
{
|
|
|
|
if (param == "-t")
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
++iter;
|
2019-06-14 01:04:59 +00:00
|
|
|
if (iter == parameters.end())
|
|
|
|
{
|
|
|
|
outputErrorAndExit("No number of threads!");
|
|
|
|
}
|
|
|
|
auto &num = *iter;
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfThreads_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of threads!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto num = param.substr(2);
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfThreads_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of threads!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (param.find("-c") == 0)
|
|
|
|
{
|
|
|
|
if (param == "-c")
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
++iter;
|
2019-06-14 01:04:59 +00:00
|
|
|
if (iter == parameters.end())
|
|
|
|
{
|
|
|
|
outputErrorAndExit("No number of connections!");
|
|
|
|
}
|
|
|
|
auto &num = *iter;
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfConnections_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of connections!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto num = param.substr(2);
|
|
|
|
try
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
numOfConnections_ = std::stoll(num);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid number of connections!");
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// else if (param == "-k")
|
|
|
|
// {
|
2019-11-21 03:27:47 +00:00
|
|
|
// keepAlive_ = true;
|
2019-06-14 01:04:59 +00:00
|
|
|
// continue;
|
|
|
|
// }
|
|
|
|
else if (param == "-q")
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
processIndication_ = false;
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
else if (param[0] != '-')
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
url_ = param;
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-21 03:27:47 +00:00
|
|
|
// std::cout << "n=" << numOfRequests_ << std::endl;
|
|
|
|
// std::cout << "t=" << numOfThreads_ << std::endl;
|
|
|
|
// std::cout << "c=" << numOfConnections_ << std::endl;
|
|
|
|
// std::cout << "q=" << processIndication_ << std::endl;
|
|
|
|
// std::cout << "url=" << url_ << std::endl;
|
2020-12-14 00:49:33 +00:00
|
|
|
if (url_.empty() || url_.compare(0, 4, "http") != 0 ||
|
|
|
|
(url_.compare(4, 3, "://") != 0 && url_.compare(4, 4, "s://") != 0))
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
outputErrorAndExit("Invalid URL");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
auto pos = url_.find("://");
|
|
|
|
auto posOfPath = url_.find("/", pos + 3);
|
2019-06-14 01:04:59 +00:00
|
|
|
if (posOfPath == std::string::npos)
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
host_ = url_;
|
|
|
|
path_ = "/";
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
host_ = url_.substr(0, posOfPath);
|
|
|
|
path_ = url_.substr(posOfPath);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-21 03:27:47 +00:00
|
|
|
// std::cout << "host=" << host_ << std::endl;
|
|
|
|
// std::cout << "path=" << path_ << std::endl;
|
2019-06-14 01:04:59 +00:00
|
|
|
doTesting();
|
|
|
|
}
|
|
|
|
|
|
|
|
void press::doTesting()
|
|
|
|
{
|
|
|
|
createRequestAndClients();
|
2019-11-21 03:27:47 +00:00
|
|
|
if (clients_.empty())
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
outputErrorAndExit("No connection!");
|
|
|
|
}
|
2019-11-21 03:27:47 +00:00
|
|
|
statistics_.startDate_ = trantor::Date::now();
|
|
|
|
for (auto &client : clients_)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
sendRequest(client);
|
|
|
|
}
|
2019-11-21 03:27:47 +00:00
|
|
|
loopPool_->wait();
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void press::createRequestAndClients()
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
loopPool_ = std::make_unique<trantor::EventLoopThreadPool>(numOfThreads_);
|
|
|
|
loopPool_->start();
|
|
|
|
for (size_t i = 0; i < numOfConnections_; ++i)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
auto client =
|
2019-11-21 03:27:47 +00:00
|
|
|
HttpClient::newHttpClient(host_, loopPool_->getNextLoop());
|
2019-06-14 01:04:59 +00:00
|
|
|
client->enableCookies();
|
2019-11-21 03:27:47 +00:00
|
|
|
clients_.push_back(client);
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void press::sendRequest(const HttpClientPtr &client)
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
auto numOfRequest = statistics_.numOfRequestsSent_++;
|
|
|
|
if (numOfRequest >= numOfRequests_)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto request = HttpRequest::newHttpRequest();
|
2019-11-21 03:27:47 +00:00
|
|
|
request->setPath(path_);
|
2019-06-14 01:04:59 +00:00
|
|
|
request->setMethod(Get);
|
|
|
|
// std::cout << "send!" << std::endl;
|
|
|
|
client->sendRequest(
|
|
|
|
request,
|
|
|
|
[this, client, request](ReqResult r, const HttpResponsePtr &resp) {
|
|
|
|
size_t goodNum, badNum;
|
|
|
|
if (r == ReqResult::Ok)
|
|
|
|
{
|
|
|
|
// std::cout << "OK" << std::endl;
|
2019-11-21 03:27:47 +00:00
|
|
|
goodNum = ++statistics_.numOfGoodResponse_;
|
|
|
|
badNum = statistics_.numOfBadResponse_;
|
|
|
|
statistics_.bytesRecieved_ += resp->body().length();
|
2019-06-14 01:04:59 +00:00
|
|
|
auto delay = trantor::Date::now().microSecondsSinceEpoch() -
|
|
|
|
request->creationDate().microSecondsSinceEpoch();
|
2019-11-21 03:27:47 +00:00
|
|
|
statistics_.totalDelay_ += delay;
|
2019-06-14 01:04:59 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-21 03:27:47 +00:00
|
|
|
goodNum = statistics_.numOfGoodResponse_;
|
|
|
|
badNum = ++statistics_.numOfBadResponse_;
|
|
|
|
if (badNum > numOfRequests_ / 10)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
outputErrorAndExit("Too many errors");
|
|
|
|
}
|
|
|
|
}
|
2019-11-21 03:27:47 +00:00
|
|
|
if (goodNum + badNum >= numOfRequests_)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
outputResults();
|
|
|
|
}
|
|
|
|
if (r == ReqResult::Ok)
|
|
|
|
sendRequest(client);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
client->getLoop()->runAfter(1, [this, client]() {
|
|
|
|
sendRequest(client);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-21 03:27:47 +00:00
|
|
|
if (processIndication_)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
auto rec = goodNum + badNum;
|
|
|
|
if (rec % 100000 == 0)
|
|
|
|
{
|
|
|
|
std::cout << rec << " responses are received" << std::endl
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void press::outputResults()
|
|
|
|
{
|
|
|
|
static std::mutex mtx;
|
|
|
|
size_t totalSent = 0;
|
|
|
|
size_t totalRecv = 0;
|
2019-11-21 03:27:47 +00:00
|
|
|
for (auto &client : clients_)
|
2019-06-14 01:04:59 +00:00
|
|
|
{
|
|
|
|
totalSent += client->bytesSent();
|
|
|
|
totalRecv += client->bytesReceived();
|
|
|
|
}
|
|
|
|
auto now = trantor::Date::now();
|
|
|
|
auto microSecs = now.microSecondsSinceEpoch() -
|
2019-11-21 03:27:47 +00:00
|
|
|
statistics_.startDate_.microSecondsSinceEpoch();
|
2019-06-14 01:04:59 +00:00
|
|
|
double seconds = (double)microSecs / 1000000.0;
|
2020-03-01 03:50:47 +00:00
|
|
|
size_t rps = static_cast<size_t>(statistics_.numOfGoodResponse_ / seconds);
|
2019-06-14 01:04:59 +00:00
|
|
|
std::cout << std::endl;
|
2019-11-21 03:27:47 +00:00
|
|
|
std::cout << "TOTALS: " << numOfConnections_ << " connect, "
|
|
|
|
<< numOfRequests_ << " requests, "
|
|
|
|
<< statistics_.numOfGoodResponse_ << " success, "
|
|
|
|
<< statistics_.numOfBadResponse_ << " fail" << std::endl;
|
2019-06-14 01:04:59 +00:00
|
|
|
|
2019-11-21 03:27:47 +00:00
|
|
|
std::cout << "TRAFFIC: "
|
|
|
|
<< statistics_.bytesRecieved_ / statistics_.numOfGoodResponse_
|
2019-06-14 01:04:59 +00:00
|
|
|
<< " avg bytes, "
|
2019-11-21 03:27:47 +00:00
|
|
|
<< (totalRecv - statistics_.bytesRecieved_) /
|
|
|
|
statistics_.numOfGoodResponse_
|
|
|
|
<< " avg overhead, " << statistics_.bytesRecieved_ << " bytes, "
|
|
|
|
<< totalRecv - statistics_.bytesRecieved_ << " overhead"
|
|
|
|
<< std::endl;
|
2019-06-14 01:04:59 +00:00
|
|
|
|
|
|
|
std::cout << std::setiosflags(std::ios::fixed) << std::setprecision(3)
|
|
|
|
<< "TIMING: " << seconds << " seconds, " << rps << " rps, "
|
2019-11-21 03:27:47 +00:00
|
|
|
<< (double)(statistics_.totalDelay_) /
|
|
|
|
statistics_.numOfGoodResponse_ / 1000
|
2019-06-14 01:04:59 +00:00
|
|
|
<< " ms avg req time" << std::endl;
|
|
|
|
|
|
|
|
std::cout << "SPEED: download " << totalRecv / seconds / 1000
|
|
|
|
<< " kBps, upload " << totalSent / seconds / 1000 << " kBps"
|
|
|
|
<< std::endl
|
|
|
|
<< std::endl;
|
|
|
|
exit(0);
|
|
|
|
}
|