blob: 9ca76df33d3c836e3cb9fc8baefbc8791aeb16ae [file] [log] [blame] [edit]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chromecast/base/thread_health_checker.h"
#include "base/functional/bind.h"
#include "base/memory/ref_counted.h"
#include "base/synchronization/waitable_event.h"
#include "base/test/test_mock_time_task_runner.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace chromecast {
namespace {
const base::TimeDelta kInterval = base::Seconds(3);
const base::TimeDelta kTimeout = base::Seconds(2);
} // namespace
class ThreadHealthCheckerTest : public ::testing::Test {
protected:
ThreadHealthCheckerTest()
: patient_(base::MakeRefCounted<base::TestMockTimeTaskRunner>()),
doctor_(base::MakeRefCounted<base::TestMockTimeTaskRunner>()),
event_(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED) {}
~ThreadHealthCheckerTest() override {}
scoped_refptr<base::TestMockTimeTaskRunner> patient_;
scoped_refptr<base::TestMockTimeTaskRunner> doctor_;
base::WaitableEvent event_;
};
#define CREATE_THREAD_HEALTH_CHECKER(name) \
ThreadHealthChecker name(patient_, doctor_, kInterval, kTimeout, \
base::BindRepeating(&base::WaitableEvent::Signal, \
base::Unretained(&event_)))
TEST_F(ThreadHealthCheckerTest, FiresTimeoutWhenTaskRunnerDoesNotFlush) {
CREATE_THREAD_HEALTH_CHECKER(thc);
// Do not flush the patient, so that the health check sentinel task won't run.
doctor_->FastForwardBy(base::Seconds(6));
EXPECT_TRUE(event_.IsSignaled());
}
TEST_F(ThreadHealthCheckerTest, DoesNotFireTimeoutWhenTaskRunnerFlushesInTime) {
CREATE_THREAD_HEALTH_CHECKER(thc);
// Advance the doctor by enough time to post the health check, but not to time
// out.
doctor_->FastForwardBy(base::Seconds(4));
// Advance the patient to let the sentinel task run.
patient_->FastForwardBy(base::Seconds(4));
// Advance the doctor by enough time such that the sentinel not running would
// cause the failure callback to run.
doctor_->FastForwardBy(base::Seconds(2));
EXPECT_FALSE(event_.IsSignaled());
}
TEST_F(ThreadHealthCheckerTest, FiresTimeoutWhenTaskRunnerFlushesTooLate) {
CREATE_THREAD_HEALTH_CHECKER(thc);
// Advance the doctor before the patient, to simulate a task in the patient
// that takes too long.
doctor_->FastForwardBy(base::Seconds(6));
patient_->FastForwardBy(base::Seconds(6));
// Flush the task runner so the health check sentinel task is run.
EXPECT_TRUE(event_.IsSignaled());
}
TEST_F(ThreadHealthCheckerTest, FiresTimeoutOnLaterIteration) {
CREATE_THREAD_HEALTH_CHECKER(thc);
// Advance the doctor enough to start the check.
doctor_->FastForwardBy(base::Seconds(4));
// Advance the patient enough to run the task.
patient_->FastForwardBy(base::Seconds(4));
// Advance the doctor enough to start the check again.
doctor_->FastForwardBy(base::Seconds(4));
EXPECT_FALSE(event_.IsSignaled());
// Advance the doctor enough for the timeout from the second check to fire.
doctor_->FastForwardBy(base::Seconds(2));
EXPECT_TRUE(event_.IsSignaled());
}
TEST_F(ThreadHealthCheckerTest, NoCrashWhenDestroyed) {
{
CREATE_THREAD_HEALTH_CHECKER(thc);
doctor_->RunUntilIdle();
}
doctor_->RunUntilIdle();
}
TEST_F(ThreadHealthCheckerTest, DropPendingEventsAfterDestruction) {
{
CREATE_THREAD_HEALTH_CHECKER(thc);
// Fast forward by enough time to have scheduled a health check.
doctor_->FastForwardBy(base::Seconds(4));
EXPECT_FALSE(event_.IsSignaled());
}
// Fast forward by enough time for the health check to have executed.
// However, we want all pending events to be dropped after the destructor is
// called, so the event should not be signalled.
doctor_->FastForwardBy(base::Seconds(2));
EXPECT_FALSE(event_.IsSignaled());
}
} // namespace chromecast