2021-01-16 02:04:44 +08:00
|
|
|
#include <chrono>
|
|
|
|
#include <random>
|
|
|
|
#include <thread>
|
|
|
|
#include <unordered_set>
|
|
|
|
|
|
|
|
#include <fmt/format.h>
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
|
|
|
|
#include "common.hpp"
|
2021-01-21 22:47:56 +08:00
|
|
|
#include "utils/logging.hpp"
|
2021-01-16 02:04:44 +08:00
|
|
|
#include "utils/thread.hpp"
|
|
|
|
#include "utils/timer.hpp"
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
google::SetUsageMessage("Memgraph E2E Replication Read-write Benchmark");
|
|
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
2021-01-21 22:47:56 +08:00
|
|
|
logging::RedirectToStderr();
|
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
auto database_endpoints = mg::e2e::replication::ParseDatabaseEndpoints(FLAGS_database_endpoints);
|
2021-01-16 02:04:44 +08:00
|
|
|
|
|
|
|
mg::Client::Init();
|
|
|
|
|
|
|
|
{
|
|
|
|
auto client = mg::e2e::replication::Connect(database_endpoints[0]);
|
|
|
|
client->Execute("MATCH (n) DETACH DELETE n;");
|
|
|
|
client->DiscardAll();
|
|
|
|
client->Execute("CREATE INDEX ON :Node(id);");
|
|
|
|
client->DiscardAll();
|
|
|
|
client->Execute("CREATE CONSTRAINT ON (n:Node) ASSERT n.id IS UNIQUE;");
|
|
|
|
client->DiscardAll();
|
|
|
|
|
|
|
|
// Sleep a bit so the constraints get replicated.
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(500));
|
|
|
|
for (const auto &database_endpoint : database_endpoints) {
|
|
|
|
auto client = mg::e2e::replication::Connect(database_endpoint);
|
|
|
|
client->Execute("SHOW CONSTRAINT INFO;");
|
|
|
|
if (const auto data = client->FetchAll()) {
|
|
|
|
const auto label_name = (*data)[0][1].ValueString();
|
|
|
|
const auto property_name = (*data)[0][2].ValueList()[0].ValueString();
|
|
|
|
if (label_name != "Node" || property_name != "id") {
|
2021-03-26 22:02:35 +08:00
|
|
|
LOG_FATAL("{} does NOT hava valid constraint created.", database_endpoint);
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
} else {
|
2021-01-21 22:47:56 +08:00
|
|
|
LOG_FATAL("Unable to get CONSTRAINT INFO from {}", database_endpoint);
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
}
|
2021-01-21 22:47:56 +08:00
|
|
|
spdlog::info("All constraints are in-place.");
|
2021-01-16 02:04:44 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < FLAGS_nodes; ++i) {
|
|
|
|
client->Execute("CREATE (:Node {id:" + std::to_string(i) + "});");
|
|
|
|
client->DiscardAll();
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
mg::e2e::replication::IntGenerator edge_generator("EdgeCreateGenerator", 0, FLAGS_nodes - 1);
|
2021-01-16 02:04:44 +08:00
|
|
|
for (int i = 0; i < FLAGS_edges; ++i) {
|
|
|
|
client->Execute("MATCH (n {id:" + std::to_string(edge_generator.Next()) +
|
2021-02-18 22:32:43 +08:00
|
|
|
"}), (m {id:" + std::to_string(edge_generator.Next()) + "}) CREATE (n)-[:Edge]->(m);");
|
2021-01-16 02:04:44 +08:00
|
|
|
client->DiscardAll();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const int num_threads = std::thread::hardware_concurrency();
|
|
|
|
std::vector<std::thread> threads;
|
|
|
|
std::vector<double> thread_duration;
|
|
|
|
threads.reserve(num_threads);
|
|
|
|
thread_duration.resize(num_threads);
|
|
|
|
|
|
|
|
for (int i = 0; i < num_threads; ++i) {
|
2021-02-18 22:32:43 +08:00
|
|
|
const auto &database_endpoint = database_endpoints[i % database_endpoints.size()];
|
|
|
|
threads.emplace_back(
|
|
|
|
[i, &database_endpoint, cluster_size = database_endpoints.size(), &local_duration = thread_duration[i]] {
|
|
|
|
auto client = mg::e2e::replication::Connect(database_endpoint);
|
|
|
|
mg::e2e::replication::IntGenerator node_update_generator(fmt::format("NodeUpdateGenerator {}", i), 0,
|
|
|
|
FLAGS_nodes - 1);
|
|
|
|
utils::Timer t;
|
2021-01-16 02:04:44 +08:00
|
|
|
|
2021-02-18 22:32:43 +08:00
|
|
|
while (true) {
|
|
|
|
local_duration = t.Elapsed().count();
|
|
|
|
if (local_duration >= FLAGS_reads_duration_limit) break;
|
|
|
|
// In the main case try to update.
|
|
|
|
if (i % cluster_size == 0) {
|
|
|
|
try {
|
|
|
|
client->Execute("MATCH (n:Node {id:" + std::to_string(node_update_generator.Next()) +
|
|
|
|
"}) SET n.id = " + std::to_string(node_update_generator.Next()) + " RETURN n.id;");
|
|
|
|
client->FetchAll();
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
// Pass.
|
|
|
|
}
|
|
|
|
} else { // In the replica case fetch all unique ids.
|
|
|
|
try {
|
|
|
|
client->Execute("MATCH (n) RETURN n.id;");
|
|
|
|
const auto data = client->FetchAll();
|
|
|
|
std::unordered_set<int64_t> unique;
|
|
|
|
for (const auto &value : *data) {
|
|
|
|
unique.insert(value[0].ValueInt());
|
|
|
|
}
|
|
|
|
if ((*data).size() != unique.size()) {
|
|
|
|
LOG_FATAL("Some ids are equal.");
|
|
|
|
}
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
LOG_FATAL(e.what());
|
|
|
|
break;
|
|
|
|
}
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
}
|
2021-02-18 22:32:43 +08:00
|
|
|
});
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for (auto &t : threads) {
|
|
|
|
if (t.joinable()) t.join();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
auto client = mg::e2e::replication::Connect(database_endpoints[0]);
|
|
|
|
client->Execute("DROP CONSTRAINT ON (n:Node) ASSERT n.id IS UNIQUE");
|
|
|
|
client->DiscardAll();
|
|
|
|
// Sleep a bit so the drop constraints get replicated.
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(500));
|
|
|
|
for (const auto &database_endpoint : database_endpoints) {
|
|
|
|
auto client = mg::e2e::replication::Connect(database_endpoint);
|
|
|
|
client->Execute("SHOW CONSTRAINT INFO;");
|
|
|
|
if (const auto data = client->FetchAll()) {
|
|
|
|
if ((*data).size() != 0) {
|
2021-01-21 22:47:56 +08:00
|
|
|
LOG_FATAL("{} still have some constraints.", database_endpoint);
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
} else {
|
2021-01-21 22:47:56 +08:00
|
|
|
LOG_FATAL("Unable to get CONSTRAINT INFO from {}", database_endpoint);
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
}
|
2021-01-21 22:47:56 +08:00
|
|
|
spdlog::info("All constraints were deleted.");
|
2021-01-16 02:04:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|