| // 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/test/test_reg_util_win.h" |
| |
| #include <memory> |
| |
| #include "base/compiler_specific.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/strings/string_util.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/test/task_environment.h" |
| #include "base/time/time.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace registry_util { |
| |
| namespace { |
| const wchar_t kTestKeyPath[] = L"Software\\Chromium\\Foo\\Baz\\TestKey"; |
| const wchar_t kTestValueName[] = L"TestValue"; |
| } // namespace |
| |
| class RegistryOverrideManagerTest : public testing::Test { |
| protected: |
| RegistryOverrideManagerTest() { |
| // We assign a fake test key path to our test RegistryOverrideManager |
| // so we don't interfere with any actual RegistryOverrideManagers running |
| // on the system. This fake path will be auto-deleted by other |
| // RegistryOverrideManagers in case we crash. |
| fake_test_key_root_ = registry_util::GenerateTempKeyPath(); |
| |
| // Ensure a clean test environment. |
| base::win::RegKey key(HKEY_CURRENT_USER); |
| key.DeleteKey(fake_test_key_root_.c_str()); |
| key.DeleteKey(kTestKeyPath); |
| } |
| |
| ~RegistryOverrideManagerTest() override { |
| base::win::RegKey key(HKEY_CURRENT_USER); |
| key.DeleteKey(fake_test_key_root_.c_str()); |
| } |
| |
| void AssertKeyExists(const std::wstring& key_path) { |
| base::win::RegKey key; |
| ASSERT_EQ(ERROR_SUCCESS, |
| key.Open(HKEY_CURRENT_USER, key_path.c_str(), KEY_READ)) |
| << key_path << " does not exist."; |
| } |
| |
| void AssertKeyAbsent(const std::wstring& key_path) { |
| base::win::RegKey key; |
| ASSERT_NE(ERROR_SUCCESS, |
| key.Open(HKEY_CURRENT_USER, key_path.c_str(), KEY_READ)) |
| << key_path << " exists but it should not."; |
| } |
| |
| void CreateKey(const std::wstring& key_path) { |
| base::win::RegKey key; |
| ASSERT_EQ(ERROR_SUCCESS, |
| key.Create(HKEY_CURRENT_USER, key_path.c_str(), KEY_ALL_ACCESS)); |
| } |
| |
| std::wstring FakeOverrideManagerPath(const base::Time& time) { |
| return fake_test_key_root_ + L"\\" + |
| base::AsWString(base::NumberToString16(time.ToInternalValue())); |
| } |
| |
| void CreateManager(const base::Time& timestamp) { |
| manager_.reset(new RegistryOverrideManager(timestamp, fake_test_key_root_)); |
| manager_->OverrideRegistry(HKEY_CURRENT_USER); |
| } |
| |
| std::wstring fake_test_key_root_; |
| std::unique_ptr<RegistryOverrideManager> manager_; |
| }; |
| |
| TEST_F(RegistryOverrideManagerTest, Basic) { |
| ASSERT_NO_FATAL_FAILURE(CreateManager(base::Time::Now())); |
| |
| base::win::RegKey create_key; |
| EXPECT_EQ(ERROR_SUCCESS, |
| create_key.Create(HKEY_CURRENT_USER, kTestKeyPath, KEY_ALL_ACCESS)); |
| EXPECT_TRUE(create_key.Valid()); |
| EXPECT_EQ(ERROR_SUCCESS, create_key.WriteValue(kTestValueName, 42)); |
| create_key.Close(); |
| |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(kTestKeyPath)); |
| |
| DWORD value; |
| base::win::RegKey read_key; |
| EXPECT_EQ(ERROR_SUCCESS, |
| read_key.Open(HKEY_CURRENT_USER, kTestKeyPath, KEY_READ)); |
| EXPECT_TRUE(read_key.Valid()); |
| EXPECT_EQ(ERROR_SUCCESS, read_key.ReadValueDW(kTestValueName, &value)); |
| EXPECT_EQ(42u, value); |
| read_key.Close(); |
| |
| manager_.reset(); |
| |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(kTestKeyPath)); |
| } |
| |
| TEST_F(RegistryOverrideManagerTest, DeleteStaleKeys) { |
| static constexpr base::Time::Exploded kTestTimeExploded = { |
| .year = 2013, .month = 11, .day_of_week = 1, .day_of_month = 4}; |
| base::Time kTestTime; |
| EXPECT_TRUE(base::Time::FromUTCExploded(kTestTimeExploded, &kTestTime)); |
| |
| std::wstring path_garbage = fake_test_key_root_ + L"\\Blah"; |
| std::wstring path_very_stale = |
| FakeOverrideManagerPath(kTestTime - base::Days(100)); |
| std::wstring path_stale = FakeOverrideManagerPath(kTestTime - base::Days(5)); |
| std::wstring path_current = |
| FakeOverrideManagerPath(kTestTime - base::Minutes(1)); |
| std::wstring path_future = |
| FakeOverrideManagerPath(kTestTime + base::Minutes(1)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_garbage)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_very_stale)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_stale)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_current)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_future)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateManager(kTestTime)); |
| manager_.reset(); |
| |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(path_garbage)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(path_very_stale)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(path_stale)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(path_current)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(path_future)); |
| } |
| |
| TEST_F(RegistryOverrideManagerTest, DoesNotUseMockTime) { |
| // This test is targeted at scenarios when multiple tests run at the same |
| // time using `RegistryOverrideManager`, new instances of |
| // `RegistryOverrideManager` will clean up any redirected registry paths that |
| // have the timestamp generated (1970) when using `base::Time::Now()` with |
| // MOCK_TIME enabled, which then cause the currently running tests to fail |
| // since their expected reg keys were deleted by the other test. |
| |
| // To fix this issue, we have updated `RegistryOverrideManager` by using |
| // `base::subtle::TimeNowIgnoringOverride()` instead of `base::Time::Now()`. |
| // So the real current time is used instead of the mock time in 1970. This can |
| // resolve related `RegistryOverrideManager` test failure when using |
| // MOCK_TIME. This test ensures we are fetching the real current time even |
| // when using MOCK_TIME. |
| |
| // Use mock time to init RegKey, which is based on 1970-01-03. |
| // The RegKey contains information about the registry for the |
| // `RegistryOverrideManager`, which also contains a time stamp, which is used |
| // to delete stale keys left over from crashed tests. |
| base::test::TaskEnvironment test_task_env( |
| base::test::TaskEnvironment::TimeSource::MOCK_TIME); |
| |
| const base::Time kTestTime = base::Time::Now(); |
| |
| std::wstring mock_time_path_stale = |
| FakeOverrideManagerPath(kTestTime - base::Days(5)); |
| std::wstring mock_time_path_current = |
| FakeOverrideManagerPath(kTestTime - base::Minutes(1)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateKey(mock_time_path_stale)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(mock_time_path_current)); |
| |
| // Use real time to init `path_real`. |
| std::wstring path_real = GenerateTempKeyPath(); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(path_real)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateManager(kTestTime)); |
| manager_.reset(); |
| |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(mock_time_path_stale)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(mock_time_path_current)); |
| // `path_real` should exist as it is initiated using real time, not mock time |
| // in 1970. |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(path_real)); |
| |
| // Use real time to init following the new set of keys with |
| // `base::subtle::TimeNowIgnoringOverride()`. |
| const base::Time kTestTime_new = base::subtle::TimeNowIgnoringOverride(); |
| std::wstring system_time_path_stale = |
| FakeOverrideManagerPath(kTestTime_new - base::Days(5)); |
| std::wstring system_time_path_current = |
| FakeOverrideManagerPath(kTestTime_new - base::Minutes(1)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateKey(system_time_path_stale)); |
| ASSERT_NO_FATAL_FAILURE(CreateKey(system_time_path_current)); |
| |
| ASSERT_NO_FATAL_FAILURE(CreateManager(kTestTime_new)); |
| manager_.reset(); |
| |
| // Check old keys created with mock time |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(mock_time_path_stale)); |
| // While old keys are created using mock time in 1970, these keys will be |
| // deleted. |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(mock_time_path_current)); |
| // `path_real` should exist as it is initiated using real time, not mock time |
| // in 1970. |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(path_real)); |
| |
| // Create a new manager with real system time. |
| const base::Time kTestTime_latest = base::subtle::TimeNowIgnoringOverride(); |
| ASSERT_NO_FATAL_FAILURE(CreateManager(kTestTime_latest)); |
| manager_.reset(); |
| |
| // Check new keys created with current time |
| ASSERT_NO_FATAL_FAILURE(AssertKeyAbsent(system_time_path_stale)); |
| ASSERT_NO_FATAL_FAILURE(AssertKeyExists(system_time_path_current)); |
| } |
| |
| } // namespace registry_util |