2017-12-19 19:40:30 +08:00
|
|
|
#include <experimental/optional>
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
2017-12-22 21:36:25 +08:00
|
|
|
#include <unordered_set>
|
2017-12-19 19:40:30 +08:00
|
|
|
#include <vector>
|
|
|
|
|
2018-01-22 23:59:40 +08:00
|
|
|
#include "gmock/gmock.h"
|
2017-12-19 19:40:30 +08:00
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
2018-01-24 19:16:14 +08:00
|
|
|
#include "communication/rpc/server.hpp"
|
2017-12-19 19:40:30 +08:00
|
|
|
#include "distributed/coordination_master.hpp"
|
|
|
|
#include "distributed/coordination_worker.hpp"
|
2018-01-15 21:03:07 +08:00
|
|
|
#include "io/network/endpoint.hpp"
|
2017-12-19 19:40:30 +08:00
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
using communication::rpc::Server;
|
2017-12-19 19:40:30 +08:00
|
|
|
using namespace distributed;
|
2018-01-24 19:16:14 +08:00
|
|
|
using namespace std::literals::chrono_literals;
|
2017-12-19 19:40:30 +08:00
|
|
|
|
|
|
|
const int kWorkerCount = 5;
|
|
|
|
const std::string kLocal = "127.0.0.1";
|
|
|
|
|
|
|
|
class WorkerInThread {
|
|
|
|
public:
|
2018-01-15 21:03:07 +08:00
|
|
|
WorkerInThread(io::network::Endpoint master_endpoint, int desired_id = -1) {
|
2017-12-19 19:40:30 +08:00
|
|
|
worker_thread_ = std::thread([this, master_endpoint, desired_id] {
|
2018-02-23 17:56:56 +08:00
|
|
|
server_.emplace(Endpoint(kLocal, 0));
|
|
|
|
coord_.emplace(*server_, master_endpoint);
|
2017-12-19 19:40:30 +08:00
|
|
|
worker_id_ = coord_->RegisterWorker(desired_id);
|
|
|
|
coord_->WaitForShutdown();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
int worker_id() const { return worker_id_; }
|
2018-02-23 17:56:56 +08:00
|
|
|
auto endpoint() const { return server_->endpoint(); }
|
2017-12-19 19:40:30 +08:00
|
|
|
auto worker_endpoint(int worker_id) { return coord_->GetEndpoint(worker_id); }
|
|
|
|
void join() { worker_thread_.join(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::thread worker_thread_;
|
2018-02-23 17:56:56 +08:00
|
|
|
std::experimental::optional<Server> server_;
|
2017-12-19 19:40:30 +08:00
|
|
|
std::experimental::optional<WorkerCoordination> coord_;
|
|
|
|
std::atomic<int> worker_id_{0};
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(Distributed, Coordination) {
|
2018-02-23 17:56:56 +08:00
|
|
|
Server master_server({kLocal, 0});
|
2017-12-19 19:40:30 +08:00
|
|
|
std::vector<std::unique_ptr<WorkerInThread>> workers;
|
2018-01-10 20:56:12 +08:00
|
|
|
{
|
2018-02-23 17:56:56 +08:00
|
|
|
MasterCoordination master_coord(master_server);
|
2018-01-10 20:56:12 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < kWorkerCount; ++i)
|
|
|
|
workers.emplace_back(
|
2018-02-23 17:56:56 +08:00
|
|
|
std::make_unique<WorkerInThread>(master_server.endpoint()));
|
2018-01-10 20:56:12 +08:00
|
|
|
|
|
|
|
// Wait till all the workers are safely initialized.
|
|
|
|
std::this_thread::sleep_for(300ms);
|
|
|
|
|
|
|
|
// Expect that all workers have a different ID.
|
|
|
|
std::unordered_set<int> worker_ids;
|
|
|
|
for (const auto &w : workers) worker_ids.insert(w->worker_id());
|
|
|
|
EXPECT_EQ(worker_ids.size(), kWorkerCount);
|
|
|
|
|
|
|
|
// Check endpoints.
|
|
|
|
for (auto &w1 : workers) {
|
|
|
|
for (auto &w2 : workers) {
|
|
|
|
EXPECT_EQ(w1->worker_endpoint(w2->worker_id()), w2->endpoint());
|
|
|
|
}
|
2017-12-19 19:40:30 +08:00
|
|
|
}
|
2018-01-10 20:56:12 +08:00
|
|
|
} // Coordinated shutdown.
|
2017-12-19 19:40:30 +08:00
|
|
|
|
|
|
|
for (auto &worker : workers) worker->join();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Distributed, DesiredAndUniqueId) {
|
2018-02-23 17:56:56 +08:00
|
|
|
Server master_server({kLocal, 0});
|
2018-01-10 20:56:12 +08:00
|
|
|
std::vector<std::unique_ptr<WorkerInThread>> workers;
|
|
|
|
{
|
2018-02-23 17:56:56 +08:00
|
|
|
MasterCoordination master_coord(master_server);
|
2017-12-19 19:40:30 +08:00
|
|
|
|
2018-01-10 20:56:12 +08:00
|
|
|
workers.emplace_back(
|
2018-02-23 17:56:56 +08:00
|
|
|
std::make_unique<WorkerInThread>(master_server.endpoint(), 42));
|
2018-01-10 20:56:12 +08:00
|
|
|
std::this_thread::sleep_for(200ms);
|
|
|
|
workers.emplace_back(
|
2018-02-23 17:56:56 +08:00
|
|
|
std::make_unique<WorkerInThread>(master_server.endpoint(), 42));
|
2018-01-10 20:56:12 +08:00
|
|
|
std::this_thread::sleep_for(200ms);
|
2017-12-19 19:40:30 +08:00
|
|
|
|
2018-01-10 20:56:12 +08:00
|
|
|
EXPECT_EQ(workers[0]->worker_id(), 42);
|
|
|
|
EXPECT_NE(workers[1]->worker_id(), 42);
|
|
|
|
}
|
2017-12-19 19:40:30 +08:00
|
|
|
|
2018-01-10 20:56:12 +08:00
|
|
|
for (auto &worker : workers) worker->join();
|
2017-12-19 19:40:30 +08:00
|
|
|
}
|
2018-01-22 23:59:40 +08:00
|
|
|
|
|
|
|
TEST(Distributed, CoordinationWorkersId) {
|
2018-02-23 17:56:56 +08:00
|
|
|
Server master_server({kLocal, 0});
|
2018-01-22 23:59:40 +08:00
|
|
|
std::vector<std::unique_ptr<WorkerInThread>> workers;
|
|
|
|
{
|
2018-02-23 17:56:56 +08:00
|
|
|
MasterCoordination master_coord(master_server);
|
2018-01-22 23:59:40 +08:00
|
|
|
|
|
|
|
workers.emplace_back(
|
2018-02-23 17:56:56 +08:00
|
|
|
std::make_unique<WorkerInThread>(master_server.endpoint(), 42));
|
2018-01-22 23:59:40 +08:00
|
|
|
std::this_thread::sleep_for(200ms);
|
|
|
|
workers.emplace_back(
|
2018-02-23 17:56:56 +08:00
|
|
|
std::make_unique<WorkerInThread>(master_server.endpoint(), 42));
|
2018-01-22 23:59:40 +08:00
|
|
|
std::this_thread::sleep_for(200ms);
|
|
|
|
|
|
|
|
std::vector<int> ids;
|
|
|
|
ids.push_back(0);
|
|
|
|
|
|
|
|
for (auto &worker : workers) ids.push_back(worker->worker_id());
|
|
|
|
EXPECT_THAT(master_coord.GetWorkerIds(),
|
|
|
|
testing::UnorderedElementsAreArray(ids));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto &worker : workers) worker->join();
|
|
|
|
}
|