|  | /* | 
|  | *  Copyright 2024 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "test/wait_until.h" | 
|  |  | 
|  | #include <memory> | 
|  |  | 
|  | #include "api/rtc_error.h" | 
|  | #include "api/test/create_time_controller.h" | 
|  | #include "api/test/rtc_error_matchers.h" | 
|  | #include "api/test/time_controller.h" | 
|  | #include "api/units/time_delta.h" | 
|  | #include "api/units/timestamp.h" | 
|  | #include "rtc_base/fake_clock.h" | 
|  | #include "rtc_base/thread.h" | 
|  | #include "system_wrappers/include/clock.h" | 
|  | #include "test/gmock.h" | 
|  | #include "test/gtest.h" | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace { | 
|  |  | 
|  | using ::testing::_; | 
|  | using ::testing::AllOf; | 
|  | using ::testing::Eq; | 
|  | using ::testing::Ge; | 
|  | using ::testing::Gt; | 
|  | using ::testing::Lt; | 
|  | using ::testing::MatchesRegex; | 
|  | using ::testing::Property; | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsTrueWhenConditionIsMet) { | 
|  | AutoThread thread; | 
|  |  | 
|  | int counter = 0; | 
|  | EXPECT_TRUE(WaitUntil([&] { return ++counter == 3; })); | 
|  |  | 
|  | // Check that functor is not called after it returned true. | 
|  | EXPECT_EQ(counter, 3); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsWhenConditionIsMet) { | 
|  | AutoThread thread; | 
|  |  | 
|  | int counter = 0; | 
|  | RTCErrorOr<int> result = WaitUntil([&] { return ++counter; }, Eq(3)); | 
|  | EXPECT_THAT(result, IsRtcOkAndHolds(3)); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsErrorWhenTimeoutIsReached) { | 
|  | AutoThread thread; | 
|  | int counter = 0; | 
|  | RTCErrorOr<int> result = | 
|  | WaitUntil([&] { return --counter; }, Eq(1), | 
|  | {.timeout = TimeDelta::Millis(10), .result_name = "counter"}); | 
|  | // Only returns the last error. Note we only are checking that the error | 
|  | // message ends with a negative number rather than a specific number to avoid | 
|  | // flakiness. | 
|  | EXPECT_THAT( | 
|  | result, | 
|  | IsRtcErrorOrWithMessage( | 
|  | _, MatchesRegex( | 
|  | "Value of: counter\nExpected: is equal to 1\nActual: -\\d+"))); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ErrorContainsMatcherExplanation) { | 
|  | AutoThread thread; | 
|  | int counter = 0; | 
|  | auto matcher = AllOf(Gt(0), Lt(10)); | 
|  | RTCErrorOr<int> result = | 
|  | WaitUntil([&] { return --counter; }, matcher, | 
|  | {.timeout = TimeDelta::Millis(10), .result_name = "counter"}); | 
|  | // Only returns the last error. Note we only are checking that the error | 
|  | // message ends with a negative number rather than a specific number to avoid | 
|  | // flakiness. | 
|  | EXPECT_THAT( | 
|  | result, | 
|  | IsRtcErrorOrWithMessage( | 
|  | _, MatchesRegex("Value of: counter\nExpected: \\(is > 0\\) and " | 
|  | "\\(is < 10\\)\nActual: -\\d+, which doesn't match " | 
|  | "\\(is > 0\\)"))); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsWhenConditionIsMetWithSimulatedClock) { | 
|  | SimulatedClock fake_clock(Timestamp::Millis(1337)); | 
|  |  | 
|  | int counter = 0; | 
|  | EXPECT_TRUE(WaitUntil( | 
|  | [&] { return ++counter == 3; }, | 
|  | {.polling_interval = TimeDelta::Millis(10), .clock = &fake_clock})); | 
|  | EXPECT_EQ(counter, 3); | 
|  | // The fake clock should have advanced at least 2 polling intervals, 20ms. | 
|  | EXPECT_THAT(fake_clock.CurrentTime(), Ge(Timestamp::Millis(1357))); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsWhenConditionIsMetWithThreadProcessingFakeClock) { | 
|  | ScopedFakeClock fake_clock; | 
|  |  | 
|  | int counter = 0; | 
|  | EXPECT_TRUE(WaitUntil( | 
|  | [&] { return ++counter == 3; }, | 
|  | {.polling_interval = TimeDelta::Millis(1), .clock = &fake_clock})); | 
|  | EXPECT_EQ(counter, 3); | 
|  | // The fake clock should have advanced at least 2ms. | 
|  | EXPECT_THAT(Timestamp::Micros(fake_clock.TimeNanos() * 1000), | 
|  | Ge(Timestamp::Millis(1339))); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsWhenConditionIsMetWithFakeClock) { | 
|  | FakeClock fake_clock; | 
|  |  | 
|  | int counter = 0; | 
|  | EXPECT_TRUE(WaitUntil( | 
|  | [&] { return ++counter == 3; }, | 
|  | {.polling_interval = TimeDelta::Millis(1), .clock = &fake_clock})); | 
|  | EXPECT_EQ(counter, 3); | 
|  | // The fake clock should have advanced at least 2ms. | 
|  | EXPECT_THAT(Timestamp::Micros(fake_clock.TimeNanos() * 1000), | 
|  | Ge(Timestamp::Millis(1339))); | 
|  | } | 
|  |  | 
|  | // No default constuctor, not assignable, move-only type. | 
|  | class CustomType { | 
|  | public: | 
|  | explicit CustomType(int value) : value_(value) {} | 
|  | CustomType(CustomType&&) = default; | 
|  | CustomType& operator=(CustomType&&) = delete; | 
|  | CustomType() = delete; | 
|  |  | 
|  | int value() const { return value_; } | 
|  |  | 
|  | private: | 
|  | const int value_; | 
|  | }; | 
|  |  | 
|  | TEST(WaitUntilTest, RequiresOnlyMoveCopyConstructionForReturnedType) { | 
|  | AutoThread thread; | 
|  |  | 
|  | int counter = 0; | 
|  | RTCErrorOr<CustomType> result = | 
|  | WaitUntil([&] { return CustomType(++counter); }, | 
|  | Property(&CustomType::value, Eq(3))); | 
|  | EXPECT_THAT(result, IsRtcOkAndHolds(Property(&CustomType::value, Eq(3)))); | 
|  | } | 
|  |  | 
|  | TEST(WaitUntilTest, ReturnsWhenConditionIsMetWithSimulatedTimeController) { | 
|  | std::unique_ptr<TimeController> time_controller = | 
|  | CreateSimulatedTimeController(); | 
|  |  | 
|  | int counter = 0; | 
|  | EXPECT_TRUE(WaitUntil([&] { return ++counter == 3; }, | 
|  | {.polling_interval = TimeDelta::Millis(1), | 
|  | .clock = time_controller.get()})); | 
|  | EXPECT_EQ(counter, 3); | 
|  |  | 
|  | // The fake clock should have advanced at least 2ms. | 
|  | EXPECT_THAT(time_controller->GetClock()->CurrentTime(), | 
|  | Ge(Timestamp::Millis(1339))); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace webrtc |