blob: bcb724ea1612f7a51286269fce77c0191c619be0 [file] [log] [blame]
// Copyright 2021 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/test/test_future.h"
#include "base/dcheck_is_on.h"
#include "base/logging.h"
#include "base/macros.h"
#include "base/run_loop.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/post_task.h"
#include "base/task/thread_pool.h"
#include "base/test/bind.h"
#include "base/test/gtest_util.h"
#include "base/test/task_environment.h"
#include "base/threading/sequenced_task_runner_handle.h"
#include "base/threading/thread_task_runner_handle.h"
#include "testing/gtest/include/gtest/gtest-spi.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace base {
namespace test {
namespace {
using AnyType = int;
constexpr int kAnyValue = 5;
constexpr int kOtherValue = 10;
struct MoveOnlyValue {
public:
MoveOnlyValue() = default;
MoveOnlyValue(const MoveOnlyValue&) = delete;
auto& operator=(const MoveOnlyValue&) = delete;
MoveOnlyValue(MoveOnlyValue&&) = default;
MoveOnlyValue& operator=(MoveOnlyValue&&) = default;
~MoveOnlyValue() = default;
int data;
};
} // namespace
class TestFutureTest : public ::testing::Test {
public:
TestFutureTest() = default;
TestFutureTest(const TestFutureTest&) = delete;
TestFutureTest& operator=(const TestFutureTest&) = delete;
~TestFutureTest() override = default;
template <typename Lambda>
void RunLater(Lambda lambda) {
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::BindLambdaForTesting(lambda));
}
void RunLater(base::OnceClosure callable) {
base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,
std::move(callable));
}
void PostDelayedTask(base::OnceClosure callable, base::TimeDelta delay) {
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
FROM_HERE, std::move(callable), delay);
}
private:
base::test::SingleThreadTaskEnvironment environment_{
base::test::TaskEnvironment::TimeSource::MOCK_TIME};
};
TEST_F(TestFutureTest, WaitShouldBlockUntilValueArrives) {
const int expected_value = 42;
TestFuture<int> future;
PostDelayedTask(base::BindOnce(future.GetCallback(), expected_value),
base::TimeDelta::FromMilliseconds(1));
future.Wait();
EXPECT_EQ(expected_value, future.Get());
}
TEST_F(TestFutureTest, WaitShouldReturnTrueWhenValueArrives) {
TestFuture<int> future;
PostDelayedTask(base::BindOnce(future.GetCallback(), kAnyValue),
base::TimeDelta::FromMilliseconds(1));
bool success = future.Wait();
EXPECT_TRUE(success);
}
TEST_F(TestFutureTest, WaitShouldReturnFalseIfTimeoutHappens) {
base::test::ScopedRunLoopTimeout timeout(
FROM_HERE, base::TimeDelta::FromMilliseconds(1));
// |ScopedRunLoopTimeout| will automatically fail the test when a timeout
// happens, so we use EXPECT_FATAL_FAILURE to handle this failure.
// EXPECT_FATAL_FAILURE only works on static objects.
static bool success;
static TestFuture<AnyType> future;
EXPECT_FATAL_FAILURE({ success = future.Wait(); }, "timed out");
EXPECT_FALSE(success);
}
TEST_F(TestFutureTest, GetShouldBlockUntilValueArrives) {
const int expected_value = 42;
TestFuture<int> future;
PostDelayedTask(base::BindOnce(future.GetCallback(), expected_value),
base::TimeDelta::FromMilliseconds(1));
int actual_value = future.Get();
EXPECT_EQ(expected_value, actual_value);
}
TEST_F(TestFutureTest, GetShouldDcheckIfTimeoutHappens) {
base::test::ScopedRunLoopTimeout timeout(
FROM_HERE, base::TimeDelta::FromMilliseconds(1));
TestFuture<AnyType> future;
EXPECT_DCHECK_DEATH_WITH((void)future.Get(), "timed out");
}
TEST_F(TestFutureTest, TakeShouldWorkWithMoveOnlyValue) {
const int expected_data = 99;
TestFuture<MoveOnlyValue> future;
RunLater(base::BindOnce(future.GetCallback(), MoveOnlyValue{expected_data}));
MoveOnlyValue actual_value = future.Take();
EXPECT_EQ(expected_data, actual_value.data);
}
TEST_F(TestFutureTest, TakeShouldDcheckIfTimeoutHappens) {
base::test::ScopedRunLoopTimeout timeout(
FROM_HERE, base::TimeDelta::FromMilliseconds(1));
TestFuture<AnyType> future;
EXPECT_DCHECK_DEATH_WITH((void)future.Take(), "timed out");
}
TEST_F(TestFutureTest, IsReadyShouldBeTrueWhenValueIsSet) {
TestFuture<AnyType> future;
EXPECT_FALSE(future.IsReady());
future.SetValue(kAnyValue);
EXPECT_TRUE(future.IsReady());
}
TEST_F(TestFutureTest, ShouldOnlyAllowSetValueToBeCalledOnce) {
TestFuture<AnyType> future;
future.SetValue(kAnyValue);
EXPECT_DCHECK_DEATH_WITH(future.SetValue(kOtherValue),
"The value of a TestFuture can only be set once.");
}
TEST_F(TestFutureTest, ShouldUnblockWhenSetValueIsInvoked) {
const int expected_value = 111;
TestFuture<int> future;
RunLater([&future]() { future.SetValue(expected_value); });
int actual_value = future.Get();
EXPECT_EQ(expected_value, actual_value);
}
TEST_F(TestFutureTest, ShouldAllowReferenceArgumentsForCallback) {
const int expected_value = 222;
TestFuture<int> future;
base::OnceCallback<void(const int&)> callback =
future.GetCallback<const int&>();
RunLater(base::BindOnce(std::move(callback), expected_value));
int actual_value = future.Get();
EXPECT_EQ(expected_value, actual_value);
}
TEST_F(TestFutureTest, ShouldAllowInvokingCallbackAfterFutureIsDestroyed) {
base::OnceCallback<void(int)> callback;
{
TestFuture<int> future;
callback = future.GetCallback();
}
std::move(callback).Run(1);
}
TEST_F(TestFutureTest, ShouldReturnTupleValue) {
const int expected_int_value = 5;
const std::string expected_string_value = "value";
TestFuture<int, std::string> future;
RunLater(base::BindOnce(future.GetCallback(), expected_int_value,
expected_string_value));
const std::tuple<int, std::string>& actual = future.Get();
EXPECT_EQ(expected_int_value, std::get<0>(actual));
EXPECT_EQ(expected_string_value, std::get<1>(actual));
}
TEST_F(TestFutureTest, ShouldAllowAccessingTupleValueThroughGetMethod) {
const int expected_int_value = 5;
const std::string expected_string_value = "value";
TestFuture<int, std::string> future;
RunLater(base::BindOnce(future.GetCallback(), expected_int_value,
expected_string_value));
ignore_result(future.Get());
EXPECT_EQ(expected_int_value, future.Get<0>());
EXPECT_EQ(expected_string_value, future.Get<1>());
}
TEST_F(TestFutureTest, ShouldAllowReferenceArgumentsForMultiArgumentCallback) {
const int expected_int_value = 5;
const std::string expected_string_value = "value";
TestFuture<int, std::string> future;
base::OnceCallback<void(int, const std::string&)> callback =
future.GetCallback<int, const std::string&>();
RunLater(base::BindOnce(std::move(callback), expected_int_value,
expected_string_value));
std::tuple<int, std::string> actual = future.Get();
EXPECT_EQ(expected_int_value, std::get<0>(actual));
EXPECT_EQ(expected_string_value, std::get<1>(actual));
}
TEST_F(TestFutureTest, SetValueShouldAllowMultipleArguments) {
const int expected_int_value = 5;
const std::string expected_string_value = "value";
TestFuture<int, std::string> future;
RunLater([&future, expected_string_value]() {
future.SetValue(expected_int_value, expected_string_value);
});
const std::tuple<int, std::string>& actual = future.Get();
EXPECT_EQ(expected_int_value, std::get<0>(actual));
EXPECT_EQ(expected_string_value, std::get<1>(actual));
}
} // namespace test
} // namespace base