2018-02-08 00:52:01 +08:00
|
|
|
#pragma once
|
|
|
|
|
2018-07-02 21:34:33 +08:00
|
|
|
#include <atomic>
|
|
|
|
#include <chrono>
|
|
|
|
#include <experimental/optional>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <map>
|
|
|
|
#include <string>
|
|
|
|
#include <thread>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
2018-01-27 01:50:16 +08:00
|
|
|
#include "json/json.hpp"
|
|
|
|
|
2018-02-15 23:28:38 +08:00
|
|
|
#include "stats/metrics.hpp"
|
2018-02-08 00:52:01 +08:00
|
|
|
#include "stats/stats.hpp"
|
2018-01-27 01:50:16 +08:00
|
|
|
#include "utils/timer.hpp"
|
|
|
|
|
2018-03-28 20:49:28 +08:00
|
|
|
#include "common.hpp"
|
|
|
|
|
2018-01-27 01:50:16 +08:00
|
|
|
const int MAX_RETRIES = 30;
|
|
|
|
|
|
|
|
DEFINE_string(address, "127.0.0.1", "Server address");
|
|
|
|
DEFINE_int32(port, 7687, "Server port");
|
|
|
|
DEFINE_int32(num_workers, 1, "Number of workers");
|
|
|
|
DEFINE_string(output, "", "Output file");
|
|
|
|
DEFINE_string(username, "", "Username for the database");
|
|
|
|
DEFINE_string(password, "", "Password for the database");
|
2018-06-20 23:44:47 +08:00
|
|
|
DEFINE_bool(use_ssl, false, "Set to true to connect with SSL to the server.");
|
2018-01-27 01:50:16 +08:00
|
|
|
DEFINE_int32(duration, 30, "Number of seconds to execute benchmark");
|
|
|
|
|
2018-02-08 00:52:01 +08:00
|
|
|
DEFINE_string(group, "unknown", "Test group name");
|
|
|
|
DEFINE_string(scenario, "unknown", "Test scenario name");
|
|
|
|
|
2018-02-18 00:05:16 +08:00
|
|
|
auto &executed_queries = stats::GetCounter("executed_queries");
|
2018-03-06 02:11:52 +08:00
|
|
|
auto &serialization_errors = stats::GetCounter("serialization_errors");
|
2018-02-15 23:28:38 +08:00
|
|
|
|
2018-01-27 01:50:16 +08:00
|
|
|
class TestClient {
|
|
|
|
public:
|
2018-03-28 20:49:28 +08:00
|
|
|
TestClient() {
|
|
|
|
Endpoint endpoint(FLAGS_address, FLAGS_port);
|
|
|
|
if (!client_.Connect(endpoint, FLAGS_username, FLAGS_password)) {
|
|
|
|
LOG(FATAL) << "Couldn't connect to " << endpoint;
|
|
|
|
}
|
|
|
|
}
|
2018-01-27 01:50:16 +08:00
|
|
|
|
|
|
|
virtual ~TestClient() {}
|
|
|
|
|
|
|
|
auto ConsumeStats() {
|
2018-05-30 19:00:25 +08:00
|
|
|
std::unique_lock<utils::SpinLock> guard(lock_);
|
2018-01-27 01:50:16 +08:00
|
|
|
auto stats = stats_;
|
|
|
|
stats_.clear();
|
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Run() {
|
|
|
|
runner_thread_ = std::thread([&] {
|
|
|
|
while (keep_running_) {
|
|
|
|
Step();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void Stop() {
|
|
|
|
keep_running_ = false;
|
|
|
|
runner_thread_.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual void Step() = 0;
|
|
|
|
|
2018-03-06 02:11:52 +08:00
|
|
|
std::experimental::optional<communication::bolt::QueryData> Execute(
|
2018-07-24 21:11:18 +08:00
|
|
|
const std::string &query, const std::map<std::string, Value> ¶ms,
|
2018-01-27 01:50:16 +08:00
|
|
|
const std::string &query_name = "") {
|
2018-03-06 02:11:52 +08:00
|
|
|
communication::bolt::QueryData result;
|
|
|
|
int retries;
|
2018-01-27 01:50:16 +08:00
|
|
|
utils::Timer timer;
|
2018-03-06 02:11:52 +08:00
|
|
|
try {
|
|
|
|
std::tie(result, retries) =
|
|
|
|
ExecuteNTimesTillSuccess(client_, query, params, MAX_RETRIES);
|
|
|
|
} catch (const utils::BasicException &e) {
|
|
|
|
serialization_errors.Bump(MAX_RETRIES);
|
|
|
|
return std::experimental::nullopt;
|
|
|
|
}
|
2018-01-27 01:50:16 +08:00
|
|
|
auto wall_time = timer.Elapsed();
|
|
|
|
auto metadata = result.metadata;
|
|
|
|
metadata["wall_time"] = wall_time.count();
|
|
|
|
{
|
2018-05-30 19:00:25 +08:00
|
|
|
std::unique_lock<utils::SpinLock> guard(lock_);
|
2018-01-27 01:50:16 +08:00
|
|
|
if (query_name != "") {
|
|
|
|
stats_[query_name].push_back(std::move(metadata));
|
|
|
|
} else {
|
|
|
|
stats_[query].push_back(std::move(metadata));
|
|
|
|
}
|
|
|
|
}
|
2018-02-15 23:28:38 +08:00
|
|
|
executed_queries.Bump();
|
2018-03-06 02:11:52 +08:00
|
|
|
serialization_errors.Bump(retries);
|
2018-01-27 01:50:16 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-05-30 19:00:25 +08:00
|
|
|
utils::SpinLock lock_;
|
2018-07-24 21:11:18 +08:00
|
|
|
std::unordered_map<std::string, std::vector<std::map<std::string, Value>>>
|
2018-01-27 01:50:16 +08:00
|
|
|
stats_;
|
|
|
|
|
|
|
|
std::atomic<bool> keep_running_{true};
|
|
|
|
std::thread runner_thread_;
|
|
|
|
|
|
|
|
private:
|
2018-06-20 23:44:47 +08:00
|
|
|
communication::ClientContext context_{FLAGS_use_ssl};
|
|
|
|
Client client_{&context_};
|
2018-01-27 01:50:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void RunMultithreadedTest(std::vector<std::unique_ptr<TestClient>> &clients) {
|
|
|
|
CHECK((int)clients.size() == FLAGS_num_workers);
|
|
|
|
|
|
|
|
// Open stream for writing stats.
|
|
|
|
std::streambuf *buf;
|
|
|
|
std::ofstream f;
|
|
|
|
if (FLAGS_output != "") {
|
|
|
|
f.open(FLAGS_output);
|
|
|
|
buf = f.rdbuf();
|
|
|
|
} else {
|
|
|
|
buf = std::cout.rdbuf();
|
|
|
|
}
|
|
|
|
std::ostream out(buf);
|
|
|
|
|
|
|
|
utils::Timer timer;
|
|
|
|
for (auto &client : clients) {
|
|
|
|
client->Run();
|
|
|
|
}
|
|
|
|
LOG(INFO) << "Starting test with " << clients.size() << " workers";
|
|
|
|
while (timer.Elapsed().count() < FLAGS_duration) {
|
2018-07-24 21:11:18 +08:00
|
|
|
std::unordered_map<std::string, std::map<std::string, Value>>
|
2018-01-27 01:50:16 +08:00
|
|
|
aggregated_stats;
|
|
|
|
|
|
|
|
using namespace std::chrono_literals;
|
2018-07-24 21:11:18 +08:00
|
|
|
std::unordered_map<std::string, std::vector<std::map<std::string, Value>>>
|
2018-01-27 01:50:16 +08:00
|
|
|
stats;
|
|
|
|
for (const auto &client : clients) {
|
|
|
|
auto client_stats = client->ConsumeStats();
|
|
|
|
for (const auto &client_query_stats : client_stats) {
|
|
|
|
auto &query_stats = stats[client_query_stats.first];
|
|
|
|
query_stats.insert(query_stats.end(), client_query_stats.second.begin(),
|
|
|
|
client_query_stats.second.end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-24 21:11:18 +08:00
|
|
|
// TODO: Here we combine pure values, json and Value which is a
|
2018-01-27 01:50:16 +08:00
|
|
|
// little bit chaotic. Think about refactoring this part to only use json
|
2018-07-24 21:11:18 +08:00
|
|
|
// and write Value to json converter.
|
2018-01-27 01:50:16 +08:00
|
|
|
const std::vector<std::string> fields = {
|
2018-03-06 02:11:52 +08:00
|
|
|
"wall_time",
|
|
|
|
"parsing_time",
|
|
|
|
"planning_time",
|
|
|
|
"plan_execution_time",
|
2018-01-27 01:50:16 +08:00
|
|
|
};
|
|
|
|
for (const auto &query_stats : stats) {
|
|
|
|
std::map<std::string, double> new_aggregated_query_stats;
|
|
|
|
for (const auto &stats : query_stats.second) {
|
|
|
|
for (const auto &field : fields) {
|
|
|
|
auto it = stats.find(field);
|
|
|
|
if (it != stats.end()) {
|
|
|
|
new_aggregated_query_stats[field] += it->second.ValueDouble();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int64_t new_count = query_stats.second.size();
|
|
|
|
|
|
|
|
auto &aggregated_query_stats = aggregated_stats[query_stats.first];
|
2018-07-24 21:11:18 +08:00
|
|
|
aggregated_query_stats.insert({"count", Value(0)});
|
2018-01-27 01:50:16 +08:00
|
|
|
auto old_count = aggregated_query_stats["count"].ValueInt();
|
|
|
|
aggregated_query_stats["count"].ValueInt() += new_count;
|
|
|
|
for (const auto &stat : new_aggregated_query_stats) {
|
2018-07-24 21:11:18 +08:00
|
|
|
auto it = aggregated_query_stats.insert({stat.first, Value(0.0)}).first;
|
2018-03-06 02:11:52 +08:00
|
|
|
it->second = (it->second.ValueDouble() * old_count + stat.second) /
|
|
|
|
(old_count + new_count);
|
Add deletion, more stats to card fraud and RWLock
Summary:
^^
this is a sample config to be used:
```
{
"num_workers": 1,
"cards_per_worker": 10001,
"pos_per_worker": 10001,
"fraud_probability": 0.01,
"hop_probability": 0.05,
"cleanup": {
"check_interval_sec": 10,
"tx_hi": 150000,
"tx_lo": 100000
},
"analytic": {
"query_interval_ms": 500,
"pos_limit": 10
}
}
```
I also added `RWLock` --- a wrapper around `pthread_rwlock`
Reviewers: mferencevic, mculinovic, florijan, teon.banek
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1280
2018-03-14 20:47:18 +08:00
|
|
|
stats::LogStat(
|
|
|
|
fmt::format("queries.{}.{}", query_stats.first, stat.first),
|
|
|
|
(stat.second / new_count));
|
2018-01-27 01:50:16 +08:00
|
|
|
}
|
Add deletion, more stats to card fraud and RWLock
Summary:
^^
this is a sample config to be used:
```
{
"num_workers": 1,
"cards_per_worker": 10001,
"pos_per_worker": 10001,
"fraud_probability": 0.01,
"hop_probability": 0.05,
"cleanup": {
"check_interval_sec": 10,
"tx_hi": 150000,
"tx_lo": 100000
},
"analytic": {
"query_interval_ms": 500,
"pos_limit": 10
}
}
```
I also added `RWLock` --- a wrapper around `pthread_rwlock`
Reviewers: mferencevic, mculinovic, florijan, teon.banek
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1280
2018-03-14 20:47:18 +08:00
|
|
|
stats::LogStat(fmt::format("queries.{}.count", query_stats.first),
|
|
|
|
new_count);
|
2018-01-27 01:50:16 +08:00
|
|
|
}
|
|
|
|
|
2018-02-15 23:28:38 +08:00
|
|
|
out << "{\"num_executed_queries\": " << executed_queries.Value() << ", "
|
2018-01-27 01:50:16 +08:00
|
|
|
<< "\"elapsed_time\": " << timer.Elapsed().count()
|
|
|
|
<< ", \"queries\": [";
|
2018-03-06 02:11:52 +08:00
|
|
|
utils::PrintIterable(
|
|
|
|
out, aggregated_stats, ", ", [](auto &stream, const auto &x) {
|
|
|
|
stream << "{\"query\": " << nlohmann::json(x.first)
|
|
|
|
<< ", \"stats\": ";
|
2018-07-24 21:11:18 +08:00
|
|
|
PrintJsonValue(stream, Value(x.second));
|
2018-03-06 02:11:52 +08:00
|
|
|
stream << "}";
|
|
|
|
});
|
2018-01-27 01:50:16 +08:00
|
|
|
out << "]}" << std::endl;
|
|
|
|
out.flush();
|
|
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
}
|
|
|
|
LOG(INFO) << "Stopping workers...";
|
|
|
|
for (auto &client : clients) {
|
|
|
|
client->Stop();
|
|
|
|
}
|
|
|
|
LOG(INFO) << "Stopped workers...";
|
|
|
|
}
|