1 // Copyright 2013 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/apps/ephemeral_app_service.h"
6
7 #include "apps/app_lifetime_monitor_factory.h"
8 #include "base/command_line.h"
9 #include "base/message_loop/message_loop.h"
10 #include "chrome/browser/apps/ephemeral_app_service_factory.h"
11 #include "chrome/browser/chrome_notification_types.h"
12 #include "chrome/browser/extensions/extension_service.h"
13 #include "chrome/browser/extensions/extension_util.h"
14 #include "chrome/browser/profiles/profile.h"
15 #include "chrome/common/chrome_switches.h"
16 #include "content/public/browser/notification_service.h"
17 #include "content/public/browser/notification_source.h"
18 #include "content/public/browser/notification_types.h"
19 #include "extensions/browser/extension_prefs.h"
20 #include "extensions/browser/extension_registry.h"
21 #include "extensions/browser/extension_system.h"
22 #include "extensions/browser/extension_util.h"
23 #include "extensions/browser/notification_types.h"
24 #include "extensions/browser/uninstall_reason.h"
25 #include "extensions/common/extension.h"
26 #include "extensions/common/extension_set.h"
27
28 using extensions::Extension;
29 using extensions::ExtensionPrefs;
30 using extensions::ExtensionRegistry;
31 using extensions::ExtensionSet;
32 using extensions::ExtensionSystem;
33
34 namespace {
35
36 // The number of seconds after startup before performing garbage collection
37 // of ephemeral apps.
38 const int kGarbageCollectAppsStartupDelay = 60;
39
40 // The number of seconds after an ephemeral app has been installed before
41 // performing garbage collection.
42 const int kGarbageCollectAppsInstallDelay = 15;
43
44 // When the number of ephemeral apps reaches this count, trigger garbage
45 // collection to trim off the least-recently used apps in excess of
46 // kMaxEphemeralAppsCount.
47 const int kGarbageCollectAppsTriggerCount = 35;
48
49 // The number of seconds after an app has stopped running before it will be
50 // disabled.
51 const int kDefaultDisableAppDelay = 1;
52
53 // The number of seconds after startup before disabling inactive ephemeral apps.
54 const int kDisableAppsOnStartupDelay = 5;
55
56 } // namespace
57
58 const int EphemeralAppService::kAppInactiveThreshold = 10;
59 const int EphemeralAppService::kAppKeepThreshold = 1;
60 const int EphemeralAppService::kMaxEphemeralAppsCount = 30;
61
62 // static
Get(Profile * profile)63 EphemeralAppService* EphemeralAppService::Get(Profile* profile) {
64 return EphemeralAppServiceFactory::GetForProfile(profile);
65 }
66
EphemeralAppService(Profile * profile)67 EphemeralAppService::EphemeralAppService(Profile* profile)
68 : profile_(profile),
69 extension_registry_observer_(this),
70 app_lifetime_monitor_observer_(this),
71 ephemeral_app_count_(-1),
72 disable_idle_app_delay_(kDefaultDisableAppDelay),
73 weak_ptr_factory_(this) {
74 registrar_.Add(this,
75 extensions::NOTIFICATION_EXTENSIONS_READY_DEPRECATED,
76 content::Source<Profile>(profile_));
77 }
78
~EphemeralAppService()79 EphemeralAppService::~EphemeralAppService() {
80 }
81
ClearCachedApps()82 void EphemeralAppService::ClearCachedApps() {
83 // Cancel any pending garbage collects.
84 garbage_collect_apps_timer_.Stop();
85
86 ExtensionRegistry* registry = ExtensionRegistry::Get(profile_);
87 DCHECK(registry);
88 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
89 DCHECK(prefs);
90 ExtensionService* service =
91 ExtensionSystem::Get(profile_)->extension_service();
92 DCHECK(service);
93
94 scoped_ptr<ExtensionSet> extensions =
95 registry->GenerateInstalledExtensionsSet();
96
97 for (ExtensionSet::const_iterator it = extensions->begin();
98 it != extensions->end();
99 ++it) {
100 std::string extension_id = (*it)->id();
101 if (!prefs->IsEphemeralApp(extension_id))
102 continue;
103
104 // Do not remove apps that are running.
105 if (!extensions::util::IsExtensionIdle(extension_id, profile_))
106 continue;
107
108 DCHECK(registry->GetExtensionById(extension_id,
109 ExtensionRegistry::EVERYTHING));
110 service->UninstallExtension(
111 extension_id,
112 extensions::UNINSTALL_REASON_ORPHANED_EPHEMERAL_EXTENSION,
113 base::Bind(&base::DoNothing),
114 NULL);
115 }
116 }
117
Observe(int type,const content::NotificationSource & source,const content::NotificationDetails & details)118 void EphemeralAppService::Observe(
119 int type,
120 const content::NotificationSource& source,
121 const content::NotificationDetails& details) {
122 switch (type) {
123 case extensions::NOTIFICATION_EXTENSIONS_READY_DEPRECATED: {
124 Init();
125 break;
126 }
127 default:
128 NOTREACHED();
129 }
130 }
131
OnExtensionWillBeInstalled(content::BrowserContext * browser_context,const extensions::Extension * extension,bool is_update,bool from_ephemeral,const std::string & old_name)132 void EphemeralAppService::OnExtensionWillBeInstalled(
133 content::BrowserContext* browser_context,
134 const extensions::Extension* extension,
135 bool is_update,
136 bool from_ephemeral,
137 const std::string& old_name) {
138 if (from_ephemeral) {
139 // An ephemeral app was just promoted to a regular installed app.
140 --ephemeral_app_count_;
141 DCHECK_GE(ephemeral_app_count_, 0);
142 HandleEphemeralAppPromoted(extension);
143 } else if (!is_update &&
144 extensions::util::IsEphemeralApp(extension->id(), profile_)) {
145 // A new ephemeral app was launched.
146 ++ephemeral_app_count_;
147 if (ephemeral_app_count_ >= kGarbageCollectAppsTriggerCount) {
148 TriggerGarbageCollect(
149 base::TimeDelta::FromSeconds(kGarbageCollectAppsInstallDelay));
150 }
151 }
152 }
153
OnExtensionUninstalled(content::BrowserContext * browser_context,const extensions::Extension * extension,extensions::UninstallReason reason)154 void EphemeralAppService::OnExtensionUninstalled(
155 content::BrowserContext* browser_context,
156 const extensions::Extension* extension,
157 extensions::UninstallReason reason) {
158 if (extensions::util::IsEphemeralApp(extension->id(), profile_)) {
159 --ephemeral_app_count_;
160 DCHECK_GE(ephemeral_app_count_, 0);
161 }
162 }
163
OnAppStop(Profile * profile,const std::string & app_id)164 void EphemeralAppService::OnAppStop(Profile* profile,
165 const std::string& app_id) {
166 if (!extensions::util::IsEphemeralApp(app_id, profile_))
167 return;
168
169 base::MessageLoop::current()->PostDelayedTask(
170 FROM_HERE,
171 base::Bind(&EphemeralAppService::DisableEphemeralApp,
172 weak_ptr_factory_.GetWeakPtr(),
173 app_id),
174 base::TimeDelta::FromSeconds(disable_idle_app_delay_));
175 }
176
OnChromeTerminating()177 void EphemeralAppService::OnChromeTerminating() {
178 garbage_collect_apps_timer_.Stop();
179
180 extension_registry_observer_.RemoveAll();
181 app_lifetime_monitor_observer_.RemoveAll();
182 }
183
Init()184 void EphemeralAppService::Init() {
185 InitEphemeralAppCount();
186
187 // Start observing.
188 extension_registry_observer_.Add(ExtensionRegistry::Get(profile_));
189 app_lifetime_monitor_observer_.Add(
190 apps::AppLifetimeMonitorFactory::GetForProfile(profile_));
191
192 // Execute startup clean up tasks (except during tests).
193 if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kTestType))
194 return;
195
196 TriggerGarbageCollect(
197 base::TimeDelta::FromSeconds(kGarbageCollectAppsStartupDelay));
198
199 base::MessageLoop::current()->PostDelayedTask(
200 FROM_HERE,
201 base::Bind(&EphemeralAppService::DisableEphemeralAppsOnStartup,
202 weak_ptr_factory_.GetWeakPtr()),
203 base::TimeDelta::FromSeconds(kDisableAppsOnStartupDelay));
204 }
205
InitEphemeralAppCount()206 void EphemeralAppService::InitEphemeralAppCount() {
207 scoped_ptr<ExtensionSet> extensions =
208 ExtensionRegistry::Get(profile_)->GenerateInstalledExtensionsSet();
209 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
210 DCHECK(prefs);
211
212 ephemeral_app_count_ = 0;
213 for (ExtensionSet::const_iterator it = extensions->begin();
214 it != extensions->end(); ++it) {
215 const Extension* extension = it->get();
216 if (prefs->IsEphemeralApp(extension->id()))
217 ++ephemeral_app_count_;
218 }
219 }
220
DisableEphemeralApp(const std::string & app_id)221 void EphemeralAppService::DisableEphemeralApp(const std::string& app_id) {
222 if (!extensions::util::IsEphemeralApp(app_id, profile_) ||
223 !extensions::util::IsExtensionIdle(app_id, profile_)) {
224 return;
225 }
226
227 // After an ephemeral app has stopped running, unload it from extension
228 // system and disable it to prevent all background activity.
229 ExtensionService* service =
230 ExtensionSystem::Get(profile_)->extension_service();
231 DCHECK(service);
232 service->DisableExtension(app_id, Extension::DISABLE_INACTIVE_EPHEMERAL_APP);
233 }
234
DisableEphemeralAppsOnStartup()235 void EphemeralAppService::DisableEphemeralAppsOnStartup() {
236 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
237 DCHECK(prefs);
238 ExtensionService* service =
239 ExtensionSystem::Get(profile_)->extension_service();
240 DCHECK(service);
241
242 // Ensure that all inactive ephemeral apps are disabled to prevent all
243 // background activity. This is done on startup to catch any apps that escaped
244 // being disabled on shutdown.
245 scoped_ptr<ExtensionSet> extensions =
246 ExtensionRegistry::Get(profile_)->GenerateInstalledExtensionsSet();
247 for (ExtensionSet::const_iterator it = extensions->begin();
248 it != extensions->end();
249 ++it) {
250 const Extension* extension = it->get();
251 if (!prefs->IsEphemeralApp(extension->id()))
252 continue;
253
254 // Only V2 apps are installed ephemerally. Remove other ephemeral app types
255 // that were cached before this policy was introduced.
256 if (!extension->is_platform_app()) {
257 service->UninstallExtension(
258 extension->id(),
259 extensions::UNINSTALL_REASON_ORPHANED_EPHEMERAL_EXTENSION,
260 base::Bind(&base::DoNothing),
261 NULL);
262 continue;
263 }
264
265 if (!prefs->HasDisableReason(extension->id(),
266 Extension::DISABLE_INACTIVE_EPHEMERAL_APP) &&
267 !prefs->IsExtensionRunning(extension->id()) &&
268 extensions::util::IsExtensionIdle(extension->id(), profile_)) {
269 service->DisableExtension(extension->id(),
270 Extension::DISABLE_INACTIVE_EPHEMERAL_APP);
271 }
272 }
273 }
274
HandleEphemeralAppPromoted(const Extension * app)275 void EphemeralAppService::HandleEphemeralAppPromoted(const Extension* app) {
276 // When ephemeral apps are promoted to regular install apps, remove the
277 // DISABLE_INACTIVE_EPHEMERAL_APP reason and enable the app if there are no
278 // other reasons.
279 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
280 DCHECK(prefs);
281
282 int disable_reasons = prefs->GetDisableReasons(app->id());
283 if (disable_reasons & Extension::DISABLE_INACTIVE_EPHEMERAL_APP) {
284 prefs->RemoveDisableReason(app->id(),
285 Extension::DISABLE_INACTIVE_EPHEMERAL_APP);
286 if (disable_reasons == Extension::DISABLE_INACTIVE_EPHEMERAL_APP)
287 prefs->SetExtensionState(app->id(), Extension::ENABLED);
288 }
289 }
290
TriggerGarbageCollect(const base::TimeDelta & delay)291 void EphemeralAppService::TriggerGarbageCollect(const base::TimeDelta& delay) {
292 if (!garbage_collect_apps_timer_.IsRunning()) {
293 garbage_collect_apps_timer_.Start(
294 FROM_HERE,
295 delay,
296 this,
297 &EphemeralAppService::GarbageCollectApps);
298 }
299 }
300
GarbageCollectApps()301 void EphemeralAppService::GarbageCollectApps() {
302 ExtensionRegistry* registry = ExtensionRegistry::Get(profile_);
303 DCHECK(registry);
304 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
305 DCHECK(prefs);
306
307 scoped_ptr<ExtensionSet> extensions =
308 registry->GenerateInstalledExtensionsSet();
309
310 int app_count = 0;
311 LaunchTimeAppMap app_launch_times;
312 std::set<std::string> remove_app_ids;
313
314 // Populate a list of ephemeral apps, ordered by their last launch time.
315 for (ExtensionSet::const_iterator it = extensions->begin();
316 it != extensions->end(); ++it) {
317 const Extension* extension = it->get();
318 if (!prefs->IsEphemeralApp(extension->id()))
319 continue;
320
321 ++app_count;
322
323 // Do not remove ephemeral apps that are running.
324 if (!extensions::util::IsExtensionIdle(extension->id(), profile_))
325 continue;
326
327 base::Time last_launch_time = prefs->GetLastLaunchTime(extension->id());
328
329 // If the last launch time is invalid, this may be because it was just
330 // installed. So use the install time. If this is also null for some reason,
331 // the app will be removed.
332 if (last_launch_time.is_null())
333 last_launch_time = prefs->GetInstallTime(extension->id());
334
335 app_launch_times.insert(std::make_pair(last_launch_time, extension->id()));
336 }
337
338 ExtensionService* service =
339 ExtensionSystem::Get(profile_)->extension_service();
340 DCHECK(service);
341 // Execute the eviction policies and remove apps marked for deletion.
342 if (!app_launch_times.empty()) {
343 GetAppsToRemove(app_count, app_launch_times, &remove_app_ids);
344
345 for (std::set<std::string>::const_iterator id = remove_app_ids.begin();
346 id != remove_app_ids.end(); ++id) {
347 // Protect against cascading uninstalls.
348 if (!registry->GetExtensionById(*id, ExtensionRegistry::EVERYTHING))
349 continue;
350
351 service->UninstallExtension(
352 *id,
353 extensions::UNINSTALL_REASON_ORPHANED_EPHEMERAL_EXTENSION,
354 base::Bind(&base::DoNothing),
355 NULL);
356 }
357 }
358 }
359
360 // static
GetAppsToRemove(int app_count,const LaunchTimeAppMap & app_launch_times,std::set<std::string> * remove_app_ids)361 void EphemeralAppService::GetAppsToRemove(
362 int app_count,
363 const LaunchTimeAppMap& app_launch_times,
364 std::set<std::string>* remove_app_ids) {
365 base::Time time_now = base::Time::Now();
366 const base::Time inactive_threshold =
367 time_now - base::TimeDelta::FromDays(kAppInactiveThreshold);
368 const base::Time keep_threshold =
369 time_now - base::TimeDelta::FromDays(kAppKeepThreshold);
370
371 // Visit the apps in order of least recently to most recently launched.
372 for (LaunchTimeAppMap::const_iterator it = app_launch_times.begin();
373 it != app_launch_times.end(); ++it) {
374 // Cannot remove apps that have been launched recently. So break when we
375 // reach the new apps.
376 if (it->first > keep_threshold)
377 break;
378
379 // Remove ephemeral apps that have been inactive for a while or if the cache
380 // is larger than the desired size.
381 if (it->first < inactive_threshold || app_count > kMaxEphemeralAppsCount) {
382 remove_app_ids->insert(it->second);
383 --app_count;
384 } else {
385 break;
386 }
387 }
388 }
389