blob: 740ec0e578567adae565827944dae8704fff8826 [file] [log] [blame]
/*
* 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