2018-04-30 15:33:09 +08:00
|
|
|
#include <mutex>
|
|
|
|
|
2018-02-21 20:03:04 +08:00
|
|
|
#include "boost/archive/binary_iarchive.hpp"
|
|
|
|
#include "boost/archive/binary_oarchive.hpp"
|
|
|
|
#include "boost/serialization/export.hpp"
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "communication/rpc/messages.hpp"
|
|
|
|
#include "communication/rpc/server.hpp"
|
2018-04-03 22:19:17 +08:00
|
|
|
#include "distributed/cluster_discovery_master.hpp"
|
|
|
|
#include "distributed/cluster_discovery_worker.hpp"
|
2018-02-21 20:03:04 +08:00
|
|
|
#include "distributed/coordination_master.hpp"
|
|
|
|
#include "distributed/coordination_worker.hpp"
|
|
|
|
#include "distributed/rpc_worker_clients.hpp"
|
|
|
|
#include "distributed/serialization.hpp"
|
|
|
|
#include "io/network/endpoint.hpp"
|
|
|
|
|
|
|
|
namespace distributed {
|
|
|
|
|
|
|
|
RPC_NO_MEMBER_MESSAGE(IncrementCounterReq);
|
|
|
|
RPC_NO_MEMBER_MESSAGE(IncrementCounterRes);
|
|
|
|
|
|
|
|
using IncrementCounterRpc =
|
|
|
|
communication::rpc::RequestResponse<IncrementCounterReq,
|
|
|
|
IncrementCounterRes>;
|
|
|
|
}; // namespace distributed
|
|
|
|
|
|
|
|
BOOST_CLASS_EXPORT(distributed::IncrementCounterReq);
|
|
|
|
BOOST_CLASS_EXPORT(distributed::IncrementCounterRes);
|
|
|
|
|
|
|
|
class RpcWorkerClientsTest : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
const io::network::Endpoint kLocalHost{"127.0.0.1", 0};
|
|
|
|
const int kWorkerCount = 2;
|
|
|
|
void SetUp() override {
|
2018-04-16 16:43:16 +08:00
|
|
|
master_coord_->SetRecoveryInfo(std::experimental::nullopt);
|
2018-02-21 20:03:04 +08:00
|
|
|
for (int i = 1; i <= kWorkerCount; ++i) {
|
2018-02-23 17:56:56 +08:00
|
|
|
workers_server_.emplace_back(
|
|
|
|
std::make_unique<communication::rpc::Server>(kLocalHost));
|
2018-02-21 20:03:04 +08:00
|
|
|
|
|
|
|
workers_coord_.emplace_back(
|
|
|
|
std::make_unique<distributed::WorkerCoordination>(
|
2018-02-23 17:56:56 +08:00
|
|
|
*workers_server_.back(), master_server_.endpoint()));
|
2018-02-21 20:03:04 +08:00
|
|
|
|
2018-04-03 22:19:17 +08:00
|
|
|
cluster_discovery_.emplace_back(
|
|
|
|
std::make_unique<distributed::ClusterDiscoveryWorker>(
|
|
|
|
*workers_server_.back(), *workers_coord_.back(),
|
|
|
|
rpc_workers_.GetClientPool(0)));
|
|
|
|
|
|
|
|
cluster_discovery_.back()->RegisterWorker(i);
|
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
workers_server_.back()->Register<distributed::IncrementCounterRpc>(
|
2018-02-21 20:03:04 +08:00
|
|
|
[this, i](const distributed::IncrementCounterReq &) {
|
2018-04-30 15:33:09 +08:00
|
|
|
std::unique_lock<std::mutex> lock(mutex_);
|
2018-02-21 20:03:04 +08:00
|
|
|
workers_cnt_[i]++;
|
|
|
|
return std::make_unique<distributed::IncrementCounterRes>();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TearDown() override {
|
|
|
|
std::vector<std::thread> wait_on_shutdown;
|
2018-04-11 22:05:49 +08:00
|
|
|
for (int i = 0; i < workers_coord_.size(); ++i) {
|
|
|
|
wait_on_shutdown.emplace_back([i, this]() {
|
|
|
|
workers_coord_[i]->WaitForShutdown();
|
|
|
|
workers_server_[i] = nullptr;
|
|
|
|
});
|
|
|
|
}
|
2018-02-21 20:03:04 +08:00
|
|
|
|
|
|
|
std::this_thread::sleep_for(300ms);
|
|
|
|
|
|
|
|
// Starts server shutdown and notifies the workers
|
|
|
|
master_coord_ = std::experimental::nullopt;
|
|
|
|
for (auto &worker : wait_on_shutdown) worker.join();
|
|
|
|
}
|
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
std::vector<std::unique_ptr<communication::rpc::Server>> workers_server_;
|
2018-02-21 20:03:04 +08:00
|
|
|
std::vector<std::unique_ptr<distributed::WorkerCoordination>> workers_coord_;
|
2018-04-03 22:19:17 +08:00
|
|
|
std::vector<std::unique_ptr<distributed::ClusterDiscoveryWorker>>
|
|
|
|
cluster_discovery_;
|
2018-04-30 15:33:09 +08:00
|
|
|
std::mutex mutex_;
|
2018-02-21 20:03:04 +08:00
|
|
|
std::unordered_map<int, int> workers_cnt_;
|
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
communication::rpc::Server master_server_{kLocalHost};
|
2018-02-21 20:03:04 +08:00
|
|
|
std::experimental::optional<distributed::MasterCoordination> master_coord_{
|
2018-04-03 22:19:17 +08:00
|
|
|
master_server_.endpoint()};
|
2018-02-21 20:03:04 +08:00
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
distributed::RpcWorkerClients rpc_workers_{*master_coord_};
|
2018-04-03 22:19:17 +08:00
|
|
|
distributed::ClusterDiscoveryMaster cluster_disocvery_{
|
|
|
|
master_server_, *master_coord_, rpc_workers_};
|
2018-02-21 20:03:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(RpcWorkerClientsTest, GetWorkerIds) {
|
|
|
|
EXPECT_THAT(rpc_workers_.GetWorkerIds(), testing::UnorderedElementsAreArray(
|
|
|
|
master_coord_->GetWorkerIds()));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RpcWorkerClientsTest, GetClientPool) {
|
|
|
|
auto &pool1 = rpc_workers_.GetClientPool(1);
|
|
|
|
auto &pool2 = rpc_workers_.GetClientPool(2);
|
|
|
|
EXPECT_NE(&pool1, &pool2);
|
|
|
|
EXPECT_EQ(&pool1, &rpc_workers_.GetClientPool(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RpcWorkerClientsTest, ExecuteOnWorker) {
|
|
|
|
auto execute = [](auto &client) -> void {
|
2018-03-12 22:17:46 +08:00
|
|
|
ASSERT_TRUE(client.template Call<distributed::IncrementCounterRpc>());
|
2018-02-21 20:03:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
rpc_workers_.ExecuteOnWorker<void>(1, execute).get();
|
|
|
|
EXPECT_EQ(workers_cnt_[0], 0);
|
|
|
|
EXPECT_EQ(workers_cnt_[1], 1);
|
|
|
|
EXPECT_EQ(workers_cnt_[2], 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(RpcWorkerClientsTest, ExecuteOnWorkers) {
|
|
|
|
auto execute = [](auto &client) -> void {
|
2018-03-12 22:17:46 +08:00
|
|
|
ASSERT_TRUE(client.template Call<distributed::IncrementCounterRpc>());
|
2018-02-21 20:03:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
// Skip master
|
|
|
|
for (auto &future : rpc_workers_.ExecuteOnWorkers<void>(0, execute))
|
|
|
|
future.get();
|
|
|
|
|
|
|
|
EXPECT_EQ(workers_cnt_[0], 0);
|
|
|
|
EXPECT_EQ(workers_cnt_[1], 1);
|
|
|
|
EXPECT_EQ(workers_cnt_[2], 1);
|
|
|
|
}
|