Roll src/third_party/WebKit d9c6159:8139f33 (svn 201974:201975)
[chromium-blink-merge.git] / android_webview / browser / renderer_host / aw_resource_dispatcher_host_delegate.cc
blob1d9e58545e05da1949e2756858dfc1101fc8536c
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 "android_webview/browser/renderer_host/aw_resource_dispatcher_host_delegate.h"
7 #include <string>
9 #include "android_webview/browser/aw_contents_io_thread_client.h"
10 #include "android_webview/browser/aw_login_delegate.h"
11 #include "android_webview/browser/aw_resource_context.h"
12 #include "android_webview/common/url_constants.h"
13 #include "base/memory/scoped_ptr.h"
14 #include "base/memory/scoped_vector.h"
15 #include "components/auto_login_parser/auto_login_parser.h"
16 #include "components/navigation_interception/intercept_navigation_delegate.h"
17 #include "content/public/browser/browser_thread.h"
18 #include "content/public/browser/resource_controller.h"
19 #include "content/public/browser/resource_dispatcher_host.h"
20 #include "content/public/browser/resource_dispatcher_host_login_delegate.h"
21 #include "content/public/browser/resource_request_info.h"
22 #include "content/public/browser/resource_throttle.h"
23 #include "net/base/load_flags.h"
24 #include "net/base/net_errors.h"
25 #include "net/http/http_response_headers.h"
26 #include "net/url_request/url_request.h"
27 #include "net/url_request/url_request_status.h"
28 #include "url/url_constants.h"
30 using android_webview::AwContentsIoThreadClient;
31 using content::BrowserThread;
32 using content::ResourceType;
33 using navigation_interception::InterceptNavigationDelegate;
35 namespace {
37 base::LazyInstance<android_webview::AwResourceDispatcherHostDelegate>
38 g_webview_resource_dispatcher_host_delegate = LAZY_INSTANCE_INITIALIZER;
40 void SetCacheControlFlag(
41 net::URLRequest* request, int flag) {
42 const int all_cache_control_flags = net::LOAD_BYPASS_CACHE |
43 net::LOAD_VALIDATE_CACHE |
44 net::LOAD_PREFERRING_CACHE |
45 net::LOAD_ONLY_FROM_CACHE;
46 DCHECK_EQ((flag & all_cache_control_flags), flag);
47 int load_flags = request->load_flags();
48 load_flags &= ~all_cache_control_flags;
49 load_flags |= flag;
50 request->SetLoadFlags(load_flags);
53 } // namespace
55 namespace android_webview {
57 // Calls through the IoThreadClient to check the embedders settings to determine
58 // if the request should be cancelled. There may not always be an IoThreadClient
59 // available for the |render_process_id|, |render_frame_id| pair (in the case of
60 // newly created pop up windows, for example) and in that case the request and
61 // the client callbacks will be deferred the request until a client is ready.
62 class IoThreadClientThrottle : public content::ResourceThrottle {
63 public:
64 IoThreadClientThrottle(int render_process_id,
65 int render_frame_id,
66 net::URLRequest* request);
67 ~IoThreadClientThrottle() override;
69 // From content::ResourceThrottle
70 void WillStartRequest(bool* defer) override;
71 void WillRedirectRequest(const net::RedirectInfo& redirect_info,
72 bool* defer) override;
73 const char* GetNameForLogging() const override;
75 void OnIoThreadClientReady(int new_render_process_id,
76 int new_render_frame_id);
77 bool MaybeBlockRequest();
78 bool ShouldBlockRequest();
79 int render_process_id() const { return render_process_id_; }
80 int render_frame_id() const { return render_frame_id_; }
82 private:
83 int render_process_id_;
84 int render_frame_id_;
85 net::URLRequest* request_;
88 IoThreadClientThrottle::IoThreadClientThrottle(int render_process_id,
89 int render_frame_id,
90 net::URLRequest* request)
91 : render_process_id_(render_process_id),
92 render_frame_id_(render_frame_id),
93 request_(request) { }
95 IoThreadClientThrottle::~IoThreadClientThrottle() {
96 DCHECK_CURRENTLY_ON(BrowserThread::IO);
97 g_webview_resource_dispatcher_host_delegate.Get().
98 RemovePendingThrottleOnIoThread(this);
101 const char* IoThreadClientThrottle::GetNameForLogging() const {
102 return "IoThreadClientThrottle";
105 void IoThreadClientThrottle::WillStartRequest(bool* defer) {
106 DCHECK_CURRENTLY_ON(BrowserThread::IO);
107 if (render_frame_id_ < 1)
108 return;
109 DCHECK(render_process_id_);
110 *defer = false;
112 // Defer all requests of a pop up that is still not associated with Java
113 // client so that the client will get a chance to override requests.
114 scoped_ptr<AwContentsIoThreadClient> io_client =
115 AwContentsIoThreadClient::FromID(render_process_id_, render_frame_id_);
116 if (io_client && io_client->PendingAssociation()) {
117 *defer = true;
118 AwResourceDispatcherHostDelegate::AddPendingThrottle(
119 render_process_id_, render_frame_id_, this);
120 } else {
121 MaybeBlockRequest();
125 void IoThreadClientThrottle::WillRedirectRequest(
126 const net::RedirectInfo& redirect_info,
127 bool* defer) {
128 WillStartRequest(defer);
131 void IoThreadClientThrottle::OnIoThreadClientReady(int new_render_process_id,
132 int new_render_frame_id) {
133 DCHECK_CURRENTLY_ON(BrowserThread::IO);
135 if (!MaybeBlockRequest()) {
136 controller()->Resume();
140 bool IoThreadClientThrottle::MaybeBlockRequest() {
141 if (ShouldBlockRequest()) {
142 controller()->CancelWithError(net::ERR_ACCESS_DENIED);
143 return true;
145 return false;
148 bool IoThreadClientThrottle::ShouldBlockRequest() {
149 scoped_ptr<AwContentsIoThreadClient> io_client =
150 AwContentsIoThreadClient::FromID(render_process_id_, render_frame_id_);
151 if (!io_client)
152 return false;
154 // Part of implementation of WebSettings.allowContentAccess.
155 if (request_->url().SchemeIs(url::kContentScheme) &&
156 io_client->ShouldBlockContentUrls()) {
157 return true;
160 // Part of implementation of WebSettings.allowFileAccess.
161 if (request_->url().SchemeIsFile() &&
162 io_client->ShouldBlockFileUrls()) {
163 // Application's assets and resources are always available.
164 return !IsAndroidSpecialFileUrl(request_->url());
167 if (io_client->ShouldBlockNetworkLoads()) {
168 if (request_->url().SchemeIs(url::kFtpScheme)) {
169 return true;
171 SetCacheControlFlag(request_, net::LOAD_ONLY_FROM_CACHE);
172 } else {
173 AwContentsIoThreadClient::CacheMode cache_mode = io_client->GetCacheMode();
174 switch(cache_mode) {
175 case AwContentsIoThreadClient::LOAD_CACHE_ELSE_NETWORK:
176 SetCacheControlFlag(request_, net::LOAD_PREFERRING_CACHE);
177 break;
178 case AwContentsIoThreadClient::LOAD_NO_CACHE:
179 SetCacheControlFlag(request_, net::LOAD_BYPASS_CACHE);
180 break;
181 case AwContentsIoThreadClient::LOAD_CACHE_ONLY:
182 SetCacheControlFlag(request_, net::LOAD_ONLY_FROM_CACHE);
183 break;
184 default:
185 break;
188 return false;
191 // static
192 void AwResourceDispatcherHostDelegate::ResourceDispatcherHostCreated() {
193 content::ResourceDispatcherHost::Get()->SetDelegate(
194 &g_webview_resource_dispatcher_host_delegate.Get());
197 AwResourceDispatcherHostDelegate::AwResourceDispatcherHostDelegate()
198 : content::ResourceDispatcherHostDelegate() {
201 AwResourceDispatcherHostDelegate::~AwResourceDispatcherHostDelegate() {
204 void AwResourceDispatcherHostDelegate::RequestBeginning(
205 net::URLRequest* request,
206 content::ResourceContext* resource_context,
207 content::AppCacheService* appcache_service,
208 ResourceType resource_type,
209 ScopedVector<content::ResourceThrottle>* throttles) {
211 AddExtraHeadersIfNeeded(request, resource_context);
213 const content::ResourceRequestInfo* request_info =
214 content::ResourceRequestInfo::ForRequest(request);
216 // We always push the throttles here. Checking the existence of io_client
217 // is racy when a popup window is created. That is because RequestBeginning
218 // is called whether or not requests are blocked via BlockRequestForRoute()
219 // however io_client may or may not be ready at the time depending on whether
220 // webcontents is created.
221 throttles->push_back(new IoThreadClientThrottle(
222 request_info->GetChildID(), request_info->GetRenderFrameID(), request));
224 if (resource_type == content::RESOURCE_TYPE_MAIN_FRAME ||
225 (resource_type == content::RESOURCE_TYPE_SUB_FRAME &&
226 !request->url().SchemeIs(url::kHttpScheme) &&
227 !request->url().SchemeIs(url::kHttpsScheme) &&
228 !request->url().SchemeIs(url::kAboutScheme))) {
229 throttles->push_back(InterceptNavigationDelegate::CreateThrottleFor(
230 request));
232 if (resource_type != content::RESOURCE_TYPE_MAIN_FRAME)
233 InterceptNavigationDelegate::UpdateUserGestureCarryoverInfo(request);
236 void AwResourceDispatcherHostDelegate::OnRequestRedirected(
237 const GURL& redirect_url,
238 net::URLRequest* request,
239 content::ResourceContext* resource_context,
240 content::ResourceResponse* response) {
241 AddExtraHeadersIfNeeded(request, resource_context);
244 void AwResourceDispatcherHostDelegate::RequestComplete(
245 net::URLRequest* request) {
246 if (request && !request->status().is_success()) {
247 const content::ResourceRequestInfo* request_info =
248 content::ResourceRequestInfo::ForRequest(request);
249 scoped_ptr<AwContentsIoThreadClient> io_client =
250 AwContentsIoThreadClient::FromID(request_info->GetChildID(),
251 request_info->GetRenderFrameID());
252 if (io_client) {
253 io_client->OnReceivedError(request);
254 } else {
255 DLOG(WARNING) << "io_client is null, onReceivedError dropped for " <<
256 request->url();
262 void AwResourceDispatcherHostDelegate::DownloadStarting(
263 net::URLRequest* request,
264 content::ResourceContext* resource_context,
265 int child_id,
266 int route_id,
267 int request_id,
268 bool is_content_initiated,
269 bool must_download,
270 ScopedVector<content::ResourceThrottle>* throttles) {
271 GURL url(request->url());
272 std::string user_agent;
273 std::string content_disposition;
274 std::string mime_type;
275 int64 content_length = request->GetExpectedContentSize();
277 request->extra_request_headers().GetHeader(
278 net::HttpRequestHeaders::kUserAgent, &user_agent);
281 net::HttpResponseHeaders* response_headers = request->response_headers();
282 if (response_headers) {
283 response_headers->GetNormalizedHeader("content-disposition",
284 &content_disposition);
285 response_headers->GetMimeType(&mime_type);
288 request->Cancel();
290 const content::ResourceRequestInfo* request_info =
291 content::ResourceRequestInfo::ForRequest(request);
293 scoped_ptr<AwContentsIoThreadClient> io_client =
294 AwContentsIoThreadClient::FromID(
295 child_id, request_info->GetRenderFrameID());
297 // POST request cannot be repeated in general, so prevent client from
298 // retrying the same request, even if it is with a GET.
299 if ("GET" == request->method() && io_client) {
300 io_client->NewDownload(url,
301 user_agent,
302 content_disposition,
303 mime_type,
304 content_length);
308 content::ResourceDispatcherHostLoginDelegate*
309 AwResourceDispatcherHostDelegate::CreateLoginDelegate(
310 net::AuthChallengeInfo* auth_info,
311 net::URLRequest* request) {
312 return new AwLoginDelegate(auth_info, request);
315 bool AwResourceDispatcherHostDelegate::HandleExternalProtocol(
316 const GURL& url,
317 int child_id,
318 int route_id,
319 bool is_main_frame,
320 ui::PageTransition page_transition,
321 bool has_user_gesture) {
322 // The AwURLRequestJobFactory implementation should ensure this method never
323 // gets called.
324 NOTREACHED();
325 return false;
328 void AwResourceDispatcherHostDelegate::OnResponseStarted(
329 net::URLRequest* request,
330 content::ResourceContext* resource_context,
331 content::ResourceResponse* response,
332 IPC::Sender* sender) {
333 const content::ResourceRequestInfo* request_info =
334 content::ResourceRequestInfo::ForRequest(request);
335 if (!request_info) {
336 DLOG(FATAL) << "Started request without associated info: " <<
337 request->url();
338 return;
341 if (request_info->GetResourceType() == content::RESOURCE_TYPE_MAIN_FRAME) {
342 // Check for x-auto-login header.
343 auto_login_parser::HeaderData header_data;
344 if (auto_login_parser::ParserHeaderInResponse(
345 request, auto_login_parser::ALLOW_ANY_REALM, &header_data)) {
346 scoped_ptr<AwContentsIoThreadClient> io_client =
347 AwContentsIoThreadClient::FromID(request_info->GetChildID(),
348 request_info->GetRenderFrameID());
349 if (io_client) {
350 io_client->NewLoginRequest(
351 header_data.realm, header_data.account, header_data.args);
357 void AwResourceDispatcherHostDelegate::RemovePendingThrottleOnIoThread(
358 IoThreadClientThrottle* throttle) {
359 DCHECK_CURRENTLY_ON(BrowserThread::IO);
360 PendingThrottleMap::iterator it = pending_throttles_.find(
361 FrameRouteIDPair(throttle->render_process_id(),
362 throttle->render_frame_id()));
363 if (it != pending_throttles_.end()) {
364 pending_throttles_.erase(it);
368 // static
369 void AwResourceDispatcherHostDelegate::OnIoThreadClientReady(
370 int new_render_process_id,
371 int new_render_frame_id) {
372 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
373 base::Bind(
374 &AwResourceDispatcherHostDelegate::OnIoThreadClientReadyInternal,
375 base::Unretained(
376 g_webview_resource_dispatcher_host_delegate.Pointer()),
377 new_render_process_id, new_render_frame_id));
380 // static
381 void AwResourceDispatcherHostDelegate::AddPendingThrottle(
382 int render_process_id,
383 int render_frame_id,
384 IoThreadClientThrottle* pending_throttle) {
385 BrowserThread::PostTask(BrowserThread::IO, FROM_HERE,
386 base::Bind(
387 &AwResourceDispatcherHostDelegate::AddPendingThrottleOnIoThread,
388 base::Unretained(
389 g_webview_resource_dispatcher_host_delegate.Pointer()),
390 render_process_id, render_frame_id, pending_throttle));
393 void AwResourceDispatcherHostDelegate::AddPendingThrottleOnIoThread(
394 int render_process_id,
395 int render_frame_id_id,
396 IoThreadClientThrottle* pending_throttle) {
397 DCHECK_CURRENTLY_ON(BrowserThread::IO);
398 pending_throttles_.insert(
399 std::pair<FrameRouteIDPair, IoThreadClientThrottle*>(
400 FrameRouteIDPair(render_process_id, render_frame_id_id),
401 pending_throttle));
404 void AwResourceDispatcherHostDelegate::OnIoThreadClientReadyInternal(
405 int new_render_process_id,
406 int new_render_frame_id) {
407 DCHECK_CURRENTLY_ON(BrowserThread::IO);
408 PendingThrottleMap::iterator it = pending_throttles_.find(
409 FrameRouteIDPair(new_render_process_id, new_render_frame_id));
411 if (it != pending_throttles_.end()) {
412 IoThreadClientThrottle* throttle = it->second;
413 throttle->OnIoThreadClientReady(new_render_process_id, new_render_frame_id);
414 pending_throttles_.erase(it);
418 void AwResourceDispatcherHostDelegate::AddExtraHeadersIfNeeded(
419 net::URLRequest* request,
420 content::ResourceContext* resource_context) {
421 const content::ResourceRequestInfo* request_info =
422 content::ResourceRequestInfo::ForRequest(request);
423 if (!request_info)
424 return;
425 if (request_info->GetResourceType() != content::RESOURCE_TYPE_MAIN_FRAME)
426 return;
428 const ui::PageTransition transition = request_info->GetPageTransition();
429 const bool is_load_url =
430 transition & ui::PAGE_TRANSITION_FROM_API;
431 const bool is_go_back_forward =
432 transition & ui::PAGE_TRANSITION_FORWARD_BACK;
433 const bool is_reload = ui::PageTransitionCoreTypeIs(
434 transition, ui::PAGE_TRANSITION_RELOAD);
435 if (is_load_url || is_go_back_forward || is_reload) {
436 AwResourceContext* awrc = static_cast<AwResourceContext*>(resource_context);
437 std::string extra_headers = awrc->GetExtraHeaders(request->url());
438 if (!extra_headers.empty()) {
439 net::HttpRequestHeaders headers;
440 headers.AddHeadersFromString(extra_headers);
441 for (net::HttpRequestHeaders::Iterator it(headers); it.GetNext(); ) {
442 request->SetExtraRequestHeaderByName(it.name(), it.value(), false);
448 } // namespace android_webview