Revert of Add button to add new FSP services to Files app. (patchset #8 id:140001...
[chromium-blink-merge.git] / chrome / browser / download / download_request_limiter.cc
blob9d2117d6c15f243758e21179371a53389004ce26
1 // Copyright (c) 2012 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/download/download_request_limiter.h"
7 #include "base/bind.h"
8 #include "base/stl_util.h"
9 #include "chrome/browser/content_settings/tab_specific_content_settings.h"
10 #include "chrome/browser/download/download_permission_request.h"
11 #include "chrome/browser/download/download_request_infobar_delegate.h"
12 #include "chrome/browser/infobars/infobar_service.h"
13 #include "chrome/browser/profiles/profile.h"
14 #include "chrome/browser/tab_contents/tab_util.h"
15 #include "chrome/browser/ui/website_settings/permission_bubble_manager.h"
16 #include "components/content_settings/core/browser/host_content_settings_map.h"
17 #include "content/public/browser/browser_context.h"
18 #include "content/public/browser/browser_thread.h"
19 #include "content/public/browser/navigation_controller.h"
20 #include "content/public/browser/navigation_entry.h"
21 #include "content/public/browser/notification_source.h"
22 #include "content/public/browser/notification_types.h"
23 #include "content/public/browser/render_process_host.h"
24 #include "content/public/browser/resource_dispatcher_host.h"
25 #include "content/public/browser/web_contents.h"
26 #include "content/public/browser/web_contents_delegate.h"
27 #include "url/gurl.h"
29 using content::BrowserThread;
30 using content::NavigationController;
31 using content::NavigationEntry;
33 // TabDownloadState ------------------------------------------------------------
35 DownloadRequestLimiter::TabDownloadState::TabDownloadState(
36 DownloadRequestLimiter* host,
37 content::WebContents* contents,
38 content::WebContents* originating_web_contents)
39 : content::WebContentsObserver(contents),
40 web_contents_(contents),
41 host_(host),
42 status_(DownloadRequestLimiter::ALLOW_ONE_DOWNLOAD),
43 download_count_(0),
44 factory_(this) {
45 content::Source<NavigationController> notification_source(
46 &contents->GetController());
47 registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_PENDING,
48 notification_source);
49 NavigationEntry* active_entry = originating_web_contents ?
50 originating_web_contents->GetController().GetActiveEntry() :
51 contents->GetController().GetActiveEntry();
52 if (active_entry)
53 initial_page_host_ = active_entry->GetURL().host();
56 DownloadRequestLimiter::TabDownloadState::~TabDownloadState() {
57 // We should only be destroyed after the callbacks have been notified.
58 DCHECK(callbacks_.empty());
60 // And we should have invalidated the back pointer.
61 DCHECK(!factory_.HasWeakPtrs());
64 void DownloadRequestLimiter::TabDownloadState::DidNavigateMainFrame(
65 const content::LoadCommittedDetails& details,
66 const content::FrameNavigateParams& params) {
67 switch (status_) {
68 case ALLOW_ONE_DOWNLOAD:
69 case PROMPT_BEFORE_DOWNLOAD:
70 // When the user reloads the page without responding to the infobar, they
71 // are expecting DownloadRequestLimiter to behave as if they had just
72 // initially navigated to this page. See http://crbug.com/171372
73 NotifyCallbacks(false);
74 host_->Remove(this, web_contents());
75 // WARNING: We've been deleted.
76 break;
77 case DOWNLOADS_NOT_ALLOWED:
78 case ALLOW_ALL_DOWNLOADS:
79 // Don't drop this information. The user has explicitly said that they
80 // do/don't want downloads from this host. If they accidentally Accepted
81 // or Canceled, tough luck, they don't get another chance. They can copy
82 // the URL into a new tab, which will make a new DownloadRequestLimiter.
83 // See also the initial_page_host_ logic in Observe() for
84 // NOTIFICATION_NAV_ENTRY_PENDING.
85 break;
86 default:
87 NOTREACHED();
91 void DownloadRequestLimiter::TabDownloadState::DidGetUserGesture() {
92 if (is_showing_prompt()) {
93 // Don't change the state if the user clicks on the page somewhere.
94 return;
97 bool promptable = (InfoBarService::FromWebContents(web_contents()) != NULL);
98 if (PermissionBubbleManager::Enabled()) {
99 promptable =
100 (PermissionBubbleManager::FromWebContents(web_contents()) != NULL);
103 // See PromptUserForDownload(): if there's no InfoBarService, then
104 // DOWNLOADS_NOT_ALLOWED is functionally equivalent to PROMPT_BEFORE_DOWNLOAD.
105 if ((status_ != DownloadRequestLimiter::ALLOW_ALL_DOWNLOADS) &&
106 (!promptable ||
107 (status_ != DownloadRequestLimiter::DOWNLOADS_NOT_ALLOWED))) {
108 // Revert to default status.
109 host_->Remove(this, web_contents());
110 // WARNING: We've been deleted.
114 void DownloadRequestLimiter::TabDownloadState::WebContentsDestroyed() {
115 // Tab closed, no need to handle closing the dialog as it's owned by the
116 // WebContents.
118 NotifyCallbacks(false);
119 host_->Remove(this, web_contents());
120 // WARNING: We've been deleted.
123 void DownloadRequestLimiter::TabDownloadState::PromptUserForDownload(
124 const DownloadRequestLimiter::Callback& callback) {
125 callbacks_.push_back(callback);
126 DCHECK(web_contents_);
127 if (is_showing_prompt())
128 return;
130 if (PermissionBubbleManager::Enabled()) {
131 PermissionBubbleManager* bubble_manager =
132 PermissionBubbleManager::FromWebContents(web_contents_);
133 if (bubble_manager) {
134 bubble_manager->AddRequest(new DownloadPermissionRequest(
135 factory_.GetWeakPtr()));
136 } else {
137 Cancel();
139 return;
142 DownloadRequestInfoBarDelegate::Create(
143 InfoBarService::FromWebContents(web_contents_), factory_.GetWeakPtr());
146 void DownloadRequestLimiter::TabDownloadState::SetContentSetting(
147 ContentSetting setting) {
148 if (!web_contents_)
149 return;
150 HostContentSettingsMap* settings =
151 DownloadRequestLimiter::GetContentSettings(web_contents_);
152 ContentSettingsPattern pattern(
153 ContentSettingsPattern::FromURL(web_contents_->GetURL()));
154 if (!settings || !pattern.IsValid())
155 return;
156 settings->SetContentSetting(
157 pattern,
158 ContentSettingsPattern::Wildcard(),
159 CONTENT_SETTINGS_TYPE_AUTOMATIC_DOWNLOADS,
160 std::string(),
161 setting);
164 void DownloadRequestLimiter::TabDownloadState::Cancel() {
165 SetContentSetting(CONTENT_SETTING_BLOCK);
166 NotifyCallbacks(false);
169 void DownloadRequestLimiter::TabDownloadState::CancelOnce() {
170 NotifyCallbacks(false);
173 void DownloadRequestLimiter::TabDownloadState::Accept() {
174 SetContentSetting(CONTENT_SETTING_ALLOW);
175 NotifyCallbacks(true);
178 DownloadRequestLimiter::TabDownloadState::TabDownloadState()
179 : web_contents_(NULL),
180 host_(NULL),
181 status_(DownloadRequestLimiter::ALLOW_ONE_DOWNLOAD),
182 download_count_(0),
183 factory_(this) {
186 bool DownloadRequestLimiter::TabDownloadState::is_showing_prompt() const {
187 return factory_.HasWeakPtrs();
190 void DownloadRequestLimiter::TabDownloadState::Observe(
191 int type,
192 const content::NotificationSource& source,
193 const content::NotificationDetails& details) {
194 DCHECK_EQ(content::NOTIFICATION_NAV_ENTRY_PENDING, type);
195 content::NavigationController* controller = &web_contents()->GetController();
196 DCHECK_EQ(controller, content::Source<NavigationController>(source).ptr());
198 // NOTE: Resetting state on a pending navigate isn't ideal. In particular it
199 // is possible that queued up downloads for the page before the pending
200 // navigation will be delivered to us after we process this request. If this
201 // happens we may let a download through that we shouldn't have. But this is
202 // rather rare, and it is difficult to get 100% right, so we don't deal with
203 // it.
204 NavigationEntry* entry = controller->GetPendingEntry();
205 if (!entry)
206 return;
208 // Redirects don't count.
209 if (ui::PageTransitionIsRedirect(entry->GetTransitionType()))
210 return;
212 if (status_ == DownloadRequestLimiter::ALLOW_ALL_DOWNLOADS ||
213 status_ == DownloadRequestLimiter::DOWNLOADS_NOT_ALLOWED) {
214 // User has either allowed all downloads or canceled all downloads. Only
215 // reset the download state if the user is navigating to a different host
216 // (or host is empty).
217 if (!initial_page_host_.empty() && !entry->GetURL().host().empty() &&
218 entry->GetURL().host() == initial_page_host_)
219 return;
222 NotifyCallbacks(false);
223 host_->Remove(this, web_contents());
226 void DownloadRequestLimiter::TabDownloadState::NotifyCallbacks(bool allow) {
227 set_download_status(allow ?
228 DownloadRequestLimiter::ALLOW_ALL_DOWNLOADS :
229 DownloadRequestLimiter::DOWNLOADS_NOT_ALLOWED);
230 std::vector<DownloadRequestLimiter::Callback> callbacks;
231 bool change_status = false;
233 // Selectively send first few notifications only if number of downloads exceed
234 // kMaxDownloadsAtOnce. In that case, we also retain the infobar instance and
235 // don't close it. If allow is false, we send all the notifications to cancel
236 // all remaining downloads and close the infobar.
237 if (!allow || (callbacks_.size() < kMaxDownloadsAtOnce)) {
238 // Null the generated weak pointer so we don't get notified again.
239 factory_.InvalidateWeakPtrs();
240 callbacks.swap(callbacks_);
241 } else {
242 std::vector<DownloadRequestLimiter::Callback>::iterator start, end;
243 start = callbacks_.begin();
244 end = callbacks_.begin() + kMaxDownloadsAtOnce;
245 callbacks.assign(start, end);
246 callbacks_.erase(start, end);
247 change_status = true;
250 for (size_t i = 0; i < callbacks.size(); ++i)
251 host_->ScheduleNotification(callbacks[i], allow);
253 if (change_status)
254 set_download_status(DownloadRequestLimiter::PROMPT_BEFORE_DOWNLOAD);
257 // DownloadRequestLimiter ------------------------------------------------------
259 HostContentSettingsMap* DownloadRequestLimiter::content_settings_ = NULL;
261 void DownloadRequestLimiter::SetContentSettingsForTesting(
262 HostContentSettingsMap* content_settings) {
263 content_settings_ = content_settings;
266 DownloadRequestLimiter::DownloadRequestLimiter()
267 : factory_(this) {
270 DownloadRequestLimiter::~DownloadRequestLimiter() {
271 // All the tabs should have closed before us, which sends notification and
272 // removes from state_map_. As such, there should be no pending callbacks.
273 DCHECK(state_map_.empty());
276 DownloadRequestLimiter::DownloadStatus
277 DownloadRequestLimiter::GetDownloadStatus(content::WebContents* web_contents) {
278 TabDownloadState* state = GetDownloadState(web_contents, NULL, false);
279 return state ? state->download_status() : ALLOW_ONE_DOWNLOAD;
282 void DownloadRequestLimiter::CanDownloadOnIOThread(
283 int render_process_host_id,
284 int render_view_id,
285 const GURL& url,
286 const std::string& request_method,
287 const Callback& callback) {
288 // This is invoked on the IO thread. Schedule the task to run on the UI
289 // thread so that we can query UI state.
290 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
291 BrowserThread::PostTask(
292 BrowserThread::UI, FROM_HERE,
293 base::Bind(&DownloadRequestLimiter::CanDownload, this,
294 render_process_host_id, render_view_id, url,
295 request_method, callback));
298 DownloadRequestLimiter::TabDownloadState*
299 DownloadRequestLimiter::GetDownloadState(
300 content::WebContents* web_contents,
301 content::WebContents* originating_web_contents,
302 bool create) {
303 DCHECK(web_contents);
304 StateMap::iterator i = state_map_.find(web_contents);
305 if (i != state_map_.end())
306 return i->second;
308 if (!create)
309 return NULL;
311 TabDownloadState* state =
312 new TabDownloadState(this, web_contents, originating_web_contents);
313 state_map_[web_contents] = state;
314 return state;
317 void DownloadRequestLimiter::CanDownload(int render_process_host_id,
318 int render_view_id,
319 const GURL& url,
320 const std::string& request_method,
321 const Callback& callback) {
322 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
324 content::WebContents* originating_contents =
325 tab_util::GetWebContentsByID(render_process_host_id, render_view_id);
326 if (!originating_contents) {
327 // The WebContents was closed, don't allow the download.
328 ScheduleNotification(callback, false);
329 return;
332 if (!originating_contents->GetDelegate()) {
333 ScheduleNotification(callback, false);
334 return;
337 // Note that because |originating_contents| might go away before
338 // OnCanDownloadDecided is invoked, we look it up by |render_process_host_id|
339 // and |render_view_id|.
340 base::Callback<void(bool)> can_download_callback = base::Bind(
341 &DownloadRequestLimiter::OnCanDownloadDecided,
342 factory_.GetWeakPtr(),
343 render_process_host_id,
344 render_view_id,
345 request_method,
346 callback);
348 originating_contents->GetDelegate()->CanDownload(
349 originating_contents->GetRenderViewHost(),
350 url,
351 request_method,
352 can_download_callback);
355 void DownloadRequestLimiter::OnCanDownloadDecided(
356 int render_process_host_id,
357 int render_view_id,
358 const std::string& request_method,
359 const Callback& orig_callback, bool allow) {
360 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
361 content::WebContents* originating_contents =
362 tab_util::GetWebContentsByID(render_process_host_id, render_view_id);
363 if (!originating_contents || !allow) {
364 ScheduleNotification(orig_callback, false);
365 return;
368 CanDownloadImpl(originating_contents,
369 request_method,
370 orig_callback);
373 HostContentSettingsMap* DownloadRequestLimiter::GetContentSettings(
374 content::WebContents* contents) {
375 return content_settings_ ? content_settings_ : Profile::FromBrowserContext(
376 contents->GetBrowserContext())->GetHostContentSettingsMap();
379 void DownloadRequestLimiter::CanDownloadImpl(
380 content::WebContents* originating_contents,
381 const std::string& request_method,
382 const Callback& callback) {
383 DCHECK(originating_contents);
385 TabDownloadState* state = GetDownloadState(
386 originating_contents, originating_contents, true);
387 switch (state->download_status()) {
388 case ALLOW_ALL_DOWNLOADS:
389 if (state->download_count() && !(state->download_count() %
390 DownloadRequestLimiter::kMaxDownloadsAtOnce))
391 state->set_download_status(PROMPT_BEFORE_DOWNLOAD);
392 ScheduleNotification(callback, true);
393 state->increment_download_count();
394 break;
396 case ALLOW_ONE_DOWNLOAD:
397 state->set_download_status(PROMPT_BEFORE_DOWNLOAD);
398 ScheduleNotification(callback, true);
399 state->increment_download_count();
400 break;
402 case DOWNLOADS_NOT_ALLOWED:
403 ScheduleNotification(callback, false);
404 break;
406 case PROMPT_BEFORE_DOWNLOAD: {
407 HostContentSettingsMap* content_settings = GetContentSettings(
408 originating_contents);
409 ContentSetting setting = CONTENT_SETTING_ASK;
410 if (content_settings)
411 setting = content_settings->GetContentSetting(
412 originating_contents->GetURL(),
413 originating_contents->GetURL(),
414 CONTENT_SETTINGS_TYPE_AUTOMATIC_DOWNLOADS,
415 std::string());
416 switch (setting) {
417 case CONTENT_SETTING_ALLOW: {
418 TabSpecificContentSettings* settings =
419 TabSpecificContentSettings::FromWebContents(
420 originating_contents);
421 if (settings)
422 settings->SetDownloadsBlocked(false);
423 ScheduleNotification(callback, true);
424 state->increment_download_count();
425 return;
427 case CONTENT_SETTING_BLOCK: {
428 TabSpecificContentSettings* settings =
429 TabSpecificContentSettings::FromWebContents(
430 originating_contents);
431 if (settings)
432 settings->SetDownloadsBlocked(true);
433 ScheduleNotification(callback, false);
434 return;
436 case CONTENT_SETTING_DEFAULT:
437 case CONTENT_SETTING_ASK:
438 case CONTENT_SETTING_SESSION_ONLY:
439 state->PromptUserForDownload(callback);
440 state->increment_download_count();
441 break;
442 case CONTENT_SETTING_NUM_SETTINGS:
443 default:
444 NOTREACHED();
445 return;
447 break;
450 default:
451 NOTREACHED();
455 void DownloadRequestLimiter::ScheduleNotification(const Callback& callback,
456 bool allow) {
457 BrowserThread::PostTask(
458 BrowserThread::IO, FROM_HERE, base::Bind(callback, allow));
461 void DownloadRequestLimiter::Remove(TabDownloadState* state,
462 content::WebContents* contents) {
463 DCHECK(ContainsKey(state_map_, contents));
464 state_map_.erase(contents);
465 delete state;