/* * * Copyright (c) 2021 Project CHIP Authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include namespace { using namespace chip; using namespace chip::System::Clock::Literals; using chip::System::Clock::Timeout; using mdns::Minimal::ActiveResolveAttempts; PeerId MakePeerId(NodeId nodeId) { PeerId peerId; return peerId.SetNodeId(nodeId).SetCompressedFabricId(123); } std::optional ScheduledPeer(NodeId id, bool first) { return std::make_optional( ActiveResolveAttempts::ScheduledAttempt(MakePeerId(id), first)); } std::optional ScheduledBrowse(const Dnssd::DiscoveryFilter & filter, const Dnssd::DiscoveryType type, bool first) { return std::make_optional( ActiveResolveAttempts::ScheduledAttempt(filter, type, first)); } TEST(TestActiveResolveAttempts, TestSinglePeerAddRemove) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); mockClock.AdvanceMonotonic(1234_ms32); // Starting up, no scheduled peers are expected EXPECT_FALSE(attempts.GetTimeUntilNextExpectedResponse().has_value()); EXPECT_FALSE(attempts.NextScheduled().has_value()); // Adding a single peer should result in it being scheduled attempts.MarkPending(MakePeerId(1)); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 1000 ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); mockClock.AdvanceMonotonic(500_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(500_ms32)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // past due date: timeout should be 0 mockClock.AdvanceMonotonic(800_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 2000 ms // sincve the timeout doubles every time EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(2000_ms32)); mockClock.AdvanceMonotonic(100_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1900_ms32)); // once complete, nothing to schedule attempts.Complete(MakePeerId(1)); EXPECT_FALSE(attempts.GetTimeUntilNextExpectedResponse().has_value()); EXPECT_FALSE(attempts.NextScheduled().has_value()); } TEST(TestActiveResolveAttempts, TestSingleBrowseAddRemove) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); Dnssd::DiscoveryFilter filter(Dnssd::DiscoveryFilterType::kLongDiscriminator, 1234); Dnssd::DiscoveryType type = Dnssd::DiscoveryType::kCommissionableNode; mockClock.AdvanceMonotonic(1234_ms32); // Starting up, no scheduled peers are expected EXPECT_FALSE(attempts.GetTimeUntilNextExpectedResponse().has_value()); EXPECT_FALSE(attempts.NextScheduled().has_value()); // Adding a single attempt should result in it being scheduled attempts.MarkPending(filter, type); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 1000 ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); mockClock.AdvanceMonotonic(500_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(500_ms32)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // past due date: timeout should be 0 mockClock.AdvanceMonotonic(800_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 2000 ms // sincve the timeout doubles every time EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(2000_ms32)); mockClock.AdvanceMonotonic(100_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1900_ms32)); // once complete, nothing to schedule attempts.CompleteAllBrowses(); EXPECT_FALSE(attempts.GetTimeUntilNextExpectedResponse().has_value()); EXPECT_FALSE(attempts.NextScheduled().has_value()); } TEST(TestActiveResolveAttempts, TestRescheduleSamePeerId) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); mockClock.AdvanceMonotonic(112233_ms32); attempts.MarkPending(MakePeerId(1)); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 1000 ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); // 2nd try goes to 2 seconds (once at least 1 second passes) mockClock.AdvanceMonotonic(1234_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(2000_ms32)); // reschedule starts fresh attempts.MarkPending(MakePeerId(1)); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); } TEST(TestActiveResolveAttempts, TestRescheduleSameFilter) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); Dnssd::DiscoveryFilter filter(Dnssd::DiscoveryFilterType::kLongDiscriminator, 1234); Dnssd::DiscoveryType type = Dnssd::DiscoveryType::kCommissionableNode; mockClock.AdvanceMonotonic(112233_ms32); attempts.MarkPending(filter, type); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // one Next schedule is called, expect to have a delay of 1000 ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); // 2nd try goes to 2 seconds (once at least 1 second passes) mockClock.AdvanceMonotonic(1234_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(2000_ms32)); // reschedule starts fresh attempts.MarkPending(filter, type); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); } TEST(TestActiveResolveAttempts, TestLRU) { // validates that the LRU logic is working System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); mockClock.AdvanceMonotonic(334455_ms32); // add a single very old peer attempts.MarkPending(MakePeerId(9999)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(9999, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); mockClock.AdvanceMonotonic(1000_ms32); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(9999, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); mockClock.AdvanceMonotonic(2000_ms32); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(9999, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // at this point, peer 9999 has a delay of 4 seconds. Fill up the rest of the table for (uint32_t i = 1; i < mdns::Minimal::ActiveResolveAttempts::kRetryQueueSize; i++) { attempts.MarkPending(MakePeerId(i)); mockClock.AdvanceMonotonic(1_ms32); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(i, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); } // +2 because: 1 element skipped, one element is the "current" that has a delay of 1000ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional( System::Clock::Milliseconds32(1000 - mdns::Minimal::ActiveResolveAttempts::kRetryQueueSize + 2))); // add another element - this should overwrite peer 9999 attempts.MarkPending(MakePeerId(mdns::Minimal::ActiveResolveAttempts::kRetryQueueSize)); mockClock.AdvanceMonotonic(32_s16); for (std::optional s = attempts.NextScheduled(); s.has_value(); s = attempts.NextScheduled()) { EXPECT_NE(s->ResolveData().peerId.GetNodeId(), 9999u); } // Still have active pending items (queue is full) EXPECT_TRUE(attempts.GetTimeUntilNextExpectedResponse().has_value()); // expire all of them. Since we double timeout every expiry, we expect a // few iteratios to be able to expire the entire queue constexpr int kMaxIterations = 10; int i = 0; for (; i < kMaxIterations; i++) { std::optional ms = attempts.GetTimeUntilNextExpectedResponse(); if (!ms.has_value()) { break; } mockClock.AdvanceMonotonic(*ms); std::optional s = attempts.NextScheduled(); while (s.has_value()) { // NOLINTNEXTLINE(bugprone-unchecked-optional-access): this is checked in the while loop EXPECT_NE(s->ResolveData().peerId.GetNodeId(), 9999u); s = attempts.NextScheduled(); } } EXPECT_LT(i, kMaxIterations); } TEST(TestActiveResolveAttempts, TestNextPeerOrdering) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); mockClock.AdvanceMonotonic(123321_ms32); // add a single peer that will be resolved quickly attempts.MarkPending(MakePeerId(1)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); mockClock.AdvanceMonotonic(20_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(980_ms32)); // expect peerid to be resolve within 1 second from now attempts.MarkPending(MakePeerId(2)); // mock that we are querying 2 as well EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(2, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); mockClock.AdvanceMonotonic(80_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(900_ms32)); // Peer 1 is done, now peer2 should be pending (in 980ms) attempts.Complete(MakePeerId(1)); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(920_ms32)); mockClock.AdvanceMonotonic(20_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(900_ms32)); // Once peer 3 is added, queue should be // - 900 ms until peer id 2 is pending // - 1000 ms until peer id 3 is pending attempts.MarkPending(MakePeerId(3)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(3, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(900_ms32)); // After the clock advance // - 400 ms until peer id 2 is pending // - 500 ms until peer id 3 is pending mockClock.AdvanceMonotonic(500_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(400_ms32)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // advancing the clock 'too long' will return both other entries, in reverse order due to how // the internal cache is built mockClock.AdvanceMonotonic(500_ms32); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(3, false)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(2, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); } TEST(TestActiveResolveAttempts, TestCombination) { System::Clock::Internal::MockClock mockClock; mdns::Minimal::ActiveResolveAttempts attempts(&mockClock); Dnssd::DiscoveryFilter filter(Dnssd::DiscoveryFilterType::kLongDiscriminator, 1234); Dnssd::DiscoveryType type = Dnssd::DiscoveryType::kCommissionableNode; // Schedule a Resolve attempts.MarkPending(MakePeerId(1)); // 20ms later, schedule a browse mockClock.AdvanceMonotonic(20_ms32); attempts.MarkPending(filter, type); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // At this point, both should reset, so we're back to 1000ms EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(1000_ms32)); // We used 20 ms, so the next time for the peer and resolve should be 980 ms mockClock.AdvanceMonotonic(20_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(980_ms32)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // Add a second Peer mockClock.AdvanceMonotonic(20_ms32); attempts.MarkPending(MakePeerId(2)); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(2, true)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // Advance to the retry time of peer 1 and the resolve mockClock.AdvanceMonotonic(960_ms32); EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional(0_ms32)); EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, false)); EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, false)); EXPECT_FALSE(attempts.NextScheduled().has_value()); // Complete all, we should see no more scheduled. attempts.Complete(MakePeerId(2)); attempts.Complete(MakePeerId(1)); attempts.CompleteAllBrowses(); EXPECT_FALSE(attempts.GetTimeUntilNextExpectedResponse().has_value()); EXPECT_FALSE(attempts.NextScheduled().has_value()); } } // namespace