1 // Copyright (c) 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.
5 #include "chrome/browser/chromeos/extensions/install_limiter.h"
10 #include "base/file_util.h"
11 #include "base/threading/sequenced_worker_pool.h"
12 #include "chrome/browser/chrome_notification_types.h"
13 #include "chrome/browser/chromeos/extensions/install_limiter_factory.h"
14 #include "content/public/browser/browser_thread.h"
15 #include "content/public/browser/notification_details.h"
16 #include "content/public/browser/notification_source.h"
18 using content::BrowserThread
;
22 int64
GetFileSizeOnBlockingPool(const base::FilePath
& file
) {
23 DCHECK(BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread());
25 // Get file size. In case of error, sets 0 as file size to let the installer
28 return base::GetFileSize(file
, &size
) ? size
: 0;
33 namespace extensions
{
35 ////////////////////////////////////////////////////////////////////////////////
36 // InstallLimiter::DeferredInstall
38 InstallLimiter::DeferredInstall::DeferredInstall(
39 const scoped_refptr
<CrxInstaller
>& installer
,
40 const base::FilePath
& path
)
41 : installer(installer
),
45 InstallLimiter::DeferredInstall::~DeferredInstall() {
48 ////////////////////////////////////////////////////////////////////////////////
51 InstallLimiter
* InstallLimiter::Get(Profile
* profile
) {
52 return InstallLimiterFactory::GetForProfile(profile
);
55 InstallLimiter::InstallLimiter() : disabled_for_test_(false) {
58 InstallLimiter::~InstallLimiter() {
61 void InstallLimiter::DisableForTest() {
62 disabled_for_test_
= true;
65 void InstallLimiter::Add(const scoped_refptr
<CrxInstaller
>& installer
,
66 const base::FilePath
& path
) {
67 // No deferred installs when disabled for test.
68 if (disabled_for_test_
) {
69 installer
->InstallCrx(path
);
73 base::PostTaskAndReplyWithResult(
74 BrowserThread::GetBlockingPool(),
76 base::Bind(&GetFileSizeOnBlockingPool
, path
),
77 base::Bind(&InstallLimiter::AddWithSize
, AsWeakPtr(), installer
, path
));
80 void InstallLimiter::AddWithSize(
81 const scoped_refptr
<CrxInstaller
>& installer
,
82 const base::FilePath
& path
,
84 const int64 kBigAppSizeThreshold
= 1048576; // 1MB
86 if (size
<= kBigAppSizeThreshold
) {
87 RunInstall(installer
, path
);
89 // Stop wait timer and let install notification drive deferred installs.
94 deferred_installs_
.push(DeferredInstall(installer
, path
));
96 // When there are no running installs, wait a bit before running deferred
97 // installs to allow small app install to take precedence, especially when a
98 // big app is the first one in the list.
99 if (running_installers_
.empty() && !wait_timer_
.IsRunning()) {
100 const int kMaxWaitTimeInMs
= 5000; // 5 seconds.
103 base::TimeDelta::FromMilliseconds(kMaxWaitTimeInMs
),
104 this, &InstallLimiter::CheckAndRunDeferrredInstalls
);
108 void InstallLimiter::CheckAndRunDeferrredInstalls() {
109 if (deferred_installs_
.empty() || !running_installers_
.empty())
112 const DeferredInstall
& deferred
= deferred_installs_
.front();
113 RunInstall(deferred
.installer
, deferred
.path
);
114 deferred_installs_
.pop();
117 void InstallLimiter::RunInstall(const scoped_refptr
<CrxInstaller
>& installer
,
118 const base::FilePath
& path
) {
120 chrome::NOTIFICATION_CRX_INSTALLER_DONE
,
121 content::Source
<CrxInstaller
>(installer
.get()));
123 installer
->InstallCrx(path
);
124 running_installers_
.insert(installer
);
127 void InstallLimiter::Observe(int type
,
128 const content::NotificationSource
& source
,
129 const content::NotificationDetails
& details
) {
130 DCHECK_EQ(chrome::NOTIFICATION_CRX_INSTALLER_DONE
, type
);
132 registrar_
.Remove(this,
133 chrome::NOTIFICATION_CRX_INSTALLER_DONE
,
136 const scoped_refptr
<CrxInstaller
> installer
=
137 content::Source
<extensions::CrxInstaller
>(source
).ptr();
138 running_installers_
.erase(installer
);
139 CheckAndRunDeferrredInstalls();
142 } // namespace extensions