2018-02-23 00:24:28 +08:00
|
|
|
#include <chrono>
|
2018-09-27 21:07:46 +08:00
|
|
|
#include <experimental/filesystem>
|
2018-02-23 00:24:28 +08:00
|
|
|
#include <iostream>
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <glog/logging.h>
|
2018-09-27 21:07:46 +08:00
|
|
|
#include <gtest/gtest.h>
|
2018-02-23 00:24:28 +08:00
|
|
|
|
2018-10-05 18:37:23 +08:00
|
|
|
#include "database/distributed/distributed_graph_db.hpp"
|
2018-08-24 16:12:04 +08:00
|
|
|
#include "query/distributed_interpreter.hpp"
|
2018-03-21 17:26:43 +08:00
|
|
|
#include "query/repl.hpp"
|
2018-09-27 21:07:46 +08:00
|
|
|
#include "utils/file.hpp"
|
2018-02-23 00:24:28 +08:00
|
|
|
#include "utils/flag_validation.hpp"
|
|
|
|
|
|
|
|
DEFINE_VALIDATED_int32(worker_count, 1,
|
|
|
|
"The number of worker nodes in cluster.",
|
|
|
|
FLAG_IN_RANGE(1, 1000));
|
|
|
|
DECLARE_int32(min_log_level);
|
2018-09-27 21:07:46 +08:00
|
|
|
DECLARE_string(durability_directory);
|
|
|
|
|
|
|
|
namespace fs = std::experimental::filesystem;
|
2018-02-23 00:24:28 +08:00
|
|
|
|
|
|
|
const std::string kLocal = "127.0.0.1";
|
|
|
|
|
|
|
|
class WorkerInThread {
|
|
|
|
public:
|
|
|
|
explicit WorkerInThread(database::Config config) : worker_(config) {
|
2018-10-16 16:58:41 +08:00
|
|
|
thread_ = std::thread([this, config] {
|
|
|
|
worker_.Start();
|
|
|
|
EXPECT_TRUE(worker_.AwaitShutdown());
|
|
|
|
});
|
2018-02-23 00:24:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
~WorkerInThread() {
|
|
|
|
if (thread_.joinable()) thread_.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
database::Worker worker_;
|
|
|
|
std::thread thread_;
|
|
|
|
};
|
|
|
|
|
2018-09-27 21:07:46 +08:00
|
|
|
fs::path GetDurabilityDirectory(const fs::path &path, int worker_id) {
|
|
|
|
if (worker_id == 0) return path / "master";
|
|
|
|
return path / fmt::format("worker{}", worker_id);
|
|
|
|
}
|
|
|
|
|
2018-02-23 00:24:28 +08:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
2018-02-23 22:49:56 +08:00
|
|
|
FLAGS_min_log_level = google::ERROR;
|
2018-02-23 00:24:28 +08:00
|
|
|
google::InitGoogleLogging(argv[0]);
|
|
|
|
|
2018-09-27 21:07:46 +08:00
|
|
|
fs::path tmp_dir =
|
|
|
|
fs::temp_directory_path() / "MG_test_manual_distributed_repl";
|
|
|
|
EXPECT_TRUE(utils::EnsureDir(tmp_dir));
|
|
|
|
|
2018-02-23 00:24:28 +08:00
|
|
|
// Start the master
|
|
|
|
database::Config master_config;
|
|
|
|
master_config.master_endpoint = {kLocal, 0};
|
2018-09-27 21:07:46 +08:00
|
|
|
master_config.durability_directory = GetDurabilityDirectory(tmp_dir, 0);
|
|
|
|
// Flag needs to be updated due to props on disk storage.
|
|
|
|
FLAGS_durability_directory = GetDurabilityDirectory(tmp_dir, 0);
|
2018-02-23 00:24:28 +08:00
|
|
|
auto master = std::make_unique<database::Master>(master_config);
|
2018-10-16 16:58:41 +08:00
|
|
|
master->Start();
|
2018-02-23 00:24:28 +08:00
|
|
|
// Allow the master to get initialized before making workers.
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(250));
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<WorkerInThread>> workers;
|
|
|
|
for (int i = 0; i < FLAGS_worker_count; ++i) {
|
|
|
|
database::Config config;
|
|
|
|
config.worker_id = i + 1;
|
|
|
|
config.master_endpoint = master->endpoint();
|
|
|
|
config.worker_endpoint = {kLocal, 0};
|
2018-09-27 21:07:46 +08:00
|
|
|
config.durability_directory = GetDurabilityDirectory(tmp_dir, i + 1);
|
|
|
|
// Flag needs to be updated due to props on disk storage.
|
|
|
|
FLAGS_durability_directory = GetDurabilityDirectory(tmp_dir, i + 1);
|
2018-02-23 00:24:28 +08:00
|
|
|
workers.emplace_back(std::make_unique<WorkerInThread>(config));
|
|
|
|
}
|
|
|
|
|
2018-11-06 22:48:18 +08:00
|
|
|
// Wait for the whole cluster to be up and running.
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
|
|
|
while (master->GetWorkerIds().size() < FLAGS_worker_count + 1) {
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
|
|
|
}
|
|
|
|
for (int i = 0; i < FLAGS_worker_count; ++i) {
|
|
|
|
while (workers[i]->worker_.GetWorkerIds().size() < FLAGS_worker_count + 1) {
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(200));
|
|
|
|
|
2018-02-23 00:24:28 +08:00
|
|
|
// Start the REPL
|
2018-08-24 16:12:04 +08:00
|
|
|
{
|
|
|
|
query::DistributedInterpreter interpreter(master.get());
|
|
|
|
query::Repl(master.get(), &interpreter);
|
|
|
|
}
|
2018-02-23 00:24:28 +08:00
|
|
|
|
2018-03-21 17:26:43 +08:00
|
|
|
master = nullptr;
|
2018-02-23 00:24:28 +08:00
|
|
|
return 0;
|
|
|
|
}
|