| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/installer/setup/install_worker.h" |
| |
| #include <memory> |
| #include <string> |
| #include <tuple> |
| |
| #include "base/rand_util.h" |
| #include "base/strings/stringprintf.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/test/scoped_os_info_override_win.h" |
| #include "base/test/test_reg_util_win.h" |
| #include "base/version.h" |
| #include "base/win/registry.h" |
| #include "base/win/win_util.h" |
| #include "base/win/windows_version.h" |
| #include "build/branding_buildflags.h" |
| #include "chrome/common/chrome_constants.h" |
| #include "chrome/install_static/install_util.h" |
| #include "chrome/install_static/test/scoped_install_details.h" |
| #include "chrome/installer/setup/install_params.h" |
| #include "chrome/installer/setup/installer_state.h" |
| #include "chrome/installer/setup/setup_util.h" |
| #include "chrome/installer/util/create_reg_key_work_item.h" |
| #include "chrome/installer/util/delete_reg_key_work_item.h" |
| #include "chrome/installer/util/delete_tree_work_item.h" |
| #include "chrome/installer/util/google_update_constants.h" |
| #include "chrome/installer/util/helper.h" |
| #include "chrome/installer/util/install_util.h" |
| #include "chrome/installer/util/installation_state.h" |
| #include "chrome/installer/util/set_reg_value_work_item.h" |
| #include "chrome/installer/util/util_constants.h" |
| #include "chrome/installer/util/work_item_list.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| using base::win::RegKey; |
| using installer::InstallationState; |
| using installer::InstallerState; |
| using installer::ProductState; |
| |
| using ::testing::_; |
| using ::testing::AnyNumber; |
| using ::testing::AtLeast; |
| using ::testing::AtMost; |
| using ::testing::Bool; |
| using ::testing::Combine; |
| using ::testing::Eq; |
| using ::testing::HasSubstr; |
| using ::testing::NiceMock; |
| using ::testing::Return; |
| using ::testing::StrCaseEq; |
| using ::testing::StrEq; |
| using ::testing::StrictMock; |
| using ::testing::TestParamInfo; |
| using ::testing::Values; |
| |
| namespace { |
| |
| // Mock classes to help with testing |
| //------------------------------------------------------------------------------ |
| |
| class MockWorkItemList : public WorkItemList { |
| public: |
| MockWorkItemList() = default; |
| |
| MOCK_METHOD5(AddCopyTreeWorkItem, |
| WorkItem*(const base::FilePath&, |
| const base::FilePath&, |
| const base::FilePath&, |
| CopyOverWriteOption, |
| const base::FilePath&)); |
| MOCK_METHOD1(AddCreateDirWorkItem, WorkItem*(const base::FilePath&)); |
| MOCK_METHOD3(AddCreateRegKeyWorkItem, |
| WorkItem*(HKEY, const std::wstring&, REGSAM)); |
| MOCK_METHOD3(AddDeleteRegKeyWorkItem, |
| WorkItem*(HKEY, const std::wstring&, REGSAM)); |
| MOCK_METHOD4( |
| AddDeleteRegValueWorkItem, |
| WorkItem*(HKEY, const std::wstring&, REGSAM, const std::wstring&)); |
| MOCK_METHOD2(AddDeleteTreeWorkItem, |
| WorkItem*(const base::FilePath&, const base::FilePath&)); |
| MOCK_METHOD4(AddMoveTreeWorkItem, |
| WorkItem*(const base::FilePath&, |
| const base::FilePath&, |
| const base::FilePath&, |
| MoveTreeOption)); |
| // Workaround for gmock problems with disambiguating between string pointers |
| // and DWORD. |
| WorkItem* AddSetRegValueWorkItem(HKEY a1, |
| const std::wstring& a2, |
| REGSAM a3, |
| const std::wstring& a4, |
| const std::wstring& a5, |
| bool a6) override { |
| return AddSetRegStringValueWorkItem(a1, a2, a3, a4, a5, a6); |
| } |
| |
| WorkItem* AddSetRegValueWorkItem(HKEY a1, |
| const std::wstring& a2, |
| REGSAM a3, |
| const std::wstring& a4, |
| DWORD a5, |
| bool a6) override { |
| return AddSetRegDwordValueWorkItem(a1, a2, a3, a4, a5, a6); |
| } |
| |
| MOCK_METHOD6(AddSetRegStringValueWorkItem, |
| WorkItem*(HKEY, |
| const std::wstring&, |
| REGSAM, |
| const std::wstring&, |
| const std::wstring&, |
| bool)); |
| MOCK_METHOD6(AddSetRegDwordValueWorkItem, |
| WorkItem*(HKEY, |
| const std::wstring&, |
| REGSAM, |
| const std::wstring&, |
| DWORD, |
| bool)); |
| }; |
| |
| class MockProductState : public ProductState { |
| public: |
| // Takes ownership of |version|. |
| void set_version(base::Version* version) { version_.reset(version); } |
| void set_brand(const std::wstring& brand) { brand_ = brand; } |
| void set_eula_accepted(DWORD eula_accepted) { |
| has_eula_accepted_ = true; |
| eula_accepted_ = eula_accepted; |
| } |
| void clear_eula_accepted() { has_eula_accepted_ = false; } |
| void set_usagestats(DWORD usagestats) { |
| has_usagestats_ = true; |
| usagestats_ = usagestats; |
| } |
| void clear_usagestats() { has_usagestats_ = false; } |
| void set_oem_install(const std::wstring& oem_install) { |
| has_oem_install_ = true; |
| oem_install_ = oem_install; |
| } |
| void clear_oem_install() { has_oem_install_ = false; } |
| void SetUninstallProgram(const base::FilePath& setup_exe) { |
| uninstall_command_ = base::CommandLine(setup_exe); |
| } |
| void AddUninstallSwitch(const std::string& option) { |
| uninstall_command_.AppendSwitch(option); |
| } |
| }; |
| |
| // Okay, so this isn't really a mock as such, but it does add setter methods |
| // to make it easier to build custom InstallationStates. |
| class MockInstallationState : public InstallationState { |
| public: |
| // Included for testing. |
| void SetProductState(bool system_install, const ProductState& product_state) { |
| ProductState& target = system_install ? system_chrome_ : user_chrome_; |
| target.CopyFrom(product_state); |
| } |
| }; |
| |
| class MockInstallerState : public InstallerState { |
| public: |
| void set_level(Level level) { InstallerState::set_level(level); } |
| |
| void set_operation(Operation operation) { operation_ = operation; } |
| |
| void set_state_key(const std::wstring& state_key) { state_key_ = state_key; } |
| }; |
| |
| void AddChromeToInstallationState(bool system_level, |
| base::Version* current_version, |
| MockInstallationState* installation_state) { |
| MockProductState product_state; |
| product_state.set_version(new base::Version(*current_version)); |
| product_state.set_brand(L"TEST"); |
| product_state.set_eula_accepted(1); |
| base::FilePath install_path = |
| installer::GetDefaultChromeInstallPath(system_level); |
| product_state.SetUninstallProgram( |
| install_path.AppendASCII(current_version->GetString()) |
| .Append(installer::kInstallerDir) |
| .Append(installer::kSetupExe)); |
| product_state.AddUninstallSwitch(installer::switches::kUninstall); |
| if (system_level) |
| product_state.AddUninstallSwitch(installer::switches::kSystemLevel); |
| |
| installation_state->SetProductState(system_level, product_state); |
| } |
| |
| MockInstallationState* BuildChromeInstallationState( |
| bool system_level, |
| base::Version* current_version) { |
| std::unique_ptr<MockInstallationState> installation_state( |
| new MockInstallationState()); |
| AddChromeToInstallationState(system_level, current_version, |
| installation_state.get()); |
| return installation_state.release(); |
| } |
| |
| MockInstallerState* BuildBasicInstallerState( |
| bool system_install, |
| const InstallationState& machine_state, |
| InstallerState::Operation operation) { |
| std::unique_ptr<MockInstallerState> installer_state(new MockInstallerState()); |
| |
| InstallerState::Level level = system_install ? InstallerState::SYSTEM_LEVEL |
| : InstallerState::USER_LEVEL; |
| installer_state->set_level(level); |
| installer_state->set_operation(operation); |
| // Hope this next one isn't checked for now. |
| installer_state->set_state_key(L"PROBABLY_INVALID_REG_PATH"); |
| return installer_state.release(); |
| } |
| |
| void AddChromeToInstallerState(const InstallationState& machine_state, |
| MockInstallerState* installer_state) { |
| // Fresh install or upgrade? |
| const ProductState* chrome = |
| machine_state.GetProductState(installer_state->system_install()); |
| if (chrome) { |
| installer_state->set_target_path_for_testing( |
| chrome->GetSetupPath().DirName().DirName().DirName()); |
| } else { |
| installer_state->set_target_path_for_testing( |
| installer::GetDefaultChromeInstallPath( |
| installer_state->system_install())); |
| } |
| } |
| |
| MockInstallerState* BuildChromeInstallerState( |
| bool system_install, |
| const InstallationState& machine_state, |
| InstallerState::Operation operation) { |
| std::unique_ptr<MockInstallerState> installer_state( |
| BuildBasicInstallerState(system_install, machine_state, operation)); |
| AddChromeToInstallerState(machine_state, installer_state.get()); |
| return installer_state.release(); |
| } |
| |
| } // namespace |
| |
| // The test fixture |
| //------------------------------------------------------------------------------ |
| |
| class InstallWorkerTest : public testing::Test { |
| public: |
| void SetUp() override { |
| current_version_ = std::make_unique<base::Version>("1.0.0.0"); |
| new_version_ = std::make_unique<base::Version>("42.0.0.0"); |
| |
| // Don't bother ensuring that these paths exist. Since we're just |
| // building the work item lists and not running them, they shouldn't |
| // actually be touched. |
| src_path_ = base::FilePath( |
| L"C:\\UnlikelyPath\\Temp\\chrome_123\\source\\Chrome-bin"); |
| setup_path_ = |
| base::FilePath(L"C:\\UnlikelyPath\\Temp\\CR_123.tmp\\setup.exe"); |
| temp_dir_ = base::FilePath(L"C:\\UnlikelyPath\\Temp\\chrome_123"); |
| } |
| |
| protected: |
| std::unique_ptr<base::Version> current_version_; |
| std::unique_ptr<base::Version> new_version_; |
| base::FilePath setup_path_; |
| base::FilePath src_path_; |
| base::FilePath temp_dir_; |
| }; |
| |
| // Tests |
| //------------------------------------------------------------------------------ |
| |
| // Chrome for Testing does not support system-level installations. |
| #if !BUILDFLAG(GOOGLE_CHROME_FOR_TESTING_BRANDING) |
| TEST_F(InstallWorkerTest, TestInstallChromeSystem) { |
| const bool system_level = true; |
| NiceMock<MockWorkItemList> work_item_list; |
| |
| const HKEY kRegRoot = system_level ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER; |
| static const wchar_t kRegKeyPath[] = L"Software\\Chromium\\test"; |
| std::unique_ptr<CreateRegKeyWorkItem> create_reg_key_work_item( |
| WorkItem::CreateCreateRegKeyWorkItem(kRegRoot, kRegKeyPath, |
| WorkItem::kWow64Default)); |
| std::unique_ptr<SetRegValueWorkItem> set_reg_value_work_item( |
| WorkItem::CreateSetRegValueWorkItem( |
| kRegRoot, kRegKeyPath, WorkItem::kWow64Default, L"", L"", false)); |
| std::unique_ptr<DeleteTreeWorkItem> delete_tree_work_item( |
| WorkItem::CreateDeleteTreeWorkItem(base::FilePath(), base::FilePath())); |
| std::unique_ptr<DeleteRegKeyWorkItem> delete_reg_key_work_item( |
| WorkItem::CreateDeleteRegKeyWorkItem(kRegRoot, kRegKeyPath, |
| WorkItem::kWow64Default)); |
| |
| std::unique_ptr<InstallationState> installation_state( |
| BuildChromeInstallationState(system_level, current_version_.get())); |
| |
| std::unique_ptr<InstallerState> installer_state( |
| BuildChromeInstallerState(system_level, *installation_state, |
| InstallerState::SINGLE_INSTALL_OR_UPDATE)); |
| |
| // Set up some expectations. |
| // TODO(robertshield): Set up some real expectations. |
| EXPECT_CALL(work_item_list, AddCopyTreeWorkItem(_, _, _, _, _)) |
| .Times(AtLeast(1)); |
| EXPECT_CALL(work_item_list, AddCreateRegKeyWorkItem(_, _, _)) |
| .WillRepeatedly(Return(create_reg_key_work_item.get())); |
| EXPECT_CALL(work_item_list, AddSetRegStringValueWorkItem(_, _, _, _, _, _)) |
| .WillRepeatedly(Return(set_reg_value_work_item.get())); |
| EXPECT_CALL(work_item_list, AddSetRegDwordValueWorkItem(_, _, _, _, _, _)) |
| .WillRepeatedly(Return(set_reg_value_work_item.get())); |
| EXPECT_CALL(work_item_list, AddDeleteTreeWorkItem(_, _)) |
| .WillRepeatedly(Return(delete_tree_work_item.get())); |
| EXPECT_CALL(work_item_list, AddDeleteRegKeyWorkItem(_, _, _)) |
| .WillRepeatedly(Return(delete_reg_key_work_item.get())); |
| |
| const base::Version current_version( |
| installer_state->GetCurrentVersion(*installation_state)); |
| installer::InstallParams install_params = { |
| *installer_state, *installation_state, setup_path_, current_version, |
| src_path_, temp_dir_, *new_version_, |
| }; |
| |
| // Set up expectations for setup.exe's on-os-upgrade handler. |
| const std::wstring update_handler_command_key = |
| base::StrCat({install_static::GetClientsKeyPath(), L"\\", |
| google_update::kRegCommandsKey, L"\\", L"on-os-upgrade"}); |
| EXPECT_CALL(work_item_list, |
| AddCreateRegKeyWorkItem(kRegRoot, update_handler_command_key, |
| KEY_WOW64_32KEY)) |
| .WillOnce(Return(create_reg_key_work_item.get())); |
| const std::wstring command_line = |
| base::StrCat({L"\"", installer_state->target_path().value(), L"\\", |
| base::ASCIIToWide(new_version_->GetString()), |
| L"\\Installer\\setup.exe\" --on-os-upgrade --system-level " |
| L"--verbose-logging %1"}); |
| EXPECT_CALL(work_item_list, |
| AddSetRegStringValueWorkItem( |
| kRegRoot, update_handler_command_key, KEY_WOW64_32KEY, |
| std::wstring(google_update::kRegCommandLineField), |
| command_line, true)) |
| .WillOnce(Return(set_reg_value_work_item.get())); |
| EXPECT_CALL( |
| work_item_list, |
| AddSetRegDwordValueWorkItem( |
| kRegRoot, update_handler_command_key, KEY_WOW64_32KEY, |
| std::wstring(google_update::kRegAutoRunOnOSUpgradeField), 1, true)) |
| .WillOnce(Return(set_reg_value_work_item.get())); |
| |
| AddInstallWorkItems(install_params, &work_item_list); |
| } |
| #endif |
| |
| // Tests for installer::AddUpdateBrandCodeWorkItem(). |
| //------------------------------------------------------------------------------ |
| |
| // Parameters for AddUpdateBrandCodeWorkItem tests: |
| // bool: is domain joined |
| // bool: is registered with MDM |
| // bool: is Windows 10 home edition |
| using AddUpdateBrandCodeWorkItemTestParams = std::tuple<bool, bool, bool>; |
| |
| // These tests run at system level. |
| static const bool kSystemLevel = true; |
| |
| class AddUpdateBrandCodeWorkItemTest |
| : public ::testing::TestWithParam<AddUpdateBrandCodeWorkItemTestParams> { |
| public: |
| AddUpdateBrandCodeWorkItemTest() |
| : is_domain_joined_(std::get<0>(GetParam())), |
| is_registered_(std::get<1>(GetParam())), |
| is_home_edition_(std::get<2>(GetParam())), |
| scoped_install_details_(kSystemLevel), |
| current_version_(new base::Version("1.0.0.0")), |
| installation_state_( |
| BuildChromeInstallationState(kSystemLevel, current_version_.get())), |
| installer_state_(BuildChromeInstallerState( |
| kSystemLevel, |
| *installation_state_, |
| InstallerState::SINGLE_INSTALL_OR_UPDATE)), |
| scoped_domain_state_(is_domain_joined_), |
| scoped_registration_state_(is_registered_), |
| scoped_os_info_override_( |
| is_home_edition_ |
| ? base::test::ScopedOSInfoOverride::Type::kWin10Home |
| : base::test::ScopedOSInfoOverride::Type::kWin10Pro) {} |
| |
| void SetUp() override { |
| // Override registry so that tests don't mess up the machine's state. |
| ASSERT_NO_FATAL_FAILURE(registry_override_.OverrideRegistry( |
| HKEY_LOCAL_MACHINE, ®istry_override_hklm_path_)); |
| } |
| |
| void SetupExpectations(const std::wstring& brand, |
| bool is_cbcm_enrolled, |
| StrictMock<MockWorkItemList>* work_item_list) { |
| if (!brand.empty()) { |
| base::win::RegKey key(installer_state_->root_key(), |
| install_static::GetClientStateKeyPath().c_str(), |
| KEY_WRITE); |
| ASSERT_TRUE(key.Valid()); |
| ASSERT_EQ( |
| 0, key.WriteValue(google_update::kRegRLZBrandField, brand.c_str())); |
| } |
| |
| if (is_cbcm_enrolled) { |
| std::wstring enrollment_token(L"ENROLLMENT_TOKEN"); |
| std::string dm_token = base::RandBytesAsString(1000); |
| for (const std::pair<std::wstring, std::wstring>& key_and_value : |
| InstallUtil::GetCloudManagementEnrollmentTokenRegistryPaths()) { |
| base::win::RegKey key(installer_state_->root_key(), |
| key_and_value.first.c_str(), KEY_WRITE); |
| ASSERT_TRUE(key.Valid()); |
| ASSERT_EQ(0, key.WriteValue(key_and_value.second.c_str(), |
| enrollment_token.c_str())); |
| } |
| base::win::RegKey key; |
| std::wstring value_name; |
| std::tie(key, value_name) = |
| InstallUtil::GetCloudManagementDmTokenLocation( |
| InstallUtil::ReadOnly(false), |
| InstallUtil::BrowserLocation(false)); |
| ASSERT_TRUE(key.Valid()); |
| ASSERT_EQ(0, key.WriteValue(value_name.c_str(), dm_token.data(), |
| base::saturated_cast<DWORD>(dm_token.size()), |
| REG_BINARY)); |
| } |
| |
| if (!installer::GetUpdatedBrandCode(brand, true).empty() && |
| (is_domain_joined_ || (is_registered_ && !is_home_edition_))) { |
| EXPECT_CALL(*work_item_list, |
| AddSetRegStringValueWorkItem(_, _, _, _, _, _)) |
| .WillOnce(Return(nullptr)); // Return value ignored. |
| } else if (!installer::GetUpdatedBrandCode(brand, false).empty() && |
| (!is_domain_joined_ && (!is_registered_ || is_home_edition_))) { |
| EXPECT_CALL(*work_item_list, |
| AddSetRegStringValueWorkItem(_, _, _, _, _, _)) |
| .WillOnce(Return(nullptr)); // Return value ignored. |
| } else if (!installer::TransformCloudManagementBrandCode(brand, |
| is_cbcm_enrolled) |
| .empty()) { |
| EXPECT_CALL(*work_item_list, |
| AddSetRegStringValueWorkItem(_, _, _, _, _, _)) |
| .WillOnce(Return(nullptr)); // Return value ignored. |
| } |
| } |
| |
| const InstallerState* installer_state() { return installer_state_.get(); } |
| |
| private: |
| const bool is_domain_joined_; |
| const bool is_registered_; |
| const bool is_home_edition_; |
| |
| install_static::ScopedInstallDetails scoped_install_details_; |
| std::unique_ptr<base::Version> current_version_; |
| std::unique_ptr<InstallationState> installation_state_; |
| std::unique_ptr<InstallerState> installer_state_; |
| registry_util::RegistryOverrideManager registry_override_; |
| std::wstring registry_override_hklm_path_; |
| base::win::ScopedDomainStateForTesting scoped_domain_state_; |
| base::win::ScopedDeviceRegisteredWithManagementForTesting |
| scoped_registration_state_; |
| base::test::ScopedOSInfoOverride scoped_os_info_override_; |
| }; |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, NoBrand) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GGRV) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GGRV", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GTPM) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GTPM", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GGLS) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GGLS", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GGRV_CBCM) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GGRV", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GGLS_CBCM) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GGLS", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GTPM_CBCM) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GTPM", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, TEST) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"TEST", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCEU) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCEU", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCEV) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCEV", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCER) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCER", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCEA) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCEA", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCEL) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCEA", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCFB) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCFB", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCGC) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCGC", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCHD) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GChD", true, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCCJ) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCCJ", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCKK) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCKK", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCLL) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCLL", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| TEST_P(AddUpdateBrandCodeWorkItemTest, GCMM) { |
| StrictMock<MockWorkItemList> work_item_list; |
| SetupExpectations(L"GCMM", false, &work_item_list); |
| installer::AddUpdateBrandCodeWorkItem(*installer_state(), &work_item_list); |
| } |
| |
| struct AddUpdateBrandCodeWorkItemTestParamToString { |
| std::string operator()( |
| const TestParamInfo<AddUpdateBrandCodeWorkItemTestParams>& info) const { |
| const char* joined = std::get<0>(info.param) ? "joined" : "notjoined"; |
| const char* registered = |
| std::get<1>(info.param) ? "registered" : "notregistered"; |
| const char* home = std::get<2>(info.param) ? "home" : "nothome"; |
| return base::StringPrintf("%s_%s_%s", joined, registered, home); |
| } |
| }; |
| |
| INSTANTIATE_TEST_SUITE_P(AddUpdateBrandCodeWorkItemTest, |
| AddUpdateBrandCodeWorkItemTest, |
| Combine(Bool(), Bool(), Bool()), |
| AddUpdateBrandCodeWorkItemTestParamToString()); |
| |
| // Test for installer::AddOldWerHelperRegistrationCleanupItems(). |
| |
| class ChromeWerDllRegistryTest : public testing::Test { |
| public: |
| ~ChromeWerDllRegistryTest() override = default; |
| |
| void SetUp() override { |
| ASSERT_NO_FATAL_FAILURE( |
| registry_overrides_.OverrideRegistry(HKEY_LOCAL_MACHINE)); |
| } |
| |
| private: |
| registry_util::RegistryOverrideManager registry_overrides_; |
| }; |
| |
| TEST_F(ChromeWerDllRegistryTest, AddOldWerHelperRegistrationCleanupItems) { |
| const base::FilePath target_path(L"C:\\TargetPath"); |
| const HKEY root_key = HKEY_LOCAL_MACHINE; |
| const std::wstring wer_registry_path = installer::GetWerHelperRegistryPath(); |
| |
| std::unique_ptr<WorkItemList> work_item_list(WorkItem::CreateWorkItemList()); |
| installer::AddWerHelperRegistration( |
| root_key, |
| installer::GetWerHelperPath(target_path, base::Version("0.0.0.1")), |
| work_item_list.get()); |
| installer::AddWerHelperRegistration( |
| root_key, |
| installer::GetWerHelperPath(target_path, base::Version("0.0.0.2")), |
| work_item_list.get()); |
| |
| const base::FilePath another_path( |
| L"C:\\AnotherPath\\0.0.0.3\\chrome_wer.dll"); |
| installer::AddWerHelperRegistration(root_key, another_path, |
| work_item_list.get()); |
| const base::FilePath path_with_invalid_version( |
| L"C:\\TargetPath\\a.b.c.d\\chrome_wer.dll"); |
| installer::AddWerHelperRegistration(root_key, path_with_invalid_version, |
| work_item_list.get()); |
| const base::FilePath path_with_another_dll( |
| L"C:\\TargetPath\\0.0.0.4\\another_wer.dll"); |
| installer::AddWerHelperRegistration(root_key, path_with_another_dll, |
| work_item_list.get()); |
| |
| ASSERT_TRUE(work_item_list->Do()); |
| { |
| base::win::RegistryValueIterator value_iter( |
| root_key, wer_registry_path.c_str(), WorkItem::kWow64Default); |
| ASSERT_TRUE(value_iter.Valid()); |
| EXPECT_EQ(value_iter.ValueCount(), 5u); |
| for (; value_iter.Valid(); ++value_iter) { |
| const std::wstring value_name(value_iter.Name()); |
| if (value_name == |
| installer::GetWerHelperPath(target_path, base::Version("0.0.0.1")) |
| .value() || |
| value_name == |
| installer::GetWerHelperPath(target_path, base::Version("0.0.0.2")) |
| .value() || |
| value_name == another_path.value() || |
| value_name == path_with_invalid_version.value() || |
| value_name == path_with_another_dll.value()) { |
| continue; |
| } |
| FAIL() << "Invalid registry value encountered: " << value_name; |
| } |
| } |
| |
| work_item_list.reset(WorkItem::CreateWorkItemList()); |
| installer::AddOldWerHelperRegistrationCleanupItems(root_key, target_path, |
| work_item_list.get()); |
| ASSERT_TRUE(work_item_list->Do()); |
| { |
| base::win::RegistryValueIterator value_iter( |
| root_key, wer_registry_path.c_str(), WorkItem::kWow64Default); |
| ASSERT_TRUE(value_iter.Valid()); |
| EXPECT_EQ(value_iter.ValueCount(), 3u); |
| for (; value_iter.Valid(); ++value_iter) { |
| const std::wstring value_name(value_iter.Name()); |
| if (value_name == another_path.value() || |
| value_name == path_with_invalid_version.value() || |
| value_name == path_with_another_dll.value()) { |
| continue; |
| } |
| FAIL() << "Invalid registry value encountered: " << value_name; |
| } |
| } |
| } |