diff --git a/src/meta/processors/zone/DivideZoneProcessor.cpp b/src/meta/processors/zone/DivideZoneProcessor.cpp index 3659a81e3de..6b43915a5ad 100644 --- a/src/meta/processors/zone/DivideZoneProcessor.cpp +++ b/src/meta/processors/zone/DivideZoneProcessor.cpp @@ -33,6 +33,7 @@ void DivideZoneProcessor::process(const cpp2::DivideZoneReq& req) { std::vector zoneNames; std::unordered_set totalHosts; + size_t totalHostsSize = 0; auto batchHolder = std::make_unique(); nebula::cpp2::ErrorCode code = nebula::cpp2::ErrorCode::SUCCEEDED; for (auto iter = zoneItems.begin(); iter != zoneItems.end(); iter++) { @@ -65,6 +66,7 @@ void DivideZoneProcessor::process(const cpp2::DivideZoneReq& req) { break; } + totalHostsSize += hosts.size(); std::copy(hosts.begin(), hosts.end(), std::inserter(totalHosts, totalHosts.end())); auto key = MetaKeyUtils::zoneKey(std::move(zone)); @@ -85,6 +87,13 @@ void DivideZoneProcessor::process(const cpp2::DivideZoneReq& req) { return; } + if (totalHostsSize != totalHosts.size()) { + LOG(ERROR) << "The host in zone list have duplicate element"; + handleErrorCode(nebula::cpp2::ErrorCode::E_INVALID_PARM); + onFinished(); + return; + } + for (auto& host : totalHosts) { auto iter = std::find(zoneHosts.begin(), zoneHosts.end(), host); if (iter == zoneHosts.end()) { diff --git a/src/meta/test/MetaClientTest.cpp b/src/meta/test/MetaClientTest.cpp index fbbfe7d3c3b..28b8231e74d 100644 --- a/src/meta/test/MetaClientTest.cpp +++ b/src/meta/test/MetaClientTest.cpp @@ -2479,6 +2479,26 @@ TEST(MetaClientTest, DivideZoneTest) { auto result = client->divideZone("default_zone", std::move(zoneItems)).get(); EXPECT_FALSE(result.ok()); } + { + std::unordered_map> zoneItems; + std::vector oneHosts = { + {"127.0.0.1", 8986}, {"127.0.0.1", 8987}, {"127.0.0.1", 8988}}; + zoneItems.emplace("one_zone", std::move(oneHosts)); + std::vector anotherHosts = {{"127.0.0.1", 8988}, {"127.0.0.1", 8989}}; + zoneItems.emplace("another_zone", std::move(anotherHosts)); + auto result = client->divideZone("default_zone", std::move(zoneItems)).get(); + EXPECT_FALSE(result.ok()); + } + { + std::unordered_map> zoneItems; + std::vector oneHosts = { + {"127.0.0.1", 8986}, {"127.0.0.1", 8987}, {"127.0.0.1", 8987}}; + zoneItems.emplace("one_zone", std::move(oneHosts)); + std::vector anotherHosts = {{"127.0.0.1", 8988}, {"127.0.0.1", 8989}}; + zoneItems.emplace("another_zone", std::move(anotherHosts)); + auto result = client->divideZone("default_zone", std::move(zoneItems)).get(); + EXPECT_FALSE(result.ok()); + } { std::unordered_map> zoneItems; std::vector hosts0 = {}; diff --git a/src/meta/test/ProcessorTest.cpp b/src/meta/test/ProcessorTest.cpp index 559872ae3b4..6eba572f768 100644 --- a/src/meta/test/ProcessorTest.cpp +++ b/src/meta/test/ProcessorTest.cpp @@ -4102,6 +4102,38 @@ TEST(ProcessorTest, DivideZoneTest) { auto resp = std::move(f).get(); ASSERT_EQ(nebula::cpp2::ErrorCode::E_INVALID_PARM, resp.get_code()); } + { + cpp2::DivideZoneReq req; + req.zone_name_ref() = "default_zone"; + std::unordered_map> zoneItems; + std::vector oneHosts = { + {"127.0.0.1", 8986}, {"127.0.0.1", 8987}, {"127.0.0.1", 8988}}; + zoneItems.emplace("one_zone", std::move(oneHosts)); + std::vector anotherHosts = {{"127.0.0.1", 8988}, {"127.0.0.1", 8989}}; + zoneItems.emplace("another_zone", std::move(anotherHosts)); + req.zone_items_ref() = std::move(zoneItems); + auto* processor = DivideZoneProcessor::instance(kv.get()); + auto f = processor->getFuture(); + processor->process(req); + auto resp = std::move(f).get(); + ASSERT_EQ(nebula::cpp2::ErrorCode::E_INVALID_PARM, resp.get_code()); + } + { + cpp2::DivideZoneReq req; + req.zone_name_ref() = "default_zone"; + std::unordered_map> zoneItems; + std::vector oneHosts = { + {"127.0.0.1", 8986}, {"127.0.0.1", 8987}, {"127.0.0.1", 8987}}; + zoneItems.emplace("one_zone", std::move(oneHosts)); + std::vector anotherHosts = {{"127.0.0.1", 8988}, {"127.0.0.1", 8989}}; + zoneItems.emplace("another_zone", std::move(anotherHosts)); + req.zone_items_ref() = std::move(zoneItems); + auto* processor = DivideZoneProcessor::instance(kv.get()); + auto f = processor->getFuture(); + processor->process(req); + auto resp = std::move(f).get(); + ASSERT_EQ(nebula::cpp2::ErrorCode::E_INVALID_PARM, resp.get_code()); + } { cpp2::DivideZoneReq req; req.zone_name_ref() = "default_zone";