// Copyright (c) 2013 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 "remoting/host/config_file_watcher.h"
#include "base/file_util.h"
#include "base/files/file_path.h"
#include "base/message_loop/message_loop.h"
#include "base/run_loop.h"
#include "remoting/base/auto_thread.h"
#include "remoting/base/auto_thread_task_runner.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gmock_mutant.h"
#include "testing/gtest/include/gtest/gtest.h"
using testing::_;
using testing::AnyNumber;
using testing::Return;
namespace remoting {
namespace {
class ConfigFileWatcherDelegate : public ConfigFileWatcher::Delegate {
public:
ConfigFileWatcherDelegate() {}
virtual ~ConfigFileWatcherDelegate() {}
MOCK_METHOD1(OnConfigUpdated, void(const std::string&));
MOCK_METHOD0(OnConfigWatcherError, void());
private:
DISALLOW_COPY_AND_ASSIGN(ConfigFileWatcherDelegate);
};
} // namespace
class ConfigFileWatcherTest : public testing::Test {
public:
ConfigFileWatcherTest();
virtual ~ConfigFileWatcherTest();
// testing::Test overrides
virtual void SetUp() OVERRIDE;
virtual void TearDown() OVERRIDE;
// Stops the config file watcher.
void StopWatcher();
protected:
base::MessageLoopForUI message_loop_;
base::RunLoop run_loop_;
ConfigFileWatcherDelegate delegate_;
// Path to the configuration file used.
base::FilePath config_file_;
// The configuration file watcher that is being tested.
scoped_ptr<ConfigFileWatcher> watcher_;
};
ConfigFileWatcherTest::ConfigFileWatcherTest() {
}
ConfigFileWatcherTest::~ConfigFileWatcherTest() {
}
void ConfigFileWatcherTest::StopWatcher() {
watcher_.reset();
}
void ConfigFileWatcherTest::SetUp() {
EXPECT_TRUE(base::CreateTemporaryFile(&config_file_));
// Arrange to run |message_loop_| until no components depend on it.
scoped_refptr<AutoThreadTaskRunner> task_runner = new AutoThreadTaskRunner(
message_loop_.message_loop_proxy(), run_loop_.QuitClosure());
scoped_refptr<AutoThreadTaskRunner> io_task_runner =
AutoThread::CreateWithType(
"IPC thread", task_runner, base::MessageLoop::TYPE_IO);
// Create an instance of the config watcher.
watcher_.reset(
new ConfigFileWatcher(task_runner, io_task_runner, config_file_));
}
void ConfigFileWatcherTest::TearDown() {
// Delete the test file.
if (!config_file_.empty())
base::DeleteFile(config_file_, false);
}
// Verifies that the initial notification is delivered.
TEST_F(ConfigFileWatcherTest, Basic) {
std::string data("test");
EXPECT_NE(base::WriteFile(config_file_, data.c_str(),
static_cast<int>(data.size())), -1);
EXPECT_CALL(delegate_, OnConfigUpdated(_))
.Times(1)
.WillOnce(InvokeWithoutArgs(this, &ConfigFileWatcherTest::StopWatcher));
EXPECT_CALL(delegate_, OnConfigWatcherError())
.Times(0);
watcher_->Watch(&delegate_);
run_loop_.Run();
}
MATCHER_P(EqualsString, s, "") {
return arg == s;
}
// Verifies that an update notification is sent when the file is changed.
TEST_F(ConfigFileWatcherTest, Update) {
EXPECT_CALL(delegate_, OnConfigUpdated(EqualsString("test")))
.Times(1)
.WillOnce(InvokeWithoutArgs(this, &ConfigFileWatcherTest::StopWatcher));
EXPECT_CALL(delegate_, OnConfigWatcherError())
.Times(0);
watcher_->Watch(&delegate_);
// Modify the watched file.
std::string data("test");
EXPECT_NE(base::WriteFile(config_file_, data.c_str(),
static_cast<int>(data.size())), -1);
run_loop_.Run();
}
} // namespace remoting