| // 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/browser/sync/test/integration/sync_app_helper.h" |
| |
| #include <list> |
| #include <map> |
| #include <memory> |
| |
| #include "chrome/browser/extensions/extension_service.h" |
| #include "chrome/browser/extensions/extension_util.h" |
| #include "chrome/browser/extensions/sync/extension_sync_util.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/sync/test/integration/extensions_helper.h" |
| #include "chrome/browser/sync/test/integration/sync_datatype_helper.h" |
| #include "chrome/browser/sync/test/integration/sync_extension_helper.h" |
| #include "chrome/browser/sync/test/integration/sync_test.h" |
| #include "chrome/common/extensions/extension_constants.h" |
| #include "chrome/common/extensions/manifest_handlers/app_launch_info.h" |
| #include "components/crx_file/id_util.h" |
| #include "extensions/browser/app_sorting.h" |
| #include "extensions/browser/extension_prefs.h" |
| #include "extensions/browser/extension_registry.h" |
| #include "extensions/browser/extension_system.h" |
| #include "extensions/browser/launch_util.h" |
| #include "extensions/browser/pending_extension_manager.h" |
| #include "extensions/common/extension_set.h" |
| |
| using extensions::AppSorting; |
| using extensions::ExtensionPrefs; |
| using extensions::ExtensionSystem; |
| |
| namespace { |
| |
| struct AppState { |
| AppState(); |
| ~AppState(); |
| bool IsValid() const; |
| bool Equals(const AppState& other) const; |
| |
| syncer::StringOrdinal app_launch_ordinal; |
| syncer::StringOrdinal page_ordinal; |
| extensions::LaunchType launch_type = extensions::LAUNCH_TYPE_INVALID; |
| GURL launch_web_url; |
| std::string description; |
| std::string name; |
| }; |
| |
| using AppStateMap = std::map<std::string, AppState>; |
| |
| AppState::AppState() = default; |
| |
| AppState::~AppState() = default; |
| |
| bool AppState::IsValid() const { |
| return page_ordinal.IsValid() && app_launch_ordinal.IsValid(); |
| } |
| |
| bool AppState::Equals(const AppState& other) const { |
| return app_launch_ordinal.Equals(other.app_launch_ordinal) && |
| page_ordinal.Equals(other.page_ordinal) && |
| launch_type == other.launch_type && |
| launch_web_url == other.launch_web_url && |
| description == other.description && name == other.name; |
| } |
| |
| // Load all the app specific values for |id| into |app_state|. |
| void LoadApp(content::BrowserContext* context, |
| const std::string& id, |
| AppState* app_state) { |
| ExtensionPrefs* prefs = ExtensionPrefs::Get(context); |
| AppSorting* app_sorting = ExtensionSystem::Get(context)->app_sorting(); |
| app_state->app_launch_ordinal = app_sorting->GetAppLaunchOrdinal(id); |
| app_state->page_ordinal = app_sorting->GetPageOrdinal(id); |
| app_state->launch_type = extensions::GetLaunchTypePrefValue(prefs, id); |
| extensions::ExtensionRegistry* registry = |
| extensions::ExtensionRegistry::Get(context); |
| const extensions::Extension* extension = registry->GetInstalledExtension(id); |
| // GetInstalledExtension(id) returns null if |id| is for a pending extension. |
| // In case of running tests against real backend servers, pending apps won't |
| // be installed. |
| if (extension) { |
| app_state->launch_web_url = |
| extensions::AppLaunchInfo::GetLaunchWebURL(extension); |
| app_state->description = extension->description(); |
| app_state->name = extension->name(); |
| } |
| } |
| |
| // Returns a map from |profile|'s installed extensions to their state. |
| AppStateMap GetAppStates(Profile* profile) { |
| AppStateMap app_state_map; |
| |
| const extensions::ExtensionSet extensions = |
| extensions::ExtensionRegistry::Get(profile) |
| ->GenerateInstalledExtensionsSet(); |
| for (const auto& extension : extensions) { |
| if (extension->is_app() && |
| extensions::sync_util::ShouldSync(profile, extension.get())) { |
| const std::string& id = extension->id(); |
| LoadApp(profile, id, &(app_state_map[id])); |
| } |
| } |
| |
| const extensions::PendingExtensionManager* pending_extension_manager = |
| extensions::PendingExtensionManager::Get(profile); |
| |
| std::list<std::string> pending_crx_ids = |
| pending_extension_manager->GetPendingIdsForUpdateCheck(); |
| |
| for (const std::string& id : pending_crx_ids) { |
| LoadApp(profile, id, &(app_state_map[id])); |
| } |
| |
| return app_state_map; |
| } |
| |
| } // namespace |
| |
| SyncAppHelper* SyncAppHelper::GetInstance() { |
| SyncAppHelper* instance = base::Singleton<SyncAppHelper>::get(); |
| instance->SetupIfNecessary(sync_datatype_helper::test()); |
| return instance; |
| } |
| |
| void SyncAppHelper::SetupIfNecessary(SyncTest* test) { |
| if (setup_completed_) { |
| return; |
| } |
| |
| for (int i = 0; i < test->num_clients(); ++i) { |
| extensions::ExtensionSystem::Get(test->GetProfile(i)) |
| ->InitForRegularProfile(true /* extensions_enabled */); |
| } |
| if (test->UseVerifier()) { |
| extensions::ExtensionSystem::Get(test->verifier()) |
| ->InitForRegularProfile(true /* extensions_enabled */); |
| } |
| |
| setup_completed_ = true; |
| } |
| |
| bool SyncAppHelper::AppStatesMatch(Profile* profile1, Profile* profile2) { |
| if (!SyncExtensionHelper::GetInstance()->ExtensionStatesMatch(profile1, |
| profile2)) { |
| return false; |
| } |
| |
| const AppStateMap& state_map1 = GetAppStates(profile1); |
| const AppStateMap& state_map2 = GetAppStates(profile2); |
| if (state_map1.size() != state_map2.size()) { |
| DVLOG(2) << "Number of Apps for profile " << profile1->GetDebugName() |
| << " does not match profile " << profile2->GetDebugName(); |
| return false; |
| } |
| |
| auto it1 = state_map1.begin(); |
| auto it2 = state_map2.begin(); |
| while (it1 != state_map1.end()) { |
| const auto& [app_id1, app_state1] = *it1; |
| const auto& [app_id2, app_state2] = *it2; |
| if (app_id1 != app_id2) { |
| DVLOG(2) << "Apps for profile " << profile1->GetDebugName() |
| << " do not match profile " << profile2->GetDebugName(); |
| return false; |
| } else if (!app_state1.IsValid()) { |
| DVLOG(2) << "Apps for profile " << profile1->GetDebugName() |
| << " are not valid."; |
| return false; |
| } else if (!app_state2.IsValid()) { |
| DVLOG(2) << "Apps for profile " << profile2->GetDebugName() |
| << " are not valid."; |
| return false; |
| } else if (!app_state1.Equals(app_state2)) { |
| // If this test is run against real backend servers then we do not expect |
| // to install pending apps. So, we don't check equality of AppStates of |
| // each app per profile. |
| DVLOG(2) << "App states for profile " << profile1->GetDebugName() |
| << " do not match profile " << profile2->GetDebugName(); |
| return false; |
| } |
| ++it1; |
| ++it2; |
| } |
| |
| return true; |
| } |
| |
| syncer::StringOrdinal SyncAppHelper::GetPageOrdinalForApp( |
| Profile* profile, |
| const std::string& name) { |
| return ExtensionSystem::Get(profile)->app_sorting()->GetPageOrdinal( |
| crx_file::id_util::GenerateId(name)); |
| } |
| |
| void SyncAppHelper::SetPageOrdinalForApp( |
| Profile* profile, |
| const std::string& name, |
| const syncer::StringOrdinal& page_ordinal) { |
| ExtensionSystem::Get(profile)->app_sorting()->SetPageOrdinal( |
| crx_file::id_util::GenerateId(name), page_ordinal); |
| } |
| |
| syncer::StringOrdinal SyncAppHelper::GetAppLaunchOrdinalForApp( |
| Profile* profile, |
| const std::string& name) { |
| return ExtensionSystem::Get(profile)->app_sorting()->GetAppLaunchOrdinal( |
| crx_file::id_util::GenerateId(name)); |
| } |
| |
| void SyncAppHelper::SetAppLaunchOrdinalForApp( |
| Profile* profile, |
| const std::string& name, |
| const syncer::StringOrdinal& app_launch_ordinal) { |
| ExtensionSystem::Get(profile)->app_sorting()->SetAppLaunchOrdinal( |
| crx_file::id_util::GenerateId(name), app_launch_ordinal); |
| } |
| |
| void SyncAppHelper::FixNTPOrdinalCollisions(Profile* profile) { |
| ExtensionSystem::Get(profile)->app_sorting()->FixNTPOrdinalCollisions(); |
| } |
| |
| SyncAppHelper::SyncAppHelper() = default; |
| |
| SyncAppHelper::~SyncAppHelper() = default; |