2018-02-05 23:28:22 +08:00
|
|
|
#include <memory>
|
2018-02-05 16:48:45 +08:00
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include "database/graph_db.hpp"
|
2018-02-14 16:44:48 +08:00
|
|
|
#include "database/graph_db_accessor.hpp"
|
2018-02-15 22:03:30 +08:00
|
|
|
#include "storage/address_types.hpp"
|
2018-02-05 16:48:45 +08:00
|
|
|
#include "transactions/engine_master.hpp"
|
|
|
|
|
|
|
|
class DistributedGraphDbTest : public ::testing::Test {
|
|
|
|
const std::string kLocal = "127.0.0.1";
|
2018-02-05 23:28:22 +08:00
|
|
|
const int kWorkerCount = 2;
|
|
|
|
|
2018-02-05 16:48:45 +08:00
|
|
|
class WorkerInThread {
|
|
|
|
public:
|
2018-02-06 23:26:29 +08:00
|
|
|
explicit WorkerInThread(database::Config config) : worker_(config) {
|
2018-02-05 16:48:45 +08:00
|
|
|
thread_ = std::thread([this, config] { worker_.WaitForShutdown(); });
|
|
|
|
}
|
|
|
|
|
|
|
|
~WorkerInThread() {
|
|
|
|
if (thread_.joinable()) thread_.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
database::Worker worker_;
|
|
|
|
std::thread thread_;
|
|
|
|
};
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void SetUp() override {
|
|
|
|
const auto kInitTime = 200ms;
|
|
|
|
|
|
|
|
database::Config master_config;
|
|
|
|
master_config.master_endpoint = {kLocal, 0};
|
2018-02-05 23:28:22 +08:00
|
|
|
master_ = std::make_unique<database::Master>(master_config);
|
2018-02-05 16:48:45 +08:00
|
|
|
std::this_thread::sleep_for(kInitTime);
|
|
|
|
|
|
|
|
auto worker_config = [this](int worker_id) {
|
|
|
|
database::Config config;
|
|
|
|
config.worker_id = worker_id;
|
|
|
|
config.master_endpoint = master_->endpoint();
|
|
|
|
config.worker_endpoint = {kLocal, 0};
|
|
|
|
return config;
|
|
|
|
};
|
|
|
|
|
2018-02-05 23:28:22 +08:00
|
|
|
for (int i = 0; i < kWorkerCount; ++i) {
|
|
|
|
workers_.emplace_back(
|
|
|
|
std::make_unique<WorkerInThread>(worker_config(i + 1)));
|
|
|
|
std::this_thread::sleep_for(kInitTime);
|
|
|
|
}
|
2018-02-05 16:48:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void TearDown() override {
|
|
|
|
// Kill master first because it will expect a shutdown response from the
|
|
|
|
// workers.
|
2018-02-05 23:28:22 +08:00
|
|
|
master_ = nullptr;
|
|
|
|
for (int i = kWorkerCount - 1; i >= 0; --i) workers_[i] = nullptr;
|
2018-02-05 16:48:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
database::Master &master() { return *master_; }
|
|
|
|
auto &master_tx_engine() {
|
|
|
|
return dynamic_cast<tx::MasterEngine &>(master_->tx_engine());
|
|
|
|
}
|
2018-02-05 23:28:22 +08:00
|
|
|
|
|
|
|
database::Worker &worker(int worker_id) {
|
|
|
|
return workers_[worker_id - 1]->worker_;
|
|
|
|
}
|
2018-02-05 16:48:45 +08:00
|
|
|
|
2018-02-14 16:44:48 +08:00
|
|
|
/// Inserts a vertex and returns it's global address. Does it in a new
|
|
|
|
/// transaction.
|
2018-02-20 21:48:36 +08:00
|
|
|
storage::VertexAddress InsertVertex(database::GraphDb &db) {
|
2018-02-14 16:44:48 +08:00
|
|
|
database::GraphDbAccessor dba{db};
|
|
|
|
auto r_val = dba.InsertVertex().GlobalAddress();
|
|
|
|
dba.Commit();
|
|
|
|
return r_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Inserts an edge (on the 'from' side) and returns it's global address.
|
2018-02-23 00:07:35 +08:00
|
|
|
auto InsertEdge(storage::VertexAddress from, storage::VertexAddress to,
|
|
|
|
const std::string &edge_type_name) {
|
|
|
|
CHECK(from.is_remote() && to.is_remote())
|
|
|
|
<< "Distributed test InsertEdge only takes global addresses";
|
|
|
|
auto db_for_vertex = [this](const auto &vertex) -> database::GraphDb & {
|
|
|
|
if (vertex.worker_id()) return worker(vertex.worker_id());
|
|
|
|
return master();
|
|
|
|
};
|
|
|
|
database::GraphDbAccessor dba(db_for_vertex(from));
|
2018-02-14 16:44:48 +08:00
|
|
|
auto from_v = dba.FindVertexChecked(from.gid(), false);
|
|
|
|
auto edge_type = dba.EdgeType(edge_type_name);
|
|
|
|
|
|
|
|
// If 'to' is on the same worker as 'from', create and send local.
|
|
|
|
if (to.worker_id() == from.worker_id()) {
|
|
|
|
auto to_v = dba.FindVertexChecked(to.gid(), false);
|
|
|
|
auto r_val = dba.InsertEdge(from_v, to_v, edge_type).GlobalAddress();
|
|
|
|
dba.Commit();
|
|
|
|
return r_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 'to' is not on the same worker as 'from'
|
|
|
|
auto edge_ga = dba.InsertOnlyEdge(from, to, edge_type,
|
|
|
|
dba.db().storage().EdgeGenerator().Next())
|
|
|
|
.GlobalAddress();
|
|
|
|
from_v.update().out_.emplace(to, edge_ga, edge_type);
|
2018-02-23 00:07:35 +08:00
|
|
|
database::GraphDbAccessor dba_to(db_for_vertex(to), dba.transaction_id());
|
2018-02-14 16:44:48 +08:00
|
|
|
auto to_v = dba_to.FindVertexChecked(to.gid(), false);
|
|
|
|
to_v.update().in_.emplace(from, edge_ga, edge_type);
|
|
|
|
|
|
|
|
dba.Commit();
|
|
|
|
return edge_ga;
|
|
|
|
}
|
|
|
|
|
2018-02-05 16:48:45 +08:00
|
|
|
private:
|
2018-02-05 23:28:22 +08:00
|
|
|
std::unique_ptr<database::Master> master_;
|
|
|
|
std::vector<std::unique_ptr<WorkerInThread>> workers_;
|
2018-02-05 16:48:45 +08:00
|
|
|
};
|