Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(382)

Side by Side Diff: chrome/browser/extensions/webstore_inline_installer.cc

Issue 7741037: Add WebstoreInlineInstaller (downloads store data, shows the install UI, and starts the install). (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Created 9 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2011 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/webstore_inline_installer.h"
6
7 #include <vector>
8
9 #include "base/string_util.h"
10 #include "base/values.h"
11 #include "chrome/browser/browser_process.h"
12 #include "chrome/browser/extensions/crx_installer.h"
13 #include "chrome/browser/extensions/extension_install_dialog.h"
14 #include "chrome/browser/profiles/profile.h"
15 #include "chrome/common/chrome_utility_messages.h"
16 #include "chrome/common/extensions/extension.h"
17 #include "chrome/common/extensions/extension_constants.h"
18 #include "content/browser/tab_contents/tab_contents.h"
19 #include "content/browser/utility_process_host.h"
20 #include "googleurl/src/gurl.h"
21 #include "net/base/escape.h"
22 #include "net/url_request/url_request_status.h"
23
24 const char kManifestKey[] = "manifest";
25 const char kIconUrlKey[] = "icon_url";
26 const char kLocalizedNameKey[] = "localized_name";
27
28 const char kWebstoreRequestError[] = "Could not fetch data from webstore";
29 const char kInvalidWebstoreResponseError[] = "Invalid webstore reponse";
30 const char kInvalidManifestError[] = "Invalid manifest";
31 const char kUserCancelledError[] = "User cancelled install";
32
33 // A flag used for WebstoreInlineInstaller::SetAutoConfirmForTests.
34 enum AutoConfirmForTest {
35 DO_NOT_SKIP = 0,
36 PROCEED,
37 ABORT
38 };
39 AutoConfirmForTest auto_confirm_for_tests = DO_NOT_SKIP;
40
41 class SafeWebstoreResponseParser : public UtilityProcessHost::Client {
42 public:
43 SafeWebstoreResponseParser(WebstoreInlineInstaller *client,
44 const std::string& webstore_data)
45 : client_(client)
46 , webstore_data_(webstore_data)
asargent_no_longer_on_chrome 2011/08/26 17:50:40 commas in weird place here and line below
Mihai Parparita -not on Chrome 2011/08/29 19:00:56 Fixed.
47 , utility_host_(NULL) {}
48
49 void Start() {
50 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
51 BrowserThread::PostTask(
52 BrowserThread::IO,
53 FROM_HERE,
54 NewRunnableMethod(this,
55 &SafeWebstoreResponseParser::StartWorkOnIOThread));
56 }
57
58 void StartWorkOnIOThread() {
59 CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
60 utility_host_ = new UtilityProcessHost(this, BrowserThread::IO);
61 utility_host_->Send(new ChromeUtilityMsg_ParseJSON(webstore_data_));
62 }
63
64 // Implementing pieces of the UtilityProcessHost::Client interface.
65 virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE {
66 bool handled = true;
67 IPC_BEGIN_MESSAGE_MAP(SafeWebstoreResponseParser, message)
68 IPC_MESSAGE_HANDLER(ChromeUtilityHostMsg_ParseJSON_Succeeded,
69 OnJSONParseSucceeded)
70 IPC_MESSAGE_HANDLER(ChromeUtilityHostMsg_ParseJSON_Failed,
71 OnJSONParseFailed)
72 IPC_MESSAGE_UNHANDLED(handled = false)
73 IPC_END_MESSAGE_MAP()
74 return handled;
75 }
76
77 void OnJSONParseSucceeded(const ListValue& wrapper) {
78 CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
79 Value* value = NULL;
80 CHECK(wrapper.Get(0, &value));
81 if (value->IsType(Value::TYPE_DICTIONARY)) {
82 parsed_webstore_data_.reset(
83 static_cast<DictionaryValue*>(value)->DeepCopy());
84 } else {
85 error_ = kInvalidWebstoreResponseError;
86 }
87
88 ReportResults();
89 }
90
91 virtual void OnJSONParseFailed(const std::string& error_message) {
92 CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
93 error_ = error_message;
94 ReportResults();
95 }
96
97 void ReportResults() {
98 CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
99
100 // The utility_host_ will take care of deleting itself after this call.
101 utility_host_ = NULL;
102
103 BrowserThread::PostTask(
104 BrowserThread::UI,
105 FROM_HERE,
106 NewRunnableMethod(this,
107 &SafeWebstoreResponseParser::ReportResultOnUIThread));
108 }
109
110 void ReportResultOnUIThread() {
111 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
112 if (error_.empty() && parsed_webstore_data_.get()) {
113 client_->OnWebstoreResponseParseSuccess(parsed_webstore_data_.release());
114 } else {
115 client_->OnWebstoreResponseParseFailure(error_);
116 }
117 }
118
119 private:
120 virtual ~SafeWebstoreResponseParser() {}
121
122 WebstoreInlineInstaller* client_;
123
124 std::string webstore_data_;
125
126 UtilityProcessHost* utility_host_;
127
128 std::string error_;
129 scoped_ptr<DictionaryValue> parsed_webstore_data_;
130 };
131
132 WebstoreInlineInstaller::WebstoreInlineInstaller(TabContents* tab_contents,
133 std::string webstore_item_id,
134 Delegate* delegate)
135 : tab_contents_(tab_contents)
136 , id_(webstore_item_id)
asargent_no_longer_on_chrome 2011/08/26 17:50:40 commas in weird place here too
Mihai Parparita -not on Chrome 2011/08/29 19:00:56 Fixed.
137 , delegate_(delegate) {}
138
139 WebstoreInlineInstaller::~WebstoreInlineInstaller() {
140 }
141
142 void WebstoreInlineInstaller::BeginInstall() {
143 GURL webstore_data_url(extension_urls::GetWebstoreItemJsonDataURL(id_));
asargent_no_longer_on_chrome 2011/08/26 17:50:40 It might be good to add a check of Extension::IdIs
Mihai Parparita -not on Chrome 2011/08/29 19:00:56 Done.
144
145 AddRef(); // Balanced in CompleteInstall.
146
147 webstore_data_url_fetcher_.reset(
148 new URLFetcher(webstore_data_url, URLFetcher::GET, this));
149 Profile* profile = Profile::FromBrowserContext(
150 tab_contents_->browser_context());
151 webstore_data_url_fetcher_->set_request_context(
152 profile->GetRequestContext());
153 webstore_data_url_fetcher_->Start();
154 }
155
156 void WebstoreInlineInstaller::SetAutoConfirmForTests(
asargent_no_longer_on_chrome 2011/08/26 17:50:40 add comment with "// static" here
Mihai Parparita -not on Chrome 2011/08/29 19:00:56 Now that I've refactored the auto-confirm code to
157 bool should_proceed) {
158 auto_confirm_for_tests = should_proceed ? PROCEED : ABORT;
159 }
160
161 void WebstoreInlineInstaller::OnURLFetchComplete(const URLFetcher* source) {
162 CHECK_EQ(webstore_data_url_fetcher_.get(), source);
163
164 if (!webstore_data_url_fetcher_->status().is_success() ||
165 webstore_data_url_fetcher_->response_code() != 200) {
166 CompleteInstall(kWebstoreRequestError);
167 return;
168 }
169
170 std::string webstore_json_data;
171 webstore_data_url_fetcher_->GetResponseAsString(&webstore_json_data);
172 webstore_data_url_fetcher_.reset();
173
174 scoped_refptr<SafeWebstoreResponseParser> parser =
175 new SafeWebstoreResponseParser(this, webstore_json_data);
176 // The parser will call us back via OnWebstoreResponseParseSucces or
177 // OnWebstoreResponseParseFailure.
178 parser->Start();
179 }
180
181 void WebstoreInlineInstaller::OnWebstoreResponseParseSuccess(
182 DictionaryValue* webstore_data) {
183 webstore_data_.reset(webstore_data);
184
185 std::string manifest;
186 if (!webstore_data->GetString(kManifestKey, &manifest)) {
187 CompleteInstall(kInvalidWebstoreResponseError);
188 return;
189 }
190
191 // Localized name is optional.
192 if (webstore_data->HasKey(kLocalizedNameKey) &&
193 !webstore_data->GetString(kLocalizedNameKey, &localized_name_)) {
194 CompleteInstall(kInvalidWebstoreResponseError);
195 return;
196 }
197
198 // Icon URL is optional.
199 GURL icon_url;
200 if (webstore_data->HasKey(kIconUrlKey)) {
201 std::string icon_url_string;
202 if (!webstore_data->GetString(kIconUrlKey, &icon_url_string)) {
203 CompleteInstall(kInvalidWebstoreResponseError);
204 return;
205 }
206 icon_url = GURL(extension_urls::GetWebstoreLaunchURL()).Resolve(
207 icon_url_string);
208 if (!icon_url.is_valid()) {
209 CompleteInstall(kInvalidWebstoreResponseError);
210 return;
211 }
212 }
213
214 scoped_refptr<WebstoreInstallHelper> helper = new WebstoreInstallHelper(
215 this,
216 manifest,
217 "", // We don't have any icon data.
218 icon_url,
219 Profile::FromBrowserContext(tab_contents_->browser_context())->
220 GetRequestContext());
221 // The helper will call us back via OnWebstoreParseSucces or
222 // OnWebstoreParseFailure.
223 helper->Start();
224 }
225
226 void WebstoreInlineInstaller::OnWebstoreResponseParseFailure(
227 const std::string& error) {
228 CompleteInstall(error);
229 }
230
231 void WebstoreInlineInstaller::OnWebstoreParseSuccess(
232 const SkBitmap& icon,
233 base::DictionaryValue* manifest) {
234 manifest_.reset(manifest);
235 icon_ = icon;
236
237 scoped_ptr<DictionaryValue> localized_manifest;
238 if (!localized_name_.empty()) {
239 localized_manifest.reset(manifest->DeepCopy());
240 localized_manifest->SetString(extension_manifest_keys::kName,
241 localized_name_);
242 }
243
244 std::string init_errors;
245 scoped_refptr<Extension> dummy_extension = Extension::CreateWithId(
246 FilePath(),
247 Extension::INTERNAL,
248 localized_manifest.get() ? *localized_manifest.get() : *manifest,
249 Extension::NO_FLAGS,
250 id_,
251 &init_errors);
252 if (!dummy_extension.get()) {
253 CompleteInstall(kInvalidManifestError);
254 return;
255 }
256
257 if (icon_.empty())
258 icon_ = Extension::GetDefaultIcon(dummy_extension->is_app());
259
260 // In tests, we may have setup to proceed or abort without putting up the real
261 // confirmation dialog.
262 if (auto_confirm_for_tests != DO_NOT_SKIP) {
263 if (auto_confirm_for_tests == PROCEED)
264 this->InstallUIProceed();
265 else
266 this->InstallUIAbort(true);
267 return;
268 }
269
270 Profile* profile = Profile::FromBrowserContext(
271 tab_contents_->browser_context());
272 ShowExtensionInstallDialog(profile,
273 this,
274 dummy_extension.get(),
275 &icon_,
276 dummy_extension->GetPermissionMessageStrings(),
277 ExtensionInstallUI::INSTALL_PROMPT);
asargent_no_longer_on_chrome 2011/08/26 17:50:40 Do you think it's worth it to extract this code to
Mihai Parparita -not on Chrome 2011/08/29 19:00:56 Added ShowExtensionInstallDialogForManifest (the s
278 }
279
280 void WebstoreInlineInstaller::OnWebstoreParseFailure(
281 InstallHelperResultCode result_code,
282 const std::string& error_message) {
283 CompleteInstall(error_message);
284 }
285
286 void WebstoreInlineInstaller::InstallUIProceed() {
287 CrxInstaller::WhitelistEntry* entry = new CrxInstaller::WhitelistEntry;
288
289 entry->parsed_manifest.reset(manifest_.get()->DeepCopy());
290 entry->localized_name = localized_name_;
291 entry->use_app_installed_bubble = true;
292 CrxInstaller::SetWhitelistEntry(id_, entry);
293
294 GURL install_url(extension_urls::GetWebstoreInstallUrl(
295 id_, g_browser_process->GetApplicationLocale()));
296
297 NavigationController& controller = tab_contents_->controller();
298 // TODO(mihaip): we pretend like the referrer is the gallery in order to pass
299 // the checks in ExtensionService::IsDownloadFromGallery. We should instead
300 // pass the real referrer, track that this is an inline install in the
301 // whitelist entry and look that up when checking that this is a valid
302 // download.
303 GURL referrer(extension_urls::GetWebstoreItemDetailURLPrefix() + id_);
304 controller.LoadURL(install_url, referrer, PageTransition::LINK);
305
306 // TODO(mihaip): the success message should happen later, when the extension
307 // is actually downloaded and installed.
asargent_no_longer_on_chrome 2011/08/26 17:50:40 Yeah, for the case where the download succeeds you
308 CompleteInstall("");
309 }
310
311 void WebstoreInlineInstaller::InstallUIAbort(bool user_initiated) {
312 CompleteInstall(kUserCancelledError);
313 }
314
315 void WebstoreInlineInstaller::CompleteInstall(const std::string& error) {
316 if (error.empty()) {
317 delegate_->OnInlineInstallSuccess();
318 } else {
319 delegate_->OnInlineInstallFailure(error);
320 }
321 Release(); // Matches the AddRef in BeginInstall.
322 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698