blob: 308cede6957103f4647be8376891a7d778e231a2 [file] [log] [blame]
/*
*
* 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 <pw_unit_test/framework.h>
#include <lib/core/StringBuilderAdapters.h>
#include <lib/dnssd/ActiveResolveAttempts.h>
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<ActiveResolveAttempts::ScheduledAttempt> ScheduledPeer(NodeId id, bool first)
{
return std::make_optional<ActiveResolveAttempts::ScheduledAttempt>(
ActiveResolveAttempts::ScheduledAttempt(MakePeerId(id), first));
}
std::optional<ActiveResolveAttempts::ScheduledAttempt> ScheduledBrowse(const Dnssd::DiscoveryFilter & filter,
const Dnssd::DiscoveryType type, bool first)
{
return std::make_optional<ActiveResolveAttempts::ScheduledAttempt>(
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<Timeout>(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<Timeout>(1000_ms32));
mockClock.AdvanceMonotonic(500_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(2000_ms32));
mockClock.AdvanceMonotonic(100_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(1000_ms32));
mockClock.AdvanceMonotonic(500_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(2000_ms32));
mockClock.AdvanceMonotonic(100_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(0_ms32));
EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, false));
EXPECT_FALSE(attempts.NextScheduled().has_value());
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(2000_ms32));
// reschedule starts fresh
attempts.MarkPending(MakePeerId(1));
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(0_ms32));
EXPECT_EQ(attempts.NextScheduled(), ScheduledPeer(1, true));
EXPECT_FALSE(attempts.NextScheduled().has_value());
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(0_ms32));
EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, false));
EXPECT_FALSE(attempts.NextScheduled().has_value());
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(2000_ms32));
// reschedule starts fresh
attempts.MarkPending(filter, type);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(0_ms32));
EXPECT_EQ(attempts.NextScheduled(), ScheduledBrowse(filter, type, true));
EXPECT_FALSE(attempts.NextScheduled().has_value());
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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::Timeout>(
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<ActiveResolveAttempts::ScheduledAttempt> 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<System::Clock::Timeout> ms = attempts.GetTimeUntilNextExpectedResponse();
if (!ms.has_value())
{
break;
}
mockClock.AdvanceMonotonic(*ms);
std::optional<ActiveResolveAttempts::ScheduledAttempt> 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<Timeout>(1000_ms32));
mockClock.AdvanceMonotonic(20_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(900_ms32));
// Peer 1 is done, now peer2 should be pending (in 980ms)
attempts.Complete(MakePeerId(1));
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(920_ms32));
mockClock.AdvanceMonotonic(20_ms32);
EXPECT_EQ(attempts.GetTimeUntilNextExpectedResponse(), std::make_optional<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(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<Timeout>(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