2021-10-26 14:53:56 +08:00
|
|
|
// Copyright 2021 Memgraph Ltd.
|
|
|
|
//
|
|
|
|
// Use of this software is governed by the Business Source License
|
|
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
|
|
//
|
|
|
|
// As of the Change Date specified in that file, in accordance with
|
|
|
|
// the Business Source License, use of this software will be governed
|
|
|
|
// by the Apache License, Version 2.0, included in the file
|
|
|
|
// licenses/APL.txt.
|
|
|
|
|
2018-08-10 20:35:43 +08:00
|
|
|
#include <array>
|
|
|
|
#include <chrono>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <queue>
|
|
|
|
#include <random>
|
|
|
|
#include <sstream>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <unordered_set>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <json/json.hpp>
|
|
|
|
|
|
|
|
#include "io/network/utils.hpp"
|
|
|
|
#include "utils/algorithm.hpp"
|
2021-01-21 22:47:56 +08:00
|
|
|
#include "utils/logging.hpp"
|
2018-08-10 20:35:43 +08:00
|
|
|
#include "utils/timer.hpp"
|
|
|
|
|
|
|
|
#include "long_running_common.hpp"
|
|
|
|
|
|
|
|
using communication::bolt::Edge;
|
|
|
|
using communication::bolt::Value;
|
|
|
|
using communication::bolt::Vertex;
|
|
|
|
|
|
|
|
class BfsPokecClient : public TestClient {
|
|
|
|
public:
|
2021-02-18 22:32:43 +08:00
|
|
|
BfsPokecClient(int id, const std::string &db) : TestClient(), rg_(id), db_(db) {
|
2018-08-10 20:35:43 +08:00
|
|
|
auto result = Execute("MATCH (n:User) RETURN count(1)", {}, "NumNodes");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(result, "Read-only query should not fail");
|
2018-08-10 20:35:43 +08:00
|
|
|
num_nodes_ = result->records[0][0].ValueInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::mt19937 rg_;
|
|
|
|
std::string db_;
|
|
|
|
int num_nodes_;
|
|
|
|
|
|
|
|
int RandomId() {
|
|
|
|
std::uniform_int_distribution<int64_t> dist(1, num_nodes_);
|
|
|
|
auto id = dist(rg_);
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BfsWithDestinationNode() {
|
|
|
|
auto start = RandomId();
|
|
|
|
auto end = RandomId();
|
|
|
|
while (start == end) {
|
|
|
|
end = RandomId();
|
|
|
|
}
|
|
|
|
if (FLAGS_db == "memgraph") {
|
|
|
|
auto result = Execute(
|
2018-08-29 16:26:27 +08:00
|
|
|
"MATCH (n:User {id: $start}), (m:User {id: $end}), "
|
|
|
|
"p = (n)-[*bfs..15]->(m) "
|
|
|
|
"RETURN extract(n in nodes(p) | n.id) AS path",
|
2018-08-10 20:35:43 +08:00
|
|
|
{{"start", start}, {"end", end}}, "Bfs");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(result, "Read-only query should not fail!");
|
2018-08-10 20:35:43 +08:00
|
|
|
} else if (FLAGS_db == "neo4j") {
|
|
|
|
auto result = Execute(
|
|
|
|
"MATCH p = shortestPath("
|
|
|
|
"(n:User {id: $start})-[*..15]->(m:User {id: $end}))"
|
|
|
|
"RETURN [x in nodes(p) | x.id] AS path;",
|
|
|
|
{{"start", start}, {"end", end}}, "Bfs");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(result, "Read-only query should not fail!");
|
2018-08-10 20:35:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BfsWithoutDestinationNode() {
|
|
|
|
auto start = RandomId();
|
|
|
|
if (FLAGS_db == "memgraph") {
|
|
|
|
auto result = Execute(
|
|
|
|
"MATCH p = (n:User {id: $start})-[*bfs..15]->(m:User) WHERE m != n "
|
2018-08-29 16:26:27 +08:00
|
|
|
"RETURN extract(n in nodes(p) | n.id) AS path",
|
2018-08-10 20:35:43 +08:00
|
|
|
{{"start", start}}, "Bfs");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(result, "Read-only query should not fail!");
|
2018-08-10 20:35:43 +08:00
|
|
|
} else {
|
|
|
|
auto result = Execute(
|
|
|
|
"MATCH p = shortestPath("
|
|
|
|
"(n:User {id: $start})-[*..15]->(m:User)) WHERE m <> n "
|
|
|
|
"RETURN [x in nodes(p) | x.id] AS path;",
|
|
|
|
{{"start", start}}, "Bfs");
|
2021-01-21 22:47:56 +08:00
|
|
|
MG_ASSERT(result, "Read-only query should not fail!");
|
2018-08-10 20:35:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
virtual void Step() override {
|
|
|
|
if (FLAGS_scenario == "with_destination_node") {
|
|
|
|
BfsWithDestinationNode();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FLAGS_scenario == "without_destination_node") {
|
|
|
|
BfsWithoutDestinationNode();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-21 22:47:56 +08:00
|
|
|
LOG_FATAL("Should not get here: unknown scenario!");
|
2018-08-10 20:35:43 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
|
|
|
|
2020-10-20 18:55:13 +08:00
|
|
|
communication::SSLInit sslInit;
|
2018-08-10 20:35:43 +08:00
|
|
|
|
|
|
|
Endpoint endpoint(FLAGS_address, FLAGS_port);
|
|
|
|
ClientContext context(FLAGS_use_ssl);
|
|
|
|
Client client(&context);
|
2018-10-19 18:28:38 +08:00
|
|
|
client.Connect(endpoint, FLAGS_username, FLAGS_password);
|
2018-08-10 20:35:43 +08:00
|
|
|
|
|
|
|
std::vector<std::unique_ptr<TestClient>> clients;
|
|
|
|
for (auto i = 0; i < FLAGS_num_workers; ++i) {
|
|
|
|
clients.emplace_back(std::make_unique<BfsPokecClient>(i, "memgraph"));
|
|
|
|
}
|
|
|
|
|
|
|
|
RunMultithreadedTest(clients);
|
|
|
|
return 0;
|
|
|
|
}
|