// Copyright 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 "chrome/browser/extensions/extension_sync_service.h" #include "base/basictypes.h" #include "base/strings/utf_string_conversions.h" #include "base/threading/thread_restrictions.h" #include "chrome/browser/extensions/bookmark_app_helper.h" #include "chrome/browser/extensions/extension_service.h" #include "chrome/browser/extensions/extension_sync_data.h" #include "chrome/browser/extensions/extension_sync_service_factory.h" #include "chrome/browser/extensions/extension_util.h" #include "chrome/browser/extensions/launch_util.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/sync/glue/sync_start_util.h" #include "chrome/common/extensions/extension_constants.h" #include "chrome/common/extensions/sync_helper.h" #include "chrome/common/web_application_info.h" #include "components/sync_driver/sync_prefs.h" #include "extensions/browser/app_sorting.h" #include "extensions/browser/extension_prefs.h" #include "extensions/browser/extension_registry.h" #include "extensions/browser/extension_util.h" #include "extensions/browser/uninstall_reason.h" #include "extensions/common/extension.h" #include "extensions/common/extension_set.h" #include "extensions/common/image_util.h" #include "sync/api/sync_change.h" #include "sync/api/sync_error_factory.h" using extensions::Extension; using extensions::ExtensionPrefs; using extensions::ExtensionRegistry; using extensions::ExtensionSet; using extensions::ExtensionSyncData; using extensions::SyncBundle; namespace { void OnWebApplicationInfoLoaded( WebApplicationInfo synced_info, base::WeakPtr extension_service, const WebApplicationInfo& loaded_info) { DCHECK_EQ(synced_info.app_url, loaded_info.app_url); if (!extension_service) return; // Use the old icons if they exist. synced_info.icons = loaded_info.icons; CreateOrUpdateBookmarkApp(extension_service.get(), &synced_info); } // Returns the pref value for "all urls enabled" for the given extension id. ExtensionSyncData::OptionalBoolean GetAllowedOnAllUrlsOptionalBoolean( const std::string& extension_id, content::BrowserContext* context) { bool allowed_on_all_urls = extensions::util::AllowedScriptingOnAllUrls(extension_id, context); // If the extension is not allowed on all urls (which is not the default), // then we have to sync the preference. if (!allowed_on_all_urls) return ExtensionSyncData::BOOLEAN_FALSE; // If the user has explicitly set a value, then we sync it. if (extensions::util::HasSetAllowedScriptingOnAllUrls(extension_id, context)) return ExtensionSyncData::BOOLEAN_TRUE; // Otherwise, unset. return ExtensionSyncData::BOOLEAN_UNSET; } // Returns true if the sync type of |extension| matches |type|. bool IsCorrectSyncType(const Extension& extension, syncer::ModelType type) { return (type == syncer::EXTENSIONS && extension.is_extension()) || (type == syncer::APPS && extension.is_app()); } syncer::SyncDataList ToSyncerSyncDataList( const std::vector& data) { syncer::SyncDataList result; result.reserve(data.size()); for (const ExtensionSyncData& item : data) result.push_back(item.GetSyncData()); return result; } } // namespace ExtensionSyncService::ExtensionSyncService(Profile* profile, ExtensionPrefs* extension_prefs, ExtensionService* extension_service) : profile_(profile), extension_prefs_(extension_prefs), extension_service_(extension_service) { SetSyncStartFlare(sync_start_util::GetFlareForSyncableService( profile_->GetPath())); extension_prefs_->app_sorting()->SetExtensionSyncService(this); } ExtensionSyncService::~ExtensionSyncService() {} // static ExtensionSyncService* ExtensionSyncService::Get( content::BrowserContext* context) { return ExtensionSyncServiceFactory::GetForBrowserContext(context); } void ExtensionSyncService::SyncUninstallExtension( const extensions::Extension& extension) { if (!extensions::util::ShouldSync(&extension, profile_)) return; // TODO(tim): If we get here and IsSyncing is false, this will cause // "back from the dead" style bugs, because sync will add-back the extension // that was uninstalled here when MergeDataAndStartSyncing is called. // See crbug.com/256795. syncer::ModelType type = extension.is_app() ? syncer::APPS : syncer::EXTENSIONS; SyncBundle* bundle = GetSyncBundle(type); if (bundle->IsSyncing()) { bundle->PushSyncDeletion(extension.id(), CreateSyncData(extension).GetSyncData()); } else if (extension_service_->is_ready() && !flare_.is_null()) { flare_.Run(type); // Tell sync to start ASAP. } } void ExtensionSyncService::SyncExtensionChangeIfNeeded( const Extension& extension) { if (!extensions::util::ShouldSync(&extension, profile_)) return; syncer::ModelType type = extension.is_app() ? syncer::APPS : syncer::EXTENSIONS; SyncBundle* bundle = GetSyncBundle(type); if (bundle->IsSyncing()) { bundle->PushSyncAddOrUpdate(extension.id(), CreateSyncData(extension).GetSyncData()); DCHECK(!ExtensionPrefs::Get(profile_)->NeedsSync(extension.id())); } else { ExtensionPrefs::Get(profile_)->SetNeedsSync(extension.id(), true); if (extension_service_->is_ready() && !flare_.is_null()) flare_.Run(type); // Tell sync to start ASAP. } } syncer::SyncMergeResult ExtensionSyncService::MergeDataAndStartSyncing( syncer::ModelType type, const syncer::SyncDataList& initial_sync_data, scoped_ptr sync_processor, scoped_ptr sync_error_factory) { CHECK(sync_processor.get()); LOG_IF(FATAL, type != syncer::EXTENSIONS && type != syncer::APPS) << "Got " << type << " ModelType"; SyncBundle* bundle = GetSyncBundle(type); bundle->StartSyncing(sync_processor.Pass()); // Apply the initial sync data, filtering out any items where we have more // recent local changes. Also tell the SyncBundle the extension IDs. for (const syncer::SyncData& sync_data : initial_sync_data) { scoped_ptr extension_sync_data( ExtensionSyncData::CreateFromSyncData(sync_data)); // If the extension has local state that needs to be synced, ignore this // change (we assume the local state is more recent). if (extension_sync_data && !ExtensionPrefs::Get(profile_)->NeedsSync(extension_sync_data->id())) { ApplySyncData(*extension_sync_data); } } // Now push those local changes to sync. // TODO(treib,kalman): We should only have to send out changes for extensions // which have NeedsSync set (i.e. |GetLocalSyncDataList(type, false)|). That // makes some sync_integration_tests fail though - figure out why and fix it! std::vector data_list = GetLocalSyncDataList(type, true); bundle->PushSyncDataList(ToSyncerSyncDataList(data_list)); for (const ExtensionSyncData& data : data_list) ExtensionPrefs::Get(profile_)->SetNeedsSync(data.id(), false); if (type == syncer::APPS) extension_prefs_->app_sorting()->FixNTPOrdinalCollisions(); return syncer::SyncMergeResult(type); } void ExtensionSyncService::StopSyncing(syncer::ModelType type) { GetSyncBundle(type)->Reset(); } syncer::SyncDataList ExtensionSyncService::GetAllSyncData( syncer::ModelType type) const { const SyncBundle* bundle = GetSyncBundle(type); if (!bundle->IsSyncing()) return syncer::SyncDataList(); std::vector sync_data_list = GetLocalSyncDataList(type, true); // Add pending data (where the local extension is either not installed yet or // outdated). std::vector pending_extensions = bundle->GetPendingData(); sync_data_list.insert(sync_data_list.begin(), pending_extensions.begin(), pending_extensions.end()); return ToSyncerSyncDataList(sync_data_list); } syncer::SyncError ExtensionSyncService::ProcessSyncChanges( const tracked_objects::Location& from_here, const syncer::SyncChangeList& change_list) { for (const syncer::SyncChange& sync_change : change_list) { scoped_ptr extension_sync_data( ExtensionSyncData::CreateFromSyncChange(sync_change)); if (extension_sync_data) ApplySyncData(*extension_sync_data); } extension_prefs_->app_sorting()->FixNTPOrdinalCollisions(); return syncer::SyncError(); } ExtensionSyncData ExtensionSyncService::CreateSyncData( const extensions::Extension& extension) const { bool enabled = extension_service_->IsExtensionEnabled(extension.id()); int disable_reasons = extension_prefs_->GetDisableReasons(extension.id()); bool incognito_enabled = extensions::util::IsIncognitoEnabled(extension.id(), profile_); bool remote_install = extension_prefs_->HasDisableReason(extension.id(), Extension::DISABLE_REMOTE_INSTALL); ExtensionSyncData::OptionalBoolean allowed_on_all_url = GetAllowedOnAllUrlsOptionalBoolean(extension.id(), profile_); if (extension.is_app()) { return ExtensionSyncData( extension, enabled, disable_reasons, incognito_enabled, remote_install, allowed_on_all_url, extension_prefs_->app_sorting()->GetAppLaunchOrdinal(extension.id()), extension_prefs_->app_sorting()->GetPageOrdinal(extension.id()), extensions::GetLaunchTypePrefValue(extension_prefs_, extension.id())); } return ExtensionSyncData( extension, enabled, disable_reasons, incognito_enabled, remote_install, allowed_on_all_url); } bool ExtensionSyncService::ApplySyncData( const ExtensionSyncData& extension_sync_data) { syncer::ModelType type = extension_sync_data.is_app() ? syncer::APPS : syncer::EXTENSIONS; SyncBundle* bundle = GetSyncBundle(type); bundle->ApplySyncData(extension_sync_data); const std::string& id = extension_sync_data.id(); if (extension_sync_data.is_app()) { if (extension_sync_data.app_launch_ordinal().IsValid() && extension_sync_data.page_ordinal().IsValid()) { extension_prefs_->app_sorting()->SetAppLaunchOrdinal( id, extension_sync_data.app_launch_ordinal()); extension_prefs_->app_sorting()->SetPageOrdinal( id, extension_sync_data.page_ordinal()); } // The corresponding validation of this value during ExtensionSyncData // population is in ExtensionSyncData::ToAppSpecifics. if (extension_sync_data.launch_type() >= extensions::LAUNCH_TYPE_FIRST && extension_sync_data.launch_type() < extensions::NUM_LAUNCH_TYPES) { extensions::SetLaunchType( profile_, id, extension_sync_data.launch_type()); } if (!extension_sync_data.bookmark_app_url().empty()) ApplyBookmarkAppSyncData(extension_sync_data); } if (!ApplyExtensionSyncDataHelper(extension_sync_data, type)) { bundle->AddPendingExtension(id, extension_sync_data); extension_service_->CheckForUpdatesSoon(); return false; } return true; } void ExtensionSyncService::ApplyBookmarkAppSyncData( const extensions::ExtensionSyncData& extension_sync_data) { DCHECK(extension_sync_data.is_app()); // Process bookmark app sync if necessary. GURL bookmark_app_url(extension_sync_data.bookmark_app_url()); if (!bookmark_app_url.is_valid() || extension_sync_data.uninstalled()) { return; } const Extension* extension = extension_service_->GetInstalledExtension(extension_sync_data.id()); // Return if there are no bookmark app details that need updating. if (extension && extension->non_localized_name() == extension_sync_data.name() && extension->description() == extension_sync_data.bookmark_app_description()) { return; } WebApplicationInfo web_app_info; web_app_info.app_url = bookmark_app_url; web_app_info.title = base::UTF8ToUTF16(extension_sync_data.name()); web_app_info.description = base::UTF8ToUTF16(extension_sync_data.bookmark_app_description()); if (!extension_sync_data.bookmark_app_icon_color().empty()) { extensions::image_util::ParseCSSColorString( extension_sync_data.bookmark_app_icon_color(), &web_app_info.generated_icon_color); } for (const auto& icon : extension_sync_data.linked_icons()) { WebApplicationInfo::IconInfo icon_info; icon_info.url = icon.url; icon_info.width = icon.size; icon_info.height = icon.size; web_app_info.icons.push_back(icon_info); } // If the bookmark app already exists, keep the old icons. if (!extension) { CreateOrUpdateBookmarkApp(extension_service_, &web_app_info); } else { GetWebApplicationInfoFromApp(profile_, extension, base::Bind(&OnWebApplicationInfoLoaded, web_app_info, extension_service_->AsWeakPtr())); } } void ExtensionSyncService::SyncOrderingChange(const std::string& extension_id) { const Extension* ext = extension_service_->GetInstalledExtension(extension_id); if (ext) SyncExtensionChangeIfNeeded(*ext); } void ExtensionSyncService::SetSyncStartFlare( const syncer::SyncableService::StartSyncFlare& flare) { flare_ = flare; } SyncBundle* ExtensionSyncService::GetSyncBundle(syncer::ModelType type) { return const_cast( const_cast(*this).GetSyncBundle(type)); } const SyncBundle* ExtensionSyncService::GetSyncBundle( syncer::ModelType type) const { return (type == syncer::APPS) ? &app_sync_bundle_ : &extension_sync_bundle_; } std::vector ExtensionSyncService::GetLocalSyncDataList( syncer::ModelType type, bool include_everything) const { // Collect the local state. FillSyncDataList will filter out extensions for // which we have pending data that should be used instead. ExtensionRegistry* registry = ExtensionRegistry::Get(profile_); std::vector data; // TODO(treib, kalman): Should we be including blacklisted/blocked extensions // here? I.e. just calling registry->GeneratedInstalledExtensionsSet()? // It would be more consistent, but the danger is that the black/blocklist // hasn't been updated on all clients by the time sync has kicked in - // so it's safest not to. Take care to add any other extension lists here // in the future if they are added. FillSyncDataList( registry->enabled_extensions(), type, include_everything, &data); FillSyncDataList( registry->disabled_extensions(), type, include_everything, &data); FillSyncDataList( registry->terminated_extensions(), type, include_everything, &data); return data; } void ExtensionSyncService::FillSyncDataList( const ExtensionSet& extensions, syncer::ModelType type, bool include_everything, std::vector* sync_data_list) const { const SyncBundle* bundle = GetSyncBundle(type); for (const scoped_refptr& extension : extensions) { if (IsCorrectSyncType(*extension, type) && extensions::util::ShouldSync(extension.get(), profile_) && !bundle->HasPendingExtensionId(extension->id()) && (include_everything || ExtensionPrefs::Get(profile_)->NeedsSync(extension->id()))) { sync_data_list->push_back(CreateSyncData(*extension)); } } } bool ExtensionSyncService::ApplyExtensionSyncDataHelper( const ExtensionSyncData& extension_sync_data, syncer::ModelType type) { const std::string& id = extension_sync_data.id(); ExtensionRegistry* registry = ExtensionRegistry::Get(profile_); const Extension* extension = registry->GetInstalledExtension(id); // TODO(bolms): we should really handle this better. The particularly bad // case is where an app becomes an extension or vice versa, and we end up with // a zombie extension that won't go away. if (extension && !IsCorrectSyncType(*extension, type)) return true; // Handle uninstalls first. if (extension_sync_data.uninstalled()) { if (!extension_service_->UninstallExtensionHelper( extension_service_, id, extensions::UNINSTALL_REASON_SYNC)) { LOG(WARNING) << "Could not uninstall extension " << id << " for sync"; } return true; } // Extension from sync was uninstalled by the user as external extensions. // Honor user choice and skip installation/enabling. if (ExtensionPrefs::Get(profile_)->IsExternalExtensionUninstalled(id)) { LOG(WARNING) << "Extension with id " << id << " from sync was uninstalled as external extension"; return true; } int version_compare_result = extension ? extension->version()->CompareTo(extension_sync_data.version()) : 0; // Set user settings. if (extension_sync_data.enabled()) { DCHECK(!extension_sync_data.disable_reasons()); // Only grant permissions if the sync data explicitly sets the disable // reasons to Extension::DISABLE_NONE (as opposed to the legacy (GrantPermissionsAndEnableExtension(extension); else extension_service_->EnableExtension(id); } else { int disable_reasons = extension_sync_data.disable_reasons(); if (extension_sync_data.remote_install()) { if (!(disable_reasons & Extension::DISABLE_REMOTE_INSTALL)) { // In the non-legacy case (>=M45) where disable reasons are synced at // all, DISABLE_REMOTE_INSTALL should be among them already. DCHECK(!extension_sync_data.supports_disable_reasons()); disable_reasons |= Extension::DISABLE_REMOTE_INSTALL; } } else if (!extension_sync_data.supports_disable_reasons()) { // Legacy case (=M45), clear any existing disable reasons first. // Otherwise sync can't remove just some of them. if (extension_sync_data.supports_disable_reasons()) ExtensionPrefs::Get(profile_)->ClearDisableReasons(id); extension_service_->DisableExtension(id, disable_reasons); } // We need to cache some information here because setting the incognito flag // invalidates the |extension| pointer (it reloads the extension). bool extension_installed = (extension != NULL); // If the target extension has already been installed ephemerally, it can // be promoted to a regular installed extension and downloading from the Web // Store is not necessary. if (extension && extensions::util::IsEphemeralApp(id, profile_)) extension_service_->PromoteEphemeralApp(extension, true); // Update the incognito flag. extensions::util::SetIsIncognitoEnabled( id, profile_, extension_sync_data.incognito_enabled()); extension = NULL; // No longer safe to use. // Update the all urls flag. if (extension_sync_data.all_urls_enabled() != ExtensionSyncData::BOOLEAN_UNSET) { bool allowed = extension_sync_data.all_urls_enabled() == ExtensionSyncData::BOOLEAN_TRUE; extensions::util::SetAllowedScriptingOnAllUrls(id, profile_, allowed); } if (extension_installed) { // If the extension is already installed, check if it's outdated. if (version_compare_result < 0) { // Extension is outdated. return false; } } else { CHECK(type == syncer::EXTENSIONS || type == syncer::APPS); if (!extension_service_->pending_extension_manager()->AddFromSync( id, extension_sync_data.update_url(), extensions::sync_helper::IsSyncable, extension_sync_data.remote_install(), extension_sync_data.installed_by_custodian())) { LOG(WARNING) << "Could not add pending extension for " << id; // This means that the extension is already pending installation, with a // non-INTERNAL location. Add to pending_sync_data, even though it will // never be removed (we'll never install a syncable version of the // extension), so that GetAllSyncData() continues to send it. } // Track pending extensions so that we can return them in GetAllSyncData(). return false; } return true; }