#include #include #include #include #include #include #include #include #include #include "storage/v2/view.hpp" using testing::UnorderedElementsAre; class ReplicationTest : public ::testing::Test { protected: std::filesystem::path storage_directory{std::filesystem::temp_directory_path() / "MG_test_unit_storage_v2_replication"}; void SetUp() override { Clear(); } void TearDown() override { Clear(); } private: void Clear() { if (!std::filesystem::exists(storage_directory)) return; std::filesystem::remove_all(storage_directory); } }; TEST_F(ReplicationTest, BasicSynchronousReplicationTest) { storage::Storage main_store( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); storage::Storage replica_store( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10000}); ASSERT_FALSE(main_store .RegisterReplica("REPLICA", io::network::Endpoint{"127.0.0.1", 10000}, storage::replication::ReplicationMode::SYNC) .HasError()); // vertex create // vertex add label // vertex set property const auto *vertex_label = "vertex_label"; const auto *vertex_property = "vertex_property"; const auto *vertex_property_value = "vertex_property_value"; std::optional vertex_gid; { auto acc = main_store.Access(); auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); ASSERT_TRUE(v.AddLabel(main_store.NameToLabel(vertex_label)).HasValue()); ASSERT_TRUE(v.SetProperty(main_store.NameToProperty(vertex_property), storage::PropertyValue(vertex_property_value)) .HasValue()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto labels = v->Labels(storage::View::OLD); ASSERT_TRUE(labels.HasValue()); ASSERT_EQ(labels->size(), 1); ASSERT_THAT(*labels, UnorderedElementsAre(replica_store.NameToLabel(vertex_label))); const auto properties = v->Properties(storage::View::OLD); ASSERT_TRUE(properties.HasValue()); ASSERT_EQ(properties->size(), 1); ASSERT_THAT(*properties, UnorderedElementsAre(std::make_pair(replica_store.NameToProperty(vertex_property), storage::PropertyValue(vertex_property_value)))); ASSERT_FALSE(acc.Commit().HasError()); } // vertex remove label { auto acc = main_store.Access(); auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_TRUE(v->RemoveLabel(main_store.NameToLabel(vertex_label)).HasValue()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto labels = v->Labels(storage::View::OLD); ASSERT_TRUE(labels.HasValue()); ASSERT_EQ(labels->size(), 0); ASSERT_FALSE(acc.Commit().HasError()); } // vertex delete { auto acc = main_store.Access(); auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_TRUE(acc.DeleteVertex(&*v).HasValue()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_FALSE(v); vertex_gid.reset(); ASSERT_FALSE(acc.Commit().HasError()); } // edge create // edge set property const auto *edge_type = "edge_type"; const auto *edge_property = "edge_property"; const auto *edge_property_value = "edge_property_value"; std::optional edge_gid; { auto acc = main_store.Access(); auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); auto edge = acc.CreateEdge(&v, &v, main_store.NameToEdgeType(edge_type)); ASSERT_TRUE(edge.HasValue()); ASSERT_TRUE(edge->SetProperty(main_store.NameToProperty(edge_property), storage::PropertyValue(edge_property_value)) .HasValue()); edge_gid.emplace(edge->Gid()); ASSERT_FALSE(acc.Commit().HasError()); } const auto find_edge = [&](const auto &edges, const storage::Gid edge_gid) -> std::optional { for (const auto &edge : edges) { if (edge.Gid() == edge_gid) { return edge; } } return std::nullopt; }; { auto acc = replica_store.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto out_edges = v->OutEdges(storage::View::OLD); ASSERT_TRUE(out_edges.HasValue()); const auto edge = find_edge(*out_edges, *edge_gid); ASSERT_EQ(edge->EdgeType(), replica_store.NameToEdgeType(edge_type)); const auto properties = edge->Properties(storage::View::OLD); ASSERT_TRUE(properties.HasValue()); ASSERT_EQ(properties->size(), 1); ASSERT_THAT(*properties, UnorderedElementsAre(std::make_pair(replica_store.NameToProperty(edge_property), storage::PropertyValue(edge_property_value)))); ASSERT_FALSE(acc.Commit().HasError()); } // delete edge { auto acc = main_store.Access(); auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); auto out_edges = v->OutEdges(storage::View::OLD); auto edge = find_edge(*out_edges, *edge_gid); ASSERT_TRUE(edge); ASSERT_TRUE(acc.DeleteEdge(&*edge).HasValue()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto out_edges = v->OutEdges(storage::View::OLD); ASSERT_TRUE(out_edges.HasValue()); ASSERT_FALSE(find_edge(*out_edges, *edge_gid)); ASSERT_FALSE(acc.Commit().HasError()); } // label index create // label property index create // existence constraint create // unique constriant create const auto *label = "label"; const auto *property = "property"; const auto *property_extra = "property_extra"; { ASSERT_TRUE(main_store.CreateIndex(main_store.NameToLabel(label))); ASSERT_TRUE(main_store.CreateIndex(main_store.NameToLabel(label), main_store.NameToProperty(property))); ASSERT_FALSE( main_store.CreateExistenceConstraint(main_store.NameToLabel(label), main_store.NameToProperty(property)) .HasError()); ASSERT_FALSE(main_store .CreateUniqueConstraint(main_store.NameToLabel(label), {main_store.NameToProperty(property), main_store.NameToProperty(property_extra)}) .HasError()); } { const auto indices = replica_store.ListAllIndices(); ASSERT_THAT(indices.label, UnorderedElementsAre(replica_store.NameToLabel(label))); ASSERT_THAT(indices.label_property, UnorderedElementsAre(std::make_pair(replica_store.NameToLabel(label), replica_store.NameToProperty(property)))); const auto constraints = replica_store.ListAllConstraints(); ASSERT_THAT(constraints.existence, UnorderedElementsAre(std::make_pair(replica_store.NameToLabel(label), replica_store.NameToProperty(property)))); ASSERT_THAT(constraints.unique, UnorderedElementsAre(std::make_pair( replica_store.NameToLabel(label), std::set{replica_store.NameToProperty(property), replica_store.NameToProperty(property_extra)}))); } // label index drop // label property index drop // existence constraint drop // unique constriant drop { ASSERT_TRUE(main_store.DropIndex(main_store.NameToLabel(label))); ASSERT_TRUE(main_store.DropIndex(main_store.NameToLabel(label), main_store.NameToProperty(property))); ASSERT_TRUE(main_store.DropExistenceConstraint(main_store.NameToLabel(label), main_store.NameToProperty(property))); ASSERT_EQ( main_store.DropUniqueConstraint(main_store.NameToLabel(label), {main_store.NameToProperty(property), main_store.NameToProperty(property_extra)}), storage::UniqueConstraints::DeletionStatus::SUCCESS); } { const auto indices = replica_store.ListAllIndices(); ASSERT_EQ(indices.label.size(), 0); ASSERT_EQ(indices.label_property.size(), 0); const auto constraints = replica_store.ListAllConstraints(); ASSERT_EQ(constraints.existence.size(), 0); ASSERT_EQ(constraints.unique.size(), 0); } } TEST_F(ReplicationTest, MultipleSynchronousReplicationTest) { storage::Storage main_store( {.durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); storage::Storage replica_store1( {.durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store1.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10000}); storage::Storage replica_store2( {.durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store2.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 20000}); ASSERT_FALSE(main_store .RegisterReplica("REPLICA1", io::network::Endpoint{"127.0.0.1", 10000}, storage::replication::ReplicationMode::SYNC) .HasError()); ASSERT_FALSE(main_store .RegisterReplica("REPLICA2", io::network::Endpoint{"127.0.0.1", 20000}, storage::replication::ReplicationMode::SYNC) .HasError()); const auto *vertex_label = "label"; const auto *vertex_property = "property"; const auto *vertex_property_value = "property_value"; std::optional vertex_gid; { auto acc = main_store.Access(); auto v = acc.CreateVertex(); ASSERT_TRUE(v.AddLabel(main_store.NameToLabel(vertex_label)).HasValue()); ASSERT_TRUE(v.SetProperty(main_store.NameToProperty(vertex_property), storage::PropertyValue(vertex_property_value)) .HasValue()); vertex_gid.emplace(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } const auto check_replica = [&](storage::Storage *replica_store) { auto acc = replica_store->Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto labels = v->Labels(storage::View::OLD); ASSERT_TRUE(labels.HasValue()); ASSERT_THAT(*labels, UnorderedElementsAre(replica_store->NameToLabel(vertex_label))); ASSERT_FALSE(acc.Commit().HasError()); }; check_replica(&replica_store1); check_replica(&replica_store2); main_store.UnregisterReplica("REPLICA2"); { auto acc = main_store.Access(); auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } // REPLICA1 should contain the new vertex { auto acc = replica_store1.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_FALSE(acc.Commit().HasError()); } // REPLICA2 should not contain the new vertex { auto acc = replica_store2.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_FALSE(v); ASSERT_FALSE(acc.Commit().HasError()); } } TEST_F(ReplicationTest, RecoveryProcess) { std::vector vertex_gids; // Force the creation of snapshot { storage::Storage main_store( {.durability = { .storage_directory = storage_directory, .recover_on_startup = true, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, .snapshot_on_exit = true, }}); { auto acc = main_store.Access(); // Create the vertex before registering a replica auto v = acc.CreateVertex(); vertex_gids.emplace_back(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } } { // Create second WAL storage::Storage main_store( {.durability = { .storage_directory = storage_directory, .recover_on_startup = true, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL}}); // Create vertices in 2 different transactions { auto acc = main_store.Access(); auto v = acc.CreateVertex(); vertex_gids.emplace_back(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = main_store.Access(); auto v = acc.CreateVertex(); vertex_gids.emplace_back(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } } storage::Storage main_store( {.durability = { .storage_directory = storage_directory, .recover_on_startup = true, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); constexpr const auto *property_name = "property_name"; constexpr const auto property_value = 1; { // Force the creation of current WAL file auto acc = main_store.Access(); for (const auto &vertex_gid : vertex_gids) { auto v = acc.FindVertex(vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_TRUE( v->SetProperty(main_store.NameToProperty(property_name), storage::PropertyValue(property_value)).HasValue()); } ASSERT_FALSE(acc.Commit().HasError()); } std::filesystem::path replica_storage_directory{std::filesystem::temp_directory_path() / "MG_test_unit_storage_v2_replication_replica"}; utils::OnScopeExit replica_directory_cleaner([&]() { std::filesystem::remove_all(replica_storage_directory); }); constexpr const auto *vertex_label = "vertex_label"; { storage::Storage replica_store( {.durability = { .storage_directory = replica_storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL}}); replica_store.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10000}); ASSERT_FALSE(main_store .RegisterReplica("REPLICA1", io::network::Endpoint{"127.0.0.1", 10000}, storage::replication::ReplicationMode::SYNC) .HasError()); ASSERT_EQ(main_store.GetReplicaState("REPLICA1"), storage::replication::ReplicaState::RECOVERY); while (main_store.GetReplicaState("REPLICA1") != storage::replication::ReplicaState::READY) { std::this_thread::sleep_for(std::chrono::milliseconds(10)); } { auto acc = main_store.Access(); for (const auto &vertex_gid : vertex_gids) { auto v = acc.FindVertex(vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_TRUE(v->AddLabel(main_store.NameToLabel(vertex_label)).HasValue()); } ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store.Access(); for (const auto &vertex_gid : vertex_gids) { auto v = acc.FindVertex(vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto labels = v->Labels(storage::View::OLD); ASSERT_TRUE(labels.HasValue()); ASSERT_THAT(*labels, UnorderedElementsAre(replica_store.NameToLabel(vertex_label))); const auto properties = v->Properties(storage::View::OLD); ASSERT_TRUE(properties.HasValue()); ASSERT_THAT(*properties, UnorderedElementsAre(std::make_pair(replica_store.NameToProperty(property_name), storage::PropertyValue(property_value)))); } ASSERT_FALSE(acc.Commit().HasError()); } } { storage::Storage replica_store( {.durability = { .storage_directory = replica_storage_directory, .recover_on_startup = true, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL}}); { auto acc = replica_store.Access(); for (const auto &vertex_gid : vertex_gids) { auto v = acc.FindVertex(vertex_gid, storage::View::OLD); ASSERT_TRUE(v); const auto labels = v->Labels(storage::View::OLD); ASSERT_TRUE(labels.HasValue()); ASSERT_THAT(*labels, UnorderedElementsAre(replica_store.NameToLabel(vertex_label))); const auto properties = v->Properties(storage::View::OLD); ASSERT_TRUE(properties.HasValue()); ASSERT_THAT(*properties, UnorderedElementsAre(std::make_pair(replica_store.NameToProperty(property_name), storage::PropertyValue(property_value)))); } ASSERT_FALSE(acc.Commit().HasError()); } } } TEST_F(ReplicationTest, BasicAsynchronousReplicationTest) { storage::Storage main_store( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); storage::Storage replica_store_async( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store_async.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 20000}); ASSERT_FALSE(main_store .RegisterReplica("REPLICA_ASYNC", io::network::Endpoint{"127.0.0.1", 20000}, storage::replication::ReplicationMode::ASYNC) .HasError()); constexpr size_t vertices_create_num = 10; std::vector created_vertices; for (size_t i = 0; i < vertices_create_num; ++i) { auto acc = main_store.Access(); auto v = acc.CreateVertex(); created_vertices.push_back(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); if (i == 0) { ASSERT_EQ(main_store.GetReplicaState("REPLICA_ASYNC"), storage::replication::ReplicaState::REPLICATING); } else { ASSERT_EQ(main_store.GetReplicaState("REPLICA_ASYNC"), storage::replication::ReplicaState::RECOVERY); } } while (main_store.GetReplicaState("REPLICA_ASYNC") != storage::replication::ReplicaState::READY) { std::this_thread::sleep_for(std::chrono::milliseconds(10)); } ASSERT_TRUE(std::all_of(created_vertices.begin(), created_vertices.end(), [&](const auto vertex_gid) { auto acc = replica_store_async.Access(); auto v = acc.FindVertex(vertex_gid, storage::View::OLD); const bool exists = v.has_value(); EXPECT_FALSE(acc.Commit().HasError()); return exists; })); } TEST_F(ReplicationTest, EpochTest) { storage::Storage main_store( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); storage::Storage replica_store1( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store1.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10000}); storage::Storage replica_store2( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store2.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10001}); ASSERT_FALSE(main_store .RegisterReplica("REPLICA1", io::network::Endpoint{"127.0.0.1", 10000}, storage::replication::ReplicationMode::SYNC) .HasError()); ASSERT_FALSE(main_store .RegisterReplica("REPLICA2", io::network::Endpoint{"127.0.0.1", 10001}, storage::replication::ReplicationMode::SYNC) .HasError()); std::optional vertex_gid; { auto acc = main_store.Access(); const auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store1.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store2.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_FALSE(acc.Commit().HasError()); } main_store.UnregisterReplica("REPLICA1"); main_store.UnregisterReplica("REPLICA2"); replica_store1.SetMainReplicationRole(); ASSERT_FALSE(replica_store1 .RegisterReplica("REPLICA2", io::network::Endpoint{"127.0.0.1", 10001}, storage::replication::ReplicationMode::SYNC) .HasError()); { auto acc = main_store.Access(); acc.CreateVertex(); ASSERT_FALSE(acc.Commit().HasError()); } { auto acc = replica_store1.Access(); auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } // Replica1 should forward it's vertex to Replica2 { auto acc = replica_store2.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_TRUE(v); ASSERT_FALSE(acc.Commit().HasError()); } replica_store1.SetReplicaRole(io::network::Endpoint{"127.0.0.1", 10000}); ASSERT_TRUE(main_store .RegisterReplica("REPLICA1", io::network::Endpoint{"127.0.0.1", 10000}, storage::replication::ReplicationMode::SYNC) .HasError()); { auto acc = main_store.Access(); const auto v = acc.CreateVertex(); vertex_gid.emplace(v.Gid()); ASSERT_FALSE(acc.Commit().HasError()); } // Replica1 is not compatible with the main so it shouldn't contain // it's newest vertex { auto acc = replica_store1.Access(); const auto v = acc.FindVertex(*vertex_gid, storage::View::OLD); ASSERT_FALSE(v); ASSERT_FALSE(acc.Commit().HasError()); } } TEST_F(ReplicationTest, ReplicationInformation) { storage::Storage main_store( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); storage::Storage replica_store1( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); const io::network::Endpoint replica1_endpoint{"127.0.0.1", 10000}; replica_store1.SetReplicaRole(replica1_endpoint); const io::network::Endpoint replica2_endpoint{"127.0.0.1", 10000}; storage::Storage replica_store2( {.items = {.properties_on_edges = true}, .durability = { .storage_directory = storage_directory, .snapshot_wal_mode = storage::Config::Durability::SnapshotWalMode::PERIODIC_SNAPSHOT_WITH_WAL, }}); replica_store2.SetReplicaRole(replica2_endpoint); const std::string replica1_name{"REPLICA1"}; ASSERT_FALSE(main_store .RegisterReplica(replica1_name, replica1_endpoint, storage::replication::ReplicationMode::SYNC, {.timeout = 2.0}) .HasError()); const std::string replica2_name{"REPLICA2"}; ASSERT_FALSE( main_store.RegisterReplica(replica2_name, replica2_endpoint, storage::replication::ReplicationMode::ASYNC) .HasError()); ASSERT_EQ(main_store.GetReplicationRole(), storage::ReplicationRole::MAIN); ASSERT_EQ(replica_store1.GetReplicationRole(), storage::ReplicationRole::REPLICA); ASSERT_EQ(replica_store2.GetReplicationRole(), storage::ReplicationRole::REPLICA); const auto replicas_info = main_store.ReplicasInfo(); ASSERT_EQ(replicas_info.size(), 2); const auto &first_info = replicas_info[0]; ASSERT_EQ(first_info.name, replica1_name); ASSERT_EQ(first_info.mode, storage::replication::ReplicationMode::SYNC); ASSERT_TRUE(first_info.timeout); ASSERT_EQ(*first_info.timeout, 2.0); ASSERT_EQ(first_info.endpoint, replica1_endpoint); ASSERT_EQ(first_info.state, storage::replication::ReplicaState::READY); const auto &second_info = replicas_info[1]; ASSERT_EQ(second_info.name, replica2_name); ASSERT_EQ(second_info.mode, storage::replication::ReplicationMode::ASYNC); ASSERT_FALSE(second_info.timeout); ASSERT_EQ(second_info.endpoint, replica2_endpoint); ASSERT_EQ(second_info.state, storage::replication::ReplicaState::READY); }