1 // Copyright 2014 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/extensions/chrome_process_manager_delegate.h" 6 7 #include "base/command_line.h" 8 #include "base/logging.h" 9 #include "chrome/browser/browser_process.h" 10 #include "chrome/browser/chrome_notification_types.h" 11 #include "chrome/browser/profiles/profile.h" 12 #include "chrome/browser/profiles/profile_manager.h" 13 #include "chrome/browser/ui/browser.h" 14 #include "chrome/browser/ui/browser_finder.h" 15 #include "chrome/common/chrome_switches.h" 16 #include "content/public/browser/notification_service.h" 17 #include "extensions/browser/extension_system.h" 18 #include "extensions/browser/process_manager.h" 19 #include "extensions/common/one_shot_event.h" 20 21 namespace extensions { 22 23 ChromeProcessManagerDelegate::ChromeProcessManagerDelegate() { 24 registrar_.Add(this, 25 chrome::NOTIFICATION_BROWSER_WINDOW_READY, 26 content::NotificationService::AllSources()); 27 registrar_.Add(this, 28 chrome::NOTIFICATION_PROFILE_CREATED, 29 content::NotificationService::AllSources()); 30 registrar_.Add(this, 31 chrome::NOTIFICATION_PROFILE_DESTROYED, 32 content::NotificationService::AllSources()); 33 } 34 35 ChromeProcessManagerDelegate::~ChromeProcessManagerDelegate() { 36 } 37 38 bool ChromeProcessManagerDelegate::IsBackgroundPageAllowed( 39 content::BrowserContext* context) const { 40 Profile* profile = static_cast<Profile*>(context); 41 42 // Disallow if the current session is a Guest mode session but the current 43 // browser context is *not* off-the-record. Such context is artificial and 44 // background page shouldn't be created in it. 45 // http://crbug.com/329498 46 return !(profile->IsGuestSession() && !profile->IsOffTheRecord()); 47 } 48 49 bool ChromeProcessManagerDelegate::DeferCreatingStartupBackgroundHosts( 50 content::BrowserContext* context) const { 51 Profile* profile = static_cast<Profile*>(context); 52 53 // The profile may not be valid yet if it is still being initialized. 54 // In that case, defer loading, since it depends on an initialized profile. 55 // Background hosts will be loaded later via NOTIFICATION_PROFILE_CREATED. 56 // http://crbug.com/222473 57 if (!g_browser_process->profile_manager()->IsValidProfile(profile)) 58 return true; 59 60 // There are no browser windows open and the browser process was 61 // started to show the app launcher. Background hosts will be loaded later 62 // via NOTIFICATION_BROWSER_WINDOW_READY. http://crbug.com/178260 63 return chrome::GetTotalBrowserCountForProfile(profile) == 0 && 64 CommandLine::ForCurrentProcess()->HasSwitch(switches::kShowAppList); 65 } 66 67 void ChromeProcessManagerDelegate::Observe( 68 int type, 69 const content::NotificationSource& source, 70 const content::NotificationDetails& details) { 71 switch (type) { 72 case chrome::NOTIFICATION_BROWSER_WINDOW_READY: { 73 Browser* browser = content::Source<Browser>(source).ptr(); 74 OnBrowserWindowReady(browser); 75 break; 76 } 77 case chrome::NOTIFICATION_PROFILE_CREATED: { 78 Profile* profile = content::Source<Profile>(source).ptr(); 79 OnProfileCreated(profile); 80 break; 81 } 82 case chrome::NOTIFICATION_PROFILE_DESTROYED: { 83 Profile* profile = content::Source<Profile>(source).ptr(); 84 OnProfileDestroyed(profile); 85 break; 86 } 87 default: 88 NOTREACHED(); 89 } 90 } 91 92 void ChromeProcessManagerDelegate::OnBrowserWindowReady(Browser* browser) { 93 Profile* profile = browser->profile(); 94 DCHECK(profile); 95 96 // If the extension system isn't ready yet the background hosts will be 97 // created automatically when it is. 98 ExtensionSystem* system = ExtensionSystem::Get(profile); 99 if (!system->ready().is_signaled()) 100 return; 101 102 // Inform the process manager for this profile that the window is ready. 103 // We continue to observe the notification in case browser windows open for 104 // a related incognito profile or other regular profiles. 105 ProcessManager* manager = system->process_manager(); 106 if (!manager) // Tests may not have a process manager. 107 return; 108 DCHECK_EQ(profile, manager->GetBrowserContext()); 109 manager->MaybeCreateStartupBackgroundHosts(); 110 111 // For incognito profiles also inform the original profile's process manager 112 // that the window is ready. This will usually be a no-op because the 113 // original profile's process manager should have been informed when the 114 // non-incognito window opened. 115 if (profile->IsOffTheRecord()) { 116 Profile* original_profile = profile->GetOriginalProfile(); 117 ProcessManager* original_manager = 118 ExtensionSystem::Get(original_profile)->process_manager(); 119 DCHECK(original_manager); 120 DCHECK_EQ(original_profile, original_manager->GetBrowserContext()); 121 original_manager->MaybeCreateStartupBackgroundHosts(); 122 } 123 } 124 125 void ChromeProcessManagerDelegate::OnProfileCreated(Profile* profile) { 126 // Incognito profiles are handled by their original profile. 127 if (profile->IsOffTheRecord()) 128 return; 129 130 // The profile can be created before the extension system is ready. 131 ProcessManager* manager = ExtensionSystem::Get(profile)->process_manager(); 132 if (!manager) 133 return; 134 135 // The profile might have been initialized asynchronously (in parallel with 136 // extension system startup). Now that initialization is complete the 137 // ProcessManager can load deferred background pages. 138 manager->MaybeCreateStartupBackgroundHosts(); 139 } 140 141 void ChromeProcessManagerDelegate::OnProfileDestroyed(Profile* profile) { 142 // Close background hosts when the last profile is closed so that they 143 // have time to shutdown various objects on different threads. The 144 // ProfileManager destructor is called too late in the shutdown sequence. 145 // http://crbug.com/15708 146 ProcessManager* manager = ExtensionSystem::Get(profile)->process_manager(); 147 if (manager) 148 manager->CloseBackgroundHosts(); 149 150 // If this profile owns an incognito profile, but it is destroyed before the 151 // incognito profile is destroyed, then close the incognito background hosts 152 // as well. This happens in a few tests. http://crbug.com/138843 153 if (!profile->IsOffTheRecord() && profile->HasOffTheRecordProfile()) { 154 ProcessManager* incognito_manager = 155 ExtensionSystem::Get(profile->GetOffTheRecordProfile()) 156 ->process_manager(); 157 if (incognito_manager) 158 incognito_manager->CloseBackgroundHosts(); 159 } 160 } 161 162 } // namespace extensions 163