OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/renderer_host/chrome_resource_dispatcher_host_delegate.
h" | 5 #include "chrome/browser/renderer_host/chrome_resource_dispatcher_host_delegate.
h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 | 8 |
9 #include "base/base64.h" | 9 #include "base/base64.h" |
10 #include "base/logging.h" | 10 #include "base/logging.h" |
11 #include "chrome/browser/browser_process.h" | 11 #include "chrome/browser/browser_process.h" |
12 #include "chrome/browser/content_settings/host_content_settings_map.h" | 12 #include "chrome/browser/content_settings/host_content_settings_map.h" |
13 #include "chrome/browser/download/download_request_limiter.h" | 13 #include "chrome/browser/download/download_request_limiter.h" |
14 #include "chrome/browser/download/download_resource_throttle.h" | 14 #include "chrome/browser/download/download_resource_throttle.h" |
15 #include "chrome/browser/download/download_util.h" | 15 #include "chrome/browser/download/download_util.h" |
16 #include "chrome/browser/extensions/api/streams_private/streams_resource_throttl
e.h" | 16 #include "chrome/browser/extensions/api/streams_private/streams_private_api.h" |
| 17 #include "chrome/browser/extensions/extension_info_map.h" |
17 #include "chrome/browser/extensions/user_script_listener.h" | 18 #include "chrome/browser/extensions/user_script_listener.h" |
18 #include "chrome/browser/external_protocol/external_protocol_handler.h" | 19 #include "chrome/browser/external_protocol/external_protocol_handler.h" |
19 #include "chrome/browser/google/google_util.h" | 20 #include "chrome/browser/google/google_util.h" |
20 #include "chrome/browser/metrics/variations/variations_http_header_provider.h" | 21 #include "chrome/browser/metrics/variations/variations_http_header_provider.h" |
21 #include "chrome/browser/net/load_timing_observer.h" | 22 #include "chrome/browser/net/load_timing_observer.h" |
22 #include "chrome/browser/net/resource_prefetch_predictor_observer.h" | 23 #include "chrome/browser/net/resource_prefetch_predictor_observer.h" |
23 #include "chrome/browser/prerender/prerender_manager.h" | 24 #include "chrome/browser/prerender/prerender_manager.h" |
24 #include "chrome/browser/prerender/prerender_tracker.h" | 25 #include "chrome/browser/prerender/prerender_tracker.h" |
25 #include "chrome/browser/prerender/prerender_util.h" | 26 #include "chrome/browser/prerender/prerender_util.h" |
| 27 #include "chrome/browser/profiles/profile.h" |
26 #include "chrome/browser/profiles/profile_io_data.h" | 28 #include "chrome/browser/profiles/profile_io_data.h" |
27 #include "chrome/browser/renderer_host/chrome_url_request_user_data.h" | 29 #include "chrome/browser/renderer_host/chrome_url_request_user_data.h" |
28 #include "chrome/browser/renderer_host/safe_browsing_resource_throttle_factory.h
" | 30 #include "chrome/browser/renderer_host/safe_browsing_resource_throttle_factory.h
" |
29 #include "chrome/browser/safe_browsing/safe_browsing_service.h" | 31 #include "chrome/browser/safe_browsing/safe_browsing_service.h" |
30 #include "chrome/browser/ui/auto_login_prompter.h" | 32 #include "chrome/browser/ui/auto_login_prompter.h" |
31 #include "chrome/browser/ui/login/login_prompt.h" | 33 #include "chrome/browser/ui/login/login_prompt.h" |
32 #include "chrome/browser/ui/sync/one_click_signin_helper.h" | 34 #include "chrome/browser/ui/sync/one_click_signin_helper.h" |
33 #include "chrome/common/chrome_notification_types.h" | 35 #include "chrome/common/chrome_notification_types.h" |
34 #include "chrome/common/extensions/extension_constants.h" | 36 #include "chrome/common/extensions/mime_types_handler.h" |
35 #include "chrome/common/extensions/user_script.h" | 37 #include "chrome/common/extensions/user_script.h" |
36 #include "chrome/common/render_messages.h" | 38 #include "chrome/common/render_messages.h" |
37 #include "content/public/browser/browser_thread.h" | 39 #include "content/public/browser/browser_thread.h" |
38 #include "content/public/browser/notification_service.h" | 40 #include "content/public/browser/notification_service.h" |
39 #include "content/public/browser/render_view_host.h" | 41 #include "content/public/browser/render_view_host.h" |
40 #include "content/public/browser/resource_context.h" | 42 #include "content/public/browser/resource_context.h" |
41 #include "content/public/browser/resource_dispatcher_host.h" | 43 #include "content/public/browser/resource_dispatcher_host.h" |
42 #include "content/public/browser/resource_request_info.h" | 44 #include "content/public/browser/resource_request_info.h" |
| 45 #include "content/public/browser/stream_handle.h" |
| 46 #include "extensions/common/constants.h" |
43 #include "net/base/load_flags.h" | 47 #include "net/base/load_flags.h" |
44 #include "net/http/http_response_headers.h" | 48 #include "net/http/http_response_headers.h" |
45 #include "net/ssl/ssl_config_service.h" | 49 #include "net/ssl/ssl_config_service.h" |
46 #include "net/url_request/url_request.h" | 50 #include "net/url_request/url_request.h" |
47 | 51 |
48 #if defined(ENABLE_MANAGED_USERS) | 52 #if defined(ENABLE_MANAGED_USERS) |
49 #include "chrome/browser/managed_mode/managed_mode_resource_throttle.h" | 53 #include "chrome/browser/managed_mode/managed_mode_resource_throttle.h" |
50 #endif | 54 #endif |
51 | 55 |
52 #if defined(USE_SYSTEM_PROTOBUF) | 56 #if defined(USE_SYSTEM_PROTOBUF) |
53 #include <google/protobuf/repeated_field.h> | 57 #include <google/protobuf/repeated_field.h> |
54 #else | 58 #else |
55 #include "third_party/protobuf/src/google/protobuf/repeated_field.h" | 59 #include "third_party/protobuf/src/google/protobuf/repeated_field.h" |
56 #endif | 60 #endif |
57 | 61 |
58 #if defined(OS_ANDROID) | 62 #if defined(OS_ANDROID) |
59 #include "components/navigation_interception/intercept_navigation_delegate.h" | 63 #include "components/navigation_interception/intercept_navigation_delegate.h" |
60 #endif | 64 #endif |
61 | 65 |
62 #if defined(OS_CHROMEOS) | 66 #if defined(OS_CHROMEOS) |
63 #include "chrome/browser/chromeos/login/merge_session_throttle.h" | 67 #include "chrome/browser/chromeos/login/merge_session_throttle.h" |
64 // TODO(oshima): Enable this for other platforms. | 68 // TODO(oshima): Enable this for other platforms. |
65 #include "chrome/browser/renderer_host/offline_resource_throttle.h" | 69 #include "chrome/browser/renderer_host/offline_resource_throttle.h" |
66 #endif | 70 #endif |
67 | 71 |
68 using content::BrowserThread; | 72 using content::BrowserThread; |
69 using content::RenderViewHost; | 73 using content::RenderViewHost; |
70 using content::ResourceDispatcherHostLoginDelegate; | 74 using content::ResourceDispatcherHostLoginDelegate; |
71 using content::ResourceRequestInfo; | 75 using content::ResourceRequestInfo; |
| 76 using extensions::Extension; |
72 | 77 |
73 namespace { | 78 namespace { |
74 | 79 |
75 void NotifyDownloadInitiatedOnUI(int render_process_id, int render_view_id) { | 80 void NotifyDownloadInitiatedOnUI(int render_process_id, int render_view_id) { |
76 RenderViewHost* rvh = RenderViewHost::FromID(render_process_id, | 81 RenderViewHost* rvh = RenderViewHost::FromID(render_process_id, |
77 render_view_id); | 82 render_view_id); |
78 if (!rvh) | 83 if (!rvh) |
79 return; | 84 return; |
80 | 85 |
81 content::NotificationService::current()->Notify( | 86 content::NotificationService::current()->Notify( |
82 chrome::NOTIFICATION_DOWNLOAD_INITIATED, | 87 chrome::NOTIFICATION_DOWNLOAD_INITIATED, |
83 content::Source<RenderViewHost>(rvh), | 88 content::Source<RenderViewHost>(rvh), |
84 content::NotificationService::NoDetails()); | 89 content::NotificationService::NoDetails()); |
85 } | 90 } |
86 | 91 |
| 92 // Goes through the extension's file browser handlers and checks if there is one |
| 93 // that can handle the |mime_type|. |
| 94 // |extension| must not be NULL. |
| 95 bool ExtensionCanHandleMimeType(const Extension* extension, |
| 96 const std::string& mime_type) { |
| 97 MimeTypesHandler* handler = MimeTypesHandler::GetHandler(extension); |
| 98 if (!handler) |
| 99 return false; |
| 100 |
| 101 return handler->CanHandleMIMEType(mime_type); |
| 102 } |
| 103 |
| 104 // Retrieves Profile for a render view host specified by |render_process_id| and |
| 105 // |render_view_id|. |
| 106 Profile* GetProfile(int render_process_id, int render_view_id) { |
| 107 content::RenderViewHost* render_view_host = |
| 108 content::RenderViewHost::FromID(render_process_id, render_view_id); |
| 109 if (!render_view_host) |
| 110 return NULL; |
| 111 |
| 112 content::WebContents* web_contents = |
| 113 content::WebContents::FromRenderViewHost(render_view_host); |
| 114 if (!web_contents) |
| 115 return NULL; |
| 116 |
| 117 content::BrowserContext* browser_context = web_contents->GetBrowserContext(); |
| 118 if (!browser_context) |
| 119 return NULL; |
| 120 |
| 121 return Profile::FromBrowserContext(browser_context); |
| 122 } |
| 123 |
| 124 void SendExecuteMimeTypeHandlerEvent(scoped_ptr<content::StreamHandle> stream, |
| 125 int render_process_id, |
| 126 int render_view_id, |
| 127 const std::string& extension_id) { |
| 128 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); |
| 129 |
| 130 Profile* profile = GetProfile(render_process_id, render_view_id); |
| 131 if (!profile) |
| 132 return; |
| 133 |
| 134 extensions::StreamsPrivateAPI::Get(profile)->ExecuteMimeTypeHandler( |
| 135 extension_id, stream.Pass()); |
| 136 } |
| 137 |
87 } // end namespace | 138 } // end namespace |
88 | 139 |
89 ChromeResourceDispatcherHostDelegate::ChromeResourceDispatcherHostDelegate( | 140 ChromeResourceDispatcherHostDelegate::ChromeResourceDispatcherHostDelegate( |
90 prerender::PrerenderTracker* prerender_tracker) | 141 prerender::PrerenderTracker* prerender_tracker) |
91 : download_request_limiter_(g_browser_process->download_request_limiter()), | 142 : download_request_limiter_(g_browser_process->download_request_limiter()), |
92 safe_browsing_(g_browser_process->safe_browsing_service()), | 143 safe_browsing_(g_browser_process->safe_browsing_service()), |
93 user_script_listener_(new extensions::UserScriptListener()), | 144 user_script_listener_(new extensions::UserScriptListener()), |
94 prerender_tracker_(prerender_tracker) { | 145 prerender_tracker_(prerender_tracker) { |
95 } | 146 } |
96 | 147 |
(...skipping 124 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
221 int request_id, | 272 int request_id, |
222 bool is_content_initiated, | 273 bool is_content_initiated, |
223 bool must_download, | 274 bool must_download, |
224 ScopedVector<content::ResourceThrottle>* throttles) { | 275 ScopedVector<content::ResourceThrottle>* throttles) { |
225 BrowserThread::PostTask( | 276 BrowserThread::PostTask( |
226 BrowserThread::UI, FROM_HERE, | 277 BrowserThread::UI, FROM_HERE, |
227 base::Bind(&NotifyDownloadInitiatedOnUI, child_id, route_id)); | 278 base::Bind(&NotifyDownloadInitiatedOnUI, child_id, route_id)); |
228 | 279 |
229 // If it's from the web, we don't trust it, so we push the throttle on. | 280 // If it's from the web, we don't trust it, so we push the throttle on. |
230 if (is_content_initiated) { | 281 if (is_content_initiated) { |
231 #if !defined(OS_ANDROID) | |
232 if (!must_download) { | |
233 ProfileIOData* io_data = | |
234 ProfileIOData::FromResourceContext(resource_context); | |
235 throttles->push_back(StreamsResourceThrottle::Create( | |
236 child_id, route_id, request, io_data->is_incognito(), | |
237 io_data->GetExtensionInfoMap())); | |
238 } | |
239 #endif | |
240 | |
241 throttles->push_back(new DownloadResourceThrottle( | 282 throttles->push_back(new DownloadResourceThrottle( |
242 download_request_limiter_, child_id, route_id, request_id, | 283 download_request_limiter_, child_id, route_id, request_id, |
243 request->method())); | 284 request->method())); |
244 } | 285 } |
245 | 286 |
246 // If this isn't a new request, we've seen this before and added the standard | 287 // If this isn't a new request, we've seen this before and added the standard |
247 // resource throttles already so no need to add it again. | 288 // resource throttles already so no need to add it again. |
248 if (!request->is_pending()) { | 289 if (!request->is_pending()) { |
249 AppendStandardResourceThrottles(request, | 290 AppendStandardResourceThrottles(request, |
250 resource_context, | 291 resource_context, |
(...skipping 130 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
381 } | 422 } |
382 } | 423 } |
383 #endif | 424 #endif |
384 | 425 |
385 bool ChromeResourceDispatcherHostDelegate::ShouldForceDownloadResource( | 426 bool ChromeResourceDispatcherHostDelegate::ShouldForceDownloadResource( |
386 const GURL& url, const std::string& mime_type) { | 427 const GURL& url, const std::string& mime_type) { |
387 // Special-case user scripts to get downloaded instead of viewed. | 428 // Special-case user scripts to get downloaded instead of viewed. |
388 return extensions::UserScript::IsURLUserScript(url, mime_type); | 429 return extensions::UserScript::IsURLUserScript(url, mime_type); |
389 } | 430 } |
390 | 431 |
| 432 bool ChromeResourceDispatcherHostDelegate::ShouldInterceptResourceAsStream( |
| 433 content::ResourceContext* resource_context, |
| 434 const GURL& url, |
| 435 const std::string& mime_type, |
| 436 GURL* security_origin, |
| 437 std::string* target_id) { |
| 438 #if !defined(OS_ANDROID) |
| 439 ProfileIOData* io_data = |
| 440 ProfileIOData::FromResourceContext(resource_context); |
| 441 bool profile_is_incognito = io_data->is_incognito(); |
| 442 const scoped_refptr<const ExtensionInfoMap> extension_info_map( |
| 443 io_data->GetExtensionInfoMap()); |
| 444 std::vector<std::string> whitelist = MimeTypesHandler::GetMIMETypeWhitelist(); |
| 445 // Go through the white-listed extensions and try to use them to intercept |
| 446 // the URL request. |
| 447 for (size_t i = 0; i < whitelist.size(); ++i) { |
| 448 const char* extension_id = whitelist[i].c_str(); |
| 449 const Extension* extension = |
| 450 extension_info_map->extensions().GetByID(extension_id); |
| 451 // The white-listed extension may not be installed, so we have to NULL check |
| 452 // |extension|. |
| 453 if (!extension || |
| 454 (profile_is_incognito && |
| 455 !extension_info_map->IsIncognitoEnabled(extension_id))) { |
| 456 continue; |
| 457 } |
| 458 |
| 459 if (ExtensionCanHandleMimeType(extension, mime_type)) { |
| 460 *security_origin = Extension::GetBaseURLFromExtensionId(extension_id); |
| 461 *target_id = extension_id; |
| 462 return true; |
| 463 } |
| 464 } |
| 465 #endif |
| 466 return false; |
| 467 } |
| 468 |
| 469 void ChromeResourceDispatcherHostDelegate::OnStreamCreated( |
| 470 content::ResourceContext* resource_context, |
| 471 int render_process_id, |
| 472 int render_view_id, |
| 473 const std::string& target_id, |
| 474 scoped_ptr<content::StreamHandle> stream) { |
| 475 #if !defined(OS_ANDROID) |
| 476 content::BrowserThread::PostTask( |
| 477 content::BrowserThread::UI, FROM_HERE, |
| 478 base::Bind(&SendExecuteMimeTypeHandlerEvent, base::Passed(&stream), |
| 479 render_process_id, render_view_id, |
| 480 target_id)); |
| 481 #endif |
| 482 } |
| 483 |
391 void ChromeResourceDispatcherHostDelegate::OnResponseStarted( | 484 void ChromeResourceDispatcherHostDelegate::OnResponseStarted( |
392 net::URLRequest* request, | 485 net::URLRequest* request, |
393 content::ResourceContext* resource_context, | 486 content::ResourceContext* resource_context, |
394 content::ResourceResponse* response, | 487 content::ResourceResponse* response, |
395 IPC::Sender* sender) { | 488 IPC::Sender* sender) { |
396 LoadTimingObserver::PopulateTimingInfo(request, response); | 489 LoadTimingObserver::PopulateTimingInfo(request, response); |
397 | 490 |
398 const ResourceRequestInfo* info = ResourceRequestInfo::ForRequest(request); | 491 const ResourceRequestInfo* info = ResourceRequestInfo::ForRequest(request); |
399 | 492 |
400 if (request->url().SchemeIsSecure()) { | 493 if (request->url().SchemeIsSecure()) { |
(...skipping 75 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
476 int child_id, route_id; | 569 int child_id, route_id; |
477 if (!prerender::PrerenderManager::DoesURLHaveValidScheme(redirect_url) && | 570 if (!prerender::PrerenderManager::DoesURLHaveValidScheme(redirect_url) && |
478 ResourceRequestInfo::ForRequest(request)->GetAssociatedRenderView( | 571 ResourceRequestInfo::ForRequest(request)->GetAssociatedRenderView( |
479 &child_id, &route_id) && | 572 &child_id, &route_id) && |
480 prerender_tracker_->IsPrerenderingOnIOThread(child_id, route_id)) { | 573 prerender_tracker_->IsPrerenderingOnIOThread(child_id, route_id)) { |
481 prerender_tracker_->TryCancel( | 574 prerender_tracker_->TryCancel( |
482 child_id, route_id, prerender::FINAL_STATUS_UNSUPPORTED_SCHEME); | 575 child_id, route_id, prerender::FINAL_STATUS_UNSUPPORTED_SCHEME); |
483 request->Cancel(); | 576 request->Cancel(); |
484 } | 577 } |
485 } | 578 } |
OLD | NEW |