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.
5 #include "chrome/browser/extensions/pending_extension_manager.h"
9 #include "base/logging.h"
10 #include "base/version.h"
11 #include "chrome/common/extensions/extension_constants.h"
12 #include "content/public/browser/browser_thread.h"
13 #include "extensions/browser/extension_prefs.h"
14 #include "extensions/browser/extension_registry.h"
15 #include "extensions/common/constants.h"
16 #include "extensions/common/extension.h"
19 using content::BrowserThread
;
23 // Install predicate used by AddFromExternalUpdateUrl().
24 bool AlwaysInstall(const extensions::Extension
* extension
) {
28 std::string
GetVersionString(const Version
& version
) {
29 return version
.IsValid() ? version
.GetString() : "invalid";
34 namespace extensions
{
36 PendingExtensionManager::PendingExtensionManager(
37 content::BrowserContext
* context
)
38 : context_(context
) {}
40 PendingExtensionManager::~PendingExtensionManager() {}
42 const PendingExtensionInfo
* PendingExtensionManager::GetById(
43 const std::string
& id
) const {
44 PendingExtensionList::const_iterator iter
;
45 for (iter
= pending_extension_list_
.begin();
46 iter
!= pending_extension_list_
.end();
55 bool PendingExtensionManager::Remove(const std::string
& id
) {
56 PendingExtensionList::iterator iter
;
57 for (iter
= pending_extension_list_
.begin();
58 iter
!= pending_extension_list_
.end();
60 if (id
== iter
->id()) {
61 pending_extension_list_
.erase(iter
);
69 bool PendingExtensionManager::IsIdPending(const std::string
& id
) const {
70 return GetById(id
) != NULL
;
73 bool PendingExtensionManager::HasPendingExtensions() const {
74 return !pending_extension_list_
.empty();
77 bool PendingExtensionManager::HasPendingExtensionFromSync() const {
78 PendingExtensionList::const_iterator iter
;
79 for (iter
= pending_extension_list_
.begin();
80 iter
!= pending_extension_list_
.end();
82 if (iter
->is_from_sync())
89 bool PendingExtensionManager::AddFromSync(
90 const std::string
& id
,
91 const GURL
& update_url
,
92 const base::Version
& version
,
93 PendingExtensionInfo::ShouldAllowInstallPredicate should_allow_install
,
95 bool installed_by_custodian
) {
96 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
98 if (ExtensionRegistry::Get(context_
)->GetExtensionById(
99 id
, ExtensionRegistry::EVERYTHING
)) {
100 LOG(ERROR
) << "Trying to add pending extension " << id
101 << " which already exists";
105 // Make sure we don't ever try to install the CWS app, because even though
106 // it is listed as a syncable app (because its values need to be synced) it
107 // should already be installed on every instance.
108 if (id
== extensions::kWebStoreAppId
) {
113 int creation_flags
= Extension::NO_FLAGS
;
114 if (installed_by_custodian
) {
115 creation_flags
|= Extension::WAS_INSTALLED_BY_CUSTODIAN
;
118 static const bool kIsFromSync
= true;
119 static const Manifest::Location kSyncLocation
= Manifest::INTERNAL
;
120 static const bool kMarkAcknowledged
= false;
122 return AddExtensionImpl(id
,
126 should_allow_install
,
134 bool PendingExtensionManager::AddFromExtensionImport(
135 const std::string
& id
,
136 const GURL
& update_url
,
137 PendingExtensionInfo::ShouldAllowInstallPredicate should_allow_install
) {
138 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
140 if (ExtensionRegistry::Get(context_
)->GetExtensionById(
141 id
, ExtensionRegistry::EVERYTHING
)) {
142 LOG(ERROR
) << "Trying to add pending extension " << id
143 << " which already exists";
147 static const bool kIsFromSync
= false;
148 static const Manifest::Location kManifestLocation
= Manifest::INTERNAL
;
149 static const bool kMarkAcknowledged
= false;
150 static const bool kRemoteInstall
= false;
152 return AddExtensionImpl(id
,
156 should_allow_install
,
164 bool PendingExtensionManager::AddFromExternalUpdateUrl(
165 const std::string
& id
,
166 const std::string
& install_parameter
,
167 const GURL
& update_url
,
168 Manifest::Location location
,
170 bool mark_acknowledged
) {
171 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
173 static const bool kIsFromSync
= false;
174 static const bool kRemoteInstall
= false;
176 const Extension
* extension
= ExtensionRegistry::Get(context_
)
177 ->GetExtensionById(id
, ExtensionRegistry::EVERYTHING
);
178 if (extension
&& location
== Manifest::GetHigherPriorityLocation(
179 location
, extension
->location())) {
180 // If the new location has higher priority than the location of an existing
181 // extension, let the update process overwrite the existing extension.
183 if (ExtensionPrefs::Get(context_
)->IsExternalExtensionUninstalled(id
))
187 LOG(DFATAL
) << "Trying to add extension " << id
188 << " by external update, but it is already installed.";
193 return AddExtensionImpl(id
,
206 bool PendingExtensionManager::AddFromExternalFile(
207 const std::string
& id
,
208 Manifest::Location install_source
,
209 const Version
& version
,
211 bool mark_acknowledged
) {
212 // TODO(skerner): AddFromSync() checks to see if the extension is
213 // installed, but this method assumes that the caller already
214 // made sure it is not installed. Make all AddFrom*() methods
216 const GURL
& kUpdateUrl
= GURL::EmptyGURL();
217 static const bool kIsFromSync
= false;
218 static const bool kRemoteInstall
= false;
220 return AddExtensionImpl(id
,
232 void PendingExtensionManager::GetPendingIdsForUpdateCheck(
233 std::list
<std::string
>* out_ids_for_update_check
) const {
234 PendingExtensionList::const_iterator iter
;
235 for (iter
= pending_extension_list_
.begin();
236 iter
!= pending_extension_list_
.end();
238 Manifest::Location install_source
= iter
->install_source();
240 // Some install sources read a CRX from the filesystem. They can
241 // not be fetched from an update URL, so don't include them in the
243 if (install_source
== Manifest::EXTERNAL_PREF
||
244 install_source
== Manifest::EXTERNAL_REGISTRY
||
245 install_source
== Manifest::EXTERNAL_POLICY
) {
249 out_ids_for_update_check
->push_back(iter
->id());
253 bool PendingExtensionManager::AddExtensionImpl(
254 const std::string
& id
,
255 const std::string
& install_parameter
,
256 const GURL
& update_url
,
257 const Version
& version
,
258 PendingExtensionInfo::ShouldAllowInstallPredicate should_allow_install
,
260 Manifest::Location install_source
,
262 bool mark_acknowledged
,
263 bool remote_install
) {
264 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI
));
266 PendingExtensionInfo
info(id
,
270 should_allow_install
,
277 if (const PendingExtensionInfo
* pending
= GetById(id
)) {
278 // Bugs in this code will manifest as sporadic incorrect extension
279 // locations in situations where multiple install sources run at the
280 // same time. For example, on first login to a chrome os machine, an
281 // extension may be requested by sync and the default extension set.
282 // The following logging will help diagnose such issues.
283 VLOG(1) << "Extension id " << id
284 << " was entered for update more than once."
285 << " old location: " << pending
->install_source()
286 << " new location: " << install_source
287 << " old version: " << GetVersionString(pending
->version())
288 << " new version: " << GetVersionString(version
);
290 // Never override an existing extension with an older version. Only
291 // extensions from local CRX files have a known version; extensions from an
292 // update URL will get the latest version.
294 // If |pending| has the same or higher precedence than |info| then don't
295 // install |info| over |pending|.
296 if (pending
->CompareTo(info
) >= 0)
299 VLOG(1) << "Overwrite existing record.";
301 std::replace(pending_extension_list_
.begin(),
302 pending_extension_list_
.end(),
306 pending_extension_list_
.push_back(info
);
312 void PendingExtensionManager::AddForTesting(
313 const PendingExtensionInfo
& pending_extension_info
) {
314 pending_extension_list_
.push_back(pending_extension_info
);
317 } // namespace extensions