[email protected] | 29679dea | 2012-03-10 03:20:28 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #ifndef CHROME_BROWSER_EXTENSIONS_WEBSTORE_INSTALLER_H_ |
| 6 | #define CHROME_BROWSER_EXTENSIONS_WEBSTORE_INSTALLER_H_ |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 7 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 8 | #include <list> |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 9 | #include <string> |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 10 | |
| 11 | #include "base/compiler_specific.h" |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 12 | #include "base/memory/ref_counted.h" |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 13 | #include "base/memory/scoped_ptr.h" |
[email protected] | d59d40c | 2012-08-08 18:18:37 | [diff] [blame] | 14 | #include "base/supports_user_data.h" |
[email protected] | c80fe5f | 2014-03-26 04:36:30 | [diff] [blame^] | 15 | #include "base/timer/timer.h" |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 16 | #include "base/values.h" |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 17 | #include "base/version.h" |
[email protected] | af6efb2 | 2012-10-12 02:23:05 | [diff] [blame] | 18 | #include "chrome/browser/extensions/extension_install_prompt.h" |
[email protected] | ed0757ec | 2012-08-02 17:23:26 | [diff] [blame] | 19 | #include "content/public/browser/browser_thread.h" |
[email protected] | 7f391888 | 2014-01-14 06:14:56 | [diff] [blame] | 20 | #include "content/public/browser/download_interrupt_reasons.h" |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 21 | #include "content/public/browser/download_item.h" |
[email protected] | 6c2381d | 2011-10-19 02:52:53 | [diff] [blame] | 22 | #include "content/public/browser/notification_observer.h" |
| 23 | #include "content/public/browser/notification_registrar.h" |
[email protected] | 2acf3a5 | 2014-02-13 00:26:00 | [diff] [blame] | 24 | #include "content/public/browser/web_contents_observer.h" |
[email protected] | 301116c6 | 2013-11-26 10:37:45 | [diff] [blame] | 25 | #include "extensions/common/manifest_handlers/shared_module_info.h" |
[email protected] | 7b4eea60 | 2013-02-08 06:19:47 | [diff] [blame] | 26 | #include "ui/gfx/image/image_skia.h" |
[email protected] | a6483d2 | 2013-07-03 22:11:00 | [diff] [blame] | 27 | #include "url/gurl.h" |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 28 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 29 | class Profile; |
| 30 | |
[email protected] | a3ef483 | 2013-02-02 05:12:33 | [diff] [blame] | 31 | namespace base { |
| 32 | class FilePath; |
| 33 | } |
| 34 | |
[email protected] | cdcb1dee | 2012-01-04 00:46:20 | [diff] [blame] | 35 | namespace content { |
[email protected] | 2acf3a5 | 2014-02-13 00:26:00 | [diff] [blame] | 36 | class WebContents; |
[email protected] | cdcb1dee | 2012-01-04 00:46:20 | [diff] [blame] | 37 | } |
| 38 | |
[email protected] | 3d61a7f | 2012-07-12 19:11:25 | [diff] [blame] | 39 | namespace extensions { |
| 40 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 41 | class Extension; |
[email protected] | 21c0104 | 2013-03-10 23:41:14 | [diff] [blame] | 42 | class Manifest; |
| 43 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 44 | // Downloads and installs extensions from the web store. |
[email protected] | 2acf3a5 | 2014-02-13 00:26:00 | [diff] [blame] | 45 | class WebstoreInstaller : public content::NotificationObserver, |
| 46 | public content::DownloadItem::Observer, |
| 47 | public content::WebContentsObserver, |
| 48 | public base::RefCountedThreadSafe< |
[email protected] | ed0757ec | 2012-08-02 17:23:26 | [diff] [blame] | 49 | WebstoreInstaller, content::BrowserThread::DeleteOnUIThread> { |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 50 | public: |
[email protected] | 084e3548 | 2013-09-25 02:46:19 | [diff] [blame] | 51 | enum InstallSource { |
[email protected] | 99dfecd | 2011-10-18 01:11:50 | [diff] [blame] | 52 | // Inline installs trigger slightly different behavior (install source |
| 53 | // is different, download referrers are the item's page in the gallery). |
[email protected] | 084e3548 | 2013-09-25 02:46:19 | [diff] [blame] | 54 | INSTALL_SOURCE_INLINE, |
| 55 | INSTALL_SOURCE_APP_LAUNCHER, |
| 56 | INSTALL_SOURCE_OTHER |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 57 | }; |
| 58 | |
[email protected] | bcd1eaf7 | 2012-10-03 05:42:29 | [diff] [blame] | 59 | enum FailureReason { |
| 60 | FAILURE_REASON_CANCELLED, |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 61 | FAILURE_REASON_DEPENDENCY_NOT_FOUND, |
| 62 | FAILURE_REASON_DEPENDENCY_NOT_SHARED_MODULE, |
[email protected] | bcd1eaf7 | 2012-10-03 05:42:29 | [diff] [blame] | 63 | FAILURE_REASON_OTHER |
| 64 | }; |
| 65 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 66 | enum ManifestCheckLevel { |
| 67 | // Do not check for any manifest equality. |
| 68 | MANIFEST_CHECK_LEVEL_NONE, |
| 69 | |
| 70 | // Only check that the expected and actual permissions have the same |
| 71 | // effective permissions. |
| 72 | MANIFEST_CHECK_LEVEL_LOOSE, |
| 73 | |
| 74 | // All data in the expected and actual manifests must match. |
| 75 | MANIFEST_CHECK_LEVEL_STRICT, |
| 76 | }; |
| 77 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 78 | class Delegate { |
| 79 | public: |
[email protected] | bcd1eaf7 | 2012-10-03 05:42:29 | [diff] [blame] | 80 | virtual void OnExtensionDownloadStarted(const std::string& id, |
| 81 | content::DownloadItem* item); |
| 82 | virtual void OnExtensionDownloadProgress(const std::string& id, |
| 83 | content::DownloadItem* item); |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 84 | virtual void OnExtensionInstallSuccess(const std::string& id) = 0; |
| 85 | virtual void OnExtensionInstallFailure(const std::string& id, |
[email protected] | bcd1eaf7 | 2012-10-03 05:42:29 | [diff] [blame] | 86 | const std::string& error, |
| 87 | FailureReason reason) = 0; |
[email protected] | 512d03f | 2012-06-26 01:06:06 | [diff] [blame] | 88 | |
| 89 | protected: |
| 90 | virtual ~Delegate() {} |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 91 | }; |
| 92 | |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 93 | // Contains information about what parts of the extension install process can |
| 94 | // be skipped or modified. If one of these is present, it means that a CRX |
| 95 | // download was initiated by WebstoreInstaller. The Approval instance should |
| 96 | // be checked further for additional details. |
[email protected] | d59d40c | 2012-08-08 18:18:37 | [diff] [blame] | 97 | struct Approval : public base::SupportsUserData::Data { |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 98 | static scoped_ptr<Approval> CreateWithInstallPrompt(Profile* profile); |
[email protected] | 8529082 | 2013-08-23 20:27:38 | [diff] [blame] | 99 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 100 | // Creates an Approval for installing a shared module. |
| 101 | static scoped_ptr<Approval> CreateForSharedModule(Profile* profile); |
| 102 | |
[email protected] | 8529082 | 2013-08-23 20:27:38 | [diff] [blame] | 103 | // Creates an Approval that will skip putting up an install confirmation |
| 104 | // prompt if the actual manifest from the extension to be installed matches |
| 105 | // |parsed_manifest|. The |strict_manifest_check| controls whether we want |
| 106 | // to require an exact manifest match, or are willing to tolerate a looser |
| 107 | // check just that the effective permissions are the same. |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 108 | static scoped_ptr<Approval> CreateWithNoInstallPrompt( |
| 109 | Profile* profile, |
| 110 | const std::string& extension_id, |
[email protected] | 8529082 | 2013-08-23 20:27:38 | [diff] [blame] | 111 | scoped_ptr<base::DictionaryValue> parsed_manifest, |
| 112 | bool strict_manifest_check); |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 113 | |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 114 | virtual ~Approval(); |
| 115 | |
| 116 | // The extension id that was approved for installation. |
| 117 | std::string extension_id; |
| 118 | |
| 119 | // The profile the extension should be installed into. |
| 120 | Profile* profile; |
| 121 | |
| 122 | // The expected manifest, before localization. |
[email protected] | 21c0104 | 2013-03-10 23:41:14 | [diff] [blame] | 123 | scoped_ptr<Manifest> manifest; |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 124 | |
| 125 | // Whether to use a bubble notification when an app is installed, instead of |
| 126 | // the default behavior of transitioning to the new tab page. |
| 127 | bool use_app_installed_bubble; |
| 128 | |
| 129 | // Whether to skip the post install UI like the extension installed bubble. |
| 130 | bool skip_post_install_ui; |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 131 | |
| 132 | // Whether to skip the install dialog once the extension has been downloaded |
| 133 | // and unpacked. One reason this can be true is that in the normal webstore |
| 134 | // installation, the dialog is shown earlier, before any download is done, |
| 135 | // so there's no need to show it again. |
| 136 | bool skip_install_dialog; |
| 137 | |
[email protected] | fc652c0 | 2013-02-04 05:16:23 | [diff] [blame] | 138 | // Whether we should enable the launcher before installing the app. |
| 139 | bool enable_launcher; |
| 140 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 141 | // Manifest check level for checking actual manifest against expected |
| 142 | // manifest. |
| 143 | ManifestCheckLevel manifest_check_level; |
[email protected] | 8529082 | 2013-08-23 20:27:38 | [diff] [blame] | 144 | |
[email protected] | af6efb2 | 2012-10-12 02:23:05 | [diff] [blame] | 145 | // Used to show the install dialog. |
| 146 | ExtensionInstallPrompt::ShowDialogCallback show_dialog_callback; |
| 147 | |
[email protected] | 7b4eea60 | 2013-02-08 06:19:47 | [diff] [blame] | 148 | // The icon to use to display the extension while it is installing. |
| 149 | gfx::ImageSkia installing_icon; |
| 150 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 151 | // A dummy extension created from |manifest|; |
| 152 | scoped_refptr<Extension> dummy_extension; |
| 153 | |
| 154 | // Required minimum version. |
[email protected] | c5e4a222 | 2014-01-03 16:06:13 | [diff] [blame] | 155 | scoped_ptr<Version> minimum_version; |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 156 | |
[email protected] | ce35418b | 2013-11-25 01:22:33 | [diff] [blame] | 157 | // Ephemeral apps (experimental) are not permanently installed in Chrome. |
| 158 | bool is_ephemeral; |
| 159 | |
[email protected] | 89019d6 | 2012-05-17 18:47:09 | [diff] [blame] | 160 | private: |
| 161 | Approval(); |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 162 | }; |
| 163 | |
| 164 | // Gets the Approval associated with the |download|, or NULL if there's none. |
| 165 | // Note that the Approval is owned by |download|. |
| 166 | static const Approval* GetAssociatedApproval( |
| 167 | const content::DownloadItem& download); |
| 168 | |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 169 | // Creates a WebstoreInstaller for downloading and installing the extension |
| 170 | // with the given |id| from the Chrome Web Store. If |delegate| is not NULL, |
| 171 | // it will be notified when the install succeeds or fails. The installer will |
| 172 | // use the specified |controller| to download the extension. Only one |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 173 | // WebstoreInstaller can use a specific controller at any given time. This |
| 174 | // also associates the |approval| with this install. |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 175 | // Note: the delegate should stay alive until being called back. |
| 176 | WebstoreInstaller(Profile* profile, |
| 177 | Delegate* delegate, |
[email protected] | 2acf3a5 | 2014-02-13 00:26:00 | [diff] [blame] | 178 | content::WebContents* web_contents, |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 179 | const std::string& id, |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 180 | scoped_ptr<Approval> approval, |
[email protected] | 084e3548 | 2013-09-25 02:46:19 | [diff] [blame] | 181 | InstallSource source); |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 182 | |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 183 | // Starts downloading and installing the extension. |
| 184 | void Start(); |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 185 | |
[email protected] | 6c2381d | 2011-10-19 02:52:53 | [diff] [blame] | 186 | // content::NotificationObserver |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 187 | virtual void Observe(int type, |
[email protected] | 6c2381d | 2011-10-19 02:52:53 | [diff] [blame] | 188 | const content::NotificationSource& source, |
| 189 | const content::NotificationDetails& details) OVERRIDE; |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 190 | |
[email protected] | e074e32 | 2012-10-30 01:12:09 | [diff] [blame] | 191 | // Removes the reference to the delegate passed in the constructor. Used when |
| 192 | // the delegate object must be deleted before this object. |
| 193 | void InvalidateDelegate(); |
| 194 | |
[email protected] | 9c265d0 | 2011-12-29 22:13:43 | [diff] [blame] | 195 | // Instead of using the default download directory, use |directory| instead. |
| 196 | // This does *not* transfer ownership of |directory|. |
[email protected] | a3ef483 | 2013-02-02 05:12:33 | [diff] [blame] | 197 | static void SetDownloadDirectoryForTests(base::FilePath* directory); |
[email protected] | 9c265d0 | 2011-12-29 22:13:43 | [diff] [blame] | 198 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 199 | private: |
[email protected] | 902a8a01 | 2013-05-04 05:04:16 | [diff] [blame] | 200 | FRIEND_TEST_ALL_PREFIXES(WebstoreInstallerTest, PlatformParams); |
[email protected] | ed0757ec | 2012-08-02 17:23:26 | [diff] [blame] | 201 | friend struct content::BrowserThread::DeleteOnThread< |
| 202 | content::BrowserThread::UI>; |
| 203 | friend class base::DeleteHelper<WebstoreInstaller>; |
[email protected] | 5f2a475 | 2012-04-27 22:18:58 | [diff] [blame] | 204 | virtual ~WebstoreInstaller(); |
| 205 | |
[email protected] | 902a8a01 | 2013-05-04 05:04:16 | [diff] [blame] | 206 | // Helper to get install URL. |
| 207 | static GURL GetWebstoreInstallURL(const std::string& extension_id, |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 208 | InstallSource source); |
[email protected] | 902a8a01 | 2013-05-04 05:04:16 | [diff] [blame] | 209 | |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 210 | // DownloadManager::DownloadUrl callback. |
[email protected] | 7f391888 | 2014-01-14 06:14:56 | [diff] [blame] | 211 | void OnDownloadStarted(content::DownloadItem* item, |
| 212 | content::DownloadInterruptReason interrupt_reason); |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 213 | |
| 214 | // DownloadItem::Observer implementation: |
| 215 | virtual void OnDownloadUpdated(content::DownloadItem* download) OVERRIDE; |
[email protected] | 7e834f0 | 2012-08-09 20:38:56 | [diff] [blame] | 216 | virtual void OnDownloadDestroyed(content::DownloadItem* download) OVERRIDE; |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 217 | |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 218 | // Downloads next pending module in |pending_modules_|. |
| 219 | void DownloadNextPendingModule(); |
| 220 | |
| 221 | // Downloads and installs a single Crx with the given |extension_id|. |
| 222 | // This function is used for both the extension Crx and dependences. |
| 223 | void DownloadCrx(const std::string& extension_id, InstallSource source); |
| 224 | |
[email protected] | f66a50a | 2011-11-02 23:53:46 | [diff] [blame] | 225 | // Starts downloading the extension to |file_path|. |
[email protected] | a3ef483 | 2013-02-02 05:12:33 | [diff] [blame] | 226 | void StartDownload(const base::FilePath& file_path); |
[email protected] | f66a50a | 2011-11-02 23:53:46 | [diff] [blame] | 227 | |
[email protected] | c80fe5f | 2014-03-26 04:36:30 | [diff] [blame^] | 228 | // Updates the InstallTracker with the latest download progress. |
| 229 | void UpdateDownloadProgress(); |
| 230 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 231 | // Reports an install |error| to the delegate for the given extension if this |
| 232 | // managed its installation. This also removes the associated PendingInstall. |
[email protected] | bcd1eaf7 | 2012-10-03 05:42:29 | [diff] [blame] | 233 | void ReportFailure(const std::string& error, FailureReason reason); |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 234 | |
| 235 | // Reports a successful install to the delegate for the given extension if |
| 236 | // this managed its installation. This also removes the associated |
| 237 | // PendingInstall. |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 238 | void ReportSuccess(); |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 239 | |
[email protected] | 0a2a724 | 2013-11-20 20:49:24 | [diff] [blame] | 240 | // Records stats regarding an interrupted webstore download item. |
| 241 | void RecordInterrupt(const content::DownloadItem* download) const; |
| 242 | |
[email protected] | 6c2381d | 2011-10-19 02:52:53 | [diff] [blame] | 243 | content::NotificationRegistrar registrar_; |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 244 | Profile* profile_; |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 245 | Delegate* delegate_; |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 246 | std::string id_; |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 247 | InstallSource install_source_; |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 248 | // The DownloadItem is owned by the DownloadManager and is valid from when |
[email protected] | 81b80bc | 2012-08-31 18:27:44 | [diff] [blame] | 249 | // OnDownloadStarted is called (with no error) until OnDownloadDestroyed(). |
[email protected] | 8adbe311 | 2012-03-27 05:42:22 | [diff] [blame] | 250 | content::DownloadItem* download_item_; |
[email protected] | c80fe5f | 2014-03-26 04:36:30 | [diff] [blame^] | 251 | // Used to periodically update the extension's download status. This will |
| 252 | // trigger at least every second, though sometimes more frequently (depending |
| 253 | // on number of modules, etc). |
| 254 | base::OneShotTimer<WebstoreInstaller> download_progress_timer_; |
[email protected] | 21a5ad6 | 2012-04-03 04:48:45 | [diff] [blame] | 255 | scoped_ptr<Approval> approval_; |
[email protected] | 98e4e52 | 2011-10-25 13:00:16 | [diff] [blame] | 256 | GURL download_url_; |
[email protected] | 669b237 | 2013-10-17 15:04:58 | [diff] [blame] | 257 | |
| 258 | // Pending modules. |
| 259 | std::list<SharedModuleInfo::ImportInfo> pending_modules_; |
| 260 | // Total extension modules we need download and install (the main module and |
| 261 | // depedences). |
| 262 | int total_modules_; |
| 263 | bool download_started_; |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 264 | }; |
| 265 | |
[email protected] | 3d61a7f | 2012-07-12 19:11:25 | [diff] [blame] | 266 | } // namespace extensions |
| 267 | |
[email protected] | 655b2b1a | 2011-10-13 17:13:06 | [diff] [blame] | 268 | #endif // CHROME_BROWSER_EXTENSIONS_WEBSTORE_INSTALLER_H_ |