extension_service.cc revision 90dce4d38c5ff5333bea97d859d4e484e27edf0c
12a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Copyright (c) 2013 The Chromium Authors. All rights reserved. 25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be 35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file. 45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_service.h" 65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <algorithm> 82a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include <iterator> 95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <set> 105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/basictypes.h" 125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h" 135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/callback.h" 145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/command_line.h" 155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/file_util.h" 165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h" 175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/metrics/histogram.h" 185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/path_service.h" 192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/prefs/pref_service.h" 205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/stl_util.h" 215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/string_util.h" 225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/stringprintf.h" 232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/strings/string_number_conversions.h" 242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/threading/sequenced_worker_pool.h" 255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/threading/thread_restrictions.h" 265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/time.h" 275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/utf_string_conversions.h" 285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/values.h" 295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/version.h" 3090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "chrome/browser/app_mode/app_mode_utils.h" 315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/browser_process.h" 322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/devtools/devtools_window.h" 335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/api/app_runtime/app_runtime_api.h" 345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/api/declarative/rules_registry_service.h" 352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/extensions/api/extension_action/extension_action_api.h" 362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/extensions/api/profile_keyed_api_factory.h" 375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/api/runtime/runtime_api.h" 382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/extensions/api/storage/settings_frontend.h" 395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/app_sync_data.h" 405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/browser_event_router.h" 415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/component_loader.h" 425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/crx_installer.h" 435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/data_deleter.h" 445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_disabled_ui.h" 455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_error_reporter.h" 465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_error_ui.h" 475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_host.h" 485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_install_ui.h" 495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_process_manager.h" 505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_sorting.h" 515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_special_storage_policy.h" 525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_sync_data.h" 535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/extension_system.h" 545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/external_install_ui.h" 555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/external_provider_impl.h" 565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/external_provider_interface.h" 575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/installed_loader.h" 585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/lazy_background_task_queue.h" 592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/extensions/management_policy.h" 605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/pending_extension_manager.h" 615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/permissions_updater.h" 625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/platform_app_launcher.h" 635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/shell_window_registry.h" 645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/unpacked_installer.h" 652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/extensions/update_observer.h" 665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/extensions/updater/extension_updater.h" 672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/plugins/chrome_plugin_service_filter.h" 685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/profiles/profile.h" 695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/profiles/profile_manager.h" 705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/themes/theme_service.h" 715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/themes/theme_service_factory.h" 725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/ui/webui/favicon_source.h" 735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/ui/webui/ntp/thumbnail_source.h" 745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/ui/webui/theme_source.h" 755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/child_process_logging.h" 765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_notification_types.h" 775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_paths.h" 785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_switches.h" 795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/chrome_version_info.h" 802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/extensions/api/plugins/plugins_handler.h" 812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/extensions/background_info.h" 825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/extension.h" 835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/extension_file_util.h" 845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/extension_manifest_constants.h" 855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/extension_messages.h" 865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/feature_switch.h" 872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/extensions/features/feature.h" 882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/extensions/incognito_handler.h" 895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/extensions/manifest.h" 90c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#include "chrome/common/extensions/manifest_handlers/app_isolation_info.h" 912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/extensions/manifest_url_handler.h" 92b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles)#include "chrome/common/extensions/permissions/permissions_data.h" 935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/pref_names.h" 942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/common/startup_metric_utils.h" 955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/common/url_constants.h" 965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/browser_thread.h" 972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "content/public/browser/devtools_agent_host.h" 985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/notification_service.h" 995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/notification_types.h" 1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/plugin_service.h" 1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/render_process_host.h" 1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "content/public/browser/site_instance.h" 1032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "content/public/browser/storage_partition.h" 1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "content/public/browser/url_data_source.h" 1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/common/pepper_plugin_info.h" 10690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "extensions/common/constants.h" 1072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "extensions/common/error_utils.h" 1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "googleurl/src/gurl.h" 1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "grit/generated_resources.h" 1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/base/registry_controlled_domains/registry_controlled_domain.h" 1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "sync/api/sync_change.h" 1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "sync/api/sync_error_factory.h" 11390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "webkit/browser/database/database_tracker.h" 11490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "webkit/browser/database/database_util.h" 1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_CHROMEOS) 1172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "chrome/browser/chromeos/extensions/install_limiter.h" 11890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "webkit/browser/fileapi/file_system_context.h" 11990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "webkit/browser/fileapi/file_system_mount_point_provider.h" 1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::BrowserContext; 1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::BrowserThread; 1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::DevToolsAgentHost; 1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::PluginService; 1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::CrxInstaller; 1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::Extension; 1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::ExtensionIdSet; 1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::ExtensionInfo; 1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::FeatureSwitch; 1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)using extensions::Manifest; 1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::PermissionMessage; 1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::PermissionMessages; 1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using extensions::PermissionSet; 1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)using extensions::UnloadedExtensionInfo; 1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace errors = extension_manifest_errors; 1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace { 1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Histogram values for logging events related to externally installed 1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// extensions. 1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)enum ExternalExtensionEvent { 1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_INSTALLED = 0, 1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_IGNORED, 1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_REENABLED, 1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_UNINSTALLED, 1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_BUCKET_BOUNDARY, 1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}; 1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Prompt the user this many times before considering an extension acknowledged. 1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static const int kMaxExtensionAcknowledgePromptCount = 3; 1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Wait this many seconds after an extensions becomes idle before updating it. 1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static const int kUpdateIdleDelay = 5; 1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Wait this many seconds before trying to garbage collect extensions again. 1582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)static const int kGarbageCollectRetryDelay = 30; 1592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char* kNaClPluginMimeType = "application/x-nacl"; 1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static bool IsSyncableExtension(const Extension& extension) { 1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension.GetSyncType() == Extension::SYNC_TYPE_EXTENSION; 1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static bool IsSyncableApp(const Extension& extension) { 1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension.GetSyncType() == Extension::SYNC_TYPE_APP; 1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} // namespace 1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::ExtensionRuntimeData::ExtensionRuntimeData() 1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) : background_page_ready(false), 1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) being_upgraded(false), 1755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) has_used_webrequest(false) { 1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::ExtensionRuntimeData::~ExtensionRuntimeData() { 1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::NaClModuleInfo::NaClModuleInfo() { 1825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::NaClModuleInfo::~NaClModuleInfo() { 1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 1865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// ExtensionService. 1885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char ExtensionService::kLocalAppSettingsDirectoryName[] = 1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) "Local App Settings"; 1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char ExtensionService::kLocalExtensionSettingsDirectoryName[] = 1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) "Local Extension Settings"; 1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char ExtensionService::kSyncAppSettingsDirectoryName[] = 1945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) "Sync App Settings"; 1955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char ExtensionService::kSyncExtensionSettingsDirectoryName[] = 1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) "Sync Extension Settings"; 1972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)const char ExtensionService::kManagedSettingsDirectoryName[] = 1982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) "Managed Extension Settings"; 1995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char ExtensionService::kStateStoreName[] = "Extension State"; 2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)const char ExtensionService::kRulesStoreName[] = "Extension Rules"; 2015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::CheckExternalUninstall(const std::string& id) { 2035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 2045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Check if the providers know about this extension. 2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ProviderCollection::const_iterator i; 2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (i = external_extension_providers_.begin(); 2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != external_extension_providers_.end(); ++i) { 2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(i->get()->IsReady()); 2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (i->get()->HasExtension(id)) 2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; // Yup, known extension, don't uninstall. 2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We get the list of external extensions to check from preferences. 2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // It is possible that an extension has preferences but is not loaded. 2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // For example, an extension that requires experimental permissions 2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // will not be loaded if the experimental command line flag is not used. 2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // In this case, do not uninstall. 2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!GetInstalledExtension(id)) { 2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We can't call UninstallExtension with an unloaded/invalid 2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension ID. 2225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Attempted uninstallation of unloaded/invalid extension " 2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "with id: " << id; 2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UninstallExtension(id, true, NULL); 2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::SetFileTaskRunnerForTesting( 2302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::SequencedTaskRunner* task_runner) { 2312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) file_task_runner_ = task_runner; 2322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 2332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::ClearProvidersForTesting() { 2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) external_extension_providers_.clear(); 2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::AddProviderForTesting( 2395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExternalProviderInterface* test_provider) { 2405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(test_provider); 2415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) external_extension_providers_.push_back( 2425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) linked_ptr<extensions::ExternalProviderInterface>(test_provider)); 2435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::OnExternalExtensionUpdateUrlFound( 2465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id, 2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const GURL& update_url, 2482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Manifest::Location location) { 2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(Extension::IdIsValid(id)); 2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = GetExtensionById(id, true); 2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension) { 2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Already installed. Skip this install if the current location has 2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // higher priority than |location|. 2562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Manifest::Location current = extension->location(); 2572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (current == Manifest::GetHigherPriorityLocation(current, location)) 2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Otherwise, overwrite the current installation. 2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Add |id| to the set of pending extensions. If it can not be added, 2635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // then there is already a pending record from a higher-priority install 2645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // source. In this case, signal that this extension will not be 2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installed by returning false. 2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pending_extension_manager()->AddFromExternalUpdateUrl( 2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, update_url, location)) { 2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) update_once_all_providers_are_ready_ = true; 2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const Extension* ExtensionService::GetInstalledApp(const GURL& url) const { 2765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = extensions_.GetExtensionOrAppByURL( 2775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionURLInfo(url)); 2785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return (extension && extension->is_app()) ? extension : NULL; 2795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsInstalledApp(const GURL& url) const { 2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return !!GetInstalledApp(url); 2835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const Extension* ExtensionService::GetIsolatedAppForRenderer( 2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int renderer_child_id) const { 2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::set<std::string> extension_ids = 2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process_map_.GetExtensionsInProcess(renderer_child_id); 2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // All apps in one process share the same partition. 2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // It is only possible for the app to have isolated storage 2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // if there is only 1 app in the process. 2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_ids.size() != 1) 2935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return NULL; 2945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::Extension* extension = 2965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_.GetByID(*(extension_ids.begin())); 297c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // We still need to check if the extension has isolated storage, 2985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // because it's common for there to be one extension in a process 299c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // without isolated storage. 300c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (extension && 301c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extensions::AppIsolationInfo::HasIsolatedStorage(extension)) 3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension; 3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return NULL; 3055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// This function is used to implement the command-line switch 3095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// --uninstall-extension, and to uninstall an extension via sync. The LOG 3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// statements within this function are used to inform the user if the uninstall 3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// cannot be done. 3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::UninstallExtensionHelper( 3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionService* extensions_service, 3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id) { 3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We can't call UninstallExtension with an invalid extension ID. 3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extensions_service->GetInstalledExtension(extension_id)) { 3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Attempted uninstallation of non-existent extension with " 3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "id: " << extension_id; 3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The following call to UninstallExtension will not allow an uninstall of a 3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // policy-controlled extension. 3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16 error; 3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extensions_service->UninstallExtension(extension_id, false, &error)) { 3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Cannot uninstall extension with id " << extension_id 3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << ": " << error; 3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::ExtensionService(Profile* profile, 3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const CommandLine* command_line, 3362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const base::FilePath& install_directory, 3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionPrefs* extension_prefs, 3382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::Blacklist* blacklist, 3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool autoupdate_enabled, 34090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) bool extensions_enabled, 34190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extensions::OneShotEvent* ready) 3422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) : extensions::Blacklist::Observer(blacklist), 3432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_(profile), 3445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_(extensions::ExtensionSystem::Get(profile)), 3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_(extension_prefs), 3462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklist_(blacklist), 3475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) settings_frontend_(extensions::SettingsFrontend::Create(profile)), 348c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) pending_extension_manager_(*this), 3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) install_directory_(install_directory), 3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_enabled_(extensions_enabled), 3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) show_extensions_prompts_(true), 3522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) install_updates_when_idle_(true), 35390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) ready_(ready), 354c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) toolbar_model_(this), 3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) menu_manager_(profile), 3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) event_routers_initialized_(false), 3575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) update_once_all_providers_are_ready_(false), 3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) browser_terminating_(false), 3592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) installs_delayed_(false), 360c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) is_first_run_(false), 361c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) app_sync_bundle_(this), 362c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_sync_bundle_(this) { 3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Figure out if extension installation should be enabled. 3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (command_line->HasSwitch(switches::kDisableExtensions) || 3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile->GetPrefs()->GetBoolean(prefs::kDisableExtensions)) { 3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_enabled_ = false; 3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, chrome::NOTIFICATION_APP_TERMINATING, 3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_PROCESS_TERMINATED, 3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, content::NOTIFICATION_RENDERER_PROCESS_CREATED, 3765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, content::NOTIFICATION_RENDERER_PROCESS_TERMINATED, 3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_HOST_DESTROYED, 3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) registrar_.Add(this, chrome::NOTIFICATION_UPGRADE_RECOMMENDED, 3822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) content::NotificationService::AllBrowserContextsAndSources()); 3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pref_change_registrar_.Init(profile->GetPrefs()); 3842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Closure callback = 3852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::OnExtensionInstallPrefChanged, 3862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Unretained(this)); 3872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) pref_change_registrar_.Add(prefs::kExtensionInstallAllowList, callback); 3882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) pref_change_registrar_.Add(prefs::kExtensionInstallDenyList, callback); 3892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) pref_change_registrar_.Add(prefs::kExtensionAllowedTypes, callback); 3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Set up the ExtensionUpdater 3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (autoupdate_enabled) { 3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int update_frequency = kDefaultUpdateFrequencySeconds; 3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (command_line->HasSwitch(switches::kExtensionsUpdateFrequency)) { 3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::StringToInt(command_line->GetSwitchValueASCII( 3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switches::kExtensionsUpdateFrequency), 3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &update_frequency); 3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) updater_.reset(new extensions::ExtensionUpdater(this, 4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs, 4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile->GetPrefs(), 4025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile, 4032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklist, 4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) update_frequency)); 4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) component_loader_.reset( 4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) new extensions::ComponentLoader(this, 4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile->GetPrefs(), 4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) g_browser_process->local_state())); 4115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extensions_enabled_) { 4132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::ExternalProviderImpl::CreateExternalProviders( 4142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) this, profile_, &external_extension_providers_); 4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Set this as the ExtensionService for extension sorting to ensure it 4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // cause syncs if required. 4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->SetExtensionService(this); 4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 421c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) is_first_run_ = !extension_prefs_->SetAlertSystemFirstRun(); 422c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(ENABLE_EXTENSIONS) 4245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_action_storage_manager_.reset( 4255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) new extensions::ExtensionActionStorageManager(profile_)); 4265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 4275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // How long is the path to the Extensions directory? 4295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_CUSTOM_COUNTS("Extensions.ExtensionRootPathLength", 4305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) install_directory_.value().length(), 0, 500, 100); 4315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ExtensionSet* ExtensionService::extensions() const { 4345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return &extensions_; 4355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ExtensionSet* ExtensionService::disabled_extensions() const { 4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return &disabled_extensions_; 4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const ExtensionSet* ExtensionService::terminated_extensions() const { 4425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return &terminated_extensions_; 4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)const ExtensionSet* ExtensionService::blacklisted_extensions() const { 4462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return &blacklisted_extensions_; 4472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 4482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 4492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)scoped_ptr<const ExtensionSet> 4502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ExtensionService::GenerateInstalledExtensionsSet() const { 4512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_ptr<ExtensionSet> installed_extensions(new ExtensionSet()); 4525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installed_extensions->InsertAll(extensions_); 4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installed_extensions->InsertAll(disabled_extensions_); 4545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installed_extensions->InsertAll(terminated_extensions_); 4552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) installed_extensions->InsertAll(blacklisted_extensions_); 4562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return installed_extensions.PassAs<const ExtensionSet>(); 4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::PendingExtensionManager* 4605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionService::pending_extension_manager() { 4615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return &pending_extension_manager_; 4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::~ExtensionService() { 4655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // No need to unload extensions here because they are profile-scoped, and the 4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // profile is in the process of being deleted. 4675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ProviderCollection::const_iterator i; 4695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (i = external_extension_providers_.begin(); 4705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != external_extension_providers_.end(); ++i) { 4715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExternalProviderInterface* provider = i->get(); 4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) provider->ServiceShutdown(); 4735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::InitEventRoutersAfterImport() { 4775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RegisterForImportFinished(); 4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::RegisterForImportFinished() { 4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!registrar_.IsRegistered(this, chrome::NOTIFICATION_IMPORT_FINISHED, 4825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_))) { 4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) registrar_.Add(this, chrome::NOTIFICATION_IMPORT_FINISHED, 4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_)); 4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::InitAfterImport() { 4892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) startup_metric_utils::ScopedSlowStartupUMA 4902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_timer("Startup.SlowStartupExtensionServiceInitAfterImport"); 4912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) component_loader_->LoadAll(); 4922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CheckForExternalUpdates(); 4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) GarbageCollectExtensions(); 4965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 4975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Idempotent, so although there is a possible race if the import 4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // process finished sometime in the middle of ProfileImpl::InitExtensions, 4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // it cannot happen twice. 5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) InitEventRouters(); 5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 5025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::InitEventRouters() { 5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (event_routers_initialized_) 5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(ENABLE_EXTENSIONS) 5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) browser_event_router_.reset(new extensions::BrowserEventRouter(profile_)); 5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif // defined(ENABLE_EXTENSIONS) 5105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) event_routers_initialized_ = true; 5115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 5125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::Shutdown() { 5145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Do nothing for now. 5155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 5165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const Extension* ExtensionService::GetExtensionById( 5185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id, bool include_disabled) const { 5195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int include_mask = INCLUDE_ENABLED; 5202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (include_disabled) { 5212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Include blacklisted extensions here because there are hundreds of 5222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // callers of this function, and many might assume that this includes those 5232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // that have been disabled due to blacklisting. 5242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) include_mask |= INCLUDE_DISABLED | INCLUDE_BLACKLISTED; 5252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 5262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return GetExtensionById(id, include_mask); 5272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 5282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 5292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)GURL ExtensionService::GetSiteForExtensionId(const std::string& extension_id) { 5302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return content::SiteInstance::GetSiteForURL( 5312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_, 5322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Extension::GetBaseURLFromExtensionId(extension_id)); 5332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 5342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 5352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)const Extension* ExtensionService::GetExtensionById( 5362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& id, int include_mask) const { 5372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::string lowercase_id = StringToLowerASCII(id); 5382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (include_mask & INCLUDE_ENABLED) { 5392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = extensions_.GetByID(lowercase_id); 5402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension) 5412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return extension; 5422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 5432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (include_mask & INCLUDE_DISABLED) { 5442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = disabled_extensions_.GetByID(lowercase_id); 5452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension) 5462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return extension; 5472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 5482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (include_mask & INCLUDE_TERMINATED) { 5492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = terminated_extensions_.GetByID(lowercase_id); 5502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension) 5512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return extension; 5522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 5532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (include_mask & INCLUDE_BLACKLISTED) { 5542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = blacklisted_extensions_.GetByID(lowercase_id); 5552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension) 5562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return extension; 5572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 5582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return NULL; 5595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 5605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 5615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::Init() { 5625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 5635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 56490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) DCHECK(!is_ready()); // Can't redo init. 5655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK_EQ(extensions_.size(), 0u); 5665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 567c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) const CommandLine* cmd_line = CommandLine::ForCurrentProcess(); 568c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (cmd_line->HasSwitch(switches::kInstallFromWebstore) || 569c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) cmd_line->HasSwitch(switches::kLimitedInstallFromWebstore)) { 570c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // The sole purpose of this launch is to install a new extension from CWS 571c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // and immediately terminate: loading already installed extensions is 572c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // unnecessary and may interfere with the inline install dialog (e.g. if an 573c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // extension listens to onStartup and opens a window). 574c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) SetReadyAndNotifyListeners(); 5752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 576c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // TODO(mek): It might be cleaner to do the FinishDelayedInstallInfo stuff 577c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // here instead of in installedloader. 578c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (g_browser_process->profile_manager() && 579c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) g_browser_process->profile_manager()->will_import()) { 580c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Do not load any component extensions, since they may conflict with the 581c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // import process. 582c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 583c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extensions::InstalledLoader(this).LoadAllExtensions(); 58490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) SetReadyAndNotifyListeners(); 585c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) RegisterForImportFinished(); 586c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } else { 58790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // In this case, LoadAllExtensions() calls OnLoadedInstalledExtensions(). 588c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) component_loader_->LoadAll(); 589c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extensions::InstalledLoader(this).LoadAllExtensions(); 59090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) SetReadyAndNotifyListeners(); 5912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 592c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // TODO(erikkay) this should probably be deferred to a future point 593c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // rather than running immediately at startup. 594c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) CheckForExternalUpdates(); 5955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 596c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // TODO(erikkay) this should probably be deferred as well. 597c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) GarbageCollectExtensions(); 598c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } 5995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 600c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (extension_prefs_->NeedsStorageGarbageCollection()) { 601c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) GarbageCollectIsolatedStorage(); 602c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_prefs_->SetNeedsStorageGarbageCollection(false); 603c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } 6045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 6055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 6065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::UpdateExtension(const std::string& id, 6082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const base::FilePath& extension_path, 6095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const GURL& download_url, 6105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CrxInstaller** out_crx_installer) { 6115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 6125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (browser_terminating_) { 6135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Skipping UpdateExtension due to browser shutdown"; 6145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Leak the temp file at extension_path. We don't want to add to the disk 6155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // I/O burden at shutdown, we can't rely on the I/O completing anyway, and 6165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // the file is in the OS temp directory which should be cleaned up for us. 6175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 6185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 6195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::PendingExtensionInfo* pending_extension_info = 6215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extension_manager()->GetById(id); 6225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = GetInstalledExtension(id); 6245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pending_extension_info && !extension) { 6255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Will not update extension " << id 6265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << " because it is not installed or pending"; 6275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Delete extension_path since we're not creating a CrxInstaller 6285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // that would do it for us. 6292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!GetFileTaskRunner()->PostTask( 6302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FROM_HERE, 6315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind( 6325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_file_util::DeleteFile, extension_path, false))) 6335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 6345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 6365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 6375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We want a silent install only for non-pending extensions and 6395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // pending extensions that have install_silently set. 6405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionInstallPrompt* client = NULL; 6415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (pending_extension_info && !pending_extension_info->install_silently()) 6425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) client = ExtensionInstallUI::CreateInstallPromptWithProfile(profile_); 6435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<CrxInstaller> installer(CrxInstaller::Create(this, client)); 6455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_expected_id(id); 6465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (pending_extension_info) { 6475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_install_source(pending_extension_info->install_source()); 6485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (pending_extension_info->install_silently()) 6495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_allow_silent_install(true); 6505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (extension) { 6515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_install_source(extension->location()); 6525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 6535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension was installed from or has migrated to the webstore, or 6545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // its auto-update URL is from the webstore, treat it as a webstore install. 6555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Note that we ignore some older extensions with blank auto-update URLs 6565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // because we are mostly concerned with restrictions on NaCl extensions, 6575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // which are newer. 6585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int creation_flags = Extension::NO_FLAGS; 6595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if ((extension && extension->from_webstore()) || 6605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) (extension && extension->UpdatesFromGallery()) || 6615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) (!extension && extension_urls::IsWebstoreUpdateUrl( 6625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extension_info->update_url()))) { 6635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) creation_flags |= Extension::FROM_WEBSTORE; 6645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 6655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Bookmark apps being updated is kind of a contradiction, but that's because 6675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // we mark the default apps as bookmark apps, and they're hosted in the web 6685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // store, thus they can get updated. See http://crbug.com/101605 for more 6695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // details. 6705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension && extension->from_bookmark()) 6715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) creation_flags |= Extension::FROM_BOOKMARK; 6725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension && extension->was_installed_by_default()) 6745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) creation_flags |= Extension::WAS_INSTALLED_BY_DEFAULT; 6755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_creation_flags(creation_flags); 6775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_delete_source(true); 6795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_download_url(download_url); 6805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_install_cause(extension_misc::INSTALL_CAUSE_UPDATE); 6815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->InstallCrx(extension_path); 6825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (out_crx_installer) 6845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *out_crx_installer = installer; 6855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 6875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 6885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::ReloadExtension(const std::string& extension_id) { 6905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int events = HasShellWindows(extension_id) ? EVENT_LAUNCHED : EVENT_NONE; 6915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ReloadExtensionWithEvents(extension_id, events); 6925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 6935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::RestartExtension(const std::string& extension_id) { 6952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ReloadExtensionWithEvents(extension_id, EVENT_RESTARTED); 6965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 6975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 6985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::ReloadExtensionWithEvents( 6995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id, 7005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int events) { 7015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 7022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 7032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If the extension is already reloading, don't reload again. 7042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension_prefs_->GetDisableReasons(extension_id) & 7052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Extension::DISABLE_RELOAD) { 7062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 7072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 7082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 7092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath path; 7105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* current_extension = GetExtensionById(extension_id, false); 7115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Disable the extension if it's loaded. It might not be loaded if it crashed. 7135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (current_extension) { 7145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension has an inspector open for its background page, detach 7155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // the inspector and hang onto a cookie for it, so that we can reattach 7165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // later. 7175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(yoz): this is not incognito-safe! 7185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionProcessManager* manager = system_->process_manager(); 7195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host = 7205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) manager->GetBackgroundHostForExtension(extension_id); 7212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (host && DevToolsAgentHost::HasFor(host->render_view_host())) { 7225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Look for an open inspector for the background page. 7232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::string devtools_cookie = DevToolsAgentHost::DisconnectRenderViewHost( 7242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) host->render_view_host()); 7252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (devtools_cookie != std::string()) 7265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) orphaned_dev_tools_[extension_id] = devtools_cookie; 7275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) path = current_extension->path(); 7305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DisableExtension(extension_id, Extension::DISABLE_RELOAD); 731c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) reloading_extensions_.insert(extension_id); 7325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 7335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) path = unloaded_extension_paths_[extension_id]; 7345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) on_load_events_[extension_id] = events; 7372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 7382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (delayed_updates_for_idle_.Contains(extension_id)) { 7392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FinishDelayedInstallation(extension_id); 7405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 7415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If we're reloading a component extension, use the component extension 7445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // loader's reloader. 7455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (component_loader_->Exists(extension_id)) { 7465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) component_loader_->Reload(extension_id); 7475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 7485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Check the installed extensions to see if what we're reloading was already 7515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installed. 7525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_ptr<ExtensionInfo> installed_extension( 7535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->GetInstalledExtensionInfo(extension_id)); 7545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (installed_extension.get() && 7555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installed_extension->extension_manifest.get()) { 7565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::InstalledLoader(this).Load(*installed_extension, false); 7575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 7585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Otherwise, the extension is unpacked (location LOAD). 7595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We should always be able to remember the extension's path. If it's not in 7605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // the map, someone failed to update |unloaded_extension_paths_|. 7615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(!path.empty()); 7625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::UnpackedInstaller::Create(this)->Load(path); 7635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 7655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::UninstallExtension( 7675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::string extension_id, 7685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool external_uninstall, 7695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16* error) { 7705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 7715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<const Extension> extension(GetInstalledExtension(extension_id)); 7735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Callers should not send us nonexistent extensions. 7755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(extension); 7765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Policy change which triggers an uninstall will always set 7785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // |external_uninstall| to true so this is the only way to uninstall 7795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // managed extensions. 7805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!external_uninstall && 7815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !system_->management_policy()->UserMayModifySettings( 7825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension.get(), error)) { 7835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 7845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_UNINSTALL_NOT_ALLOWED, 7855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 7865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 7875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 7885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extract the data we need for sync now, but don't actually sync until we've 7915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // completed the uninstallation. 7925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::SyncChange sync_change; 7935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (app_sync_bundle_.HandlesApp(*extension)) { 7945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change = app_sync_bundle_.CreateSyncChangeToDelete(extension); 7955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (extension_sync_bundle_.HandlesExtension(*extension)) { 7965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change = extension_sync_bundle_.CreateSyncChangeToDelete(extension); 7975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 7985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 7995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsUnacknowledgedExternalExtension(extension)) { 8005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.ExternalExtensionEvent", 8015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_UNINSTALLED, 8025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_BUCKET_BOUNDARY); 8035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 8045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.UninstallType", 8055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension->GetType(), 100); 8065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RecordPermissionMessagesHistogram( 8075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension, "Extensions.Permissions_Uninstall"); 8085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Unload before doing more cleanup to ensure that nothing is hanging on to 8105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // any of these resources. 8115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadExtension(extension_id, extension_misc::UNLOAD_REASON_UNINSTALL); 8125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->OnExtensionUninstalled(extension_id, extension->location(), 8145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) external_uninstall); 8155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Tell the backend to start deleting installed extensions on the file thread. 8172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!Manifest::IsUnpackedLocation(extension->location())) { 8182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!GetFileTaskRunner()->PostTask( 8192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FROM_HERE, 8205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind( 8215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_file_util::UninstallExtension, 8225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) install_directory_, 8235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_id))) 8245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 8255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 8265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) GURL launch_web_url_origin(extension->launch_web_url()); 8285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) launch_web_url_origin = launch_web_url_origin.GetOrigin(); 829c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool is_storage_isolated = 830c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extensions::AppIsolationInfo::HasIsolatedStorage(extension); 8315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (is_storage_isolated) { 8332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) BrowserContext::AsyncObliterateStoragePartition( 8342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_, 8352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetSiteForExtensionId(extension_id), 8362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::OnNeedsToGarbageCollectIsolatedStorage, 8372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AsWeakPtr())); 8382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 8392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->is_hosted_app() && 8402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !profile_->GetExtensionSpecialStoragePolicy()-> 8412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) IsStorageProtected(launch_web_url_origin)) { 8422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::DataDeleter::StartDeleting( 8432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_, extension_id, launch_web_url_origin); 8442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 8452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::DataDeleter::StartDeleting(profile_, extension_id, 8462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->url()); 8475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 8485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UntrackTerminatedExtension(extension_id); 8505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Notify interested parties that we've uninstalled this extension. 8525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 8535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_UNINSTALLED, 8545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 8555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 8565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (app_sync_bundle_.HasExtensionId(extension_id) && 8585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change.sync_data().GetDataType() == syncer::APPS) { 8595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.ProcessDeletion(extension_id, sync_change); 8605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (extension_sync_bundle_.HasExtensionId(extension_id) && 8615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change.sync_data().GetDataType() == syncer::EXTENSIONS) { 8625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.ProcessDeletion(extension_id, sync_change); 8635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 8645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_updates_for_idle_.Remove(extension_id); 8662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_installs_.Remove(extension_id); 8675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Track the uninstallation. 8695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.ExtensionUninstalled", 1, 2); 8705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 8725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 8735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsExtensionEnabled( 8755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id) const { 8765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extensions_.Contains(extension_id) || 8775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) terminated_extensions_.Contains(extension_id)) { 8785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 8795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 8805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (disabled_extensions_.Contains(extension_id) || 8822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.Contains(extension_id)) { 8835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 8842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 8855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension hasn't been loaded yet, check the prefs for it. Assume 8875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // enabled unless otherwise noted. 8885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return !extension_prefs_->IsExtensionDisabled(extension_id) && 8892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !extension_prefs_->IsExternalExtensionUninstalled(extension_id); 8905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 8915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsExternalExtensionUninstalled( 8935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id) const { 8945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_prefs_->IsExternalExtensionUninstalled(extension_id); 8955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 8965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 8972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool ExtensionService::IsExtensionEnabledForLauncher( 8982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& extension_id) const { 8992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return IsExtensionEnabled(extension_id) && 9002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !GetTerminatedExtension(extension_id); 9012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 9022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 9035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::EnableExtension(const std::string& extension_id) { 9045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 9055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsExtensionEnabled(extension_id)) 9075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 9085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetExtensionState(extension_id, Extension::ENABLED); 9105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->ClearDisableReasons(extension_id); 9115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = disabled_extensions_.GetByID(extension_id); 9135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This can happen if sync enables an extension that is not 9145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installed yet. 9155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension) 9165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 9175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsUnacknowledgedExternalExtension(extension)) { 9195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.ExternalExtensionEvent", 9205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_REENABLED, 9215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_BUCKET_BOUNDARY); 9225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AcknowledgeExternalExtension(extension->id()); 9235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 9245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Move it over to the enabled list. 9265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_.Insert(make_scoped_refptr(extension)); 9275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) disabled_extensions_.Remove(extension->id()); 9285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NotifyExtensionLoaded(extension); 9305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Notify listeners that the extension was enabled. 9325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 9335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_ENABLED, 9345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 9355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 9365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 9385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 9395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::DisableExtension( 9415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id, 9425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Extension::DisableReason disable_reason) { 9435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 9445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The extension may have been disabled already. 9465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!IsExtensionEnabled(extension_id)) 9475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 9485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = GetInstalledExtension(extension_id); 9505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // |extension| can be NULL if sync disables an extension that is not 9515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installed yet. 9525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension && 9535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !system_->management_policy()->UserMayModifySettings(extension, NULL)) { 9545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 9555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 9565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetExtensionState(extension_id, Extension::DISABLED); 9585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AddDisableReason(extension_id, disable_reason); 9595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) int include_mask = INCLUDE_EVERYTHING & ~INCLUDE_DISABLED; 9612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension = GetExtensionById(extension_id, include_mask); 9625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension) 9635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 9645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 965c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Reset the background_page_ready flag 966c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (extensions::BackgroundInfo::HasBackgroundPage(extension)) 967c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_runtime_data_[extension->id()].background_page_ready = false; 968c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 9695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Move it over to the disabled list. Don't send a second unload notification 9705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // for terminated extensions being disabled. 9715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) disabled_extensions_.Insert(make_scoped_refptr(extension)); 9725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extensions_.Contains(extension->id())) { 9735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_.Remove(extension->id()); 9745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NotifyExtensionUnloaded(extension, extension_misc::UNLOAD_REASON_DISABLE); 9755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 9765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) terminated_extensions_.Remove(extension->id()); 9775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 9785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 9805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 9815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::GrantPermissionsAndEnableExtension( 9832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension) { 9842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GrantPermissions(extension); 9855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RecordPermissionMessagesHistogram( 9865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension, "Extensions.Permissions_ReEnable"); 9875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetDidExtensionEscalatePermissions(extension, false); 9885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EnableExtension(extension->id()); 9895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 9905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::GrantPermissions(const Extension* extension) { 9925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(extension); 9935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::PermissionsUpdater perms_updater(profile()); 9942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) perms_updater.GrantActivePermissions(extension); 9955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 9965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 9975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 9985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::RecordPermissionMessagesHistogram( 99990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) const Extension* extension, const char* histogram) { 10002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Since this is called from multiple sources, and since the histogram macros 10012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // use statics, we need to manually lookup the histogram ourselves. 10022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::HistogramBase* counter = base::LinearHistogram::FactoryGet( 10035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) histogram, 10045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1, 10055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PermissionMessage::kEnumBoundary, 10065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PermissionMessage::kEnumBoundary + 1, 10072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::HistogramBase::kUmaTargetedHistogramFlag); 10085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 100990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) PermissionMessages permissions = 101090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extensions::PermissionsData::GetPermissionMessages(extension); 10115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (permissions.empty()) { 10125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) counter->Add(PermissionMessage::kNone); 10135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 10145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (PermissionMessages::iterator it = permissions.begin(); 10155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) it != permissions.end(); ++it) 10165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) counter->Add(it->id()); 10175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 10195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::NotifyExtensionLoaded(const Extension* extension) { 10215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The ChromeURLRequestContexts need to be first to know that the extension 10225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // was loaded, otherwise a race can arise where a renderer that is created 10235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // for the extension may try to load an extension URL with an extension id 10245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // that the request context doesn't yet know about. The profile is responsible 10255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // for ensuring its URLRequestContexts appropriately discover the loaded 10265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension. 10275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->RegisterExtensionWithRequestContexts(extension); 10285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Tell renderers about the new extension, unless it's a theme (renderers 10305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // don't need to know about themes). 10315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension->is_theme()) { 10325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (content::RenderProcessHost::iterator i( 10335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost::AllHostsIterator()); 10345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !i.IsAtEnd(); i.Advance()) { 10355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost* host = i.GetCurrentValue(); 10365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile* host_profile = 10375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile::FromBrowserContext(host->GetBrowserContext()); 10385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (host_profile->GetOriginalProfile() == 10395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_->GetOriginalProfile()) { 10405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<ExtensionMsg_Loaded_Params> loaded_extensions( 10415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 1, ExtensionMsg_Loaded_Params(extension)); 10425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) host->Send( 10435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) new ExtensionMsg_Loaded(loaded_extensions)); 10445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Tell subsystems that use the EXTENSION_LOADED notification about the new 10495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension. 10505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // 10515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // NOTE: It is important that this happen after notifying the renderers about 10525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // the new extensions so that if we navigate to an extension URL in 10535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // NOTIFICATION_EXTENSION_LOADED, the renderer is guaranteed to know about it. 10545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 10555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_LOADED, 10565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 10575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 10585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Tell a random-ass collection of other subsystems about the new extension. 10605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(aa): What should we do with all this goop? Can it move into the 10615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // relevant objects via EXTENSION_LOADED? 10625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_->GetExtensionSpecialStoragePolicy()-> 10645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) GrantRightsForExtension(extension); 10655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateActiveExtensionsInCrashReporter(); 10675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension has permission to load chrome://favicon/ resources we need 10695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // to make sure that the FaviconSource is registered with the 10705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // ChromeURLDataManager. 107190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (extensions::PermissionsData::HasHostPermission( 107290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extension, GURL(chrome::kChromeUIFaviconURL))) { 10735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) FaviconSource* favicon_source = new FaviconSource(profile_, 10745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) FaviconSource::FAVICON); 10752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) content::URLDataSource::Add(profile_, favicon_source); 10765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if !defined(OS_ANDROID) 10795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Same for chrome://theme/ resources. 108090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (extensions::PermissionsData::HasHostPermission( 108190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extension, GURL(chrome::kChromeUIThemeURL))) { 10825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ThemeSource* theme_source = new ThemeSource(profile_); 10832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) content::URLDataSource::Add(profile_, theme_source); 10845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 10865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Same for chrome://thumb/ resources. 108890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (extensions::PermissionsData::HasHostPermission( 108990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extension, GURL(chrome::kChromeUIThumbnailURL))) { 10905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ThumbnailSource* thumbnail_source = new ThumbnailSource(profile_); 10912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) content::URLDataSource::Add(profile_, thumbnail_source); 10925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 10935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 10942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(ENABLE_PLUGINS) 10955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(mpcomplete): This ends up affecting all profiles. See crbug.com/80757. 10965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool plugins_changed = false; 10972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extensions::PluginInfo::HasPlugins(extension)) { 10982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const extensions::PluginInfo::PluginVector* plugins = 10992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::PluginInfo::GetPlugins(extension); 11002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) CHECK(plugins); 11015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) plugins_changed = true; 11022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (extensions::PluginInfo::PluginVector::const_iterator plugin = 11032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugins->begin(); 11042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugin != plugins->end(); ++plugin) { 11052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) PluginService::GetInstance()->RefreshPlugins(); 11062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) PluginService::GetInstance()->AddExtraPluginPath(plugin->path); 11072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ChromePluginServiceFilter* filter = 11082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ChromePluginServiceFilter::GetInstance(); 11092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (plugin->is_public) { 11102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) filter->RestrictPluginToProfileAndOrigin( 11112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugin->path, profile_, GURL()); 11122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 11132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) filter->RestrictPluginToProfileAndOrigin( 11142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugin->path, profile_, extension->url()); 11152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 11165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool nacl_modules_changed = false; 11205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < extension->nacl_modules().size(); ++i) { 11215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension::NaClModuleInfo& module = extension->nacl_modules()[i]; 11225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RegisterNaClModule(module.url, module.mime_type); 11235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) nacl_modules_changed = true; 11245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (nacl_modules_changed) 11275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdatePluginListWithNaClModules(); 11285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (plugins_changed || nacl_modules_changed) 11305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()->PurgePluginListCache(profile_, false); 11312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif // defined(ENABLE_PLUGINS) 11325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 11335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::NotifyExtensionUnloaded( 11355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension, 11365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_misc::UnloadedExtensionReason reason) { 11375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadedExtensionInfo details(extension, reason); 11385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 11395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_UNLOADED, 11405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 11415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<UnloadedExtensionInfo>(&details)); 11425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(ENABLE_THEMES) 11442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If the current theme is being unloaded, tell ThemeService to revert back 11452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // to the default theme. 11462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (reason != extension_misc::UNLOAD_REASON_UPDATE && extension->is_theme()) { 11472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ThemeService* theme_service = ThemeServiceFactory::GetForProfile(profile_); 11482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->id() == theme_service->GetThemeID()) 11492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) theme_service->UseDefaultTheme(); 11502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 11512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif 11522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 11535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (content::RenderProcessHost::iterator i( 11545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost::AllHostsIterator()); 11555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !i.IsAtEnd(); i.Advance()) { 11565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost* host = i.GetCurrentValue(); 11575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile* host_profile = 11585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile::FromBrowserContext(host->GetBrowserContext()); 11595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (host_profile->GetOriginalProfile() == profile_->GetOriginalProfile()) 11605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) host->Send(new ExtensionMsg_Unloaded(extension->id())); 11615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->UnregisterExtensionWithRequestContexts(extension->id(), reason); 11645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_->GetExtensionSpecialStoragePolicy()-> 11655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RevokeRightsForExtension(extension); 11665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_CHROMEOS) 11682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Revoke external file access for the extension from its file system context. 11692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // It is safe to access the extension's storage partition at this point. The 11702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // storage partition may get destroyed only after the extension gets unloaded. 11712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GURL site = extensions::ExtensionSystem::Get(profile_)->extension_service()-> 11722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetSiteForExtensionId(extension->id()); 11735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) fileapi::FileSystemContext* filesystem_context = 11742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) BrowserContext::GetStoragePartitionForSite(profile_, site)-> 11755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) GetFileSystemContext(); 11765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (filesystem_context && filesystem_context->external_provider()) { 11775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) filesystem_context->external_provider()-> 11785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RevokeAccessForExtension(extension->id()); 11795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 11815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateActiveExtensionsInCrashReporter(); 11835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 11842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(ENABLE_PLUGINS) 11855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool plugins_changed = false; 11862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extensions::PluginInfo::HasPlugins(extension)) { 11872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const extensions::PluginInfo::PluginVector* plugins = 11882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::PluginInfo::GetPlugins(extension); 11895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) plugins_changed = true; 11902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (extensions::PluginInfo::PluginVector::const_iterator plugin = 11912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugins->begin(); 11922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) plugin != plugins->end(); ++plugin) { 11932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) PluginService::GetInstance()->ForcePluginShutdown(plugin->path); 11942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) PluginService::GetInstance()->RefreshPlugins(); 11952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) PluginService::GetInstance()->RemoveExtraPluginPath(plugin->path); 11962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ChromePluginServiceFilter::GetInstance()->UnrestrictPlugin(plugin->path); 11972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 11985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 11995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool nacl_modules_changed = false; 12015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < extension->nacl_modules().size(); ++i) { 12025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension::NaClModuleInfo& module = extension->nacl_modules()[i]; 12035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnregisterNaClModule(module.url); 12045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) nacl_modules_changed = true; 12055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 12065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (nacl_modules_changed) 12085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdatePluginListWithNaClModules(); 12095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (plugins_changed || nacl_modules_changed) 12115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()->PurgePluginListCache(profile_, false); 12122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif // defined(ENABLE_PLUGINS) 12135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)Profile* ExtensionService::profile() { 12165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return profile_; 12175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::ExtensionPrefs* ExtensionService::extension_prefs() { 12205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_prefs_; 12215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::SettingsFrontend* ExtensionService::settings_frontend() { 12245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return settings_frontend_.get(); 12255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::ContentSettingsStore* ExtensionService::GetContentSettingsStore() { 12285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_prefs()->content_settings_store(); 12295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::is_ready() { 123290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) return ready_->is_signaled(); 12335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)base::SequencedTaskRunner* ExtensionService::GetFileTaskRunner() { 12362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (file_task_runner_) 12372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return file_task_runner_; 12382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 12392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // We should be able to interrupt any part of extension install process during 12402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // shutdown. SKIP_ON_SHUTDOWN ensures that not started extension install tasks 12412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // will be ignored/deleted while we will block on started tasks. 12422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::string token("ext_install-"); 12432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) token.append(profile_->GetPath().AsUTF8Unsafe()); 12442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) file_task_runner_ = BrowserThread::GetBlockingPool()-> 12452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetSequencedTaskRunnerWithShutdownBehavior( 12462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) BrowserThread::GetBlockingPool()->GetNamedSequenceToken(token), 12472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::SequencedWorkerPool::SKIP_ON_SHUTDOWN); 12482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return file_task_runner_; 12492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 12502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 12515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::ExtensionUpdater* ExtensionService::updater() { 12525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return updater_.get(); 12535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::CheckManagementPolicy() { 12565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<std::string> to_be_removed; 12572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 12585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Loop through extensions list, unload installed extensions. 12595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator iter = extensions_.begin(); 12605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != extensions_.end(); ++iter) { 12615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = (*iter); 12625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!system_->management_policy()->UserMayLoad(extension, NULL)) 12635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) to_be_removed.push_back(extension->id()); 12645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 12655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // UnloadExtension will change the extensions_ list. So, we should 12675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // call it outside the iterator loop. 12685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < to_be_removed.size(); ++i) 12695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadExtension(to_be_removed[i], extension_misc::UNLOAD_REASON_DISABLE); 12705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::CheckForUpdatesSoon() { 12735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (updater()) { 12745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (AreAllExternalProvidersReady()) { 12755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) updater()->CheckSoon(); 12765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 12775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Sync can start updating before all the external providers are ready 12785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // during startup. Start the update as soon as those providers are ready, 12795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // but not before. 12805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) update_once_all_providers_are_ready_ = true; 12815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 12825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 12835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "CheckForUpdatesSoon() called with auto-update turned off"; 12845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 12855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 12865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)syncer::SyncMergeResult ExtensionService::MergeDataAndStartSyncing( 12885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::ModelType type, 12895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const syncer::SyncDataList& initial_sync_data, 12905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_ptr<syncer::SyncChangeProcessor> sync_processor, 12915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_ptr<syncer::SyncErrorFactory> sync_error_factory) { 12925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(sync_processor.get()); 12935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(sync_error_factory.get()); 12945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 12955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switch (type) { 12965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case syncer::EXTENSIONS: 12975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.SetupSync(sync_processor.release(), 12985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_error_factory.release(), 12995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) initial_sync_data); 13005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 13015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case syncer::APPS: 13035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.SetupSync(sync_processor.release(), 13045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_error_factory.release(), 13055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) initial_sync_data); 13065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 13075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) default: 13095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(FATAL) << "Got " << type << " ModelType"; 13105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Process local extensions. 13135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(yoz): Determine whether pending extensions should be considered too. 13145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // See crbug.com/104399. 13155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::SyncDataList sync_data_list = GetAllSyncData(type); 13165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::SyncChangeList sync_change_list; 13175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (syncer::SyncDataList::const_iterator i = sync_data_list.begin(); 13185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != sync_data_list.end(); 13195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ++i) { 13205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switch (type) { 13215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case syncer::EXTENSIONS: 13225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change_list.push_back( 13235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.CreateSyncChange(*i)); 13245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 13255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case syncer::APPS: 13265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) sync_change_list.push_back(app_sync_bundle_.CreateSyncChange(*i)); 13275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 13285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) default: 13295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(FATAL) << "Got " << type << " ModelType"; 13305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::EXTENSIONS) { 13355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.ProcessSyncChangeList(sync_change_list); 13365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (type == syncer::APPS) { 13375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.ProcessSyncChangeList(sync_change_list); 13385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return syncer::SyncMergeResult(type); 13415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 13425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::StopSyncing(syncer::ModelType type) { 13445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::APPS) { 13455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.Reset(); 13465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (type == syncer::EXTENSIONS) { 13475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.Reset(); 13485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 13505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)syncer::SyncDataList ExtensionService::GetAllSyncData( 13525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::ModelType type) const { 13535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::EXTENSIONS) 13545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_sync_bundle_.GetAllSyncData(); 13555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::APPS) 13565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return app_sync_bundle_.GetAllSyncData(); 13575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We should only get sync data for extensions and apps. 13595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 13605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return syncer::SyncDataList(); 13625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 13635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)syncer::SyncError ExtensionService::ProcessSyncChanges( 13655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const tracked_objects::Location& from_here, 13665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const syncer::SyncChangeList& change_list) { 13675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (syncer::SyncChangeList::const_iterator i = change_list.begin(); 13685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != change_list.end(); 13695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ++i) { 13705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::ModelType type = i->sync_data().GetDataType(); 13715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::EXTENSIONS) { 13725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.ProcessSyncChange( 13735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionSyncData(*i)); 13745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (type == syncer::APPS) { 13755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.ProcessSyncChange(extensions::AppSyncData(*i)); 13765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 13785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs()->extension_sorting()->FixNTPOrdinalCollisions(); 13805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return syncer::SyncError(); 13825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 13835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::ExtensionSyncData ExtensionService::GetExtensionSyncData( 13855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension& extension) const { 13865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extensions::ExtensionSyncData(extension, 13875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IsExtensionEnabled(extension.id()), 13885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IsIncognitoEnabled(extension.id())); 13895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 13905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 13915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extensions::AppSyncData ExtensionService::GetAppSyncData( 13925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension& extension) const { 13935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extensions::AppSyncData( 13945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension, 13955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IsExtensionEnabled(extension.id()), 13965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IsIncognitoEnabled(extension.id()), 13975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->GetAppLaunchOrdinal( 13985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension.id()), 13995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->GetPageOrdinal(extension.id())); 14005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)std::vector<extensions::ExtensionSyncData> 14035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionService::GetExtensionSyncDataList() const { 14045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<extensions::ExtensionSyncData> extension_sync_list; 14055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.GetExtensionSyncDataListHelper(extensions_, 14065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_sync_list); 14075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.GetExtensionSyncDataListHelper(disabled_extensions_, 14085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_sync_list); 14095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.GetExtensionSyncDataListHelper(terminated_extensions_, 14105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_sync_list); 14115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<extensions::ExtensionSyncData> pending_extensions = 14135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.GetPendingData(); 14145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_list.insert(extension_sync_list.begin(), 14155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extensions.begin(), 14165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extensions.end()); 14175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_sync_list; 14195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)std::vector<extensions::AppSyncData> ExtensionService::GetAppSyncDataList() 14225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const { 14235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<extensions::AppSyncData> app_sync_list; 14245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.GetAppSyncDataListHelper(extensions_, &app_sync_list); 14255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.GetAppSyncDataListHelper(disabled_extensions_, 14265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &app_sync_list); 14275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.GetAppSyncDataListHelper(terminated_extensions_, 14285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &app_sync_list); 14295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<extensions::AppSyncData> pending_apps = 14315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.GetPendingData(); 14325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_list.insert(app_sync_list.begin(), 14335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_apps.begin(), 14345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_apps.end()); 14355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return app_sync_list; 14375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ProcessExtensionSyncData( 14405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::ExtensionSyncData& extension_sync_data) { 14415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!ProcessExtensionSyncDataHelper(extension_sync_data, 14425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::EXTENSIONS)) { 14435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.AddPendingExtension(extension_sync_data.id(), 14445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_data); 14455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CheckForUpdatesSoon(); 14465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 14475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 14485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 14505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ProcessAppSyncData( 14535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::AppSyncData& app_sync_data) { 14545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id = app_sync_data.id(); 14555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (app_sync_data.app_launch_ordinal().IsValid() && 14575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_data.page_ordinal().IsValid()) { 14585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->SetAppLaunchOrdinal( 14595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, 14605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_data.app_launch_ordinal()); 14615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->SetPageOrdinal( 14625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, 14635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_data.page_ordinal()); 14645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 14655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!ProcessExtensionSyncDataHelper(app_sync_data.extension_sync_data(), 14675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::APPS)) { 14685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.AddPendingApp(id, app_sync_data); 14695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CheckForUpdatesSoon(); 14705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 14715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 14725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 14745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsCorrectSyncType(const Extension& extension, 14775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::ModelType type) const { 14785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::EXTENSIONS && 14795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension.GetSyncType() == Extension::SYNC_TYPE_EXTENSION) { 14805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 14815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 14825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (type == syncer::APPS && 14845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension.GetSyncType() == Extension::SYNC_TYPE_APP) { 14855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 14865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 14875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 14895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 14905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ProcessExtensionSyncDataHelper( 14925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::ExtensionSyncData& extension_sync_data, 14935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) syncer::ModelType type) { 14945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id = extension_sync_data.id(); 14955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = GetInstalledExtension(id); 14965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 14975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(bolms): we should really handle this better. The particularly bad 14985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // case is where an app becomes an extension or vice versa, and we end up with 14995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // a zombie extension that won't go away. 15005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension && !IsCorrectSyncType(*extension, type)) 15015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 15025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Handle uninstalls first. 15045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_sync_data.uninstalled()) { 15055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::string error; 15065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!UninstallExtensionHelper(this, id)) { 15075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Could not uninstall extension " << id 15085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << " for sync"; 15095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 15115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extension from sync was uninstalled by the user as external extensions. 15145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Honor user choice and skip installation/enabling. 15155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsExternalExtensionUninstalled(id)) { 15165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Extension with id " << id 15175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << " from sync was uninstalled as external extension"; 15185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 15195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Set user settings. 15222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If the extension has been disabled from sync, it may not have 15232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // been installed yet, so we don't know if the disable reason was a 1524c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // permissions increase. That will be updated once CheckPermissionsIncrease 15252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // is called for it. 15265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_sync_data.enabled()) 15275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EnableExtension(id); 15285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) else 15292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DisableExtension(id, Extension::DISABLE_UNKNOWN_FROM_SYNC); 15305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We need to cache some version information here because setting the 15325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // incognito flag invalidates the |extension| pointer (it reloads the 15335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension). 15345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool extension_installed = (extension != NULL); 15355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int result = extension ? 15365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension->version()->CompareTo(extension_sync_data.version()) : 0; 15375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SetIsIncognitoEnabled(id, extension_sync_data.incognito_enabled()); 15385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension = NULL; // No longer safe to use. 15395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_installed) { 15415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension is already installed, check if it's outdated. 15425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (result < 0) { 15435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extension is outdated. 15445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 15455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 15475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(akalin): Replace silent update with a list of enabled 15485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // permissions. 15495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const bool kInstallSilently = true; 15505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(type == syncer::EXTENSIONS || type == syncer::APPS); 15525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionFilter filter = 15535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) (type == syncer::APPS) ? IsSyncableApp : IsSyncableExtension; 15545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pending_extension_manager()->AddFromSync( 15565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, 15575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_data.update_url(), 15585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) filter, 15595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) kInstallSilently)) { 15605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Could not add pending extension for " << id; 15615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This means that the extension is already pending installation, with a 15625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // non-INTERNAL location. Add to pending_sync_data, even though it will 15635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // never be removed (we'll never install a syncable version of the 15645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension), so that GetAllSyncData() continues to send it. 15655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Track pending extensions so that we can return them in GetAllSyncData(). 15675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 15685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 15695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 15715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 15725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsIncognitoEnabled( 15745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id) const { 15752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension = GetInstalledExtension(extension_id); 15762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension && !extension->can_be_incognito_enabled()) 15772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 15785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If this is an existing component extension we always allow it to 15795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // work in incognito mode. 15802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension && extension->location() == Manifest::COMPONENT) 15815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 15825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Check the prefs. 15845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension_prefs_->IsIncognitoEnabled(extension_id); 15855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 15865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SetIsIncognitoEnabled( 15885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id, bool enabled) { 15895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = GetInstalledExtension(extension_id); 15902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension && !extension->can_be_incognito_enabled()) 15912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 15922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension && extension->location() == Manifest::COMPONENT) { 15935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This shouldn't be called for component extensions unless they are 15945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // syncable. 15955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(extension->IsSyncable()); 15965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 15975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If we are here, make sure the we aren't trying to change the value. 15985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK_EQ(enabled, IsIncognitoEnabled(extension_id)); 15995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 16015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 16025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Broadcast unloaded and loaded events to update browser state. Only bother 16045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // if the value changed and the extension is actually enabled, since there is 16055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // no UI otherwise. 16065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool old_enabled = extension_prefs_->IsIncognitoEnabled(extension_id); 16075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (enabled == old_enabled) 16085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 16095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetIsIncognitoEnabled(extension_id, enabled); 16115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool extension_is_enabled = extensions_.Contains(extension_id); 16135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // When we reload the extension the ID may be invalidated if we've passed it 16155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // by const ref everywhere. Make a copy to be safe. 16165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::string id = extension_id; 16175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_is_enabled) 16185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ReloadExtension(id); 16195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Reloading the extension invalidates the |extension| pointer. 16215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension = GetInstalledExtension(id); 16225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension) 16235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 16245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::CanCrossIncognito(const Extension* extension) const { 16275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We allow the extension to see events and data from another profile iff it 16285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // uses "spanning" behavior and it has incognito access. "split" mode 16295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extensions only see events for a matching profile. 16305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(extension); 16315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return IsIncognitoEnabled(extension->id()) && 16322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !extensions::IncognitoInfo::IsSplitMode(extension); 16335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::CanLoadInIncognito(const Extension* extension) const { 16365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension->is_hosted_app()) 16375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 16385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Packaged apps and regular extensions need to be enabled specifically for 16395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // incognito (and split mode should be set). 16402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return extensions::IncognitoInfo::IsSplitMode(extension) && 16415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IsIncognitoEnabled(extension->id()); 16425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::OnExtensionMoved( 16455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& moved_extension_id, 16465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& predecessor_extension_id, 16475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& successor_extension_id) { 16485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->extension_sorting()->OnExtensionMoved( 16495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) moved_extension_id, 16505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) predecessor_extension_id, 16515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) successor_extension_id); 16525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = GetInstalledExtension(moved_extension_id); 16545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension) 16555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 16565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::AllowFileAccess(const Extension* extension) const { 16595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return (CommandLine::ForCurrentProcess()->HasSwitch( 16605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switches::kDisableExtensionsFileAccessCheck) || 16615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AllowFileAccess(extension->id())); 16625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SetAllowFileAccess(const Extension* extension, 16655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool allow) { 16665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Reload to update browser state. Only bother if the value changed and the 16675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension is actually enabled, since there is no UI otherwise. 16685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool old_allow = AllowFileAccess(extension); 16695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (allow == old_allow) 16705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 16715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetAllowFileAccess(extension->id(), allow); 16735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool extension_is_enabled = extensions_.Contains(extension->id()); 16755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_is_enabled) 16765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ReloadExtension(extension->id()); 16775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 16785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Some extensions will autoupdate themselves externally from Chrome. These 16805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// are typically part of some larger client application package. To support 16815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// these, the extension will register its location in the the preferences file 16825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// (and also, on Windows, in the registry) and this code will periodically 16835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// check that location for a .crx file, which it will then install locally if 16845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// a new version is available. 16855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Errors are reported through ExtensionErrorReporter. Succcess is not 16865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// reported. 16875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::CheckForExternalUpdates() { 16885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 16895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Note that this installation is intentionally silent (since it didn't 16915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // go through the front-end). Extensions that are registered in this 16925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // way are effectively considered 'pre-bundled', and so implicitly 16935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // trusted. In general, if something has HKLM or filesystem access, 16945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // they could install an extension manually themselves anyway. 16955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 16965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Ask each external extension provider to give us a call back for each 16975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension they know about. See OnExternalExtension(File|UpdateUrl)Found. 16985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ProviderCollection::const_iterator i; 16995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (i = external_extension_providers_.begin(); 17005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != external_extension_providers_.end(); ++i) { 17015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExternalProviderInterface* provider = i->get(); 17025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) provider->VisitRegisteredExtension(); 17035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Do any required work that we would have done after completion of all 17065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // providers. 17075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (external_extension_providers_.empty()) { 17085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) OnAllExternalProvidersReady(); 17095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 17115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::OnExternalProviderReady( 17135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::ExternalProviderInterface* provider) { 17145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 17155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(provider->IsReady()); 17165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // An external provider has finished loading. We only take action 17185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // if all of them are finished. So we check them first. 17195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (AreAllExternalProvidersReady()) 17205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) OnAllExternalProvidersReady(); 17215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 17225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::AreAllExternalProvidersReady() const { 17245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ProviderCollection::const_iterator i; 17255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (i = external_extension_providers_.begin(); 17265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) i != external_extension_providers_.end(); ++i) { 17275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!i->get()->IsReady()) 17285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 17295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 17315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 17325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::OnAllExternalProvidersReady() { 17345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 17355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::TimeDelta elapsed = base::Time::Now() - profile_->GetStartTime(); 17365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_TIMES("Extension.ExternalProvidersReadyAfter", elapsed); 17375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Install any pending extensions. 17395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (update_once_all_providers_are_ready_ && updater()) { 17405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) update_once_all_providers_are_ready_ = false; 17412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) updater()->CheckNow(extensions::ExtensionUpdater::CheckParams()); 17425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Uninstall all the unclaimed extensions. 17455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_ptr<extensions::ExtensionPrefs::ExtensionsInfo> extensions_info( 17465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->GetInstalledExtensionsInfo()); 17475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < extensions_info->size(); ++i) { 17485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionInfo* info = extensions_info->at(i).get(); 17492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (Manifest::IsExternalLocation(info->extension_location)) 17505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CheckExternalUninstall(info->extension_id); 17515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IdentifyAlertableExtensions(); 17535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 17545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::IdentifyAlertableExtensions() { 17565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 17575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Build up the lists of extensions that require acknowledgment. If this is 17595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // the first time, grandfather extensions that would have caused 17605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // notification. 17615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_error_ui_.reset(ExtensionErrorUI::Create(this)); 17625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool did_show_alert = false; 17645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (PopulateExtensionErrorUI(extension_error_ui_.get())) { 1765c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (!is_first_run_) { 17665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 17675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) did_show_alert = extension_error_ui_->ShowErrorInBubbleView(); 17685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 17695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // First run. Just acknowledge all the extensions, silently, by 17705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // shortcutting the display of the UI and going straight to the 17715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // callback for pressing the Accept button. 17725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) HandleExtensionAlertAccept(); 17735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 17755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateExternalExtensionAlert(); 17775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!did_show_alert) 17795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_error_ui_.reset(); 17805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 17815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 17825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::PopulateExtensionErrorUI( 17835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionErrorUI* extension_error_ui) { 17845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool needs_alert = false; 17852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 17862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Extensions that are blacklisted. 17872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (ExtensionSet::const_iterator it = blacklisted_extensions_.begin(); 17882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != blacklisted_extensions_.end(); ++it) { 17892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::string id = (*it)->id(); 17902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension_prefs_->IsBlacklistedExtensionAcknowledged(id)) { 17912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_error_ui->AddBlacklistedExtension(id); 17922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) needs_alert = true; 17932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 17942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 17952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 17965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator iter = extensions_.begin(); 17975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != extensions_.end(); ++iter) { 17985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* e = *iter; 17992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 18002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Extensions disabled by policy. Note: this no longer includes blacklisted 18012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // extensions, though we still show the same UI. 18022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!system_->management_policy()->UserMayLoad(e, NULL)) { 18035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension_prefs_->IsBlacklistedExtensionAcknowledged(e->id())) { 18045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_error_ui->AddBlacklistedExtension(e->id()); 18055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) needs_alert = true; 18065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 18105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return needs_alert; 18115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::HandleExtensionAlertClosed() { 18145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const ExtensionIdSet* extension_ids = 18155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_error_ui_->get_blacklisted_extension_ids(); 18165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionIdSet::const_iterator iter = extension_ids->begin(); 18175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != extension_ids->end(); ++iter) { 18185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AcknowledgeBlacklistedExtension(*iter); 18195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_error_ui_.reset(); 18212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 18222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 18232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::HandleExtensionAlertAccept() { 1824c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_error_ui_->Close(); 18255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::AcknowledgeExternalExtension(const std::string& id) { 18285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AcknowledgeExternalExtension(id); 18295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateExternalExtensionAlert(); 18305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsUnacknowledgedExternalExtension( 18335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension) { 18345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!FeatureSwitch::prompt_for_external_extensions()->IsEnabled()) 18355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 18365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return (Manifest::IsExternalLocation(extension->location()) && 18385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !extension_prefs_->IsExternalExtensionAcknowledged(extension->id()) && 18395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !(extension_prefs_->GetDisableReasons(extension->id()) & 18405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Extension::DISABLE_SIDELOAD_WIPEOUT)); 18415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::HandleExtensionAlertDetails() { 18445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_error_ui_->ShowExtensions(); 1845c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // ShowExtensions may cause the error UI to close synchronously, e.g. if it 1846c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // causes a navigation. 1847c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (extension_error_ui_) 1848c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_error_ui_->Close(); 18495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UpdateExternalExtensionAlert() { 18525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!FeatureSwitch::prompt_for_external_extensions()->IsEnabled()) 18535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 18545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = NULL; 18565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator iter = disabled_extensions_.begin(); 18575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != disabled_extensions_.end(); ++iter) { 18585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* e = *iter; 18595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsUnacknowledgedExternalExtension(e)) { 18605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension = e; 18615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 18625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension) { 18665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extensions::HasExternalInstallError(this)) { 18675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_prefs_->IncrementAcknowledgePromptCount(extension->id()) > 18685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) kMaxExtensionAcknowledgePromptCount) { 18695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Stop prompting for this extension, and check if there's another 18705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // one that needs prompting. 18715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AcknowledgeExternalExtension(extension->id()); 18725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateExternalExtensionAlert(); 18735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.ExternalExtensionEvent", 18745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_IGNORED, 18755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) EXTERNAL_EXTENSION_BUCKET_BOUNDARY); 18765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 18775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 1878c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (is_first_run_) 1879c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_prefs_->SetExternalInstallFirstRun(extension->id()); 1880c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // first_run is true if the extension was installed during a first run 1881c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // (even if it's post-first run now). 1882c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool first_run = extension_prefs_->IsExternalInstallFirstRun( 1883c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension->id()); 1884c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extensions::AddExternalInstallError(this, extension, first_run); 18855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 18875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::RemoveExternalInstallError(this); 18885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 18895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 18905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UnloadExtension( 18925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id, 18935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_misc::UnloadedExtensionReason reason) { 18945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Make sure the extension gets deleted after we return from this function. 18952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) int include_mask = INCLUDE_EVERYTHING & ~INCLUDE_TERMINATED; 18965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<const Extension> extension( 18972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetExtensionById(extension_id, include_mask)); 18985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 18995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This method can be called via PostTask, so the extension may have been 19005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // unloaded by the time this runs. 19015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension) { 19025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // In case the extension may have crashed/uninstalled. Allow the profile to 19035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // clean up its RequestContexts. 19045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->UnregisterExtensionWithRequestContexts(extension_id, reason); 19055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 19065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 19075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 190890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // If uninstalling let RuntimeEventRouter know. 190990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (reason == extension_misc::UNLOAD_REASON_UNINSTALL) 191090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extensions::RuntimeEventRouter::OnExtensionUninstalled( 191190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) profile_, extension_id); 191290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) 19135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Keep information about the extension so that we can reload it later 19145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // even if it's not permanently installed. 19155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) unloaded_extension_paths_[extension->id()] = extension->path(); 19165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Clean up if the extension is meant to be enabled after a reload. 1918c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) reloading_extensions_.erase(extension->id()); 19195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Clean up runtime data. 19215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_.erase(extension_id); 19225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (disabled_extensions_.Contains(extension->id())) { 19245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadedExtensionInfo details(extension, reason); 19255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) details.already_disabled = true; 19265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) disabled_extensions_.Remove(extension->id()); 19275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 19285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_UNLOADED, 19295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 19305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<UnloadedExtensionInfo>(&details)); 19315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Make sure the profile cleans up its RequestContexts when an already 19325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // disabled extension is unloaded (since they are also tracking the disabled 19335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extensions). 19345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->UnregisterExtensionWithRequestContexts(extension_id, reason); 19355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 19365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 19375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Remove the extension from our list. 19395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_.Remove(extension->id()); 19405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NotifyExtensionUnloaded(extension.get(), reason); 19425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 19435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UnloadAllExtensions() { 19455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_->GetExtensionSpecialStoragePolicy()->RevokeRightsForAllExtensions(); 19465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions_.Clear(); 19485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) disabled_extensions_.Clear(); 19495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) terminated_extensions_.Clear(); 19505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_.clear(); 19515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(erikkay) should there be a notification for this? We can't use 19535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // EXTENSION_UNLOADED since that implies that the extension has been disabled 19545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // or uninstalled, and UnloadAll is just part of shutdown. 19555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 19565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::ReloadExtensions() { 19585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadAllExtensions(); 19595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) component_loader_->LoadAll(); 19605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::InstalledLoader(this).LoadAllExtensions(); 196190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // Don't call SetReadyAndNotifyListeners() since tests call this multiple 196290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // times. 19635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 19645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::GarbageCollectExtensions() { 19665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_prefs_->pref_service()->ReadOnly()) 19675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 19685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (pending_extension_manager()->HasPendingExtensions()) { 19702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Don't garbage collect while there are pending installations, which may 19712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // be using the temporary installation directory. Try to garbage collect 19722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // again later. 197390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) base::MessageLoop::current()->PostDelayedTask( 19742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FROM_HERE, 19752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::GarbageCollectExtensions, AsWeakPtr()), 19762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::TimeDelta::FromSeconds(kGarbageCollectRetryDelay)); 19772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 19782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 19792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 19805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_ptr<extensions::ExtensionPrefs::ExtensionsInfo> info( 19815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->GetInstalledExtensionsInfo()); 19825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::multimap<std::string, base::FilePath> extension_paths; 19845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < info->size(); ++i) 19855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_paths.insert(std::make_pair(info->at(i)->extension_id, 19865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) info->at(i)->extension_path)); 19875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) info = extension_prefs_->GetAllDelayedInstallInfo(); 19895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (size_t i = 0; i < info->size(); ++i) 19905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_paths.insert(std::make_pair(info->at(i)->extension_id, 19915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) info->at(i)->extension_path)); 19925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 19932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!GetFileTaskRunner()->PostTask( 19942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FROM_HERE, 19955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind( 19965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &extension_file_util::GarbageCollectExtensions, 19975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) install_directory_, 19985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_paths))) { 19995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 20005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 20015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(ENABLE_THEMES) 20035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Also garbage-collect themes. We check |profile_| to be 20045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // defensive; in the future, we may call GarbageCollectExtensions() 20055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // from somewhere other than Init() (e.g., in a timer). 20065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (profile_) { 20075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ThemeServiceFactory::GetForProfile(profile_)->RemoveUnusedThemes(); 20085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 20095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 20105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 20115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SyncExtensionChangeIfNeeded(const Extension& extension) { 20135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (app_sync_bundle_.HandlesApp(extension)) { 20145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) app_sync_bundle_.SyncChangeIfNeeded(extension); 20155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (extension_sync_bundle_.HandlesExtension(extension)) { 20165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_sync_bundle_.SyncChangeIfNeeded(extension); 20175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 20185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 20195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2020c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void ExtensionService::SetReadyAndNotifyListeners() { 202190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) ready_->Signal(); 20225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 20235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSIONS_READY, 20245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 20255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::NoDetails()); 20265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 20275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2028c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void ExtensionService::OnLoadedInstalledExtensions() { 2029c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (updater_) 2030c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) updater_->Start(); 2031c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 2032c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) OnBlacklistUpdated(); 2033c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)} 2034c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 20355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::AddExtension(const Extension* extension) { 20365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(jstritar): We may be able to get rid of this branch by overriding the 20375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // default extension state to DISABLED when the --disable-extensions flag 20385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // is set (http://crbug.com/29067). 20395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extensions_enabled() && 20405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !extension->is_theme() && 20412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location() != Manifest::COMPONENT && 20422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !Manifest::IsExternalLocation(extension->location())) { 20435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 20445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 20455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2046c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool is_extension_upgrade = false; 2047c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (const Extension* old = GetInstalledExtension(extension->id())) { 2048c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) is_extension_upgrade = true; 2049c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) DCHECK_NE(extension, old); 2050c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Other than for unpacked extensions, CrxInstaller should have guaranteed 2051c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // that we aren't downgrading. 2052c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (!Manifest::IsUnpackedLocation(extension->location())) 2053c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) CHECK_GE(extension->version()->CompareTo(*(old->version())), 0); 2054c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } 2055c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) SetBeingUpgraded(extension, is_extension_upgrade); 20565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The extension is now loaded, remove its data from unloaded extension map. 20585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) unloaded_extension_paths_.erase(extension->id()); 20595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If a terminated extension is loaded, remove it from the terminated list. 20615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UntrackTerminatedExtension(extension->id()); 20625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension was disabled for a reload, then enable it. 2064c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool reloading = reloading_extensions_.erase(extension->id()) > 0; 20655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2066c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Check if the extension's privileges have changed and mark the 2067c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // extension disabled if necessary. 2068c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) CheckPermissionsIncrease(extension, is_extension_upgrade); 20695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2070c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (is_extension_upgrade && !reloading) { 2071c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // To upgrade an extension in place, unload the old one and then load the 2072c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // new one. ReloadExtension disables the extension, which is sufficient. 2073c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) UnloadExtension(extension->id(), extension_misc::UNLOAD_REASON_UPDATE); 2074c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } 20755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension_prefs_->IsExtensionBlacklisted(extension->id())) { 20772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Only prefs is checked for the blacklist. We rely on callers to check the 20782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // blacklist before calling into here, e.g. CrxInstaller checks before 20792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // installation, we check when loading installed extensions. 20802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.Insert(extension); 2081c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } else if (!reloading && 2082c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_prefs_->IsExtensionDisabled(extension->id())) { 20832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) disabled_extensions_.Insert(extension); 20845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 20855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 20865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_UPDATE_DISABLED, 20875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 20885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 20895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 20902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Show the extension disabled error if a permissions increase was the 20912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // only reason it was disabled. 20922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension_prefs_->GetDisableReasons(extension->id()) == 20935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Extension::DISABLE_PERMISSIONS_INCREASE) { 20945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::AddExtensionDisabledError(this, extension); 20955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2096c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) } else if (reloading) { 2097c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Replace the old extension with the new version. 2098c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) CHECK(!disabled_extensions_.Insert(extension)); 2099c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) EnableExtension(extension->id()); 2100c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) DoPostLoadTasks(extension); 21012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 21022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // All apps that are displayed in the launcher are ordered by their ordinals 21032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // so we must ensure they have valid ordinals. 21042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->RequiresSortOrdinal()) { 21052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension->ShouldDisplayInNewTabPage()) { 21062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->extension_sorting()->MarkExtensionAsHidden( 21072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->id()); 21082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 21092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->extension_sorting()->EnsureValidOrdinals( 21102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->id(), syncer::StringOrdinal()); 21112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 21122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 21132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions_.Insert(extension); 21142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) SyncExtensionChangeIfNeeded(*extension); 21152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NotifyExtensionLoaded(extension); 21162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DoPostLoadTasks(extension); 21175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2118c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) SetBeingUpgraded(extension, false); 21192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 21202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 21212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::AddComponentExtension(const Extension* extension) { 21222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string old_version_string( 21232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->GetVersionString(extension->id())); 21242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Version old_version(old_version_string); 21255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!old_version.IsValid() || !old_version.Equals(*extension->version())) { 21272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) VLOG(1) << "Component extension " << extension->name() << " (" 21282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) << extension->id() << ") installing/upgrading from '" 21292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) << old_version_string << "' to " << extension->version()->GetString(); 21302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 21312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AddNewOrUpdatedExtension(extension, 21322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Extension::ENABLED_COMPONENT, 21332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) syncer::StringOrdinal()); 21342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 21355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 21365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AddExtension(extension); 21385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 21395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2140c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void ExtensionService::UpdateActivePermissions(const Extension* extension) { 21415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension has used the optional permissions API, it will have a 21425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // custom set of active permissions defined in the extension prefs. Here, 21435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // we update the extension's active permissions based on the prefs. 21445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<PermissionSet> active_permissions = 21455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs()->GetActivePermissions(extension->id()); 21465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 2147c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (active_permissions) { 21485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We restrict the active permissions to be within the bounds defined in the 21495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension's manifest. 21505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // a) active permissions must be a subset of optional + default permissions 21515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // b) active permissions must contains all default permissions 21525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<PermissionSet> total_permissions = 21535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PermissionSet::CreateUnion( 2154b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles) extensions::PermissionsData::GetRequiredPermissions(extension), 2155b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles) extensions::PermissionsData::GetOptionalPermissions(extension)); 21565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Make sure the active permissions contain no more than optional + default. 21585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<PermissionSet> adjusted_active = 21595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PermissionSet::CreateIntersection( 21605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) total_permissions.get(), active_permissions.get()); 21615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Make sure the active permissions contain the default permissions. 21635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) adjusted_active = PermissionSet::CreateUnion( 2164b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles) extensions::PermissionsData::GetRequiredPermissions(extension), 2165b2df76ea8fec9e32f6f3718986dba0d95315b29cTorne (Richard Coles) adjusted_active.get()); 21665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::PermissionsUpdater perms_updater(profile()); 21685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) perms_updater.UpdateActivePermissions(extension, adjusted_active); 21695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2170c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)} 2171c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 2172c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void ExtensionService::CheckPermissionsIncrease(const Extension* extension, 2173c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool is_extension_upgrade) { 2174c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) UpdateActivePermissions(extension); 21755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 21765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We keep track of all permissions the user has granted each extension. 21775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This allows extensions to gracefully support backwards compatibility 21785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // by including unknown permissions in their manifests. When the user 21795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installs the extension, only the recognized permissions are recorded. 21805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // When the unknown permissions become recognized (e.g., through browser 21815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // upgrade), we can prompt the user to accept these new permissions. 21825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extensions can also silently upgrade to less permissions, and then 21835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // silently upgrade to a version that adds these permissions back. 21845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // 21855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // For example, pretend that Chrome 10 includes a permission "omnibox" 21865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // for an API that adds suggestions to the omnibox. An extension can 21875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // maintain backwards compatibility while still having "omnibox" in the 21885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // manifest. If a user installs the extension on Chrome 9, the browser 21895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // will record the permissions it recognized, not including "omnibox." 21905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // When upgrading to Chrome 10, "omnibox" will be recognized and Chrome 21915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // will disable the extension and prompt the user to approve the increase 21925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // in privileges. The extension could then release a new version that 21935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // removes the "omnibox" permission. When the user upgrades, Chrome will 21945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // still remember that "omnibox" had been granted, so that if the 21955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension once again includes "omnibox" in an upgrade, the extension 21965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // can upgrade without requiring this user's approval. 21975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int disable_reasons = extension_prefs_->GetDisableReasons(extension->id()); 21985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 219990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) bool auto_grant_permission = 220090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) (!is_extension_upgrade && extension->was_installed_by_default()) || 220190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) chrome::IsRunningInForcedAppMode(); 2202c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // Silently grant all active permissions to default apps only on install. 2203c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // After install they should behave like other apps. 220490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // Silently grant all active permissions to apps install in kiosk mode on both 220590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) // install and update. 220690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (auto_grant_permission) 2207c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) GrantPermissions(extension); 2208c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 2209c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool is_privilege_increase = false; 2210c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // We only need to compare the granted permissions to the current permissions 2211c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) // if the extension is not allowed to silently increase its permissions. 221290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (!extensions::PermissionsData::CanSilentlyIncreasePermissions(extension) && 221390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) !auto_grant_permission) { 22145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Add all the recognized permissions if the granted permissions list 22155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // hasn't been initialized yet. 22165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<PermissionSet> granted_permissions = 22175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->GetGrantedPermissions(extension->id()); 22185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(granted_permissions.get()); 22195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Here, we check if an extension's privileges have increased in a manner 22215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // that requires the user's approval. This could occur because the browser 22225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // upgraded and recognized additional privileges, or an extension upgrades 22235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // to a version that requires additional privileges. 22245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) is_privilege_increase = 22255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) granted_permissions->HasLessPrivilegesThan( 22265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension->GetActivePermissions()); 22275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 22285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (is_extension_upgrade) { 22305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension was already disabled, suppress any alerts for becoming 22315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // disabled on permissions increase. 2232c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool previously_disabled = 2233c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) extension_prefs_->IsExtensionDisabled(extension->id()); 22342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Legacy disabled extensions do not have a disable reason. Infer that if 22352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // there was no permission increase, it was likely disabled by the user. 22362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (previously_disabled && disable_reasons == Extension::DISABLE_NONE && 2237c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) !extension_prefs_->DidExtensionEscalatePermissions(extension->id())) { 22382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) disable_reasons |= Extension::DISABLE_USER_ACTION; 22392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 22402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Extensions that came to us disabled from sync need a similar inference, 22412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // except based on the new version's permissions. 22422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (previously_disabled && 22432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) disable_reasons == Extension::DISABLE_UNKNOWN_FROM_SYNC) { 22442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Remove the DISABLE_UNKNOWN_FROM_SYNC reason. 22452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->ClearDisableReasons(extension->id()); 22462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!is_privilege_increase) 22472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) disable_reasons |= Extension::DISABLE_USER_ACTION; 22485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2249c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) disable_reasons &= ~Extension::DISABLE_UNKNOWN_FROM_SYNC; 22505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 22515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extension has changed permissions significantly. Disable it. A 22535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // notification should be sent by the caller. 22545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (is_privilege_increase) { 22552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) disable_reasons |= Extension::DISABLE_PERMISSIONS_INCREASE; 22565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension_prefs_->DidExtensionEscalatePermissions(extension->id())) { 22575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RecordPermissionMessagesHistogram( 22585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension, "Extensions.Permissions_AutoDisable"); 22595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 22605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetExtensionState(extension->id(), Extension::DISABLED); 22615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->SetDidExtensionEscalatePermissions(extension, true); 22625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 2263c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (disable_reasons != Extension::DISABLE_NONE) { 22642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->AddDisableReason( 22652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->id(), 2266c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) static_cast<Extension::DisableReason>(disable_reasons)); 22675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 22685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 22695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UpdateActiveExtensionsInCrashReporter() { 22715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::set<std::string> extension_ids; 22725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator iter = extensions_.begin(); 22735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != extensions_.end(); ++iter) { 22745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = *iter; 22752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension->is_theme() && extension->location() != Manifest::COMPONENT) 22765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_ids.insert(extension->id()); 22775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 22785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) child_process_logging::SetActiveExtensions(extension_ids); 22805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 22815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::ScheduleLaunchOnLoad(const std::string& extension_id) { 22832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) on_load_events_[extension_id] = EVENT_LAUNCHED; 22842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 22852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 22865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::OnExtensionInstalled( 22875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension, 22885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const syncer::StringOrdinal& page_ordinal, 22895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool has_requirement_errors, 22905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool wait_for_idle) { 22915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 22925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 22935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id = extension->id(); 22945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool initial_enable = ShouldEnableOnInstall(extension); 22955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const extensions::PendingExtensionInfo* pending_extension_info = NULL; 22965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if ((pending_extension_info = pending_extension_manager()->GetById(id))) { 22975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pending_extension_info->ShouldAllowInstall(*extension)) { 22985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extension_manager()->Remove(id); 22995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "ShouldAllowInstall() returned false for " 23015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << id << " of type " << extension->GetType() 23022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) << " and update URL " 23032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) << extensions::ManifestURL::GetUpdateURL(extension).spec() 23045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "; not installing"; 23055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 23075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_INSTALL_NOT_ALLOWED, 23085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 23095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const Extension>(extension)); 23105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Delete the extension directory since we're not going to 23125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // load it. 23132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!GetFileTaskRunner()->PostTask( 23142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FROM_HERE, 23155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind(&extension_file_util::DeleteFile, 23165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension->path(), true))) { 23175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 23185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 23205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pending_extension_manager()->Remove(id); 23235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 23245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // We explicitly want to re-enable an uninstalled external 23255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // extension; if we're here, that means the user is manually 23265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installing the extension. 23275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (IsExternalExtensionUninstalled(id)) { 23285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) initial_enable = true; 23295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Unsupported requirements overrides the management policy. 23335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (has_requirement_errors) { 23345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) initial_enable = false; 23355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->AddDisableReason( 23365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, Extension::DISABLE_UNSUPPORTED_REQUIREMENT); 23375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension was disabled because of unsupported requirements but 23385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // now supports all requirements after an update and there are not other 23395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // disable reasons, enable it. 23405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else if (extension_prefs_->GetDisableReasons(id) == 23415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Extension::DISABLE_UNSUPPORTED_REQUIREMENT) { 23425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) initial_enable = true; 23435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_prefs_->ClearDisableReasons(id); 23445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!GetInstalledExtension(extension->id())) { 23475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.InstallType", 23485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension->GetType(), 100); 23492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.InstallSource", 23502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location(), Manifest::NUM_LOCATIONS); 23515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) RecordPermissionMessagesHistogram( 23525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension, "Extensions.Permissions_Install"); 23532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 23542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.UpdateType", 23552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->GetType(), 100); 23562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.UpdateSource", 23572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location(), Manifest::NUM_LOCATIONS); 23585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Certain extension locations are specific enough that we can 23615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // auto-acknowledge any extension that came from one of them. 23622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->location() == Manifest::EXTERNAL_POLICY_DOWNLOAD) 23635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AcknowledgeExternalExtension(extension->id()); 23642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension::State initial_state = 23652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) initial_enable ? Extension::ENABLED : Extension::DISABLED; 23662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (ShouldDelayExtensionUpdate(id, wait_for_idle)) { 23672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->SetDelayedInstallInfo(extension, initial_state, 23682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) page_ordinal); 23695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Transfer ownership of |extension|. 23712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_updates_for_idle_.Insert(extension); 23725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Notify extension of available update. 23745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::RuntimeEventRouter::DispatchOnUpdateAvailableEvent( 23755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_, id, extension->manifest()->value()); 23762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 23772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Notify observers that app update is available. 23782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FOR_EACH_OBSERVER(extensions::UpdateObserver, update_observers_, 23792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) OnAppUpdateAvailable(extension->id())); 23805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 23815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (installs_delayed()) { 23842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->SetDelayedInstallInfo(extension, initial_state, 23852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) page_ordinal); 23862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_installs_.Insert(extension); 23872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 23882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AddNewOrUpdatedExtension(extension, initial_state, page_ordinal); 23895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 23902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 23915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::AddNewOrUpdatedExtension( 23932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* extension, 23942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Extension::State initial_state, 23952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const syncer::StringOrdinal& page_ordinal) { 23962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 23975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 23982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->OnExtensionInstalled( 23992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension, 24002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) initial_state, 24012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) page_ordinal); 24025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FinishInstallation(extension); 24045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::MaybeFinishDelayedInstallation( 24075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& extension_id) { 24085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Check if the extension already got updated. 24092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!delayed_updates_for_idle_.Contains(extension_id)) 24105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 24112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Check if the extension is idle. 24125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!IsExtensionIdle(extension_id)) 24135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 24145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FinishDelayedInstallation(extension_id); 24165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::FinishDelayedInstallation( 24192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& extension_id) { 24202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_refptr<const Extension> extension( 24212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetPendingExtensionUpdate(extension_id)); 24225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(extension); 24232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_updates_for_idle_.Remove(extension_id); 24245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension_prefs_->FinishDelayedInstallInfo(extension_id)) 24265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED(); 24275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FinishInstallation(extension); 24292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 24302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 24312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::FinishInstallation(const Extension* extension) { 243290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) const extensions::Extension* existing_extension = 243390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) GetInstalledExtension(extension->id()); 243490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) bool is_update = false; 243590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) std::string old_name; 243690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) if (existing_extension) { 243790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) is_update = true; 243890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) old_name = existing_extension->name(); 243990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) } 244090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extensions::InstalledExtensionInfo details(extension, is_update, old_name); 24415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 24425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_INSTALLED, 24435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 2444c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) content::Details<const extensions::InstalledExtensionInfo>(&details)); 24455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) bool unacknowledged_external = IsUnacknowledgedExternalExtension(extension); 24472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 24482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Unpacked extensions default to allowing file access, but if that has been 24492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // overridden, don't reset the value. 24502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (Manifest::ShouldAlwaysAllowFileAccess(extension->location()) && 24512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) !extension_prefs_->HasAllowFileAccessSetting(extension->id())) { 24522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->SetAllowFileAccess(extension->id(), true); 24532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 24542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 24552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AddExtension(extension); 24562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 24572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(ENABLE_THEMES) 24582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // We do this here since AddExtension() is always called on browser startup, 24592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // and we only really care about the last theme installed. 24602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If that ever changes and we have to move this code somewhere 24612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // else, it should be somewhere that's not in the startup path. 24622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->is_theme() && extensions_.GetByID(extension->id())) { 24632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK_EQ(extensions_.GetByID(extension->id()), extension); 24642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Now that the theme extension is visible from outside the 24652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // ExtensionService, notify the ThemeService about the 24662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // newly-installed theme. 24672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ThemeServiceFactory::GetForProfile(profile_)->SetTheme(extension); 24682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 24692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif 24705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If this is a new external extension that was disabled, alert the user 24722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // so he can reenable it. We do this last so that it has already been 24732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // added to our list of extensions. 24742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (unacknowledged_external) { 24755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UpdateExternalExtensionAlert(); 24762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("Extensions.ExternalExtensionEvent", 24772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) EXTERNAL_EXTENSION_INSTALLED, 24782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) EXTERNAL_EXTENSION_BUCKET_BOUNDARY); 24792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 24805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)const Extension* ExtensionService::GetPendingExtensionUpdate( 24832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& id) const { 24842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return delayed_updates_for_idle_.GetByID(id); 24855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::TrackTerminatedExtension(const Extension* extension) { 24885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!terminated_extensions_.Contains(extension->id())) 24895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) terminated_extensions_.Insert(make_scoped_refptr(extension)); 24905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnloadExtension(extension->id(), extension_misc::UNLOAD_REASON_TERMINATE); 24925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UntrackTerminatedExtension(const std::string& id) { 24955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::string lowercase_id = StringToLowerASCII(id); 24965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) terminated_extensions_.Remove(lowercase_id); 24975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 24985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 24995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const Extension* ExtensionService::GetTerminatedExtension( 25005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id) const { 25012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return GetExtensionById(id, INCLUDE_TERMINATED); 25025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 25035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const Extension* ExtensionService::GetInstalledExtension( 25055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id) const { 25062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) int include_mask = INCLUDE_ENABLED | 25072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) INCLUDE_DISABLED | 25082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) INCLUDE_TERMINATED | 25092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) INCLUDE_BLACKLISTED; 25102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return GetExtensionById(id, include_mask); 25115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 25125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ExtensionBindingsAllowed(const GURL& url) { 25145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Allow bindings for all packaged extensions and component hosted apps. 25155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = extensions_.GetExtensionOrAppByURL( 25165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionURLInfo(url)); 25175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return extension && (!extension->is_hosted_app() || 25182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location() == Manifest::COMPONENT); 25195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 25205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ShouldBlockUrlInBrowserTab(GURL* url) { 25225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* extension = extensions_.GetExtensionOrAppByURL( 25235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionURLInfo(*url)); 25245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension && extension->is_platform_app()) { 25255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *url = GURL(chrome::kExtensionInvalidRequestURL); 25265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 25275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 25285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 25305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 25315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::OnExternalExtensionFileFound( 25335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& id, 25345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Version* version, 25352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const base::FilePath& path, 25362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Manifest::Location location, 25375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int creation_flags, 25385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool mark_acknowledged) { 25395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 25405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CHECK(Extension::IdIsValid(id)); 25415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_prefs_->IsExternalExtensionUninstalled(id)) 25425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 25435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Before even bothering to unpack, check and see if we already have this 25455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // version. This is important because these extensions are going to get 25465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installed on every startup. 25475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* existing = GetExtensionById(id, true); 25485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (existing) { 25505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // The default apps will have the location set as INTERNAL. Since older 25515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // default apps are installed as EXTERNAL, we override them. However, if the 25525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // app is already installed as internal, then do the version check. 25535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // TODO(grv) : Remove after Q1-2013. 25545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool is_default_apps_migration = 25552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) (location == Manifest::INTERNAL && 25562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Manifest::IsExternalLocation(existing->location())); 25575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!is_default_apps_migration) { 25595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(version); 25605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switch (existing->version()->CompareTo(*version)) { 25625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case -1: // existing version is older, we should upgrade 25635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 25645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case 0: // existing version is same, do nothing 25655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 25665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case 1: // existing version is newer, uh-oh 25675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) LOG(WARNING) << "Found external version of extension " << id 25685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "that is older than current version. Current version " 25695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "is: " << existing->VersionString() << ". New " 25705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << "version is: " << version->GetString() 25715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) << ". Keeping current version."; 25725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 25735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 25745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 25755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 25765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // If the extension is already pending, don't start an install. 25785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pending_extension_manager()->AddFromExternalFile( 25795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) id, location, *version)) { 25805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 25815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 25825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // no client (silent install) 25845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) scoped_refptr<CrxInstaller> installer(CrxInstaller::Create(this, NULL)); 25855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_install_source(location); 25865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_expected_id(id); 25875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_expected_version(*version); 25885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_install_cause(extension_misc::INSTALL_CAUSE_EXTERNAL_FILE); 25895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->set_creation_flags(creation_flags); 25902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#if defined(OS_CHROMEOS) 25912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::InstallLimiter::Get(profile_)->Add(installer, path); 25922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#else 25935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) installer->InstallCrx(path); 25942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#endif 25955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 25965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Depending on the source, a new external extension might not need a user 25975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // notification on installation. For such extensions, mark them acknowledged 25985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // now to suppress the notification. 25995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (mark_acknowledged) 26005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AcknowledgeExternalExtension(id); 26015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 26035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 26045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::ReportExtensionLoadError( 26062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const base::FilePath& extension_path, 26075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string &error, 26085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool be_noisy) { 26095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 26105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_LOAD_ERROR, 26115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(profile_), 26125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<const std::string>(&error)); 26135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::string path_str = UTF16ToUTF8(extension_path.LossyDisplayName()); 26155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) string16 message = UTF8ToUTF16(base::StringPrintf( 26165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) "Could not load extension from '%s'. %s", 26175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) path_str.c_str(), error.c_str())); 26185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionErrorReporter::GetInstance()->ReportError(message, be_noisy); 26195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 26205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::DidCreateRenderViewForBackgroundPage( 26225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host) { 26235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) OrphanedDevTools::iterator iter = 26245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) orphaned_dev_tools_.find(host->extension_id()); 26255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (iter == orphaned_dev_tools_.end()) 26265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 26275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DevToolsAgentHost::ConnectRenderViewHost(iter->second, 26292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) host->render_view_host()); 26305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) orphaned_dev_tools_.erase(iter); 26315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 26325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::Observe(int type, 26345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const content::NotificationSource& source, 26355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const content::NotificationDetails& details) { 26365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) switch (type) { 26375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case chrome::NOTIFICATION_APP_TERMINATING: 26385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Shutdown has started. Don't start any more extension installs. 26395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // (We cannot use ExtensionService::Shutdown() for this because it 26405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // happens too late in browser teardown.) 26415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) browser_terminating_ = true; 26425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 26435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case chrome::NOTIFICATION_EXTENSION_PROCESS_TERMINATED: { 26445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (profile_ != 26455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<Profile>(source).ptr()->GetOriginalProfile()) { 26465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 26475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 26485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host = 26505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<extensions::ExtensionHost>(details).ptr(); 26515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Mark the extension as terminated and Unload it. We want it to 26535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // be in a consistent state: either fully working or not loaded 26545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // at all, but never half-crashed. We do it in a PostTask so 26555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // that other handlers of this notification will still have 26565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // access to the Extension and ExtensionHost. 265790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) base::MessageLoop::current()->PostTask( 26585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) FROM_HERE, 26595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind( 26605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) &ExtensionService::TrackTerminatedExtension, 26615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AsWeakPtr(), 26625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) host->extension())); 26635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 26645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 26655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case content::NOTIFICATION_RENDERER_PROCESS_CREATED: { 26665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost* process = 26675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<content::RenderProcessHost>(source).ptr(); 26685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile* host_profile = 26695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile::FromBrowserContext(process->GetBrowserContext()); 26705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!profile_->IsSameProfile(host_profile->GetOriginalProfile())) 26715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 26725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extensions need to know the channel for API restrictions. 26745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process->Send(new ExtensionMsg_SetChannel( 26755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::Feature::GetCurrentChannel())); 26765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Valid extension function names, used to setup bindings in renderer. 26785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<std::string> function_names; 26795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionFunctionDispatcher::GetAllFunctionNames(&function_names); 26805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process->Send(new ExtensionMsg_SetFunctionNames(function_names)); 26815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Scripting whitelist. This is modified by tests and must be communicated 26835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // to renderers. 26845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process->Send(new ExtensionMsg_SetScriptingWhitelist( 26855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) *Extension::GetScriptingWhitelist())); 26865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 26875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Loaded extensions. 26885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<ExtensionMsg_Loaded_Params> loaded_extensions; 26895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator iter = extensions_.begin(); 26905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != extensions_.end(); ++iter) { 26915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Renderers don't need to know about themes. 26925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!(*iter)->is_theme()) 26935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) loaded_extensions.push_back(ExtensionMsg_Loaded_Params(*iter)); 26945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 26955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process->Send(new ExtensionMsg_Loaded(loaded_extensions)); 26965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 26975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 26985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case content::NOTIFICATION_RENDERER_PROCESS_TERMINATED: { 26995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::RenderProcessHost* process = 27005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<content::RenderProcessHost>(source).ptr(); 27015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile* host_profile = 27025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) Profile::FromBrowserContext(process->GetBrowserContext()); 27035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!profile_->IsSameProfile(host_profile->GetOriginalProfile())) 27045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 27055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process_map_.RemoveAllFromProcess(process->GetID()); 27075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) BrowserThread::PostTask( 27085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) BrowserThread::IO, FROM_HERE, 27095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind(&ExtensionInfoMap::UnregisterAllExtensionsInProcess, 27105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->info_map(), 27115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) process->GetID())); 27125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 27135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case chrome::NOTIFICATION_IMPORT_FINISHED: { 27155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) InitAfterImport(); 27165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 27175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) case chrome::NOTIFICATION_EXTENSION_HOST_DESTROYED: { 27195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host = 27205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Details<extensions::ExtensionHost>(details).ptr(); 27212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::string extension_id = host->extension_id(); 27222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (delayed_updates_for_idle_.Contains(extension_id)) { 27232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // We were waiting for this extension to become idle, it now might have, 27242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // so maybe finish installation. 272590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) base::MessageLoop::current()->PostDelayedTask( 27265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) FROM_HERE, 27272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::MaybeFinishDelayedInstallation, 27285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) AsWeakPtr(), extension_id), 27295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::TimeDelta::FromSeconds(kUpdateIdleDelay)); 27305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 27325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) case chrome::NOTIFICATION_UPGRADE_RECOMMENDED: { 27342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Notify extensions that chrome update is available. 27352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::RuntimeEventRouter::DispatchOnBrowserUpdateAvailableEvent( 27362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_); 27372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 27382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Notify observers that chrome update is available. 27392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FOR_EACH_OBSERVER(extensions::UpdateObserver, update_observers_, 27402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) OnChromeUpdateAvailable()); 27412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) break; 27422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 27435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) default: 27455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NOTREACHED() << "Unexpected notification type."; 27465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::OnExtensionInstallPrefChanged() { 27505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) IdentifyAlertableExtensions(); 27515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) CheckManagementPolicy(); 27525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::HasApps() const { 27555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return !GetAppIds().empty(); 27565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionIdSet ExtensionService::GetAppIds() const { 27595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionIdSet result; 27605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionSet::const_iterator it = extensions_.begin(); 27615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) it != extensions_.end(); ++it) { 27622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if ((*it)->is_app() && (*it)->location() != Manifest::COMPONENT) 27635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) result.insert((*it)->id()); 27645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 27655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return result; 27675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsBackgroundPageReady(const Extension* extension) const { 27702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extensions::BackgroundInfo::HasPersistentBackgroundPage(extension)) 27715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 27725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionRuntimeDataMap::const_iterator it = 27735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_.find(extension->id()); 27745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return it == extension_runtime_data_.end() ? false : 27755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) it->second.background_page_ready; 27765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SetBackgroundPageReady(const Extension* extension) { 27792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(extensions::BackgroundInfo::HasBackgroundPage(extension)); 27805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_[extension->id()].background_page_ready = true; 27815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::current()->Notify( 27825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) chrome::NOTIFICATION_EXTENSION_BACKGROUND_PAGE_READY, 27835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::Source<const Extension>(extension), 27845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) content::NotificationService::NoDetails()); 27855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 27865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::InspectBackgroundPage(const Extension* extension) { 27885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(extension); 27895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ExtensionProcessManager* pm = system_->process_manager(); 27915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::LazyBackgroundTaskQueue* queue = 27922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) system_->lazy_background_task_queue(); 27935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 27945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host = 27955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) pm->GetBackgroundHostForExtension(extension->id()); 27965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (host) { 27975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) InspectExtensionHost(host); 27985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } else { 27995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) queue->AddPendingTask( 28005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) profile_, extension->id(), 28015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind(&ExtensionService::InspectExtensionHost, 28025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Unretained(this))); 28035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 28045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsBeingUpgraded(const Extension* extension) const { 28075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionRuntimeDataMap::const_iterator it = 28085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_.find(extension->id()); 28095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return it == extension_runtime_data_.end() ? false : 28105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) it->second.being_upgraded; 28115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SetBeingUpgraded(const Extension* extension, 2814c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) bool value) { 28155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_[extension->id()].being_upgraded = value; 28165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::HasUsedWebRequest(const Extension* extension) const { 28195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionRuntimeDataMap::const_iterator it = 28205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_.find(extension->id()); 28215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return it == extension_runtime_data_.end() ? false : 28225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) it->second.has_used_webrequest; 28235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::SetHasUsedWebRequest(const Extension* extension, 28265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) bool value) { 28275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_runtime_data_[extension->id()].has_used_webrequest = value; 28285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::RegisterNaClModule(const GURL& url, 28315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const std::string& mime_type) { 28325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NaClModuleInfo info; 28335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) info.url = url; 28345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) info.mime_type = mime_type; 28355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(FindNaClModule(url) == nacl_module_list_.end()); 28375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) nacl_module_list_.push_front(info); 28385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UnregisterNaClModule(const GURL& url) { 28415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) NaClModuleInfoList::iterator iter = FindNaClModule(url); 28425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DCHECK(iter != nacl_module_list_.end()); 28435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) nacl_module_list_.erase(iter); 28445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::UpdatePluginListWithNaClModules() { 28475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // An extension has been added which has a nacl_module component, which means 28485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // there is a MIME type that module wants to handle, so we need to add that 28495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // MIME type to plugins which handle NaCl modules in order to allow the 28505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // individual modules to handle these types. 28512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::FilePath path; 28525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!PathService::Get(chrome::FILE_NACL_PLUGIN, &path)) 28535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 28545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const content::PepperPluginInfo* pepper_info = 28555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()->GetRegisteredPpapiPluginInfo(path); 28565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!pepper_info) 28575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 28585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::vector<webkit::WebPluginMimeType>::const_iterator mime_iter; 28605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Check each MIME type the plugins handle for the NaCl MIME type. 28615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (mime_iter = pepper_info->mime_types.begin(); 28625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) mime_iter != pepper_info->mime_types.end(); ++mime_iter) { 28635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (mime_iter->mime_type == kNaClPluginMimeType) { 28645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This plugin handles "application/x-nacl". 28655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()-> 28675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UnregisterInternalPlugin(pepper_info->path); 28685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) webkit::WebPluginInfo info = pepper_info->ToWebPluginInfo(); 28705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (ExtensionService::NaClModuleInfoList::const_iterator iter = 28725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) nacl_module_list_.begin(); 28735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != nacl_module_list_.end(); ++iter) { 28745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Add the MIME type specified in the extension to this NaCl plugin, 28755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // With an extra "nacl" argument to specify the location of the NaCl 28765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // manifest file. 28775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) webkit::WebPluginMimeType mime_type_info; 28785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) mime_type_info.mime_type = iter->mime_type; 28795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) mime_type_info.additional_param_names.push_back(UTF8ToUTF16("nacl")); 28805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) mime_type_info.additional_param_values.push_back( 28815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) UTF8ToUTF16(iter->url.spec())); 28825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) info.mime_types.push_back(mime_type_info); 28835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 28845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()->RefreshPlugins(); 28865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) PluginService::GetInstance()->RegisterInternalPlugin(info, true); 28875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // This plugin has been modified, no need to check the rest of its 28885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // types, but continue checking other plugins. 28895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) break; 28905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 28915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 28925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 28935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 28945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ExtensionService::NaClModuleInfoList::iterator 28955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) ExtensionService::FindNaClModule(const GURL& url) { 28965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) for (NaClModuleInfoList::iterator iter = nacl_module_list_.begin(); 28975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) iter != nacl_module_list_.end(); ++iter) { 28985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (iter->url == url) 28995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return iter; 29005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 29015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return nacl_module_list_.end(); 29025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::DoPostLoadTasks(const Extension* extension) { 29055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) std::map<std::string, int>::iterator it = 29065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) on_load_events_.find(extension->id()); 29075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (it == on_load_events_.end()) 29085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 29095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) int events_to_fire = it->second; 29115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::LazyBackgroundTaskQueue* queue = 29125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) system_->lazy_background_task_queue(); 29135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (queue->ShouldEnqueueTask(profile(), extension)) { 29145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (events_to_fire & EVENT_LAUNCHED) 29155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) queue->AddPendingTask(profile(), extension->id(), 29165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) base::Bind(&ExtensionService::LaunchApplication)); 29172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (events_to_fire & EVENT_RESTARTED) { 29185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) queue->AddPendingTask(profile(), extension->id(), 291990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) base::Bind(&ExtensionService::RestartApplication)); 29202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 29215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 29225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) on_load_events_.erase(it); 29245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 29275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::LaunchApplication( 29285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* extension_host) { 29295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension_host) 29305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 29315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if !defined(OS_ANDROID) 29335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::LaunchPlatformApp(extension_host->profile(), 29345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extension_host->extension(), 29352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NULL, base::FilePath()); 29365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 29375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static 29405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::RestartApplication( 29415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* extension_host) { 29425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (!extension_host) 29435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return; 29445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if !defined(OS_ANDROID) 294690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extensions::RestartPlatformApp( 294790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles) extension_host->profile(), extension_host->extension()); 29485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif 29495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::HasShellWindows(const std::string& extension_id) { 29525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) const Extension* current_extension = GetExtensionById(extension_id, false); 29535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return current_extension && current_extension->is_platform_app() && 29545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !extensions::ShellWindowRegistry::Get(profile_)-> 29555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) GetShellWindowsForApp(extension_id).empty(); 29565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ExtensionService::InspectExtensionHost( 29595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host) { 29605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (host) 29615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) DevToolsWindow::OpenDevToolsWindow(host->render_view_host()); 29625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::ShouldEnableOnInstall(const Extension* extension) { 29655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // Extensions installed by policy can't be disabled. So even if a previous 29665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // installation disabled the extension, make sure it is now enabled. 29675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (system_->management_policy()->MustRemainEnabled(extension, NULL)) 29685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 29695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (extension_prefs_->IsExtensionDisabled(extension->id())) 29715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 29725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) if (FeatureSwitch::prompt_for_external_extensions()->IsEnabled()) { 29745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) // External extensions are initially disabled. We prompt the user before 29752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // enabling them. Hosted apps are excepted because they are not dangerous 29762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // (they need to be launched by the user anyway). 29772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extension->GetType() != Manifest::TYPE_HOSTED_APP && 29782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) Manifest::IsExternalLocation(extension->location()) && 29795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) !extension_prefs_->IsExternalExtensionAcknowledged(extension->id())) { 29805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return false; 29815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 29825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) } 29835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) return true; 29855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 29865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) 29875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ExtensionService::IsExtensionIdle(const std::string& extension_id) const { 29882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ExtensionProcessManager* process_manager = system_->process_manager(); 29892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(process_manager); 29905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) extensions::ExtensionHost* host = 29912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) process_manager->GetBackgroundHostForExtension(extension_id); 29922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (host) 29932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 29942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return process_manager->GetRenderViewHostsForExtension(extension_id).empty(); 29952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 29962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 29972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)bool ExtensionService::ShouldDelayExtensionUpdate( 29982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& extension_id, 29992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) bool wait_for_idle) const { 30002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const char kOnUpdateAvailableEvent[] = "runtime.onUpdateAvailable"; 30012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If delayed updates are globally disabled, or just for this extension, 30032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // don't delay. 30042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!install_updates_when_idle_ || !wait_for_idle) 30052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 30062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const Extension* old = GetInstalledExtension(extension_id); 30082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // If there is no old extension, this is not an update, so don't delay. 30092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!old) 30102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return false; 30112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (extensions::BackgroundInfo::HasPersistentBackgroundPage(old)) { 30132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Delay installation if the extension listens for the onUpdateAvailable 30142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // event. 30152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return system_->event_router()->ExtensionHasEventListener( 30162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_id, kOnUpdateAvailableEvent); 30172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } else { 30182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Delay installation if the extension is not idle. 30192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return !IsExtensionIdle(extension_id); 30202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 30222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::GarbageCollectIsolatedStorage() { 30242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_ptr<base::hash_set<base::FilePath> > active_paths( 30252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) new base::hash_set<base::FilePath>()); 30262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (ExtensionSet::const_iterator it = extensions_.begin(); 30272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != extensions_.end(); ++it) { 3028c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) if (extensions::AppIsolationInfo::HasIsolatedStorage(*it)) { 30292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) active_paths->insert( 30302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) BrowserContext::GetStoragePartitionForSite( 30312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_, 30322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GetSiteForExtensionId((*it)->id()))->GetPath()); 30332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(!installs_delayed()); 30372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) set_installs_delayed(true); 30382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) BrowserContext::GarbageCollectStoragePartitions( 30392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) profile_, active_paths.Pass(), 30402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::OnGarbageCollectIsolatedStorageFinished, 30412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AsWeakPtr())); 30422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 30432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::OnGarbageCollectIsolatedStorageFinished() { 30452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) set_installs_delayed(false); 30462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (ExtensionSet::const_iterator it = delayed_installs_.begin(); 30472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != delayed_installs_.end(); 30482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ++it) { 30492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FinishDelayedInstallation((*it)->id()); 30502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (ExtensionSet::const_iterator it = delayed_updates_for_idle_.begin(); 30522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != delayed_updates_for_idle_.end(); 30532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) ++it) { 30542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) MaybeFinishDelayedInstallation((*it)->id()); 30552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) delayed_installs_.Clear(); 30572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 30582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::OnNeedsToGarbageCollectIsolatedStorage() { 30602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension_prefs_->SetNeedsStorageGarbageCollection(true); 30612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 30622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::OnBlacklistUpdated() { 30642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklist_->GetBlacklistedIDs( 30652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) GenerateInstalledExtensionsSet()->GetIDs(), 30662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) base::Bind(&ExtensionService::ManageBlacklist, 30672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AsWeakPtr(), 30682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.GetIDs())); 30692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 30702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::ManageBlacklist( 30722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::set<std::string>& old_blacklisted_ids, 30732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::set<std::string>& new_blacklisted_ids) { 30742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 30752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::set<std::string> no_longer_blacklisted; 30772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::set_difference(old_blacklisted_ids.begin(), old_blacklisted_ids.end(), 30782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) new_blacklisted_ids.begin(), new_blacklisted_ids.end(), 30792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::inserter(no_longer_blacklisted, 30802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) no_longer_blacklisted.begin())); 30812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::set<std::string> not_yet_blacklisted; 30822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::set_difference(new_blacklisted_ids.begin(), new_blacklisted_ids.end(), 30832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) old_blacklisted_ids.begin(), old_blacklisted_ids.end(), 30842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) std::inserter(not_yet_blacklisted, 30852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) not_yet_blacklisted.begin())); 30862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 30872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (std::set<std::string>::iterator it = no_longer_blacklisted.begin(); 30882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != no_longer_blacklisted.end(); ++it) { 30892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_refptr<const Extension> extension = 30902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.GetByID(*it); 30912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(extension); 30922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension) 30932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) continue; 30942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.Remove(*it); 30952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AddExtension(extension); 30962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("ExtensionBlacklist.UnblacklistInstalled", 30972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location(), Manifest::NUM_LOCATIONS); 30982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 30992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 31002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (std::set<std::string>::iterator it = not_yet_blacklisted.begin(); 31012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) it != not_yet_blacklisted.end(); ++it) { 31022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) scoped_refptr<const Extension> extension = GetInstalledExtension(*it); 31032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(extension); 31042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) if (!extension) 31052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) continue; 31062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) blacklisted_extensions_.Insert(extension); 31072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UnloadExtension(*it, extension_misc::UNLOAD_REASON_BLACKLIST); 31082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) UMA_HISTOGRAM_ENUMERATION("ExtensionBlacklist.BlacklistInstalled", 31092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extension->location(), Manifest::NUM_LOCATIONS); 31102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 31112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 31122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) IdentifyAlertableExtensions(); 31132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 31142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 31152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::AddUpdateObserver(extensions::UpdateObserver* observer) { 31162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) update_observers_.AddObserver(observer); 31172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 31182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 31192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void ExtensionService::RemoveUpdateObserver( 31202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) extensions::UpdateObserver* observer) { 31212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) update_observers_.RemoveObserver(observer); 31225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} 3123