Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(18)

Side by Side Diff: chrome/browser/extensions/extension_service.cc

Issue 5742008: Clean up threading model of external extension providers (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: final rebase Created 9 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "chrome/browser/extensions/extension_service.h" 5 #include "chrome/browser/extensions/extension_service.h"
6 6
7 #include <algorithm> 7 #include <algorithm>
8 8
9 #include "base/basictypes.h" 9 #include "base/basictypes.h"
10 #include "base/command_line.h" 10 #include "base/command_line.h"
11 #include "base/file_util.h" 11 #include "base/file_util.h"
12 #include "base/metrics/histogram.h" 12 #include "base/metrics/histogram.h"
13 #include "base/stl_util-inl.h" 13 #include "base/stl_util-inl.h"
14 #include "base/string16.h" 14 #include "base/string16.h"
15 #include "base/string_number_conversions.h" 15 #include "base/string_number_conversions.h"
16 #include "base/string_util.h" 16 #include "base/string_util.h"
17 #include "base/stringprintf.h" 17 #include "base/stringprintf.h"
18 #include "base/threading/thread_restrictions.h" 18 #include "base/threading/thread_restrictions.h"
19 #include "base/time.h" 19 #include "base/time.h"
20 #include "base/utf_string_conversions.h" 20 #include "base/utf_string_conversions.h"
21 #include "base/values.h" 21 #include "base/values.h"
22 #include "base/values_util.h"
23 #include "base/version.h" 22 #include "base/version.h"
24 #include "chrome/browser/browser_process.h" 23 #include "chrome/browser/browser_process.h"
25 #include "chrome/browser/themes/browser_theme_provider.h" 24 #include "chrome/browser/themes/browser_theme_provider.h"
26 #include "chrome/browser/browser_thread.h" 25 #include "chrome/browser/browser_thread.h"
27 #include "chrome/browser/debugger/devtools_manager.h" 26 #include "chrome/browser/debugger/devtools_manager.h"
28 #include "chrome/browser/dom_ui/shown_sections_handler.h" 27 #include "chrome/browser/dom_ui/shown_sections_handler.h"
29 #include "chrome/browser/extensions/crx_installer.h" 28 #include "chrome/browser/extensions/crx_installer.h"
30 #include "chrome/browser/extensions/default_apps.h" 29 #include "chrome/browser/extensions/default_apps.h"
31 #include "chrome/browser/extensions/extension_accessibility_api.h" 30 #include "chrome/browser/extensions/extension_accessibility_api.h"
32 #include "chrome/browser/extensions/extension_bookmarks_module.h" 31 #include "chrome/browser/extensions/extension_bookmarks_module.h"
33 #include "chrome/browser/extensions/extension_browser_event_router.h" 32 #include "chrome/browser/extensions/extension_browser_event_router.h"
34 #include "chrome/browser/extensions/extension_cookies_api.h" 33 #include "chrome/browser/extensions/extension_cookies_api.h"
35 #include "chrome/browser/extensions/extension_data_deleter.h" 34 #include "chrome/browser/extensions/extension_data_deleter.h"
36 #include "chrome/browser/extensions/extension_dom_ui.h" 35 #include "chrome/browser/extensions/extension_dom_ui.h"
37 #include "chrome/browser/extensions/extension_error_reporter.h" 36 #include "chrome/browser/extensions/extension_error_reporter.h"
38 #include "chrome/browser/extensions/extension_history_api.h" 37 #include "chrome/browser/extensions/extension_history_api.h"
39 #include "chrome/browser/extensions/extension_host.h" 38 #include "chrome/browser/extensions/extension_host.h"
40 #include "chrome/browser/extensions/extension_management_api.h" 39 #include "chrome/browser/extensions/extension_management_api.h"
41 #include "chrome/browser/extensions/extension_process_manager.h" 40 #include "chrome/browser/extensions/extension_process_manager.h"
42 #include "chrome/browser/extensions/extension_processes_api.h" 41 #include "chrome/browser/extensions/extension_processes_api.h"
43 #include "chrome/browser/extensions/extension_updater.h" 42 #include "chrome/browser/extensions/extension_updater.h"
44 #include "chrome/browser/extensions/extension_webnavigation_api.h" 43 #include "chrome/browser/extensions/extension_webnavigation_api.h"
45 #include "chrome/browser/extensions/external_extension_provider.h" 44 #include "chrome/browser/extensions/external_extension_provider_interface.h"
46 #include "chrome/browser/extensions/external_policy_extension_provider.h" 45 #include "chrome/browser/extensions/external_extension_provider_impl.h"
47 #include "chrome/browser/extensions/external_pref_extension_provider.h"
48 #include "chrome/browser/net/chrome_url_request_context.h" 46 #include "chrome/browser/net/chrome_url_request_context.h"
49 #include "chrome/browser/prefs/pref_service.h" 47 #include "chrome/browser/prefs/pref_service.h"
50 #include "chrome/browser/profiles/profile.h" 48 #include "chrome/browser/profiles/profile.h"
51 #include "chrome/browser/search_engines/template_url_model.h" 49 #include "chrome/browser/search_engines/template_url_model.h"
52 #include "chrome/common/child_process_logging.h" 50 #include "chrome/common/child_process_logging.h"
53 #include "chrome/common/chrome_switches.h" 51 #include "chrome/common/chrome_switches.h"
54 #include "chrome/common/extensions/extension.h" 52 #include "chrome/common/extensions/extension.h"
55 #include "chrome/common/extensions/extension_constants.h" 53 #include "chrome/common/extensions/extension_constants.h"
56 #include "chrome/common/extensions/extension_error_utils.h" 54 #include "chrome/common/extensions/extension_error_utils.h"
57 #include "chrome/common/extensions/extension_file_util.h" 55 #include "chrome/common/extensions/extension_file_util.h"
58 #include "chrome/common/extensions/extension_l10n_util.h" 56 #include "chrome/common/extensions/extension_l10n_util.h"
59 #include "chrome/common/extensions/extension_resource.h" 57 #include "chrome/common/extensions/extension_resource.h"
60 #include "chrome/common/notification_service.h" 58 #include "chrome/common/notification_service.h"
61 #include "chrome/common/notification_type.h" 59 #include "chrome/common/notification_type.h"
62 #include "chrome/common/json_value_serializer.h" 60 #include "chrome/common/json_value_serializer.h"
63 #include "chrome/common/pref_names.h" 61 #include "chrome/common/pref_names.h"
64 #include "chrome/common/url_constants.h" 62 #include "chrome/common/url_constants.h"
65 #include "googleurl/src/gurl.h" 63 #include "googleurl/src/gurl.h"
66 #include "net/base/registry_controlled_domain.h" 64 #include "net/base/registry_controlled_domain.h"
67 #include "webkit/database/database_tracker.h" 65 #include "webkit/database/database_tracker.h"
68 #include "webkit/database/database_util.h" 66 #include "webkit/database/database_util.h"
69 67
70 #if defined(OS_WIN)
71 #include "chrome/browser/extensions/external_registry_extension_provider_win.h"
72 #endif
73
74 using base::Time; 68 using base::Time;
75 69
76 namespace errors = extension_manifest_errors; 70 namespace errors = extension_manifest_errors;
77 71
78 namespace { 72 namespace {
79 73
80 #if defined(OS_LINUX) 74 #if defined(OS_LINUX)
81 static const int kOmniboxIconPaddingLeft = 2; 75 static const int kOmniboxIconPaddingLeft = 2;
82 static const int kOmniboxIconPaddingRight = 2; 76 static const int kOmniboxIconPaddingRight = 2;
83 #elif defined(OS_MACOSX) 77 #elif defined(OS_MACOSX)
(...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after
177 } 171 }
178 172
179 // ExtensionService. 173 // ExtensionService.
180 174
181 const char* ExtensionService::kInstallDirectoryName = "Extensions"; 175 const char* ExtensionService::kInstallDirectoryName = "Extensions";
182 const char* ExtensionService::kCurrentVersionFileName = "Current Version"; 176 const char* ExtensionService::kCurrentVersionFileName = "Current Version";
183 177
184 // Implements IO for the ExtensionService. 178 // Implements IO for the ExtensionService.
185 179
186 class ExtensionServiceBackend 180 class ExtensionServiceBackend
187 : public base::RefCountedThreadSafe<ExtensionServiceBackend>, 181 : public base::RefCountedThreadSafe<ExtensionServiceBackend> {
188 public ExternalExtensionProvider::Visitor {
189 public: 182 public:
190 // |install_directory| is a path where to look for extensions to load. 183 // |install_directory| is a path where to look for extensions to load.
191 ExtensionServiceBackend(PrefService* prefs, 184 explicit ExtensionServiceBackend(const FilePath& install_directory);
192 const FilePath& install_directory);
193 185
194 // Loads a single extension from |path| where |path| is the top directory of 186 // Loads a single extension from |path| where |path| is the top directory of
195 // a specific extension where its manifest file lives. 187 // a specific extension where its manifest file lives.
196 // Errors are reported through ExtensionErrorReporter. On success, 188 // Errors are reported through ExtensionErrorReporter. On success,
197 // OnExtensionLoaded() is called. 189 // OnExtensionLoaded() is called.
198 // TODO(erikkay): It might be useful to be able to load a packed extension 190 // TODO(erikkay): It might be useful to be able to load a packed extension
199 // (presumably into memory) without installing it. 191 // (presumably into memory) without installing it.
200 void LoadSingleExtension(const FilePath &path, 192 void LoadSingleExtension(const FilePath &path,
201 scoped_refptr<ExtensionService> frontend); 193 scoped_refptr<ExtensionService> frontend);
202 194
203 // Check externally updated extensions for updates and install if necessary.
204 // Errors are reported through ExtensionErrorReporter. Succcess is not
205 // reported.
206 void CheckForExternalUpdates(scoped_refptr<ExtensionService> frontend);
207
208 // For the extension in |version_path| with |id|, check to see if it's an
209 // externally managed extension. If so, tell the frontend to uninstall it.
210 void CheckExternalUninstall(scoped_refptr<ExtensionService> frontend,
211 const std::string& id);
212
213 // Clear all ExternalExtensionProviders.
214 void ClearProvidersForTesting();
215
216 // Adds an ExternalExtensionProvider for the service to use during testing.
217 // Takes ownership of |test_provider|.
218 void AddProviderForTesting(ExternalExtensionProvider* test_provider);
219
220 // ExternalExtensionProvider::Visitor implementation.
221 virtual void OnExternalExtensionFileFound(const std::string& id,
222 const Version* version,
223 const FilePath& path,
224 Extension::Location location);
225
226 virtual void OnExternalExtensionUpdateUrlFound(const std::string& id,
227 const GURL& update_url,
228 Extension::Location location);
229
230 virtual void UpdateExternalPolicyExtensionProvider(
231 scoped_refptr<RefCountedList> forcelist);
232
233 private: 195 private:
234 friend class base::RefCountedThreadSafe<ExtensionServiceBackend>; 196 friend class base::RefCountedThreadSafe<ExtensionServiceBackend>;
235 197
236 virtual ~ExtensionServiceBackend(); 198 virtual ~ExtensionServiceBackend();
237 199
238 // Finish installing the extension in |crx_path| after it has been unpacked to 200 // Finish installing the extension in |crx_path| after it has been unpacked to
239 // |unpacked_path|. If |expected_id| is not empty, it's verified against the 201 // |unpacked_path|. If |expected_id| is not empty, it's verified against the
240 // extension's manifest before installation. If |silent| is true, there will 202 // extension's manifest before installation. If |silent| is true, there will
241 // be no install confirmation dialog. |from_gallery| indicates whether the 203 // be no install confirmation dialog. |from_gallery| indicates whether the
242 // crx was installed from our gallery, which results in different UI. 204 // crx was installed from our gallery, which results in different UI.
(...skipping 11 matching lines...) Expand all
254 // This is a naked pointer which is set by each entry point. 216 // This is a naked pointer which is set by each entry point.
255 // The entry point is responsible for ensuring lifetime. 217 // The entry point is responsible for ensuring lifetime.
256 ExtensionService* frontend_; 218 ExtensionService* frontend_;
257 219
258 // The top-level extensions directory being installed to. 220 // The top-level extensions directory being installed to.
259 FilePath install_directory_; 221 FilePath install_directory_;
260 222
261 // Whether errors result in noisy alerts. 223 // Whether errors result in noisy alerts.
262 bool alert_on_error_; 224 bool alert_on_error_;
263 225
264 // A collection of external extension providers. Each provider reads
265 // a source of external extension information. Examples include the
266 // windows registry and external_extensions.json.
267 typedef std::vector<linked_ptr<ExternalExtensionProvider> >
268 ProviderCollection;
269 ProviderCollection external_extension_providers_;
270 linked_ptr<ExternalPolicyExtensionProvider>
271 external_policy_extension_provider_;
272
273 // Set to true by OnExternalExtensionUpdateUrlFound() when an external
274 // extension URL is found. Used in CheckForExternalUpdates() to see
275 // if an update check is needed to install pending extensions.
276 bool external_extension_added_;
277
278 DISALLOW_COPY_AND_ASSIGN(ExtensionServiceBackend); 226 DISALLOW_COPY_AND_ASSIGN(ExtensionServiceBackend);
279 }; 227 };
280 228
281 ExtensionServiceBackend::ExtensionServiceBackend( 229 ExtensionServiceBackend::ExtensionServiceBackend(
282 PrefService* prefs,
283 const FilePath& install_directory) 230 const FilePath& install_directory)
284 : frontend_(NULL), 231 : frontend_(NULL),
285 install_directory_(install_directory), 232 install_directory_(install_directory),
286 alert_on_error_(false), 233 alert_on_error_(false) {
287 external_extension_added_(false) {
288 // TODO(aa): This ends up doing blocking IO on the UI thread because it reads
289 // pref data in the ctor and that is called on the UI thread. Would be better
290 // to re-read data each time we list external extensions, anyway.
291 external_extension_providers_.push_back(
292 linked_ptr<ExternalExtensionProvider>(
293 new ExternalPrefExtensionProvider()));
294 #if defined(OS_WIN)
295 external_extension_providers_.push_back(
296 linked_ptr<ExternalExtensionProvider>(
297 new ExternalRegistryExtensionProvider()));
298 #endif
299 // The policy-controlled extension provider is also stored in a member
300 // variable so that UpdateExternalPolicyExtensionProvider can access it and
301 // update its extension list later.
302 external_policy_extension_provider_.reset(
303 new ExternalPolicyExtensionProvider(
304 prefs->GetList(prefs::kExtensionInstallForceList)));
305 external_extension_providers_.push_back(external_policy_extension_provider_);
306 } 234 }
307 235
308 ExtensionServiceBackend::~ExtensionServiceBackend() { 236 ExtensionServiceBackend::~ExtensionServiceBackend() {
309 } 237 }
310 238
311 void ExtensionServiceBackend::LoadSingleExtension( 239 void ExtensionServiceBackend::LoadSingleExtension(
312 const FilePath& path_in, scoped_refptr<ExtensionService> frontend) { 240 const FilePath& path_in, scoped_refptr<ExtensionService> frontend) {
313 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 241 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
314 242
315 frontend_ = frontend; 243 frontend_ = frontend;
(...skipping 29 matching lines...) Expand all
345 const FilePath& extension_path, const std::string &error) { 273 const FilePath& extension_path, const std::string &error) {
346 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 274 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
347 BrowserThread::PostTask( 275 BrowserThread::PostTask(
348 BrowserThread::UI, FROM_HERE, 276 BrowserThread::UI, FROM_HERE,
349 NewRunnableMethod( 277 NewRunnableMethod(
350 frontend_, 278 frontend_,
351 &ExtensionService::ReportExtensionLoadError, extension_path, 279 &ExtensionService::ReportExtensionLoadError, extension_path,
352 error, NotificationType::EXTENSION_INSTALL_ERROR, alert_on_error_)); 280 error, NotificationType::EXTENSION_INSTALL_ERROR, alert_on_error_));
353 } 281 }
354 282
355 // Some extensions will autoupdate themselves externally from Chrome. These 283 void ExtensionService::CheckExternalUninstall(const std::string& id) {
356 // are typically part of some larger client application package. To support 284 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
357 // these, the extension will register its location in the the preferences file
358 // (and also, on Windows, in the registry) and this code will periodically
359 // check that location for a .crx file, which it will then install locally if
360 // a new version is available.
361 void ExtensionServiceBackend::CheckForExternalUpdates(
362 scoped_refptr<ExtensionService> frontend) {
363 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
364
365 // Note that this installation is intentionally silent (since it didn't
366 // go through the front-end). Extensions that are registered in this
367 // way are effectively considered 'pre-bundled', and so implicitly
368 // trusted. In general, if something has HKLM or filesystem access,
369 // they could install an extension manually themselves anyway.
370 alert_on_error_ = false;
371 frontend_ = frontend;
372 external_extension_added_ = false;
373
374 // Ask each external extension provider to give us a call back for each
375 // extension they know about. See OnExternalExtension(File|UpdateUrl)Found.
376 ProviderCollection::const_iterator i;
377 for (i = external_extension_providers_.begin();
378 i != external_extension_providers_.end(); ++i) {
379 ExternalExtensionProvider* provider = i->get();
380 provider->VisitRegisteredExtension(this);
381 }
382
383 if (external_extension_added_ && frontend->updater()) {
384 BrowserThread::PostTask(
385 BrowserThread::UI, FROM_HERE,
386 NewRunnableMethod(
387 frontend->updater(), &ExtensionUpdater::CheckNow));
388 }
389 }
390
391 void ExtensionServiceBackend::CheckExternalUninstall(
392 scoped_refptr<ExtensionService> frontend, const std::string& id) {
393 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
394 285
395 // Check if the providers know about this extension. 286 // Check if the providers know about this extension.
396 ProviderCollection::const_iterator i; 287 ProviderCollection::const_iterator i;
397 for (i = external_extension_providers_.begin(); 288 for (i = external_extension_providers_.begin();
398 i != external_extension_providers_.end(); ++i) { 289 i != external_extension_providers_.end(); ++i) {
290 DCHECK(i->get()->IsReady());
399 if (i->get()->HasExtension(id)) 291 if (i->get()->HasExtension(id))
400 return; // Yup, known extension, don't uninstall. 292 return; // Yup, known extension, don't uninstall.
401 } 293 }
402 294
403 // This is an external extension that we don't have registered. Uninstall. 295 // This is an external extension that we don't have registered. Uninstall.
404 BrowserThread::PostTask( 296 UninstallExtension(id, true);
405 BrowserThread::UI, FROM_HERE,
406 NewRunnableMethod(
407 frontend.get(), &ExtensionService::UninstallExtension, id, true));
408 } 297 }
409 298
410 void ExtensionServiceBackend::UpdateExternalPolicyExtensionProvider( 299 void ExtensionService::ClearProvidersForTesting() {
411 scoped_refptr<RefCountedList> forcelist) {
412 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
413 external_policy_extension_provider_->SetPreferences(forcelist->Get());
414 }
415
416 void ExtensionServiceBackend::ClearProvidersForTesting() {
417 external_extension_providers_.clear(); 300 external_extension_providers_.clear();
418 } 301 }
419 302
420 void ExtensionServiceBackend::AddProviderForTesting( 303 void ExtensionService::AddProviderForTesting(
421 ExternalExtensionProvider* test_provider) { 304 ExternalExtensionProviderInterface* test_provider) {
422 DCHECK(test_provider); 305 DCHECK(test_provider);
423 external_extension_providers_.push_back( 306 external_extension_providers_.push_back(
424 linked_ptr<ExternalExtensionProvider>(test_provider)); 307 linked_ptr<ExternalExtensionProviderInterface>(test_provider));
425 } 308 }
426 309
427 void ExtensionServiceBackend::OnExternalExtensionFileFound( 310 void ExtensionService::OnExternalExtensionUpdateUrlFound(
428 const std::string& id, const Version* version, const FilePath& path,
429 Extension::Location location) {
430 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
431
432 DCHECK(version);
433 BrowserThread::PostTask(
434 BrowserThread::UI, FROM_HERE,
435 NewRunnableMethod(
436 frontend_, &ExtensionService::OnExternalExtensionFileFound, id,
437 version->GetString(), path, location));
438 }
439
440 void ExtensionServiceBackend::OnExternalExtensionUpdateUrlFound(
441 const std::string& id, 311 const std::string& id,
442 const GURL& update_url, 312 const GURL& update_url,
443 Extension::Location location) { 313 Extension::Location location) {
444 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 314 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
445 315
446 if (frontend_->GetExtensionById(id, true)) { 316 if (GetExtensionById(id, true)) {
447 // Already installed. Do not change the update URL that the extension set. 317 // Already installed. Do not change the update URL that the extension set.
448 return; 318 return;
449 } 319 }
450 320 AddPendingExtensionFromExternalUpdateUrl(id, update_url, location);
451 BrowserThread::PostTask(
452 BrowserThread::UI, FROM_HERE,
453 NewRunnableMethod(
454 frontend_,
455 &ExtensionService::AddPendingExtensionFromExternalUpdateUrl,
456 id, update_url, location));
457 external_extension_added_ |= true; 321 external_extension_added_ |= true;
458 } 322 }
459 323
460 bool ExtensionService::IsDownloadFromGallery(const GURL& download_url, 324 bool ExtensionService::IsDownloadFromGallery(const GURL& download_url,
461 const GURL& referrer_url) { 325 const GURL& referrer_url) {
462 // Special-case the themes mini-gallery. 326 // Special-case the themes mini-gallery.
463 // TODO(erikkay) When that gallery goes away, remove this code. 327 // TODO(erikkay) When that gallery goes away, remove this code.
464 if (IsDownloadFromMiniGallery(download_url) && 328 if (IsDownloadFromMiniGallery(download_url) &&
465 StartsWithASCII(referrer_url.spec(), 329 StartsWithASCII(referrer_url.spec(),
466 extension_urls::kMiniGalleryBrowsePrefix, false)) { 330 extension_urls::kMiniGalleryBrowsePrefix, false)) {
(...skipping 96 matching lines...) Expand 10 before | Expand all | Expand 10 after
563 extensions_enabled_ = false; 427 extensions_enabled_ = false;
564 } else if (profile->GetPrefs()->GetBoolean(prefs::kDisableExtensions)) { 428 } else if (profile->GetPrefs()->GetBoolean(prefs::kDisableExtensions)) {
565 extensions_enabled_ = false; 429 extensions_enabled_ = false;
566 } 430 }
567 431
568 registrar_.Add(this, NotificationType::EXTENSION_PROCESS_TERMINATED, 432 registrar_.Add(this, NotificationType::EXTENSION_PROCESS_TERMINATED,
569 NotificationService::AllSources()); 433 NotificationService::AllSources());
570 pref_change_registrar_.Init(profile->GetPrefs()); 434 pref_change_registrar_.Init(profile->GetPrefs());
571 pref_change_registrar_.Add(prefs::kExtensionInstallAllowList, this); 435 pref_change_registrar_.Add(prefs::kExtensionInstallAllowList, this);
572 pref_change_registrar_.Add(prefs::kExtensionInstallDenyList, this); 436 pref_change_registrar_.Add(prefs::kExtensionInstallDenyList, this);
573 pref_change_registrar_.Add(prefs::kExtensionInstallForceList, this);
574 437
575 // Set up the ExtensionUpdater 438 // Set up the ExtensionUpdater
576 if (autoupdate_enabled) { 439 if (autoupdate_enabled) {
577 int update_frequency = kDefaultUpdateFrequencySeconds; 440 int update_frequency = kDefaultUpdateFrequencySeconds;
578 if (command_line->HasSwitch(switches::kExtensionsUpdateFrequency)) { 441 if (command_line->HasSwitch(switches::kExtensionsUpdateFrequency)) {
579 base::StringToInt(command_line->GetSwitchValueASCII( 442 base::StringToInt(command_line->GetSwitchValueASCII(
580 switches::kExtensionsUpdateFrequency), 443 switches::kExtensionsUpdateFrequency),
581 &update_frequency); 444 &update_frequency);
582 } 445 }
583 updater_ = new ExtensionUpdater(this, 446 updater_ = new ExtensionUpdater(this,
584 profile->GetPrefs(), 447 profile->GetPrefs(),
585 update_frequency); 448 update_frequency);
586 } 449 }
587 450
588 backend_ = new ExtensionServiceBackend(profile->GetPrefs(), 451 backend_ = new ExtensionServiceBackend(install_directory_);
589 install_directory_); 452
453 ExternalExtensionProviderImpl::CreateExternalProviders(
454 this, profile_, &external_extension_providers_);
590 455
591 // Use monochrome icons for Omnibox icons. 456 // Use monochrome icons for Omnibox icons.
592 omnibox_popup_icon_manager_.set_monochrome(true); 457 omnibox_popup_icon_manager_.set_monochrome(true);
593 omnibox_icon_manager_.set_monochrome(true); 458 omnibox_icon_manager_.set_monochrome(true);
594 omnibox_icon_manager_.set_padding(gfx::Insets(0, kOmniboxIconPaddingLeft, 459 omnibox_icon_manager_.set_padding(gfx::Insets(0, kOmniboxIconPaddingLeft,
595 0, kOmniboxIconPaddingRight)); 460 0, kOmniboxIconPaddingRight));
596 } 461 }
597 462
598 const ExtensionList* ExtensionService::extensions() const { 463 const ExtensionList* ExtensionService::extensions() const {
599 return &extensions_; 464 return &extensions_;
(...skipping 14 matching lines...) Expand all
614 bool ExtensionService::HasInstalledExtensions() { 479 bool ExtensionService::HasInstalledExtensions() {
615 return !(extensions_.empty() && disabled_extensions_.empty()); 480 return !(extensions_.empty() && disabled_extensions_.empty());
616 } 481 }
617 482
618 ExtensionService::~ExtensionService() { 483 ExtensionService::~ExtensionService() {
619 DCHECK(!profile_); // Profile should have told us it's going away. 484 DCHECK(!profile_); // Profile should have told us it's going away.
620 UnloadAllExtensions(); 485 UnloadAllExtensions();
621 if (updater_.get()) { 486 if (updater_.get()) {
622 updater_->Stop(); 487 updater_->Stop();
623 } 488 }
489 ProviderCollection::const_iterator i;
490 for (i = external_extension_providers_.begin();
491 i != external_extension_providers_.end(); ++i) {
492 ExternalExtensionProviderInterface* provider = i->get();
493 provider->ServiceShutdown();
494 }
624 } 495 }
625 496
626 void ExtensionService::InitEventRouters() { 497 void ExtensionService::InitEventRouters() {
627 if (event_routers_initialized_) 498 if (event_routers_initialized_)
628 return; 499 return;
629 500
630 ExtensionHistoryEventRouter::GetInstance()->ObserveProfile(profile_); 501 ExtensionHistoryEventRouter::GetInstance()->ObserveProfile(profile_);
631 ExtensionAccessibilityEventRouter::GetInstance()->ObserveProfile(profile_); 502 ExtensionAccessibilityEventRouter::GetInstance()->ObserveProfile(profile_);
632 ExtensionBrowserEventRouter::GetInstance()->Init(profile_); 503 ExtensionBrowserEventRouter::GetInstance()->Init(profile_);
633 ExtensionBookmarkEventRouter::GetInstance()->Observe( 504 ExtensionBookmarkEventRouter::GetInstance()->Observe(
(...skipping 564 matching lines...) Expand 10 before | Expand all | Expand 10 after
1198 error, 1069 error,
1199 NotificationType::EXTENSION_INSTALL_ERROR, 1070 NotificationType::EXTENSION_INSTALL_ERROR,
1200 false); 1071 false);
1201 return; 1072 return;
1202 } 1073 }
1203 1074
1204 if (write_to_prefs) 1075 if (write_to_prefs)
1205 extension_prefs_->UpdateManifest(extension); 1076 extension_prefs_->UpdateManifest(extension);
1206 1077
1207 OnExtensionLoaded(extension); 1078 OnExtensionLoaded(extension);
1208
1209 if (Extension::IsExternalLocation(info.extension_location)) {
1210 BrowserThread::PostTask(
1211 BrowserThread::FILE, FROM_HERE,
1212 NewRunnableMethod(
1213 backend_.get(),
1214 &ExtensionServiceBackend::CheckExternalUninstall,
1215 scoped_refptr<ExtensionService>(this),
1216 info.extension_id));
1217 }
1218 } 1079 }
1219 1080
1220 void ExtensionService::NotifyExtensionLoaded(const Extension* extension) { 1081 void ExtensionService::NotifyExtensionLoaded(const Extension* extension) {
1221 // The ChromeURLRequestContexts need to be first to know that the extension 1082 // The ChromeURLRequestContexts need to be first to know that the extension
1222 // was loaded, otherwise a race can arise where a renderer that is created 1083 // was loaded, otherwise a race can arise where a renderer that is created
1223 // for the extension may try to load an extension URL with an extension id 1084 // for the extension may try to load an extension URL with an extension id
1224 // that the request context doesn't yet know about. The profile is responsible 1085 // that the request context doesn't yet know about. The profile is responsible
1225 // for ensuring its URLRequestContexts appropriately discover the loaded 1086 // for ensuring its URLRequestContexts appropriately discover the loaded
1226 // extension. 1087 // extension.
1227 if (profile_) { 1088 if (profile_) {
(...skipping 225 matching lines...) Expand 10 before | Expand all | Expand 10 after
1453 NotificationType::EXTENSION_USER_SCRIPTS_UPDATED, 1314 NotificationType::EXTENSION_USER_SCRIPTS_UPDATED,
1454 Source<Profile>(profile_), 1315 Source<Profile>(profile_),
1455 Details<const Extension>(extension)); 1316 Details<const Extension>(extension));
1456 } 1317 }
1457 1318
1458 bool ExtensionService::GetBrowserActionVisibility(const Extension* extension) { 1319 bool ExtensionService::GetBrowserActionVisibility(const Extension* extension) {
1459 return extension_prefs_->GetBrowserActionVisibility(extension); 1320 return extension_prefs_->GetBrowserActionVisibility(extension);
1460 } 1321 }
1461 1322
1462 void ExtensionService::SetBrowserActionVisibility(const Extension* extension, 1323 void ExtensionService::SetBrowserActionVisibility(const Extension* extension,
1463 bool visible) { 1324 bool visible) {
1464 extension_prefs_->SetBrowserActionVisibility(extension, visible); 1325 extension_prefs_->SetBrowserActionVisibility(extension, visible);
1465 } 1326 }
1466 1327
1328 // Some extensions will autoupdate themselves externally from Chrome. These
1329 // are typically part of some larger client application package. To support
1330 // these, the extension will register its location in the the preferences file
1331 // (and also, on Windows, in the registry) and this code will periodically
1332 // check that location for a .crx file, which it will then install locally if
1333 // a new version is available.
1334 // Errors are reported through ExtensionErrorReporter. Succcess is not
1335 // reported.
1467 void ExtensionService::CheckForExternalUpdates() { 1336 void ExtensionService::CheckForExternalUpdates() {
1468 BrowserThread::PostTask( 1337 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1469 BrowserThread::FILE, FROM_HERE, 1338
1470 NewRunnableMethod( 1339 // Note that this installation is intentionally silent (since it didn't
1471 backend_.get(), &ExtensionServiceBackend::CheckForExternalUpdates, 1340 // go through the front-end). Extensions that are registered in this
1472 scoped_refptr<ExtensionService>(this))); 1341 // way are effectively considered 'pre-bundled', and so implicitly
1342 // trusted. In general, if something has HKLM or filesystem access,
1343 // they could install an extension manually themselves anyway.
1344 external_extension_added_ = false;
1345
1346 // Ask each external extension provider to give us a call back for each
1347 // extension they know about. See OnExternalExtension(File|UpdateUrl)Found.
1348 ProviderCollection::const_iterator i;
1349 for (i = external_extension_providers_.begin();
1350 i != external_extension_providers_.end(); ++i) {
1351 ExternalExtensionProviderInterface* provider = i->get();
1352 provider->VisitRegisteredExtension();
1353 }
1354
1355 // Uninstall of unclaimed extensions will happen after all the providers
1356 // had reported ready. Trigger uninstall even if there are no providers
1357 // installed:
1358 OnExternalProviderReady();
1473 } 1359 }
1474 1360
1475 void ExtensionService::UpdateExternalPolicyExtensionProvider() { 1361 void ExtensionService::OnExternalProviderReady() {
1476 const ListValue* list_pref = 1362 // An external provider has finished loading. We only take action
1477 profile_->GetPrefs()->GetList(prefs::kExtensionInstallForceList); 1363 // if all of them are finished. So we check them first.
1478 ListValue* list_copy = NULL; 1364 ProviderCollection::const_iterator i;
1479 if (list_pref) 1365 for (i = external_extension_providers_.begin();
1480 list_copy = static_cast<ListValue*>(list_pref->DeepCopy()); 1366 i != external_extension_providers_.end(); ++i) {
1481 BrowserThread::PostTask( 1367 ExternalExtensionProviderInterface* provider = i->get();
1482 BrowserThread::FILE, FROM_HERE, 1368 if (!provider->IsReady()) {
1483 NewRunnableMethod( 1369 return;
1484 backend_.get(), 1370 }
1485 &ExtensionServiceBackend::UpdateExternalPolicyExtensionProvider, 1371 }
1486 scoped_refptr<RefCountedList>( 1372
1487 new RefCountedList(list_copy)))); 1373 // All the providers are ready. Install any pending extensions.
1374 if (external_extension_added_ && updater()) {
1375 external_extension_added_ = false;
1376 updater()->CheckNow();
1377 }
1378
1379 // Uninstall all the unclaimed extensions.
1380 scoped_ptr<ExtensionPrefs::ExtensionsInfo> extensions_info(
1381 extension_prefs_->GetInstalledExtensionsInfo());
1382 for (size_t i = 0; i < extensions_info->size(); ++i) {
1383 ExtensionInfo* info = extensions_info->at(i).get();
1384 if (Extension::IsExternalLocation(info->extension_location))
1385 CheckExternalUninstall(info->extension_id);
1386 }
1488 } 1387 }
1489 1388
1490 void ExtensionService::UnloadExtension( 1389 void ExtensionService::UnloadExtension(
1491 const std::string& extension_id, 1390 const std::string& extension_id,
1492 UnloadedExtensionInfo::Reason reason) { 1391 UnloadedExtensionInfo::Reason reason) {
1493 // Make sure the extension gets deleted after we return from this function. 1392 // Make sure the extension gets deleted after we return from this function.
1494 scoped_refptr<const Extension> extension( 1393 scoped_refptr<const Extension> extension(
1495 GetExtensionByIdInternal(extension_id, true, true)); 1394 GetExtensionByIdInternal(extension_id, true, true));
1496 1395
1497 // This method can be called via PostTask, so the extension may have been 1396 // This method can be called via PostTask, so the extension may have been
(...skipping 414 matching lines...) Expand 10 before | Expand all | Expand 10 after
1912 const SkBitmap& ExtensionService::GetOmniboxIcon( 1811 const SkBitmap& ExtensionService::GetOmniboxIcon(
1913 const std::string& extension_id) { 1812 const std::string& extension_id) {
1914 return omnibox_icon_manager_.GetIcon(extension_id); 1813 return omnibox_icon_manager_.GetIcon(extension_id);
1915 } 1814 }
1916 1815
1917 const SkBitmap& ExtensionService::GetOmniboxPopupIcon( 1816 const SkBitmap& ExtensionService::GetOmniboxPopupIcon(
1918 const std::string& extension_id) { 1817 const std::string& extension_id) {
1919 return omnibox_popup_icon_manager_.GetIcon(extension_id); 1818 return omnibox_popup_icon_manager_.GetIcon(extension_id);
1920 } 1819 }
1921 1820
1922 void ExtensionService::ClearProvidersForTesting() {
1923 BrowserThread::PostTask(
1924 BrowserThread::FILE, FROM_HERE,
1925 NewRunnableMethod(
1926 backend_.get(), &ExtensionServiceBackend::ClearProvidersForTesting));
1927 }
1928
1929 void ExtensionService::AddProviderForTesting(
1930 ExternalExtensionProvider* test_provider) {
1931 BrowserThread::PostTask(
1932 BrowserThread::FILE, FROM_HERE,
1933 NewRunnableMethod(
1934 backend_.get(), &ExtensionServiceBackend::AddProviderForTesting,
1935 test_provider));
1936 }
1937
1938 void ExtensionService::OnExternalExtensionFileFound( 1821 void ExtensionService::OnExternalExtensionFileFound(
1939 const std::string& id, 1822 const std::string& id,
1940 const std::string& version, 1823 const Version* version,
1941 const FilePath& path, 1824 const FilePath& path,
1942 Extension::Location location) { 1825 Extension::Location location) {
1943 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 1826 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1944 if (extension_prefs_->IsExtensionKilled(id)) 1827 if (extension_prefs_->IsExtensionKilled(id))
1945 return; 1828 return;
1946 1829
1830 DCHECK(version);
1831
1947 // Before even bothering to unpack, check and see if we already have this 1832 // Before even bothering to unpack, check and see if we already have this
1948 // version. This is important because these extensions are going to get 1833 // version. This is important because these extensions are going to get
1949 // installed on every startup. 1834 // installed on every startup.
1950 const Extension* existing = GetExtensionById(id, true); 1835 const Extension* existing = GetExtensionById(id, true);
1951 scoped_ptr<Version> other(Version::GetVersionFromString(version));
1952 if (existing) { 1836 if (existing) {
1953 switch (existing->version()->CompareTo(*other)) { 1837 switch (existing->version()->CompareTo(*version)) {
1954 case -1: // existing version is older, we should upgrade 1838 case -1: // existing version is older, we should upgrade
1955 break; 1839 break;
1956 case 0: // existing version is same, do nothing 1840 case 0: // existing version is same, do nothing
1957 return; 1841 return;
1958 case 1: // existing version is newer, uh-oh 1842 case 1: // existing version is newer, uh-oh
1959 LOG(WARNING) << "Found external version of extension " << id 1843 LOG(WARNING) << "Found external version of extension " << id
1960 << "that is older than current version. Current version " 1844 << "that is older than current version. Current version "
1961 << "is: " << existing->VersionString() << ". New version " 1845 << "is: " << existing->VersionString() << ". New version "
1962 << "is: " << version << ". Keeping current version."; 1846 << "is: " << version << ". Keeping current version.";
1963 return; 1847 return;
(...skipping 73 matching lines...) Expand 10 before | Expand all | Expand 10 after
2037 host->extension()->id(), 1921 host->extension()->id(),
2038 UnloadedExtensionInfo::DISABLE)); 1922 UnloadedExtensionInfo::DISABLE));
2039 break; 1923 break;
2040 } 1924 }
2041 1925
2042 case NotificationType::PREF_CHANGED: { 1926 case NotificationType::PREF_CHANGED: {
2043 std::string* pref_name = Details<std::string>(details).ptr(); 1927 std::string* pref_name = Details<std::string>(details).ptr();
2044 if (*pref_name == prefs::kExtensionInstallAllowList || 1928 if (*pref_name == prefs::kExtensionInstallAllowList ||
2045 *pref_name == prefs::kExtensionInstallDenyList) { 1929 *pref_name == prefs::kExtensionInstallDenyList) {
2046 CheckAdminBlacklist(); 1930 CheckAdminBlacklist();
2047 } else if (*pref_name == prefs::kExtensionInstallForceList) {
2048 UpdateExternalPolicyExtensionProvider();
2049 CheckForExternalUpdates();
2050 // TODO(gfeher): Also check for external extensions that can be
2051 // uninstalled because they were removed from the pref.
2052 // (crbug.com/63667)
2053 } else { 1931 } else {
2054 NOTREACHED() << "Unexpected preference name."; 1932 NOTREACHED() << "Unexpected preference name.";
2055 } 1933 }
2056 break; 1934 break;
2057 } 1935 }
2058 1936
2059 default: 1937 default:
2060 NOTREACHED() << "Unexpected notification type."; 1938 NOTREACHED() << "Unexpected notification type.";
2061 } 1939 }
2062 } 1940 }
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
2095 } 1973 }
2096 1974
2097 void ExtensionService::SetBeingUpgraded(const Extension* extension, 1975 void ExtensionService::SetBeingUpgraded(const Extension* extension,
2098 bool value) { 1976 bool value) {
2099 extension_runtime_data_[extension->id()].being_upgraded = value; 1977 extension_runtime_data_[extension->id()].being_upgraded = value;
2100 } 1978 }
2101 1979
2102 PropertyBag* ExtensionService::GetPropertyBag(const Extension* extension) { 1980 PropertyBag* ExtensionService::GetPropertyBag(const Extension* extension) {
2103 return &extension_runtime_data_[extension->id()].property_bag; 1981 return &extension_runtime_data_[extension->id()].property_bag;
2104 } 1982 }
OLDNEW
« no previous file with comments | « chrome/browser/extensions/extension_service.h ('k') | chrome/browser/extensions/extension_service_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698