diff options
author | Greg Thompson <grt@chromium.org> | 2015-06-17 09:29:53 -0400 |
---|---|---|
committer | Greg Thompson <grt@chromium.org> | 2015-06-17 13:31:34 +0000 |
commit | 8a5ccc0e0ad6b2237b2fcfffcb0ab24fe97bc77b (patch) | |
tree | 6a7f94e803eb224fac6d4b5375238f80a4e0c70c | |
parent | 22f53c432c2154b39b027b81597db82b68c26268 (diff) | |
download | chromium_src-8a5ccc0e0ad6b2237b2fcfffcb0ab24fe97bc77b.zip chromium_src-8a5ccc0e0ad6b2237b2fcfffcb0ab24fe97bc77b.tar.gz chromium_src-8a5ccc0e0ad6b2237b2fcfffcb0ab24fe97bc77b.tar.bz2 |
Beacons for tracking default browser status.
BUG=488247
R=gab@chromium.org,wfh@chromium.org
Review URL: https://codereview.chromium.org/1146843003
Cr-Commit-Position: refs/heads/master@{#332423}
(cherry picked from commit 53cc88da9a258bc4a34c4bff50025ee044c2e64d)
TBR=grt@chromium.org
Review URL: https://codereview.chromium.org/1188963003.
Cr-Commit-Position: refs/branch-heads/2357@{#474}
Cr-Branched-From: 59d4494849b405682265ed5d3f5164573b9a939b-refs/heads/master@{#323860}
-rw-r--r-- | chrome/chrome_installer.gypi | 3 | ||||
-rw-r--r-- | chrome/chrome_installer_util.gypi | 2 | ||||
-rw-r--r-- | chrome/installer/setup/install.cc | 15 | ||||
-rw-r--r-- | chrome/installer/setup/install_worker.cc | 2 | ||||
-rw-r--r-- | chrome/installer/util/BUILD.gn | 2 | ||||
-rw-r--r-- | chrome/installer/util/beacons.cc | 151 | ||||
-rw-r--r-- | chrome/installer/util/beacons.h | 126 | ||||
-rw-r--r-- | chrome/installer/util/beacons_unittest.cc | 277 | ||||
-rw-r--r-- | chrome/installer/util/shell_util.cc | 8 | ||||
-rw-r--r-- | chrome/installer/util/shell_util.h | 7 | ||||
-rw-r--r-- | chrome/installer/util/test_app_registration_data.cc | 27 | ||||
-rw-r--r-- | chrome/installer/util/test_app_registration_data.h | 20 |
12 files changed, 633 insertions, 7 deletions
diff --git a/chrome/chrome_installer.gypi b/chrome/chrome_installer.gypi index b3e1a17..24e4bbd 100644 --- a/chrome/chrome_installer.gypi +++ b/chrome/chrome_installer.gypi @@ -97,6 +97,7 @@ 'installer/setup/compat_checks_unittest.cc', 'installer/setup/setup_constants.cc', 'installer/util/advanced_firewall_manager_win_unittest.cc', + 'installer/util/beacons_unittest.cc', 'installer/util/callback_work_item_unittest.cc', 'installer/util/channel_info_unittest.cc', 'installer/util/copy_tree_work_item_unittest.cc', @@ -135,6 +136,8 @@ 'installer/util/self_cleaning_temp_dir_unittest.cc', 'installer/util/set_reg_value_work_item_unittest.cc', 'installer/util/shell_util_unittest.cc', + 'installer/util/test_app_registration_data.cc', + 'installer/util/test_app_registration_data.h', 'installer/util/uninstall_metrics_unittest.cc', 'installer/util/wmi_unittest.cc', 'installer/util/work_item_list_unittest.cc', diff --git a/chrome/chrome_installer_util.gypi b/chrome/chrome_installer_util.gypi index 8c9d875..eb62cf4 100644 --- a/chrome/chrome_installer_util.gypi +++ b/chrome/chrome_installer_util.gypi @@ -20,6 +20,8 @@ 'installer/util/app_registration_data.h', 'installer/util/auto_launch_util.cc', 'installer/util/auto_launch_util.h', + 'installer/util/beacons.cc', + 'installer/util/beacons.h', 'installer/util/browser_distribution.cc', 'installer/util/browser_distribution.h', 'installer/util/callback_work_item.cc', diff --git a/chrome/installer/setup/install.cc b/chrome/installer/setup/install.cc index 68ecef4..58272b7 100644 --- a/chrome/installer/setup/install.cc +++ b/chrome/installer/setup/install.cc @@ -25,6 +25,7 @@ #include "chrome/installer/setup/install_worker.h" #include "chrome/installer/setup/setup_constants.h" #include "chrome/installer/util/auto_launch_util.h" +#include "chrome/installer/util/beacons.h" #include "chrome/installer/util/browser_distribution.h" #include "chrome/installer/util/create_reg_key_work_item.h" #include "chrome/installer/util/delete_after_reboot_helper.h" @@ -582,6 +583,13 @@ InstallStatus InstallOrUpdateProduct( installer_state.target_path()); } } + + if (!installer_state.system_install()) { + DCHECK_EQ(chrome_product->distribution(), + BrowserDistribution::GetDistribution()); + UpdateDefaultBrowserBeaconForPath( + installer_state.target_path().Append(installer::kChromeExe)); + } } installer_state.UpdateStage(installer::REMOVING_OLD_VERSIONS); @@ -617,6 +625,11 @@ void HandleOsUpgradeForBrowser(const installer::InstallerState& installer_state, CreateOrUpdateShortcuts( chrome_exe, chrome, prefs, level, INSTALL_SHORTCUT_REPLACE_EXISTING); RegisterChromeOnMachine(installer_state, chrome, false); + + UpdateOsUpgradeBeacon(installer_state.system_install(), + BrowserDistribution::GetDistribution()); + if (!installer_state.system_install()) + UpdateDefaultBrowserBeaconForPath(chrome_exe); } } @@ -644,6 +657,8 @@ void HandleActiveSetupForBrowser(const base::FilePath& installation_root, base::FilePath chrome_exe(installation_root.Append(kChromeExe)); CreateOrUpdateShortcuts( chrome_exe, chrome, prefs, CURRENT_USER, install_operation); + + UpdateDefaultBrowserBeaconForPath(chrome_exe); } } // namespace installer diff --git a/chrome/installer/setup/install_worker.cc b/chrome/installer/setup/install_worker.cc index 5f7449f..7b40d98 100644 --- a/chrome/installer/setup/install_worker.cc +++ b/chrome/installer/setup/install_worker.cc @@ -61,7 +61,7 @@ namespace { // on user login by way of Active Setup. Increase this value if the work done // in setup_main.cc's handling of kConfigureUserSettings changes and should be // executed again for all users. -const wchar_t kActiveSetupVersion[] = L"24,0,0,0"; +const wchar_t kActiveSetupVersion[] = L"43,0,0,0"; // Although the UUID of the ChromeFrame class is used for the "current" value, // this is done only as a convenience; there is no need for the GUID of the Low diff --git a/chrome/installer/util/BUILD.gn b/chrome/installer/util/BUILD.gn index 8dca7ce..3017bd9 100644 --- a/chrome/installer/util/BUILD.gn +++ b/chrome/installer/util/BUILD.gn @@ -79,6 +79,8 @@ static_library("util") { "app_registration_data.h", "auto_launch_util.cc", "auto_launch_util.h", + "beacons.cc", + "beacons.h", "browser_distribution.cc", "browser_distribution.h", "callback_work_item.cc", diff --git a/chrome/installer/util/beacons.cc b/chrome/installer/util/beacons.cc new file mode 100644 index 0000000..9a06503 --- /dev/null +++ b/chrome/installer/util/beacons.cc @@ -0,0 +1,151 @@ +// Copyright 2015 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 "chrome/installer/util/beacons.h" + +#include "base/win/registry.h" +#include "base/win/win_util.h" +#include "chrome/installer/util/app_registration_data.h" +#include "chrome/installer/util/browser_distribution.h" +#include "chrome/installer/util/install_util.h" +#include "chrome/installer/util/shell_util.h" + +void UpdateDefaultBrowserBeaconForPath(const base::FilePath& chrome_exe) { + // Getting Chrome's default state causes the beacon to be updated via a call + // to UpdateDefaultBrowserBeaconWithState below. + ignore_result(ShellUtil::GetChromeDefaultStateFromPath(chrome_exe)); +} + +void UpdateDefaultBrowserBeaconWithState( + const base::FilePath& chrome_exe, + BrowserDistribution* distribution, + ShellUtil::DefaultState default_state) { + const bool system_install = !InstallUtil::IsPerUserInstall(chrome_exe); + const AppRegistrationData& registration_data = + distribution->GetAppRegistrationData(); + switch (default_state) { + case ShellUtil::NOT_DEFAULT: + installer_util::MakeFirstNotDefaultBeacon(system_install, + registration_data)->Update(); + break; + case ShellUtil::IS_DEFAULT: + installer_util::MakeLastWasDefaultBeacon(system_install, + registration_data)->Update(); + installer_util::MakeFirstNotDefaultBeacon(system_install, + registration_data)->Remove(); + break; + case ShellUtil::UNKNOWN_DEFAULT: + break; + } +} + +void UpdateOsUpgradeBeacon(bool system_install, + BrowserDistribution* distribution) { + installer_util::MakeLastOsUpgradeBeacon( + system_install, distribution->GetAppRegistrationData())->Update(); +} + +namespace installer_util { + +scoped_ptr<Beacon> MakeLastOsUpgradeBeacon( + bool system_install, + const AppRegistrationData& registration_data) { + return make_scoped_ptr(new Beacon(L"LastOsUpgrade", Beacon::BeaconType::LAST, + Beacon::BeaconScope::PER_INSTALL, + system_install, registration_data)); +} + +scoped_ptr<Beacon> MakeLastWasDefaultBeacon( + bool system_install, + const AppRegistrationData& registration_data) { + return make_scoped_ptr(new Beacon(L"LastWasDefault", Beacon::BeaconType::LAST, + Beacon::BeaconScope::PER_USER, + system_install, registration_data)); +} + +scoped_ptr<Beacon> MakeFirstNotDefaultBeacon( + bool system_install, + const AppRegistrationData& registration_data) { + return make_scoped_ptr(new Beacon( + L"FirstNotDefault", Beacon::BeaconType::FIRST, + Beacon::BeaconScope::PER_USER, system_install, registration_data)); +} + +// Beacon ---------------------------------------------------------------------- + +Beacon::Beacon(base::StringPiece16 name, + BeaconType type, + BeaconScope scope, + bool system_install, + const AppRegistrationData& registration_data) + : type_(type), + root_(system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER), + scope_(scope) { + Initialize(name, system_install, registration_data); +} + +Beacon::~Beacon() { +} + +void Beacon::Update() { + const REGSAM kAccess = KEY_WOW64_32KEY | KEY_QUERY_VALUE | KEY_SET_VALUE; + base::win::RegKey key; + + // Nothing to update if the key couldn't be created. This should only be the + // case for a developer build. + if (key.Create(root_, key_path_.c_str(), kAccess) != ERROR_SUCCESS) + return; + + // Nothing to do if this beacon is tracking the first occurrence of an event + // that has already been recorded. + if (type_ == BeaconType::FIRST && key.HasValue(value_name_.c_str())) + return; + + int64_t now(base::Time::Now().ToInternalValue()); + key.WriteValue(value_name_.c_str(), &now, sizeof(now), REG_QWORD); +} + +void Beacon::Remove() { + const REGSAM kAccess = KEY_WOW64_32KEY | KEY_SET_VALUE; + base::win::RegKey key; + + if (key.Open(root_, key_path_.c_str(), kAccess) == ERROR_SUCCESS) + key.DeleteValue(value_name_.c_str()); +} + +base::Time Beacon::Get() { + const REGSAM kAccess = KEY_WOW64_32KEY | KEY_QUERY_VALUE; + base::win::RegKey key; + int64_t now; + + if (key.Open(root_, key_path_.c_str(), kAccess) != ERROR_SUCCESS || + key.ReadInt64(value_name_.c_str(), &now) != ERROR_SUCCESS) { + return base::Time(); + } + + return base::Time::FromInternalValue(now); +} + +void Beacon::Initialize(base::StringPiece16 name, + bool system_install, + const AppRegistrationData& registration_data) { + // When possible, beacons are located in the app's ClientState key. Per-user + // beacons for a per-machine install are located in a beacon-specific sub-key + // of the app's ClientStateMedium key. + if (!system_install || scope_ == BeaconScope::PER_INSTALL) { + key_path_ = registration_data.GetStateKey(); + value_name_ = name.as_string(); + } else { + key_path_ = registration_data.GetStateMediumKey(); + key_path_.push_back(L'\\'); + key_path_.append(name.data(), name.size()); + // This should never fail. If it does, the beacon will be written in the + // key's default value, which is okay since the majority case is likely a + // machine with a single user. + if (!base::win::GetUserSidString(&value_name_)) + NOTREACHED(); + } +} + +} // namespace installer_util diff --git a/chrome/installer/util/beacons.h b/chrome/installer/util/beacons.h new file mode 100644 index 0000000..bec393e --- /dev/null +++ b/chrome/installer/util/beacons.h @@ -0,0 +1,126 @@ +// Copyright 2015 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. + +#ifndef CHROME_INSTALLER_UTIL_BEACONS_H_ +#define CHROME_INSTALLER_UTIL_BEACONS_H_ + +#include <windows.h> + +#include "base/macros.h" +#include "base/memory/scoped_ptr.h" +#include "base/strings/string16.h" +#include "base/strings/string_piece.h" +#include "base/time/time.h" +#include "chrome/installer/util/shell_util.h" + +class AppRegistrationData; +class BrowserDistribution; + +namespace base { +class FilePath; +} + +// Checks the default state of the browser for the current user and updates the +// appropriate beacon (last was default or first not default). +void UpdateDefaultBrowserBeaconForPath(const base::FilePath& chrome_exe); + +// Updates the last was default or first not default beacon for the current user +// based on |default_state|. +void UpdateDefaultBrowserBeaconWithState(const base::FilePath& chrome_exe, + BrowserDistribution* distribution, + ShellUtil::DefaultState default_state); + +// Updates the last OS upgrade beacon for the install. +void UpdateOsUpgradeBeacon(bool system_install, + BrowserDistribution* distribution); + +namespace installer_util { + +class Beacon; + +// Returns a Beacon representing the last time the machine's OS was ugpraded. +scoped_ptr<Beacon> MakeLastOsUpgradeBeacon( + bool system_install, + const AppRegistrationData& registration_data); + +// Returns a Beacon representing the last time Chrome was the user's default +// browser. +scoped_ptr<Beacon> MakeLastWasDefaultBeacon( + bool system_install, + const AppRegistrationData& registration_data); + +// Returns a Beacon representing the first time Chrome was not the user's +// default browser. +scoped_ptr<Beacon> MakeFirstNotDefaultBeacon( + bool system_install, + const AppRegistrationData& registration_data); + +// A named beacon stored in the registry representing the first or last time at +// which some event took place. A beacon may apply to a per-user event or a +// per-install event. In general, beacons should be created via factory methods +// such as those above. +class Beacon { + public: + enum class BeaconType { + // A beacon that marks the first occurrence of an event. + FIRST, + // A beacon that marks the last occurrence of an event. + LAST, + }; + + enum class BeaconScope { + // A beacon that applies to a per-user event. + PER_USER, + // A beacon that applies to a per-install event. + PER_INSTALL, + }; + + // Creates a beacon named |name| for the product identified by + // |registration_data| installed as either a per-user or a per-machine app + // according to |system_install|. + Beacon(base::StringPiece16 name, + BeaconType type, + BeaconScope scope, + bool system_install, + const AppRegistrationData& registration_data); + ~Beacon(); + + // Updates the beacon. For a type LAST beacon, the current time is written + // unconditionally. For a type FIRST beacon, the beacon is only updated if it + // does not already exist. + void Update(); + + // Removes the beacon. + void Remove(); + + // Returns the beacon's value or a null time if not found. + base::Time Get(); + + private: + // Initializes the key_path_ and value_name_ fields of the beacon. + void Initialize(base::StringPiece16 name, + bool system_install, + const AppRegistrationData& registration_data); + + // The type of beacon. + const BeaconType type_; + + // The root key in the registry where this beacon is stored. + const HKEY root_; + + // The scope of the beacon. + const BeaconScope scope_; + + // The path to the registry key holding the beacon. + base::string16 key_path_; + + // The name of the registry value holding the beacon. + base::string16 value_name_; + + DISALLOW_COPY_AND_ASSIGN(Beacon); +}; + +} // namespace installer_util + +#endif // CHROME_INSTALLER_UTIL_BEACONS_H_ diff --git a/chrome/installer/util/beacons_unittest.cc b/chrome/installer/util/beacons_unittest.cc new file mode 100644 index 0000000..2465dd3 --- /dev/null +++ b/chrome/installer/util/beacons_unittest.cc @@ -0,0 +1,277 @@ +// Copyright 2015 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 "chrome/installer/util/beacons.h" + +#include "base/base_paths.h" +#include "base/memory/scoped_vector.h" +#include "base/path_service.h" +#include "base/test/scoped_path_override.h" +#include "base/test/test_reg_util_win.h" +#include "base/test/test_timeouts.h" +#include "base/threading/platform_thread.h" +#include "base/win/registry.h" +#include "base/win/win_util.h" +#include "chrome/installer/util/browser_distribution.h" +#include "chrome/installer/util/test_app_registration_data.h" +#include "chrome/installer/util/util_constants.h" +#include "testing/gtest/include/gtest/gtest.h" + +using ::testing::Bool; +using ::testing::Combine; +using ::testing::Values; +using BeaconType = installer_util::Beacon::BeaconType; +using BeaconScope = installer_util::Beacon::BeaconScope; + +namespace installer_util { + +// A test fixture that exercises a beacon. +class BeaconTest : public ::testing::TestWithParam< + ::testing::tuple<BeaconType, BeaconScope, bool>> { + protected: + static const base::char16 kBeaconName[]; + + BeaconTest() + : beacon_type_(::testing::get<0>(GetParam())), + beacon_scope_(::testing::get<1>(GetParam())), + system_install_(::testing::get<2>(GetParam())), + beacon_(kBeaconName, + beacon_type_, + beacon_scope_, + system_install_, + app_registration_data_) { + // Override the registry so that tests can freely push state to it. + registry_override_manager_.OverrideRegistry(HKEY_CURRENT_USER); + registry_override_manager_.OverrideRegistry(HKEY_LOCAL_MACHINE); + } + + TestAppRegistrationData app_registration_data_; + BeaconType beacon_type_; + BeaconScope beacon_scope_; + bool system_install_; + Beacon beacon_; + + private: + registry_util::RegistryOverrideManager registry_override_manager_; +}; + +// static +const base::char16 BeaconTest::kBeaconName[] = L"TestBeacon"; + +// Nothing in the regsitry, so the beacon should not exist. +TEST_P(BeaconTest, GetNonExistant) { + ASSERT_TRUE(beacon_.Get().is_null()); +} + +// Updating and then getting the beacon should return a value, and that it is +// within range. +TEST_P(BeaconTest, UpdateAndGet) { + base::Time before(base::Time::Now()); + beacon_.Update(); + base::Time after(base::Time::Now()); + base::Time beacon_time(beacon_.Get()); + ASSERT_FALSE(beacon_time.is_null()); + ASSERT_LE(before, beacon_time); + ASSERT_GE(after, beacon_time); +} + +// Tests that updating a first beacon only updates it the first time, but doing +// so for a last beacon always updates. +TEST_P(BeaconTest, UpdateTwice) { + beacon_.Update(); + base::Time beacon_time(beacon_.Get()); + + base::PlatformThread::Sleep(TestTimeouts::tiny_timeout()); + + beacon_.Update(); + if (beacon_type_ == BeaconType::FIRST) { + ASSERT_EQ(beacon_time, beacon_.Get()); + } else { + ASSERT_NE(beacon_time, beacon_.Get()); + } +} + +// Tests that the beacon is written into the proper location in the registry. +TEST_P(BeaconTest, Location) { + beacon_.Update(); + HKEY right_root = system_install_ ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER; + HKEY wrong_root = system_install_ ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE; + base::string16 right_key; + base::string16 wrong_key; + base::string16 value_name; + + if (beacon_scope_ == BeaconScope::PER_INSTALL || !system_install_) { + value_name = kBeaconName; + right_key = app_registration_data_.GetStateKey(); + wrong_key = app_registration_data_.GetStateMediumKey(); + } else { + ASSERT_TRUE(base::win::GetUserSidString(&value_name)); + right_key = + app_registration_data_.GetStateMediumKey() + L"\\" + kBeaconName; + wrong_key = app_registration_data_.GetStateKey(); + } + + // Keys should not exist in the wrong root or in the right root but wrong key. + ASSERT_FALSE(base::win::RegKey(wrong_root, right_key.c_str(), + KEY_READ).Valid()) << right_key; + ASSERT_FALSE(base::win::RegKey(wrong_root, wrong_key.c_str(), + KEY_READ).Valid()) << wrong_key; + ASSERT_FALSE(base::win::RegKey(right_root, wrong_key.c_str(), + KEY_READ).Valid()) << wrong_key; + // The right key should exist. + base::win::RegKey key(right_root, right_key.c_str(), KEY_READ); + ASSERT_TRUE(key.Valid()) << right_key; + // And should have the value. + ASSERT_TRUE(key.HasValue(value_name.c_str())) << value_name; +} + +// Run the tests for all combinations of beacon type, scope, and install level. +INSTANTIATE_TEST_CASE_P(BeaconTest, + BeaconTest, + Combine(Values(BeaconType::FIRST, BeaconType::LAST), + Values(BeaconScope::PER_USER, + BeaconScope::PER_INSTALL), + Bool())); + +enum class DistributionVariant { + SYSTEM_LEVEL, + USER_LEVEL, + SXS, +}; + +class DefaultBrowserBeaconTest + : public ::testing::TestWithParam<DistributionVariant> { + protected: + using Super = ::testing::TestWithParam<DistributionVariant>; + + DefaultBrowserBeaconTest() + : system_install_(GetParam() == DistributionVariant::SYSTEM_LEVEL), + chrome_sxs_(GetParam() == DistributionVariant::SXS), + chrome_exe_(GetChromePathForParams()), + distribution_(nullptr) {} + + void SetUp() override { + Super::SetUp(); + + // Override FILE_EXE so that various InstallUtil functions will consider + // this to be a user/system Chrome or Chrome SxS. + path_overrides_.push_back(new base::ScopedPathOverride( + base::FILE_EXE, chrome_exe_, true /* is_absolute */, + false /* !create */)); + + // Override these paths with their own values so that they can be found + // after the registry override manager is in place. Getting them would + // otherwise fail since the underlying calls to the OS need to see the real + // contents of the registry. + static const int kPathKeys[] = { + base::DIR_PROGRAM_FILES, + base::DIR_PROGRAM_FILESX86, + base::DIR_LOCAL_APP_DATA, + }; + for (int key : kPathKeys) { + base::FilePath temp; + PathService::Get(key, &temp); + path_overrides_.push_back(new base::ScopedPathOverride(key, temp)); + } + + // Override the registry so that tests can freely push state to it. + registry_override_manager_.OverrideRegistry(HKEY_CURRENT_USER); + registry_override_manager_.OverrideRegistry(HKEY_LOCAL_MACHINE); + + distribution_ = BrowserDistribution::GetDistribution(); + } + + bool system_install_; + bool chrome_sxs_; + base::FilePath chrome_exe_; + BrowserDistribution* distribution_; + + private: + base::FilePath GetChromePathForParams() const { + base::FilePath chrome_exe; + int dir_key = base::DIR_LOCAL_APP_DATA; + + if (system_install_) { +#if defined(_WIN64) + static const int kSystemKey = base::DIR_PROGRAM_FILESX86; +#else + static const int kSystemKey = base::DIR_PROGRAM_FILES; +#endif + dir_key = kSystemKey; + } + PathService::Get(dir_key, &chrome_exe); +#if defined(GOOGLE_CHROME_BUILD) + chrome_exe = chrome_exe.Append(installer::kGoogleChromeInstallSubDir1); + if (chrome_sxs_) { + chrome_exe = chrome_exe.Append( + base::string16(installer::kGoogleChromeInstallSubDir2) + + installer::kSxSSuffix); + } else { + chrome_exe = chrome_exe.Append(installer::kGoogleChromeInstallSubDir2); + } +#else + chrome_exe = chrome_exe.AppendASCII("Chromium"); +#endif + chrome_exe = chrome_exe.Append(installer::kInstallBinaryDir); + return chrome_exe.Append(installer::kChromeExe); + } + + ScopedVector<base::ScopedPathOverride> path_overrides_; + registry_util::RegistryOverrideManager registry_override_manager_; +}; + +// Tests that the default browser beacons work as expected. +TEST_P(DefaultBrowserBeaconTest, All) { + scoped_ptr<Beacon> last_was_default(MakeLastWasDefaultBeacon( + system_install_, distribution_->GetAppRegistrationData())); + scoped_ptr<Beacon> first_not_default(MakeFirstNotDefaultBeacon( + system_install_, distribution_->GetAppRegistrationData())); + + ASSERT_TRUE(last_was_default->Get().is_null()); + ASSERT_TRUE(first_not_default->Get().is_null()); + + // Chrome is not default. + UpdateDefaultBrowserBeaconWithState(chrome_exe_, distribution_, + ShellUtil::NOT_DEFAULT); + ASSERT_TRUE(last_was_default->Get().is_null()); + ASSERT_FALSE(first_not_default->Get().is_null()); + + // Then it is. + UpdateDefaultBrowserBeaconWithState(chrome_exe_, distribution_, + ShellUtil::IS_DEFAULT); + ASSERT_FALSE(last_was_default->Get().is_null()); + ASSERT_TRUE(first_not_default->Get().is_null()); + + // It still is. + UpdateDefaultBrowserBeaconWithState(chrome_exe_, distribution_, + ShellUtil::IS_DEFAULT); + ASSERT_FALSE(last_was_default->Get().is_null()); + ASSERT_TRUE(first_not_default->Get().is_null()); + + // Now it's not again. + UpdateDefaultBrowserBeaconWithState(chrome_exe_, distribution_, + ShellUtil::NOT_DEFAULT); + ASSERT_FALSE(last_was_default->Get().is_null()); + ASSERT_FALSE(first_not_default->Get().is_null()); + + // And it still isn't. + UpdateDefaultBrowserBeaconWithState(chrome_exe_, distribution_, + ShellUtil::NOT_DEFAULT); + ASSERT_FALSE(last_was_default->Get().is_null()); + ASSERT_FALSE(first_not_default->Get().is_null()); +} + +INSTANTIATE_TEST_CASE_P(SystemLevelChrome, + DefaultBrowserBeaconTest, + Values(DistributionVariant::SYSTEM_LEVEL)); +INSTANTIATE_TEST_CASE_P(UserLevelChrome, + DefaultBrowserBeaconTest, + Values(DistributionVariant::USER_LEVEL)); +#if 0 && defined(GOOGLE_CHROME_BUILD) +// Disabled for now since InstallUtil::IsChromeSxSProcess makes this impossible. +INSTANTIATE_TEST_CASE_P(ChromeSxS, DefaultBrowserBeaconTest, + Values(DistributionVariant::SXS)); +#endif + +} // namespace installer_util diff --git a/chrome/installer/util/shell_util.cc b/chrome/installer/util/shell_util.cc index 469d10d..917278e 100644 --- a/chrome/installer/util/shell_util.cc +++ b/chrome/installer/util/shell_util.cc @@ -42,6 +42,7 @@ #include "base/win/windows_version.h" #include "chrome/common/chrome_constants.h" #include "chrome/common/chrome_switches.h" +#include "chrome/installer/util/beacons.h" #include "chrome/installer/util/browser_distribution.h" #include "chrome/installer/util/install_util.h" #include "chrome/installer/util/l10n_string_util.h" @@ -1933,9 +1934,10 @@ ShellUtil::DefaultState ShellUtil::GetChromeDefaultStateFromPath( // flag. There is doubtless some other key we can hook into to cause "Repair" // to show up in Add/Remove programs for us. static const wchar_t* const kChromeProtocols[] = { L"http", L"https" }; - return ProbeProtocolHandlers(chrome_exe, - kChromeProtocols, - arraysize(kChromeProtocols)); + DefaultState default_state = ProbeProtocolHandlers( + chrome_exe, kChromeProtocols, arraysize(kChromeProtocols)); + UpdateDefaultBrowserBeaconWithState(chrome_exe, distribution, default_state); + return default_state; } ShellUtil::DefaultState ShellUtil::GetChromeDefaultProtocolClientState( diff --git a/chrome/installer/util/shell_util.h b/chrome/installer/util/shell_util.h index 27e51e4..80521d7 100644 --- a/chrome/installer/util/shell_util.h +++ b/chrome/installer/util/shell_util.h @@ -411,11 +411,12 @@ class ShellUtil { // Windows prior to Windows 8. static bool CanMakeChromeDefaultUnattended(); - // Returns the DefaultState of Chrome for HTTP and HTTPS. + // Returns the DefaultState of Chrome for HTTP and HTTPS and updates the + // default browser beacons as appropriate. static DefaultState GetChromeDefaultState(); - // Returns the DefaultState of the Chrome instance with the specified path - // for HTTP and HTTPs. + // Returns the DefaultState of the Chrome instance with the specified path for + // HTTP and HTTPs and updates the default browser beacons as appropriate. static DefaultState GetChromeDefaultStateFromPath( const base::FilePath& chrome_exe); diff --git a/chrome/installer/util/test_app_registration_data.cc b/chrome/installer/util/test_app_registration_data.cc new file mode 100644 index 0000000..b370df3 --- /dev/null +++ b/chrome/installer/util/test_app_registration_data.cc @@ -0,0 +1,27 @@ +// Copyright 2015 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 "chrome/installer/util/test_app_registration_data.h" + +TestAppRegistrationData::TestAppRegistrationData() { +} + +TestAppRegistrationData::~TestAppRegistrationData() { +} + +base::string16 TestAppRegistrationData::GetAppGuid() const { + return L"test_app_guid"; +} + +base::string16 TestAppRegistrationData::GetStateKey() const { + return L"Software\\Chromium\\ClientState\\test_app_guid"; +} + +base::string16 TestAppRegistrationData::GetStateMediumKey() const { + return L"Software\\Chromium\\ClientStateMedium\\test_app_guid"; +} + +base::string16 TestAppRegistrationData::GetVersionKey() const { + return L"Software\\Chromium\\Clients\\test_app_guid"; +} diff --git a/chrome/installer/util/test_app_registration_data.h b/chrome/installer/util/test_app_registration_data.h new file mode 100644 index 0000000..9b5fb86 --- /dev/null +++ b/chrome/installer/util/test_app_registration_data.h @@ -0,0 +1,20 @@ +// Copyright 2015 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. + +#ifndef CHROME_INSTALLER_UTIL_TEST_APP_REGISTRATION_DATA_H_ +#define CHROME_INSTALLER_UTIL_TEST_APP_REGISTRATION_DATA_H_ + +#include "chrome/installer/util/app_registration_data.h" + +class TestAppRegistrationData : public AppRegistrationData { + public: + TestAppRegistrationData(); + ~TestAppRegistrationData() override; + base::string16 GetAppGuid() const override; + base::string16 GetStateKey() const override; + base::string16 GetStateMediumKey() const override; + base::string16 GetVersionKey() const override; +}; + +#endif // CHROME_INSTALLER_UTIL_TEST_APP_REGISTRATION_DATA_H_ |