2024-03-06 16:16:46 +08:00
|
|
|
// Copyright 2024 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.
|
|
|
|
|
|
|
|
#include "nuraft/coordinator_cluster_state.hpp"
|
2024-03-21 14:41:26 +08:00
|
|
|
#include "io/network/endpoint.hpp"
|
2024-03-06 16:16:46 +08:00
|
|
|
#include "replication_coordination_glue/role.hpp"
|
|
|
|
#include "utils/file.hpp"
|
2024-03-15 21:41:02 +08:00
|
|
|
#include "utils/uuid.hpp"
|
2024-03-06 16:16:46 +08:00
|
|
|
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "json/json.hpp"
|
|
|
|
|
|
|
|
#include "libnuraft/nuraft.hxx"
|
|
|
|
|
|
|
|
using memgraph::coordination::CoordinatorClusterState;
|
2024-03-15 21:41:02 +08:00
|
|
|
using memgraph::coordination::CoordinatorInstanceState;
|
|
|
|
using memgraph::coordination::CoordinatorToCoordinatorConfig;
|
2024-03-21 14:41:26 +08:00
|
|
|
using memgraph::coordination::CoordinatorToReplicaConfig;
|
2024-03-06 16:16:46 +08:00
|
|
|
using memgraph::coordination::RaftLogAction;
|
2024-03-21 14:41:26 +08:00
|
|
|
using memgraph::coordination::ReplicationInstanceState;
|
|
|
|
using memgraph::io::network::Endpoint;
|
2024-03-06 16:16:46 +08:00
|
|
|
using memgraph::replication_coordination_glue::ReplicationMode;
|
|
|
|
using memgraph::replication_coordination_glue::ReplicationRole;
|
2024-03-15 21:41:02 +08:00
|
|
|
using memgraph::utils::UUID;
|
2024-03-06 16:16:46 +08:00
|
|
|
using nuraft::buffer;
|
|
|
|
using nuraft::ptr;
|
|
|
|
|
|
|
|
class CoordinatorClusterStateTest : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
void SetUp() override {}
|
|
|
|
|
|
|
|
void TearDown() override {}
|
|
|
|
|
|
|
|
std::filesystem::path test_folder_{std::filesystem::temp_directory_path() /
|
|
|
|
"MG_tests_unit_coordinator_cluster_state"};
|
|
|
|
};
|
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, RegisterReplicationInstance) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
|
|
|
|
|
|
|
auto config =
|
2024-03-21 14:41:26 +08:00
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance3",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10112},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10001}},
|
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
2024-03-15 21:41:02 +08:00
|
|
|
.ssl = std::nullopt};
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
auto instances = cluster_state.GetReplicationInstances();
|
|
|
|
ASSERT_EQ(instances.size(), 1);
|
|
|
|
ASSERT_EQ(instances[0].config, config);
|
|
|
|
ASSERT_EQ(instances[0].status, ReplicationRole::REPLICA);
|
|
|
|
ASSERT_EQ(cluster_state.GetCoordinatorInstances().size(), 0);
|
|
|
|
|
|
|
|
ASSERT_TRUE(cluster_state.IsReplica("instance3"));
|
2024-03-06 16:16:46 +08:00
|
|
|
}
|
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, UnregisterReplicationInstance) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
|
|
|
|
|
|
|
auto config =
|
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance3",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10112},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10001}},
|
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
|
|
|
.ssl = std::nullopt};
|
|
|
|
|
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
|
|
|
cluster_state.DoAction("instance3", RaftLogAction::UNREGISTER_REPLICATION_INSTANCE);
|
|
|
|
|
|
|
|
ASSERT_EQ(cluster_state.GetReplicationInstances().size(), 0);
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, SetInstanceToMain) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
2024-03-06 16:16:46 +08:00
|
|
|
{
|
2024-03-21 14:41:26 +08:00
|
|
|
auto config =
|
2024-03-15 21:41:02 +08:00
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance3",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10112},
|
2024-03-21 14:41:26 +08:00
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
2024-03-15 21:41:02 +08:00
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
2024-03-21 14:41:26 +08:00
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10001}},
|
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
2024-03-06 16:16:46 +08:00
|
|
|
.ssl = std::nullopt};
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
2024-03-06 16:16:46 +08:00
|
|
|
}
|
|
|
|
{
|
2024-03-21 14:41:26 +08:00
|
|
|
auto config =
|
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance2",
|
2024-03-15 21:41:02 +08:00
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10111},
|
2024-03-21 14:41:26 +08:00
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7688},
|
2024-03-15 21:41:02 +08:00
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
2024-03-21 14:41:26 +08:00
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
2024-03-15 21:41:02 +08:00
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10010}},
|
2024-03-21 14:41:26 +08:00
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
2024-03-06 16:16:46 +08:00
|
|
|
.ssl = std::nullopt};
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction("instance3", RaftLogAction::SET_INSTANCE_AS_MAIN);
|
|
|
|
auto const repl_instances = cluster_state.GetReplicationInstances();
|
|
|
|
ASSERT_EQ(repl_instances.size(), 2);
|
|
|
|
ASSERT_EQ(repl_instances[0].status, ReplicationRole::REPLICA);
|
|
|
|
ASSERT_EQ(repl_instances[1].status, ReplicationRole::MAIN);
|
|
|
|
ASSERT_TRUE(cluster_state.MainExists());
|
|
|
|
ASSERT_TRUE(cluster_state.IsMain("instance3"));
|
|
|
|
ASSERT_FALSE(cluster_state.IsMain("instance2"));
|
|
|
|
ASSERT_TRUE(cluster_state.IsReplica("instance2"));
|
|
|
|
ASSERT_FALSE(cluster_state.IsReplica("instance3"));
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, SetInstanceToReplica) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
2024-03-06 16:16:46 +08:00
|
|
|
{
|
2024-03-21 14:41:26 +08:00
|
|
|
auto config =
|
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance3",
|
2024-03-15 21:41:02 +08:00
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10112},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
2024-03-21 14:41:26 +08:00
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
2024-03-15 21:41:02 +08:00
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10001}},
|
2024-03-21 14:41:26 +08:00
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
2024-03-06 16:16:46 +08:00
|
|
|
.ssl = std::nullopt};
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
2024-03-06 16:16:46 +08:00
|
|
|
}
|
|
|
|
{
|
2024-03-21 14:41:26 +08:00
|
|
|
auto config =
|
2024-03-15 21:41:02 +08:00
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance2",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10111},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7688},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
2024-03-21 14:41:26 +08:00
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
2024-03-15 21:41:02 +08:00
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10010}},
|
2024-03-21 14:41:26 +08:00
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
2024-03-06 16:16:46 +08:00
|
|
|
.ssl = std::nullopt};
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction(config, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
cluster_state.DoAction("instance3", RaftLogAction::SET_INSTANCE_AS_MAIN);
|
|
|
|
cluster_state.DoAction("instance3", RaftLogAction::SET_INSTANCE_AS_REPLICA);
|
|
|
|
cluster_state.DoAction("instance2", RaftLogAction::SET_INSTANCE_AS_MAIN);
|
|
|
|
auto const repl_instances = cluster_state.GetReplicationInstances();
|
|
|
|
ASSERT_EQ(repl_instances.size(), 2);
|
|
|
|
ASSERT_EQ(repl_instances[0].status, ReplicationRole::MAIN);
|
|
|
|
ASSERT_EQ(repl_instances[1].status, ReplicationRole::REPLICA);
|
|
|
|
ASSERT_TRUE(cluster_state.MainExists());
|
|
|
|
ASSERT_TRUE(cluster_state.IsMain("instance2"));
|
|
|
|
ASSERT_FALSE(cluster_state.IsMain("instance3"));
|
|
|
|
ASSERT_TRUE(cluster_state.IsReplica("instance3"));
|
|
|
|
ASSERT_FALSE(cluster_state.IsReplica("instance2"));
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, UpdateUUID) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
|
|
|
auto uuid = UUID();
|
|
|
|
cluster_state.DoAction(uuid, RaftLogAction::UPDATE_UUID);
|
|
|
|
ASSERT_EQ(cluster_state.GetUUID(), uuid);
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, AddCoordinatorInstance) {
|
2024-03-18 21:02:21 +08:00
|
|
|
CoordinatorToCoordinatorConfig config{.coordinator_id = 1,
|
2024-03-15 21:41:02 +08:00
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.coordinator_server = Endpoint{"127.0.0.1", 10111}};
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
CoordinatorClusterState cluster_state;
|
|
|
|
cluster_state.DoAction(config, RaftLogAction::ADD_COORDINATOR_INSTANCE);
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
auto instances = cluster_state.GetCoordinatorInstances();
|
|
|
|
ASSERT_EQ(instances.size(), 1);
|
|
|
|
ASSERT_EQ(instances[0].config, config);
|
|
|
|
}
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
TEST_F(CoordinatorClusterStateTest, ReplicationInstanceStateSerialization) {
|
|
|
|
ReplicationInstanceState instance_state{
|
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance3",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10112},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10001}},
|
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
|
|
|
.ssl = std::nullopt},
|
|
|
|
ReplicationRole::MAIN};
|
|
|
|
|
|
|
|
nlohmann::json j = instance_state;
|
|
|
|
ReplicationInstanceState deserialized_instance_state = j.get<ReplicationInstanceState>();
|
|
|
|
|
|
|
|
EXPECT_EQ(instance_state.config, deserialized_instance_state.config);
|
|
|
|
EXPECT_EQ(instance_state.status, deserialized_instance_state.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(CoordinatorClusterStateTest, CoordinatorInstanceStateSerialization) {
|
|
|
|
CoordinatorInstanceState instance_state{
|
2024-03-18 21:02:21 +08:00
|
|
|
CoordinatorToCoordinatorConfig{.coordinator_id = 1,
|
2024-03-15 21:41:02 +08:00
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.coordinator_server = Endpoint{"127.0.0.1", 10111}}};
|
|
|
|
nlohmann::json j = instance_state;
|
|
|
|
CoordinatorInstanceState deserialized_instance_state = j.get<CoordinatorInstanceState>();
|
|
|
|
ASSERT_EQ(instance_state, deserialized_instance_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(CoordinatorClusterStateTest, Marshalling) {
|
|
|
|
CoordinatorClusterState cluster_state;
|
2024-03-18 21:02:21 +08:00
|
|
|
CoordinatorToCoordinatorConfig config{.coordinator_id = 1,
|
2024-03-15 21:41:02 +08:00
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7687},
|
|
|
|
.coordinator_server = Endpoint{"127.0.0.1", 10111}};
|
|
|
|
|
|
|
|
cluster_state.DoAction(config, RaftLogAction::ADD_COORDINATOR_INSTANCE);
|
|
|
|
|
|
|
|
auto config2 =
|
|
|
|
CoordinatorToReplicaConfig{.instance_name = "instance2",
|
|
|
|
.mgt_server = Endpoint{"127.0.0.1", 10111},
|
|
|
|
.bolt_server = Endpoint{"127.0.0.1", 7688},
|
|
|
|
.replication_client_info = {.instance_name = "instance_name",
|
|
|
|
.replication_mode = ReplicationMode::ASYNC,
|
|
|
|
.replication_server = Endpoint{"127.0.0.1", 10010}},
|
|
|
|
.instance_health_check_frequency_sec = std::chrono::seconds{1},
|
|
|
|
.instance_down_timeout_sec = std::chrono::seconds{5},
|
|
|
|
.instance_get_uuid_frequency_sec = std::chrono::seconds{10},
|
|
|
|
.ssl = std::nullopt};
|
|
|
|
cluster_state.DoAction(config2, RaftLogAction::REGISTER_REPLICATION_INSTANCE);
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
ptr<buffer> data{};
|
|
|
|
cluster_state.Serialize(data);
|
2024-03-06 16:16:46 +08:00
|
|
|
|
2024-03-15 21:41:02 +08:00
|
|
|
auto deserialized_cluster_state = CoordinatorClusterState::Deserialize(*data);
|
|
|
|
ASSERT_EQ(cluster_state, deserialized_cluster_state);
|
2024-03-06 16:16:46 +08:00
|
|
|
}
|