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

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: " 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) 2010 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"
(...skipping 25 matching lines...) Expand all
36 #include "chrome/browser/extensions/extension_dom_ui.h" 36 #include "chrome/browser/extensions/extension_dom_ui.h"
37 #include "chrome/browser/extensions/extension_error_reporter.h" 37 #include "chrome/browser/extensions/extension_error_reporter.h"
38 #include "chrome/browser/extensions/extension_history_api.h" 38 #include "chrome/browser/extensions/extension_history_api.h"
39 #include "chrome/browser/extensions/extension_host.h" 39 #include "chrome/browser/extensions/extension_host.h"
40 #include "chrome/browser/extensions/extension_management_api.h" 40 #include "chrome/browser/extensions/extension_management_api.h"
41 #include "chrome/browser/extensions/extension_process_manager.h" 41 #include "chrome/browser/extensions/extension_process_manager.h"
42 #include "chrome/browser/extensions/extension_processes_api.h" 42 #include "chrome/browser/extensions/extension_processes_api.h"
43 #include "chrome/browser/extensions/extension_updater.h" 43 #include "chrome/browser/extensions/extension_updater.h"
44 #include "chrome/browser/extensions/extension_webnavigation_api.h" 44 #include "chrome/browser/extensions/extension_webnavigation_api.h"
45 #include "chrome/browser/extensions/external_extension_provider.h" 45 #include "chrome/browser/extensions/external_extension_provider.h"
46 #include "chrome/browser/extensions/external_policy_extension_provider.h" 46 #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" 47 #include "chrome/browser/net/chrome_url_request_context.h"
49 #include "chrome/browser/prefs/pref_service.h" 48 #include "chrome/browser/prefs/pref_service.h"
50 #include "chrome/browser/profiles/profile.h" 49 #include "chrome/browser/profiles/profile.h"
51 #include "chrome/browser/search_engines/template_url_model.h" 50 #include "chrome/browser/search_engines/template_url_model.h"
52 #include "chrome/common/child_process_logging.h" 51 #include "chrome/common/child_process_logging.h"
53 #include "chrome/common/chrome_switches.h" 52 #include "chrome/common/chrome_switches.h"
54 #include "chrome/common/extensions/extension.h" 53 #include "chrome/common/extensions/extension.h"
55 #include "chrome/common/extensions/extension_constants.h" 54 #include "chrome/common/extensions/extension_constants.h"
56 #include "chrome/common/extensions/extension_error_utils.h" 55 #include "chrome/common/extensions/extension_error_utils.h"
57 #include "chrome/common/extensions/extension_file_util.h" 56 #include "chrome/common/extensions/extension_file_util.h"
58 #include "chrome/common/extensions/extension_l10n_util.h" 57 #include "chrome/common/extensions/extension_l10n_util.h"
59 #include "chrome/common/extensions/extension_resource.h" 58 #include "chrome/common/extensions/extension_resource.h"
60 #include "chrome/common/notification_service.h" 59 #include "chrome/common/notification_service.h"
61 #include "chrome/common/notification_type.h" 60 #include "chrome/common/notification_type.h"
62 #include "chrome/common/json_value_serializer.h" 61 #include "chrome/common/json_value_serializer.h"
63 #include "chrome/common/pref_names.h" 62 #include "chrome/common/pref_names.h"
64 #include "chrome/common/url_constants.h" 63 #include "chrome/common/url_constants.h"
65 #include "googleurl/src/gurl.h" 64 #include "googleurl/src/gurl.h"
66 #include "net/base/registry_controlled_domain.h" 65 #include "net/base/registry_controlled_domain.h"
67 #include "webkit/database/database_tracker.h" 66 #include "webkit/database/database_tracker.h"
68 #include "webkit/database/database_util.h" 67 #include "webkit/database/database_util.h"
69 68
70 #if defined(OS_WIN)
71 #include "chrome/browser/extensions/external_registry_extension_provider_win.h"
72 #endif
73
74 using base::Time; 69 using base::Time;
75 70
76 namespace errors = extension_manifest_errors; 71 namespace errors = extension_manifest_errors;
77 72
78 namespace { 73 namespace {
79 74
80 #if defined(OS_LINUX) 75 #if defined(OS_LINUX)
81 static const int kOmniboxIconPaddingLeft = 2; 76 static const int kOmniboxIconPaddingLeft = 2;
82 static const int kOmniboxIconPaddingRight = 2; 77 static const int kOmniboxIconPaddingRight = 2;
83 #elif defined(OS_MACOSX) 78 #elif defined(OS_MACOSX)
(...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after
177 } 172 }
178 173
179 // ExtensionService. 174 // ExtensionService.
180 175
181 const char* ExtensionService::kInstallDirectoryName = "Extensions"; 176 const char* ExtensionService::kInstallDirectoryName = "Extensions";
182 const char* ExtensionService::kCurrentVersionFileName = "Current Version"; 177 const char* ExtensionService::kCurrentVersionFileName = "Current Version";
183 178
184 // Implements IO for the ExtensionService. 179 // Implements IO for the ExtensionService.
185 180
186 class ExtensionServiceBackend 181 class ExtensionServiceBackend
187 : public base::RefCountedThreadSafe<ExtensionServiceBackend>, 182 : public base::RefCountedThreadSafe<ExtensionServiceBackend> {
188 public ExternalExtensionProvider::Visitor {
189 public: 183 public:
190 // |install_directory| is a path where to look for extensions to load. 184 // |install_directory| is a path where to look for extensions to load.
191 ExtensionServiceBackend(PrefService* prefs, 185 explicit ExtensionServiceBackend(const FilePath& install_directory);
192 const FilePath& install_directory);
193 186
194 // Loads a single extension from |path| where |path| is the top directory of 187 // Loads a single extension from |path| where |path| is the top directory of
195 // a specific extension where its manifest file lives. 188 // a specific extension where its manifest file lives.
196 // Errors are reported through ExtensionErrorReporter. On success, 189 // Errors are reported through ExtensionErrorReporter. On success,
197 // OnExtensionLoaded() is called. 190 // OnExtensionLoaded() is called.
198 // TODO(erikkay): It might be useful to be able to load a packed extension 191 // TODO(erikkay): It might be useful to be able to load a packed extension
199 // (presumably into memory) without installing it. 192 // (presumably into memory) without installing it.
200 void LoadSingleExtension(const FilePath &path, 193 void LoadSingleExtension(const FilePath &path,
201 scoped_refptr<ExtensionService> frontend); 194 scoped_refptr<ExtensionService> frontend);
202 195
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: 196 private:
234 friend class base::RefCountedThreadSafe<ExtensionServiceBackend>; 197 friend class base::RefCountedThreadSafe<ExtensionServiceBackend>;
235 198
236 virtual ~ExtensionServiceBackend(); 199 virtual ~ExtensionServiceBackend();
237 200
238 // Finish installing the extension in |crx_path| after it has been unpacked to 201 // 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 202 // |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 203 // extension's manifest before installation. If |silent| is true, there will
241 // be no install confirmation dialog. |from_gallery| indicates whether the 204 // be no install confirmation dialog. |from_gallery| indicates whether the
242 // crx was installed from our gallery, which results in different UI. 205 // 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. 217 // This is a naked pointer which is set by each entry point.
255 // The entry point is responsible for ensuring lifetime. 218 // The entry point is responsible for ensuring lifetime.
256 ExtensionService* frontend_; 219 ExtensionService* frontend_;
257 220
258 // The top-level extensions directory being installed to. 221 // The top-level extensions directory being installed to.
259 FilePath install_directory_; 222 FilePath install_directory_;
260 223
261 // Whether errors result in noisy alerts. 224 // Whether errors result in noisy alerts.
262 bool alert_on_error_; 225 bool alert_on_error_;
263 226
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); 227 DISALLOW_COPY_AND_ASSIGN(ExtensionServiceBackend);
279 }; 228 };
280 229
281 ExtensionServiceBackend::ExtensionServiceBackend( 230 ExtensionServiceBackend::ExtensionServiceBackend(
282 PrefService* prefs,
283 const FilePath& install_directory) 231 const FilePath& install_directory)
284 : frontend_(NULL), 232 : frontend_(NULL),
285 install_directory_(install_directory), 233 install_directory_(install_directory),
286 alert_on_error_(false), 234 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 } 235 }
307 236
308 ExtensionServiceBackend::~ExtensionServiceBackend() { 237 ExtensionServiceBackend::~ExtensionServiceBackend() {
309 } 238 }
310 239
311 void ExtensionServiceBackend::LoadSingleExtension( 240 void ExtensionServiceBackend::LoadSingleExtension(
312 const FilePath& path_in, scoped_refptr<ExtensionService> frontend) { 241 const FilePath& path_in, scoped_refptr<ExtensionService> frontend) {
313 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 242 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
314 243
315 frontend_ = frontend; 244 frontend_ = frontend;
(...skipping 29 matching lines...) Expand all
345 const FilePath& extension_path, const std::string &error) { 274 const FilePath& extension_path, const std::string &error) {
346 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 275 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
347 BrowserThread::PostTask( 276 BrowserThread::PostTask(
348 BrowserThread::UI, FROM_HERE, 277 BrowserThread::UI, FROM_HERE,
349 NewRunnableMethod( 278 NewRunnableMethod(
350 frontend_, 279 frontend_,
351 &ExtensionService::ReportExtensionLoadError, extension_path, 280 &ExtensionService::ReportExtensionLoadError, extension_path,
352 error, NotificationType::EXTENSION_INSTALL_ERROR, alert_on_error_)); 281 error, NotificationType::EXTENSION_INSTALL_ERROR, alert_on_error_));
353 } 282 }
354 283
355 // Some extensions will autoupdate themselves externally from Chrome. These 284 void ExtensionService::CheckExternalUninstall(const std::string& id) {
356 // are typically part of some larger client application package. To support 285 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 286
395 // Check if the providers know about this extension. 287 // Check if the providers know about this extension.
396 ProviderCollection::const_iterator i; 288 ProviderCollection::const_iterator i;
397 for (i = external_extension_providers_.begin(); 289 for (i = external_extension_providers_.begin();
398 i != external_extension_providers_.end(); ++i) { 290 i != external_extension_providers_.end(); ++i) {
291 DCHECK(i->get()->IsReady());
399 if (i->get()->HasExtension(id)) 292 if (i->get()->HasExtension(id))
400 return; // Yup, known extension, don't uninstall. 293 return; // Yup, known extension, don't uninstall.
401 } 294 }
402 295
403 // This is an external extension that we don't have registered. Uninstall. 296 // This is an external extension that we don't have registered. Uninstall.
404 BrowserThread::PostTask( 297 UninstallExtension(id, true);
405 BrowserThread::UI, FROM_HERE,
406 NewRunnableMethod(
407 frontend.get(), &ExtensionService::UninstallExtension, id, true));
408 } 298 }
409 299
410 void ExtensionServiceBackend::UpdateExternalPolicyExtensionProvider( 300 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(); 301 external_extension_providers_.clear();
418 } 302 }
419 303
420 void ExtensionServiceBackend::AddProviderForTesting( 304 void ExtensionService::AddProviderForTesting(
421 ExternalExtensionProvider* test_provider) { 305 ExternalExtensionProvider* test_provider) {
422 DCHECK(test_provider); 306 DCHECK(test_provider);
423 external_extension_providers_.push_back( 307 external_extension_providers_.push_back(
424 linked_ptr<ExternalExtensionProvider>(test_provider)); 308 linked_ptr<ExternalExtensionProvider>(test_provider));
425 } 309 }
426 310
427 void ExtensionServiceBackend::OnExternalExtensionFileFound( 311 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, 312 const std::string& id,
442 const GURL& update_url, 313 const GURL& update_url,
443 Extension::Location location) { 314 Extension::Location location) {
444 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); 315 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
445 316
446 if (frontend_->GetExtensionById(id, true)) { 317 if (GetExtensionById(id, true)) {
447 // Already installed. Do not change the update URL that the extension set. 318 // Already installed. Do not change the update URL that the extension set.
448 return; 319 return;
449 } 320 }
450 321 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; 322 external_extension_added_ |= true;
458 } 323 }
459 324
460 bool ExtensionService::IsDownloadFromGallery(const GURL& download_url, 325 bool ExtensionService::IsDownloadFromGallery(const GURL& download_url,
461 const GURL& referrer_url) { 326 const GURL& referrer_url) {
462 // Special-case the themes mini-gallery. 327 // Special-case the themes mini-gallery.
463 // TODO(erikkay) When that gallery goes away, remove this code. 328 // TODO(erikkay) When that gallery goes away, remove this code.
464 if (IsDownloadFromMiniGallery(download_url) && 329 if (IsDownloadFromMiniGallery(download_url) &&
465 StartsWithASCII(referrer_url.spec(), 330 StartsWithASCII(referrer_url.spec(),
466 extension_urls::kMiniGalleryBrowsePrefix, false)) { 331 extension_urls::kMiniGalleryBrowsePrefix, false)) {
(...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after
562 extensions_enabled_ = false; 427 extensions_enabled_ = false;
563 } else if (profile->GetPrefs()->GetBoolean(prefs::kDisableExtensions)) { 428 } else if (profile->GetPrefs()->GetBoolean(prefs::kDisableExtensions)) {
564 extensions_enabled_ = false; 429 extensions_enabled_ = false;
565 } 430 }
566 431
567 registrar_.Add(this, NotificationType::EXTENSION_PROCESS_TERMINATED, 432 registrar_.Add(this, NotificationType::EXTENSION_PROCESS_TERMINATED,
568 NotificationService::AllSources()); 433 NotificationService::AllSources());
569 pref_change_registrar_.Init(profile->GetPrefs()); 434 pref_change_registrar_.Init(profile->GetPrefs());
570 pref_change_registrar_.Add(prefs::kExtensionInstallAllowList, this); 435 pref_change_registrar_.Add(prefs::kExtensionInstallAllowList, this);
571 pref_change_registrar_.Add(prefs::kExtensionInstallDenyList, this); 436 pref_change_registrar_.Add(prefs::kExtensionInstallDenyList, this);
572 pref_change_registrar_.Add(prefs::kExtensionInstallForceList, this);
573 437
574 // Set up the ExtensionUpdater 438 // Set up the ExtensionUpdater
575 if (autoupdate_enabled) { 439 if (autoupdate_enabled) {
576 int update_frequency = kDefaultUpdateFrequencySeconds; 440 int update_frequency = kDefaultUpdateFrequencySeconds;
577 if (command_line->HasSwitch(switches::kExtensionsUpdateFrequency)) { 441 if (command_line->HasSwitch(switches::kExtensionsUpdateFrequency)) {
578 base::StringToInt(command_line->GetSwitchValueASCII( 442 base::StringToInt(command_line->GetSwitchValueASCII(
579 switches::kExtensionsUpdateFrequency), 443 switches::kExtensionsUpdateFrequency),
580 &update_frequency); 444 &update_frequency);
581 } 445 }
582 updater_ = new ExtensionUpdater(this, 446 updater_ = new ExtensionUpdater(this,
583 profile->GetPrefs(), 447 profile->GetPrefs(),
584 update_frequency); 448 update_frequency);
585 } 449 }
586 450
587 backend_ = new ExtensionServiceBackend(profile->GetPrefs(), 451 backend_ = new ExtensionServiceBackend(install_directory_);
588 install_directory_); 452
453 ExternalExtensionProviderImpl::CreateExternalProviders(
454 this, profile_, &external_extension_providers_);
589 455
590 // Use monochrome icons for Omnibox icons. 456 // Use monochrome icons for Omnibox icons.
591 omnibox_popup_icon_manager_.set_monochrome(true); 457 omnibox_popup_icon_manager_.set_monochrome(true);
592 omnibox_icon_manager_.set_monochrome(true); 458 omnibox_icon_manager_.set_monochrome(true);
593 omnibox_icon_manager_.set_padding(gfx::Insets(0, kOmniboxIconPaddingLeft, 459 omnibox_icon_manager_.set_padding(gfx::Insets(0, kOmniboxIconPaddingLeft,
594 0, kOmniboxIconPaddingRight)); 460 0, kOmniboxIconPaddingRight));
595 } 461 }
596 462
597 const ExtensionList* ExtensionService::extensions() const { 463 const ExtensionList* ExtensionService::extensions() const {
598 return &extensions_; 464 return &extensions_;
(...skipping 10 matching lines...) Expand all
609 bool ExtensionService::HasInstalledExtensions() { 475 bool ExtensionService::HasInstalledExtensions() {
610 return !(extensions_.empty() && disabled_extensions_.empty()); 476 return !(extensions_.empty() && disabled_extensions_.empty());
611 } 477 }
612 478
613 ExtensionService::~ExtensionService() { 479 ExtensionService::~ExtensionService() {
614 DCHECK(!profile_); // Profile should have told us it's going away. 480 DCHECK(!profile_); // Profile should have told us it's going away.
615 UnloadAllExtensions(); 481 UnloadAllExtensions();
616 if (updater_.get()) { 482 if (updater_.get()) {
617 updater_->Stop(); 483 updater_->Stop();
618 } 484 }
485 ProviderCollection::const_iterator i;
486 for (i = external_extension_providers_.begin();
487 i != external_extension_providers_.end(); ++i) {
488 ExternalExtensionProvider* provider = i->get();
489 provider->ServiceShutdown();
490 }
619 } 491 }
620 492
621 void ExtensionService::InitEventRouters() { 493 void ExtensionService::InitEventRouters() {
622 if (event_routers_initialized_) 494 if (event_routers_initialized_)
623 return; 495 return;
624 496
625 ExtensionHistoryEventRouter::GetInstance()->ObserveProfile(profile_); 497 ExtensionHistoryEventRouter::GetInstance()->ObserveProfile(profile_);
626 ExtensionAccessibilityEventRouter::GetInstance()->ObserveProfile(profile_); 498 ExtensionAccessibilityEventRouter::GetInstance()->ObserveProfile(profile_);
627 ExtensionBrowserEventRouter::GetInstance()->Init(profile_); 499 ExtensionBrowserEventRouter::GetInstance()->Init(profile_);
628 ExtensionBookmarkEventRouter::GetInstance()->Observe( 500 ExtensionBookmarkEventRouter::GetInstance()->Observe(
(...skipping 564 matching lines...) Expand 10 before | Expand all | Expand 10 after
1193 error, 1065 error,
1194 NotificationType::EXTENSION_INSTALL_ERROR, 1066 NotificationType::EXTENSION_INSTALL_ERROR,
1195 false); 1067 false);
1196 return; 1068 return;
1197 } 1069 }
1198 1070
1199 if (write_to_prefs) 1071 if (write_to_prefs)
1200 extension_prefs_->UpdateManifest(extension); 1072 extension_prefs_->UpdateManifest(extension);
1201 1073
1202 OnExtensionLoaded(extension); 1074 OnExtensionLoaded(extension);
1203
1204 if (Extension::IsExternalLocation(info.extension_location)) {
1205 BrowserThread::PostTask(
1206 BrowserThread::FILE, FROM_HERE,
1207 NewRunnableMethod(
1208 backend_.get(),
1209 &ExtensionServiceBackend::CheckExternalUninstall,
1210 scoped_refptr<ExtensionService>(this),
1211 info.extension_id));
1212 }
1213 } 1075 }
1214 1076
1215 void ExtensionService::NotifyExtensionLoaded(const Extension* extension) { 1077 void ExtensionService::NotifyExtensionLoaded(const Extension* extension) {
1216 // The ChromeURLRequestContexts need to be first to know that the extension 1078 // The ChromeURLRequestContexts need to be first to know that the extension
1217 // was loaded, otherwise a race can arise where a renderer that is created 1079 // was loaded, otherwise a race can arise where a renderer that is created
1218 // for the extension may try to load an extension URL with an extension id 1080 // for the extension may try to load an extension URL with an extension id
1219 // that the request context doesn't yet know about. The profile is responsible 1081 // that the request context doesn't yet know about. The profile is responsible
1220 // for ensuring its URLRequestContexts appropriately discover the loaded 1082 // for ensuring its URLRequestContexts appropriately discover the loaded
1221 // extension. 1083 // extension.
1222 if (profile_) { 1084 if (profile_) {
(...skipping 229 matching lines...) Expand 10 before | Expand all | Expand 10 after
1452 1314
1453 bool ExtensionService::GetBrowserActionVisibility(const Extension* extension) { 1315 bool ExtensionService::GetBrowserActionVisibility(const Extension* extension) {
1454 return extension_prefs_->GetBrowserActionVisibility(extension); 1316 return extension_prefs_->GetBrowserActionVisibility(extension);
1455 } 1317 }
1456 1318
1457 void ExtensionService::SetBrowserActionVisibility(const Extension* extension, 1319 void ExtensionService::SetBrowserActionVisibility(const Extension* extension,
1458 bool visible) { 1320 bool visible) {
1459 extension_prefs_->SetBrowserActionVisibility(extension, visible); 1321 extension_prefs_->SetBrowserActionVisibility(extension, visible);
1460 } 1322 }
1461 1323
1324 // Some extensions will autoupdate themselves externally from Chrome. These
1325 // are typically part of some larger client application package. To support
1326 // these, the extension will register its location in the the preferences file
1327 // (and also, on Windows, in the registry) and this code will periodically
1328 // check that location for a .crx file, which it will then install locally if
1329 // a new version is available.
1330 // Errors are reported through ExtensionErrorReporter. Succcess is not
1331 // reported.
1462 void ExtensionService::CheckForExternalUpdates() { 1332 void ExtensionService::CheckForExternalUpdates() {
1463 BrowserThread::PostTask( 1333 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1464 BrowserThread::FILE, FROM_HERE, 1334
1465 NewRunnableMethod( 1335 // Note that this installation is intentionally silent (since it didn't
1466 backend_.get(), &ExtensionServiceBackend::CheckForExternalUpdates, 1336 // go through the front-end). Extensions that are registered in this
1467 scoped_refptr<ExtensionService>(this))); 1337 // way are effectively considered 'pre-bundled', and so implicitly
1338 // trusted. In general, if something has HKLM or filesystem access,
1339 // they could install an extension manually themselves anyway.
1340 external_extension_added_ = false;
1341
1342 // Ask each external extension provider to give us a call back for each
1343 // extension they know about. See OnExternalExtension(File|UpdateUrl)Found.
1344 ProviderCollection::const_iterator i;
1345 for (i = external_extension_providers_.begin();
1346 i != external_extension_providers_.end(); ++i) {
1347 ExternalExtensionProvider* provider = i->get();
1348 provider->VisitRegisteredExtension();
1349 }
1350
1351 // Uninstall of unclaimed extensions will happen after all the providers
1352 // had reported ready. Trigger uninstall even if there are no providers
1353 // installed:
1354 OnExternalProviderReady();
1468 } 1355 }
1469 1356
1470 void ExtensionService::UpdateExternalPolicyExtensionProvider() { 1357 void ExtensionService::OnExternalProviderReady() {
1471 const ListValue* list_pref = 1358 // An external provider has finished loading. We only take action
1472 profile_->GetPrefs()->GetList(prefs::kExtensionInstallForceList); 1359 // if all of them are finished. So first, we all of them.
Sam Kerner (Chrome) 2011/01/04 14:06:00 Comment missing a word?
gfeher 2011/01/04 23:37:09 Sorry. Done.
1473 ListValue* list_copy = NULL; 1360 ProviderCollection::const_iterator i;
1474 if (list_pref) 1361 for (i = external_extension_providers_.begin();
1475 list_copy = static_cast<ListValue*>(list_pref->DeepCopy()); 1362 i != external_extension_providers_.end(); ++i) {
1476 BrowserThread::PostTask( 1363 ExternalExtensionProvider* provider = i->get();
1477 BrowserThread::FILE, FROM_HERE, 1364 if (!provider->IsReady()) {
1478 NewRunnableMethod( 1365 return;
1479 backend_.get(), 1366 }
1480 &ExtensionServiceBackend::UpdateExternalPolicyExtensionProvider, 1367 }
1481 scoped_refptr<RefCountedList>( 1368
1482 new RefCountedList(list_copy)))); 1369 // All the providers are ready. Install any pending extensions.
1370 if (external_extension_added_ && updater()) {
1371 external_extension_added_ = false;
1372 updater()->CheckNow();
1373 }
1374
1375 // Uninstall all the unclaimed extensions.
1376 scoped_ptr<ExtensionPrefs::ExtensionsInfo> extensions_info(
1377 extension_prefs_->GetInstalledExtensionsInfo());
1378 for (size_t i = 0; i < extensions_info->size(); ++i) {
1379 ExtensionInfo* info = extensions_info->at(i).get();
1380 if (Extension::IsExternalLocation(info->extension_location))
1381 CheckExternalUninstall(info->extension_id);
1382 }
1483 } 1383 }
1484 1384
1485 void ExtensionService::UnloadExtension( 1385 void ExtensionService::UnloadExtension(
1486 const std::string& extension_id, 1386 const std::string& extension_id,
1487 UnloadedExtensionInfo::Reason reason) { 1387 UnloadedExtensionInfo::Reason reason) {
1488 // Make sure the extension gets deleted after we return from this function. 1388 // Make sure the extension gets deleted after we return from this function.
1489 scoped_refptr<const Extension> extension( 1389 scoped_refptr<const Extension> extension(
1490 GetExtensionByIdInternal(extension_id, true, true)); 1390 GetExtensionByIdInternal(extension_id, true, true));
1491 1391
1492 // This method can be called via PostTask, so the extension may have been 1392 // This method can be called via PostTask, so the extension may have been
(...skipping 390 matching lines...) Expand 10 before | Expand all | Expand 10 after
1883 const SkBitmap& ExtensionService::GetOmniboxIcon( 1783 const SkBitmap& ExtensionService::GetOmniboxIcon(
1884 const std::string& extension_id) { 1784 const std::string& extension_id) {
1885 return omnibox_icon_manager_.GetIcon(extension_id); 1785 return omnibox_icon_manager_.GetIcon(extension_id);
1886 } 1786 }
1887 1787
1888 const SkBitmap& ExtensionService::GetOmniboxPopupIcon( 1788 const SkBitmap& ExtensionService::GetOmniboxPopupIcon(
1889 const std::string& extension_id) { 1789 const std::string& extension_id) {
1890 return omnibox_popup_icon_manager_.GetIcon(extension_id); 1790 return omnibox_popup_icon_manager_.GetIcon(extension_id);
1891 } 1791 }
1892 1792
1893 void ExtensionService::ClearProvidersForTesting() {
1894 BrowserThread::PostTask(
1895 BrowserThread::FILE, FROM_HERE,
1896 NewRunnableMethod(
1897 backend_.get(), &ExtensionServiceBackend::ClearProvidersForTesting));
1898 }
1899
1900 void ExtensionService::AddProviderForTesting(
1901 ExternalExtensionProvider* test_provider) {
1902 BrowserThread::PostTask(
1903 BrowserThread::FILE, FROM_HERE,
1904 NewRunnableMethod(
1905 backend_.get(), &ExtensionServiceBackend::AddProviderForTesting,
1906 test_provider));
1907 }
1908
1909 void ExtensionService::OnExternalExtensionFileFound( 1793 void ExtensionService::OnExternalExtensionFileFound(
1910 const std::string& id, 1794 const std::string& id,
1911 const std::string& version, 1795 const Version* version,
1912 const FilePath& path, 1796 const FilePath& path,
1913 Extension::Location location) { 1797 Extension::Location location) {
1914 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); 1798 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1915 if (extension_prefs_->IsExtensionKilled(id)) 1799 if (extension_prefs_->IsExtensionKilled(id))
1916 return; 1800 return;
1917 1801
1802 DCHECK(version);
1803
1918 // Before even bothering to unpack, check and see if we already have this 1804 // Before even bothering to unpack, check and see if we already have this
1919 // version. This is important because these extensions are going to get 1805 // version. This is important because these extensions are going to get
1920 // installed on every startup. 1806 // installed on every startup.
1921 const Extension* existing = GetExtensionById(id, true); 1807 const Extension* existing = GetExtensionById(id, true);
1922 scoped_ptr<Version> other(Version::GetVersionFromString(version));
1923 if (existing) { 1808 if (existing) {
1924 switch (existing->version()->CompareTo(*other)) { 1809 switch (existing->version()->CompareTo(*version)) {
1925 case -1: // existing version is older, we should upgrade 1810 case -1: // existing version is older, we should upgrade
1926 break; 1811 break;
1927 case 0: // existing version is same, do nothing 1812 case 0: // existing version is same, do nothing
1928 return; 1813 return;
1929 case 1: // existing version is newer, uh-oh 1814 case 1: // existing version is newer, uh-oh
1930 LOG(WARNING) << "Found external version of extension " << id 1815 LOG(WARNING) << "Found external version of extension " << id
1931 << "that is older than current version. Current version " 1816 << "that is older than current version. Current version "
1932 << "is: " << existing->VersionString() << ". New version " 1817 << "is: " << existing->VersionString() << ". New version "
1933 << "is: " << version << ". Keeping current version."; 1818 << "is: " << version << ". Keeping current version.";
1934 return; 1819 return;
(...skipping 72 matching lines...) Expand 10 before | Expand all | Expand 10 after
2007 host->extension()->id(), 1892 host->extension()->id(),
2008 UnloadedExtensionInfo::DISABLE)); 1893 UnloadedExtensionInfo::DISABLE));
2009 break; 1894 break;
2010 } 1895 }
2011 1896
2012 case NotificationType::PREF_CHANGED: { 1897 case NotificationType::PREF_CHANGED: {
2013 std::string* pref_name = Details<std::string>(details).ptr(); 1898 std::string* pref_name = Details<std::string>(details).ptr();
2014 if (*pref_name == prefs::kExtensionInstallAllowList || 1899 if (*pref_name == prefs::kExtensionInstallAllowList ||
2015 *pref_name == prefs::kExtensionInstallDenyList) { 1900 *pref_name == prefs::kExtensionInstallDenyList) {
2016 CheckAdminBlacklist(); 1901 CheckAdminBlacklist();
2017 } else if (*pref_name == prefs::kExtensionInstallForceList) {
2018 UpdateExternalPolicyExtensionProvider();
2019 CheckForExternalUpdates();
2020 // TODO(gfeher): Also check for external extensions that can be
2021 // uninstalled because they were removed from the pref.
2022 // (crbug.com/63667)
2023 } else { 1902 } else {
2024 NOTREACHED() << "Unexpected preference name."; 1903 NOTREACHED() << "Unexpected preference name.";
2025 } 1904 }
2026 break; 1905 break;
2027 } 1906 }
2028 1907
2029 default: 1908 default:
2030 NOTREACHED() << "Unexpected notification type."; 1909 NOTREACHED() << "Unexpected notification type.";
2031 } 1910 }
2032 } 1911 }
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
2065 } 1944 }
2066 1945
2067 void ExtensionService::SetBeingUpgraded(const Extension* extension, 1946 void ExtensionService::SetBeingUpgraded(const Extension* extension,
2068 bool value) { 1947 bool value) {
2069 extension_runtime_data_[extension->id()].being_upgraded = value; 1948 extension_runtime_data_[extension->id()].being_upgraded = value;
2070 } 1949 }
2071 1950
2072 PropertyBag* ExtensionService::GetPropertyBag(const Extension* extension) { 1951 PropertyBag* ExtensionService::GetPropertyBag(const Extension* extension) {
2073 return &extension_runtime_data_[extension->id()].property_bag; 1952 return &extension_runtime_data_[extension->id()].property_bag;
2074 } 1953 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698