blob: e16d072f9dfd0978ad8ac8d9dbd5ed3890e7c5a2 [file] [log] [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/functional/callback_helpers.h"
#include <functional>
#include <type_traits>
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/test/gtest_util.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace {
TEST(CallbackHelpersTest, IsBaseCallback) {
// Check that base::{Once,Repeating}Closures and references to them are
// considered base::{Once,Repeating}Callbacks.
static_assert(base::IsBaseCallback<base::OnceClosure>::value, "");
static_assert(base::IsBaseCallback<base::RepeatingClosure>::value, "");
static_assert(base::IsBaseCallback<base::OnceClosure&&>::value, "");
static_assert(base::IsBaseCallback<const base::RepeatingClosure&>::value, "");
// Check that base::{Once, Repeating}Callbacks with a given RunType and
// references to them are considered base::{Once, Repeating}Callbacks.
static_assert(base::IsBaseCallback<base::OnceCallback<int(int)>>::value, "");
static_assert(base::IsBaseCallback<base::RepeatingCallback<int(int)>>::value,
"");
static_assert(base::IsBaseCallback<base::OnceCallback<int(int)>&&>::value,
"");
static_assert(
base::IsBaseCallback<const base::RepeatingCallback<int(int)>&>::value,
"");
// Check that POD types are not considered base::{Once, Repeating}Callbacks.
static_assert(!base::IsBaseCallback<bool>::value, "");
static_assert(!base::IsBaseCallback<int>::value, "");
static_assert(!base::IsBaseCallback<double>::value, "");
// Check that the closely related std::function is not considered a
// base::{Once, Repeating}Callback.
static_assert(!base::IsBaseCallback<std::function<void()>>::value, "");
static_assert(!base::IsBaseCallback<const std::function<void()>&>::value, "");
static_assert(!base::IsBaseCallback<std::function<void()>&&>::value, "");
}
TEST(CallbackHelpersTest, IsOnceCallback) {
// Check that base::OnceClosures and references to them are considered
// base::OnceCallbacks, but base::RepeatingClosures are not.
static_assert(base::IsOnceCallback<base::OnceClosure>::value, "");
static_assert(!base::IsOnceCallback<base::RepeatingClosure>::value, "");
static_assert(base::IsOnceCallback<base::OnceClosure&&>::value, "");
static_assert(!base::IsOnceCallback<const base::RepeatingClosure&>::value,
"");
// Check that base::OnceCallbacks with a given RunType and references to them
// are considered base::OnceCallbacks, but base::RepeatingCallbacks are not.
static_assert(base::IsOnceCallback<base::OnceCallback<int(int)>>::value, "");
static_assert(!base::IsOnceCallback<base::RepeatingCallback<int(int)>>::value,
"");
static_assert(base::IsOnceCallback<base::OnceCallback<int(int)>&&>::value,
"");
static_assert(
!base::IsOnceCallback<const base::RepeatingCallback<int(int)>&>::value,
"");
// Check that POD types are not considered base::OnceCallbacks.
static_assert(!base::IsOnceCallback<bool>::value, "");
static_assert(!base::IsOnceCallback<int>::value, "");
static_assert(!base::IsOnceCallback<double>::value, "");
// Check that the closely related std::function is not considered a
// base::OnceCallback.
static_assert(!base::IsOnceCallback<std::function<void()>>::value, "");
static_assert(!base::IsOnceCallback<const std::function<void()>&>::value, "");
static_assert(!base::IsOnceCallback<std::function<void()>&&>::value, "");
// Check that the result of BindOnce is a OnceCallback.
auto cb = base::BindOnce([](int* count) { ++*count; });
static_assert(base::IsOnceCallback<decltype(cb)>::value, "");
}
void Increment(int* value) {
(*value)++;
}
TEST(CallbackHelpersTest, ScopedClosureRunnerHasClosure) {
base::ScopedClosureRunner runner1;
EXPECT_FALSE(runner1);
base::ScopedClosureRunner runner2{base::DoNothing()};
EXPECT_TRUE(runner2);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerExitScope) {
int run_count = 0;
{
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count));
EXPECT_EQ(0, run_count);
}
EXPECT_EQ(1, run_count);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerRelease) {
int run_count = 0;
base::OnceClosure c;
{
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count));
c = runner.Release();
EXPECT_EQ(0, run_count);
}
EXPECT_EQ(0, run_count);
std::move(c).Run();
EXPECT_EQ(1, run_count);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerReplaceClosure) {
int run_count_1 = 0;
int run_count_2 = 0;
{
base::ScopedClosureRunner runner;
runner.ReplaceClosure(base::BindOnce(&Increment, &run_count_1));
runner.ReplaceClosure(base::BindOnce(&Increment, &run_count_2));
EXPECT_EQ(0, run_count_1);
EXPECT_EQ(0, run_count_2);
}
EXPECT_EQ(0, run_count_1);
EXPECT_EQ(1, run_count_2);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerRunAndResetNonNull) {
int run_count_3 = 0;
{
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count_3));
EXPECT_EQ(0, run_count_3);
runner.RunAndReset();
EXPECT_EQ(1, run_count_3);
}
EXPECT_EQ(1, run_count_3);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerRunAndResetNull) {
base::ScopedClosureRunner runner;
runner.RunAndReset(); // Should not crash.
}
TEST(CallbackHelpersTest, ScopedClosureRunnerMoveConstructor) {
int run_count = 0;
{
std::unique_ptr<base::ScopedClosureRunner> runner(
new base::ScopedClosureRunner(base::BindOnce(&Increment, &run_count)));
base::ScopedClosureRunner runner2(std::move(*runner));
runner.reset();
EXPECT_EQ(0, run_count);
}
EXPECT_EQ(1, run_count);
}
TEST(CallbackHelpersTest, ScopedClosureRunnerMoveAssignment) {
int run_count_1 = 0;
int run_count_2 = 0;
{
base::ScopedClosureRunner runner(base::BindOnce(&Increment, &run_count_1));
{
base::ScopedClosureRunner runner2(
base::BindOnce(&Increment, &run_count_2));
runner = std::move(runner2);
EXPECT_EQ(1, run_count_1);
EXPECT_EQ(0, run_count_2);
}
EXPECT_EQ(1, run_count_1);
EXPECT_EQ(0, run_count_2);
}
EXPECT_EQ(1, run_count_1);
EXPECT_EQ(1, run_count_2);
}
TEST(CallbackHelpersTest, SplitOnceCallback_EmptyCallback) {
base::OnceCallback<void(int*)> cb = base::NullCallback();
EXPECT_FALSE(cb);
auto split = base::SplitOnceCallback(std::move(cb));
static_assert(std::is_same<decltype(split),
std::pair<base::OnceCallback<void(int*)>,
base::OnceCallback<void(int*)>>>::value,
"");
EXPECT_FALSE(split.first);
EXPECT_FALSE(split.second);
}
TEST(CallbackHelpersTest, SplitOnceCallback_FirstCallback) {
int count = 0;
base::OnceCallback<void(int*)> cb =
base::BindOnce([](int* count) { ++*count; });
auto split = base::SplitOnceCallback(std::move(cb));
static_assert(std::is_same<decltype(split),
std::pair<base::OnceCallback<void(int*)>,
base::OnceCallback<void(int*)>>>::value,
"");
EXPECT_EQ(0, count);
std::move(split.first).Run(&count);
EXPECT_EQ(1, count);
#if GTEST_HAS_DEATH_TEST
EXPECT_CHECK_DEATH(std::move(split.second).Run(&count));
#endif // GTEST_HAS_DEATH_TEST
}
TEST(CallbackHelpersTest, SplitOnceCallback_SecondCallback) {
int count = 0;
base::OnceCallback<void(int*)> cb =
base::BindOnce([](int* count) { ++*count; });
auto split = base::SplitOnceCallback(std::move(cb));
static_assert(std::is_same<decltype(split),
std::pair<base::OnceCallback<void(int*)>,
base::OnceCallback<void(int*)>>>::value,
"");
EXPECT_EQ(0, count);
std::move(split.second).Run(&count);
EXPECT_EQ(1, count);
EXPECT_CHECK_DEATH(std::move(split.first).Run(&count));
}
TEST(CallbackHelpersTest, SplitSplitOnceCallback_FirstSplit) {
int count = 0;
base::OnceCallback<void(int*)> cb =
base::BindOnce([](int* count) { ++*count; });
auto split = base::SplitOnceCallback(std::move(cb));
base::OnceCallback<void(int*)> cb1 = std::move(split.first);
split = base::SplitOnceCallback(std::move(split.second));
base::OnceCallback<void(int*)> cb2 = std::move(split.first);
base::OnceCallback<void(int*)> cb3 = std::move(split.second);
EXPECT_EQ(0, count);
std::move(cb1).Run(&count);
EXPECT_EQ(1, count);
EXPECT_CHECK_DEATH(std::move(cb3).Run(&count));
}
TEST(CallbackHelpersTest, SplitSplitOnceCallback_SecondSplit) {
int count = 0;
base::OnceCallback<void(int*)> cb =
base::BindOnce([](int* count) { ++*count; });
auto split = base::SplitOnceCallback(std::move(cb));
base::OnceCallback<void(int*)> cb1 = std::move(split.first);
split = base::SplitOnceCallback(std::move(split.second));
base::OnceCallback<void(int*)> cb2 = std::move(split.first);
base::OnceCallback<void(int*)> cb3 = std::move(split.second);
EXPECT_EQ(0, count);
std::move(cb2).Run(&count);
EXPECT_EQ(1, count);
EXPECT_CHECK_DEATH(std::move(cb1).Run(&count));
}
TEST(CallbackHelpersTest, IgnoreArgs) {
int count = 0;
base::RepeatingClosure repeating_closure =
base::BindRepeating(&Increment, &count);
base::OnceClosure once_closure = base::BindOnce(&Increment, &count);
base::RepeatingCallback<void(int)> repeating_int_cb =
base::IgnoreArgs<int>(repeating_closure);
EXPECT_EQ(0, count);
repeating_int_cb.Run(42);
EXPECT_EQ(1, count);
repeating_int_cb.Run(42);
EXPECT_EQ(2, count);
base::OnceCallback<void(int)> once_int_cb =
base::IgnoreArgs<int>(std::move(once_closure));
EXPECT_EQ(2, count);
std::move(once_int_cb).Run(42);
EXPECT_EQ(3, count);
}
TEST(CallbackHelpersTest, IgnoreArgs_EmptyCallback) {
base::RepeatingCallback<void(int)> repeating_int_cb =
base::IgnoreArgs<int>(base::RepeatingClosure());
EXPECT_FALSE(repeating_int_cb);
base::OnceCallback<void(int)> once_int_cb =
base::IgnoreArgs<int>(base::OnceClosure());
EXPECT_FALSE(once_int_cb);
}
} // namespace