| 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 "content/browser/renderer_host/buffered_resource_handler.h" | 5 #include "content/browser/renderer_host/buffered_resource_handler.h" |
| 6 | 6 |
| 7 #include <vector> | 7 #include <vector> |
| 8 | 8 |
| 9 #include "base/bind.h" | 9 #include "base/bind.h" |
| 10 #include "base/logging.h" | 10 #include "base/logging.h" |
| (...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 65 ResourceHandler* handler, | 65 ResourceHandler* handler, |
| 66 ResourceDispatcherHostImpl* host, | 66 ResourceDispatcherHostImpl* host, |
| 67 net::URLRequest* request) | 67 net::URLRequest* request) |
| 68 : LayeredResourceHandler(handler), | 68 : LayeredResourceHandler(handler), |
| 69 host_(host), | 69 host_(host), |
| 70 request_(request), | 70 request_(request), |
| 71 read_buffer_size_(0), | 71 read_buffer_size_(0), |
| 72 bytes_read_(0), | 72 bytes_read_(0), |
| 73 sniff_content_(false), | 73 sniff_content_(false), |
| 74 wait_for_plugins_(false), | 74 wait_for_plugins_(false), |
| 75 deferred_waiting_for_plugins_(false), |
| 75 buffering_(false), | 76 buffering_(false), |
| 76 next_handler_needs_response_started_(false), | 77 next_handler_needs_response_started_(false), |
| 77 next_handler_needs_will_read_(false), | 78 next_handler_needs_will_read_(false), |
| 78 finished_(false) { | 79 finished_(false) { |
| 79 } | 80 } |
| 80 | 81 |
| 81 bool BufferedResourceHandler::OnResponseStarted( | 82 bool BufferedResourceHandler::OnResponseStarted( |
| 82 int request_id, | 83 int request_id, |
| 83 ResourceResponse* response) { | 84 ResourceResponse* response, |
| 85 bool* defer) { |
| 84 response_ = response; | 86 response_ = response; |
| 87 |
| 85 if (!DelayResponse()) | 88 if (!DelayResponse()) |
| 86 return CompleteResponseStarted(request_id); | 89 return CompleteResponseStarted(request_id, defer); |
| 90 |
| 91 if (wait_for_plugins_) { |
| 92 deferred_waiting_for_plugins_ = true; |
| 93 *defer = true; |
| 94 } |
| 87 return true; | 95 return true; |
| 88 } | 96 } |
| 89 | 97 |
| 90 void BufferedResourceHandler::OnRequestClosed() { | 98 void BufferedResourceHandler::OnRequestClosed() { |
| 91 request_ = NULL; | 99 request_ = NULL; |
| 92 next_handler_->OnRequestClosed(); | 100 next_handler_->OnRequestClosed(); |
| 93 } | 101 } |
| 94 | 102 |
| 95 // We'll let the original event handler provide a buffer, and reuse it for | 103 // We'll let the original event handler provide a buffer, and reuse it for |
| 96 // subsequent reads until we're done buffering. | 104 // subsequent reads until we're done buffering. |
| 97 bool BufferedResourceHandler::OnWillRead(int request_id, net::IOBuffer** buf, | 105 bool BufferedResourceHandler::OnWillRead(int request_id, net::IOBuffer** buf, |
| 98 int* buf_size, int min_size) { | 106 int* buf_size, int min_size) { |
| 99 if (buffering_) { | 107 if (buffering_) { |
| 100 DCHECK(!my_buffer_.get()); | 108 DCHECK(!my_buffer_.get()); |
| 101 my_buffer_ = new net::IOBuffer(net::kMaxBytesToSniff); | 109 my_buffer_ = new net::IOBuffer(net::kMaxBytesToSniff); |
| 102 *buf = my_buffer_.get(); | 110 *buf = my_buffer_.get(); |
| 103 *buf_size = net::kMaxBytesToSniff; | 111 *buf_size = net::kMaxBytesToSniff; |
| 104 return true; | 112 return true; |
| 105 } | 113 } |
| 106 | 114 |
| 107 if (finished_) | 115 if (finished_) |
| 108 return false; | 116 return false; |
| 109 | 117 |
| 110 if (!next_handler_->OnWillRead(request_id, buf, buf_size, min_size)) { | 118 if (!next_handler_->OnWillRead(request_id, buf, buf_size, min_size)) |
| 111 return false; | 119 return false; |
| 112 } | 120 |
| 113 read_buffer_ = *buf; | 121 read_buffer_ = *buf; |
| 114 read_buffer_size_ = *buf_size; | 122 read_buffer_size_ = *buf_size; |
| 115 DCHECK_GE(read_buffer_size_, net::kMaxBytesToSniff * 2); | 123 DCHECK_GE(read_buffer_size_, net::kMaxBytesToSniff * 2); |
| 116 bytes_read_ = 0; | 124 bytes_read_ = 0; |
| 117 return true; | 125 return true; |
| 118 } | 126 } |
| 119 | 127 |
| 120 bool BufferedResourceHandler::OnReadCompleted(int request_id, int* bytes_read) { | 128 bool BufferedResourceHandler::OnReadCompleted(int request_id, int* bytes_read, |
| 121 ResourceRequestInfoImpl* info = | 129 bool* defer) { |
| 122 ResourceRequestInfoImpl::ForRequest(request_); | |
| 123 | |
| 124 if (sniff_content_) { | 130 if (sniff_content_) { |
| 125 if (KeepBuffering(*bytes_read)) | 131 if (KeepBuffering(*bytes_read)) { |
| 132 if (wait_for_plugins_) |
| 133 *defer = deferred_waiting_for_plugins_ = true; |
| 126 return true; | 134 return true; |
| 135 } |
| 127 | 136 |
| 128 *bytes_read = bytes_read_; | 137 *bytes_read = bytes_read_; |
| 129 | 138 |
| 130 // Done buffering, send the pending ResponseStarted event. | 139 // Done buffering, send the pending ResponseStarted event. |
| 131 if (!CompleteResponseStarted(request_id)) | 140 if (!CompleteResponseStarted(request_id, defer)) |
| 132 return false; | 141 return false; |
| 133 | 142 if (*defer) |
| 134 // The next handler might have paused the request in OnResponseStarted. | |
| 135 if (info->pause_count()) | |
| 136 return true; | 143 return true; |
| 137 } else if (wait_for_plugins_) { | 144 } else if (wait_for_plugins_) { |
| 145 *defer = deferred_waiting_for_plugins_ = true; |
| 138 return true; | 146 return true; |
| 139 } | 147 } |
| 140 | 148 |
| 141 if (!ForwardPendingEventsToNextHandler(request_id)) | 149 if (!ForwardPendingEventsToNextHandler(request_id, defer)) |
| 142 return false; | 150 return false; |
| 143 if (info->pause_count()) | 151 if (*defer) |
| 144 return true; | 152 return true; |
| 145 | 153 |
| 146 // Release the reference that we acquired at OnWillRead. | 154 // Release the reference that we acquired at OnWillRead. |
| 147 read_buffer_ = NULL; | 155 read_buffer_ = NULL; |
| 148 return next_handler_->OnReadCompleted(request_id, bytes_read); | 156 return next_handler_->OnReadCompleted(request_id, bytes_read, defer); |
| 149 } | 157 } |
| 150 | 158 |
| 151 BufferedResourceHandler::~BufferedResourceHandler() {} | 159 BufferedResourceHandler::~BufferedResourceHandler() {} |
| 152 | 160 |
| 153 bool BufferedResourceHandler::DelayResponse() { | 161 bool BufferedResourceHandler::DelayResponse() { |
| 154 std::string mime_type; | 162 std::string mime_type; |
| 155 request_->GetMimeType(&mime_type); | 163 request_->GetMimeType(&mime_type); |
| 156 | 164 |
| 157 std::string content_type_options; | 165 std::string content_type_options; |
| 158 request_->GetResponseHeaderByName("x-content-type-options", | 166 request_->GetResponseHeaderByName("x-content-type-options", |
| (...skipping 77 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 236 } | 244 } |
| 237 | 245 |
| 238 buffering_ = false; | 246 buffering_ = false; |
| 239 | 247 |
| 240 if (wait_for_plugins_) | 248 if (wait_for_plugins_) |
| 241 return true; | 249 return true; |
| 242 | 250 |
| 243 return false; | 251 return false; |
| 244 } | 252 } |
| 245 | 253 |
| 246 bool BufferedResourceHandler::CompleteResponseStarted(int request_id) { | 254 bool BufferedResourceHandler::CompleteResponseStarted(int request_id, |
| 255 bool* defer) { |
| 247 ResourceRequestInfoImpl* info = | 256 ResourceRequestInfoImpl* info = |
| 248 ResourceRequestInfoImpl::ForRequest(request_); | 257 ResourceRequestInfoImpl::ForRequest(request_); |
| 249 std::string mime_type; | 258 std::string mime_type; |
| 250 request_->GetMimeType(&mime_type); | 259 request_->GetMimeType(&mime_type); |
| 251 | 260 |
| 252 // Check if this is an X.509 certificate, if yes, let it be handled | |
| 253 // by X509UserCertResourceHandler. | |
| 254 if (mime_type == "application/x-x509-user-cert") { | 261 if (mime_type == "application/x-x509-user-cert") { |
| 255 // This is entirely similar to how DownloadResourceThrottle works except we | 262 // Let X.509 certs be handled by the X509UserCertResourceHandler. |
| 263 |
| 264 // This is entirely similar to how DownloadResourceHandler works except we |
| 256 // are doing it for an X.509 client certificates. | 265 // are doing it for an X.509 client certificates. |
| 257 // TODO(darin): This does not belong here! | 266 // TODO(darin): This does not belong here! |
| 258 | 267 |
| 259 if (response_->headers && // Can be NULL if FTP. | 268 if (response_->headers && // Can be NULL if FTP. |
| 260 response_->headers->response_code() / 100 != 2) { | 269 response_->headers->response_code() / 100 != 2) { |
| 261 // The response code indicates that this is an error page, but we are | 270 // The response code indicates that this is an error page, but we are |
| 262 // expecting an X.509 user certificate. We follow Firefox here and show | 271 // expecting an X.509 user certificate. We follow Firefox here and show |
| 263 // our own error page instead of handling the error page as a | 272 // our own error page instead of handling the error page as a |
| 264 // certificate. | 273 // certificate. |
| 265 // TODO(abarth): We should abstract the response_code test, but this kind | 274 // TODO(abarth): We should abstract the response_code test, but this kind |
| 266 // of check is scattered throughout our codebase. | 275 // of check is scattered throughout our codebase. |
| 267 request_->CancelWithError(net::ERR_FILE_NOT_FOUND); | 276 request_->CancelWithError(net::ERR_FILE_NOT_FOUND); |
| 268 return false; | 277 return false; |
| 269 } | 278 } |
| 270 | 279 |
| 271 X509UserCertResourceHandler* x509_cert_handler = | 280 X509UserCertResourceHandler* x509_cert_handler = |
| 272 new X509UserCertResourceHandler(request_, | 281 new X509UserCertResourceHandler(request_, |
| 273 info->GetChildID(), | 282 info->GetChildID(), |
| 274 info->GetRouteID()); | 283 info->GetRouteID()); |
| 275 if (!UseAlternateResourceHandler(request_id, x509_cert_handler)) | 284 if (!UseAlternateResourceHandler(request_id, x509_cert_handler, defer)) |
| 276 return false; | 285 return false; |
| 277 } | 286 } else if (info->allow_download() && ShouldDownload(NULL)) { |
| 287 // Forward the data to the download thread. |
| 278 | 288 |
| 279 // Check to see if we should forward the data from this request to the | |
| 280 // download thread. | |
| 281 // TODO(paulg): Only download if the context from the renderer allows it. | |
| 282 if (info->allow_download() && ShouldDownload(NULL)) { | |
| 283 if (response_->headers && // Can be NULL if FTP. | 289 if (response_->headers && // Can be NULL if FTP. |
| 284 response_->headers->response_code() / 100 != 2) { | 290 response_->headers->response_code() / 100 != 2) { |
| 285 // The response code indicates that this is an error page, but we don't | 291 // The response code indicates that this is an error page, but we don't |
| 286 // know how to display the content. We follow Firefox here and show our | 292 // know how to display the content. We follow Firefox here and show our |
| 287 // own error page instead of triggering a download. | 293 // own error page instead of triggering a download. |
| 288 // TODO(abarth): We should abstract the response_code test, but this kind | 294 // TODO(abarth): We should abstract the response_code test, but this kind |
| 289 // of check is scattered throughout our codebase. | 295 // of check is scattered throughout our codebase. |
| 290 request_->CancelWithError(net::ERR_FILE_NOT_FOUND); | 296 request_->CancelWithError(net::ERR_FILE_NOT_FOUND); |
| 291 return false; | 297 return false; |
| 292 } | 298 } |
| 293 | 299 |
| 294 info->set_is_download(true); | 300 info->set_is_download(true); |
| 295 | 301 |
| 296 scoped_refptr<ResourceHandler> handler( | 302 scoped_refptr<ResourceHandler> handler( |
| 297 host_->CreateResourceHandlerForDownload( | 303 host_->CreateResourceHandlerForDownload( |
| 298 request_, | 304 request_, |
| 299 info->GetContext(), | 305 info->GetContext(), |
| 300 info->GetChildID(), | 306 info->GetChildID(), |
| 301 info->GetRouteID(), | 307 info->GetRouteID(), |
| 302 info->GetRequestID(), | 308 info->GetRequestID(), |
| 303 true, // is_content_initiated | 309 true, // is_content_initiated |
| 304 DownloadSaveInfo(), | 310 DownloadSaveInfo(), |
| 305 DownloadResourceHandler::OnStartedCallback())); | 311 DownloadResourceHandler::OnStartedCallback())); |
| 306 | 312 |
| 307 if (!UseAlternateResourceHandler(request_id, handler)) | 313 if (!UseAlternateResourceHandler(request_id, handler, defer)) |
| 308 return false; | 314 return false; |
| 309 } | 315 } |
| 310 | 316 |
| 311 if (info->pause_count()) | 317 if (*defer) |
| 312 return true; | 318 return true; |
| 313 | 319 |
| 314 return next_handler_->OnResponseStarted(request_id, response_); | 320 return next_handler_->OnResponseStarted(request_id, response_, defer); |
| 315 } | 321 } |
| 316 | 322 |
| 317 bool BufferedResourceHandler::ShouldWaitForPlugins() { | 323 bool BufferedResourceHandler::ShouldWaitForPlugins() { |
| 318 bool need_plugin_list; | 324 bool need_plugin_list; |
| 319 if (!ShouldDownload(&need_plugin_list) || !need_plugin_list) | 325 if (!ShouldDownload(&need_plugin_list) || !need_plugin_list) |
| 320 return false; | 326 return false; |
| 321 | 327 |
| 322 // We don't want to keep buffering as our buffer will fill up. | |
| 323 ResourceRequestInfoImpl* info = | |
| 324 ResourceRequestInfoImpl::ForRequest(request_); | |
| 325 host_->PauseRequest(info->GetChildID(), info->GetRequestID(), true); | |
| 326 | |
| 327 // Get the plugins asynchronously. | 328 // Get the plugins asynchronously. |
| 328 PluginServiceImpl::GetInstance()->GetPlugins( | 329 PluginServiceImpl::GetInstance()->GetPlugins( |
| 329 base::Bind(&BufferedResourceHandler::OnPluginsLoaded, this)); | 330 base::Bind(&BufferedResourceHandler::OnPluginsLoaded, this)); |
| 330 return true; | 331 return true; |
| 331 } | 332 } |
| 332 | 333 |
| 333 // This test mirrors the decision that WebKit makes in | 334 // This test mirrors the decision that WebKit makes in |
| 334 // WebFrameLoaderClient::dispatchDecidePolicyForMIMEType. | 335 // WebFrameLoaderClient::dispatchDecidePolicyForMIMEType. |
| 335 bool BufferedResourceHandler::ShouldDownload(bool* need_plugin_list) { | 336 bool BufferedResourceHandler::ShouldDownload(bool* need_plugin_list) { |
| 336 if (need_plugin_list) | 337 if (need_plugin_list) |
| (...skipping 35 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 372 } | 373 } |
| 373 } else { | 374 } else { |
| 374 DCHECK(!stale); | 375 DCHECK(!stale); |
| 375 } | 376 } |
| 376 | 377 |
| 377 return !found; | 378 return !found; |
| 378 } | 379 } |
| 379 | 380 |
| 380 bool BufferedResourceHandler::UseAlternateResourceHandler( | 381 bool BufferedResourceHandler::UseAlternateResourceHandler( |
| 381 int request_id, | 382 int request_id, |
| 382 ResourceHandler* handler) { | 383 ResourceHandler* handler, |
| 384 bool* defer) { |
| 383 // Inform the original ResourceHandler that this will be handled entirely by | 385 // Inform the original ResourceHandler that this will be handled entirely by |
| 384 // the new ResourceHandler. | 386 // the new ResourceHandler. |
| 385 // TODO(darin): We should probably check the return values of these. | 387 // TODO(darin): We should probably check the return values of these. |
| 386 next_handler_->OnResponseStarted(request_id, response_); | 388 bool defer_ignored = false; |
| 389 next_handler_->OnResponseStarted(request_id, response_, &defer_ignored); |
| 390 DCHECK(!defer_ignored); |
| 387 net::URLRequestStatus status(net::URLRequestStatus::HANDLED_EXTERNALLY, 0); | 391 net::URLRequestStatus status(net::URLRequestStatus::HANDLED_EXTERNALLY, 0); |
| 388 next_handler_->OnResponseCompleted(request_id, status, std::string()); | 392 next_handler_->OnResponseCompleted(request_id, status, std::string()); |
| 389 | 393 |
| 390 // Remove the non-owning pointer to the CrossSiteResourceHandler, if any, | 394 // Remove the non-owning pointer to the CrossSiteResourceHandler, if any, |
| 391 // from the extra request info because the CrossSiteResourceHandler (part of | 395 // from the extra request info because the CrossSiteResourceHandler (part of |
| 392 // the original ResourceHandler chain) will be deleted by the next statement. | 396 // the original ResourceHandler chain) will be deleted by the next statement. |
| 393 ResourceRequestInfoImpl* info = | 397 ResourceRequestInfoImpl* info = |
| 394 ResourceRequestInfoImpl::ForRequest(request_); | 398 ResourceRequestInfoImpl::ForRequest(request_); |
| 395 info->set_cross_site_handler(NULL); | 399 info->set_cross_site_handler(NULL); |
| 396 | 400 |
| 397 // This is handled entirely within the new ResourceHandler, so just reset the | 401 // This is handled entirely within the new ResourceHandler, so just reset the |
| 398 // original ResourceHandler. | 402 // original ResourceHandler. |
| 399 next_handler_ = handler; | 403 next_handler_ = handler; |
| 400 | 404 |
| 401 next_handler_needs_response_started_ = true; | 405 next_handler_needs_response_started_ = true; |
| 402 next_handler_needs_will_read_ = true; | 406 next_handler_needs_will_read_ = true; |
| 403 | 407 |
| 404 return ForwardPendingEventsToNextHandler(request_id); | 408 return ForwardPendingEventsToNextHandler(request_id, defer); |
| 405 } | 409 } |
| 406 | 410 |
| 407 bool BufferedResourceHandler::ForwardPendingEventsToNextHandler( | 411 bool BufferedResourceHandler::ForwardPendingEventsToNextHandler(int request_id, |
| 408 int request_id) { | 412 bool* defer) { |
| 409 ResourceRequestInfoImpl* info = | |
| 410 ResourceRequestInfoImpl::ForRequest(request_); | |
| 411 if (info->pause_count()) | |
| 412 return true; | |
| 413 | |
| 414 if (next_handler_needs_response_started_) { | 413 if (next_handler_needs_response_started_) { |
| 415 if (!next_handler_->OnResponseStarted(request_id, response_)) | 414 if (!next_handler_->OnResponseStarted(request_id, response_, defer)) |
| 416 return false; | 415 return false; |
| 417 // If the request was paused during OnResponseStarted, we need to avoid | 416 // If the request was deferred during OnResponseStarted, we need to avoid |
| 418 // calling OnResponseStarted again. | 417 // calling OnResponseStarted again. |
| 419 next_handler_needs_response_started_ = false; | 418 next_handler_needs_response_started_ = false; |
| 420 if (info->pause_count()) | 419 if (*defer) |
| 421 return true; | 420 return true; |
| 422 } | 421 } |
| 423 | 422 |
| 424 if (next_handler_needs_will_read_) { | 423 if (next_handler_needs_will_read_) { |
| 425 CopyReadBufferToNextHandler(request_id); | 424 CopyReadBufferToNextHandler(request_id); |
| 426 // If the request was paused during OnWillRead, we need to be sure to try | |
| 427 // calling OnWillRead again. | |
| 428 if (info->pause_count()) | |
| 429 return true; | |
| 430 next_handler_needs_will_read_ = false; | 425 next_handler_needs_will_read_ = false; |
| 431 } | 426 } |
| 432 return true; | 427 return true; |
| 433 } | 428 } |
| 434 | 429 |
| 435 void BufferedResourceHandler::CopyReadBufferToNextHandler(int request_id) { | 430 void BufferedResourceHandler::CopyReadBufferToNextHandler(int request_id) { |
| 436 if (!bytes_read_) | 431 if (!bytes_read_) |
| 437 return; | 432 return; |
| 438 | 433 |
| 439 net::IOBuffer* buf = NULL; | 434 net::IOBuffer* buf = NULL; |
| 440 int buf_len = 0; | 435 int buf_len = 0; |
| 441 if (next_handler_->OnWillRead(request_id, &buf, &buf_len, bytes_read_)) { | 436 if (next_handler_->OnWillRead(request_id, &buf, &buf_len, bytes_read_)) { |
| 442 CHECK((buf_len >= bytes_read_) && (bytes_read_ >= 0)); | 437 CHECK((buf_len >= bytes_read_) && (bytes_read_ >= 0)); |
| 443 memcpy(buf->data(), read_buffer_->data(), bytes_read_); | 438 memcpy(buf->data(), read_buffer_->data(), bytes_read_); |
| 444 } | 439 } |
| 445 } | 440 } |
| 446 | 441 |
| 447 void BufferedResourceHandler::OnPluginsLoaded( | 442 void BufferedResourceHandler::OnPluginsLoaded( |
| 448 const std::vector<webkit::WebPluginInfo>& plugins) { | 443 const std::vector<webkit::WebPluginInfo>& plugins) { |
| 444 bool needs_resume = deferred_waiting_for_plugins_; |
| 445 deferred_waiting_for_plugins_ = false; |
| 446 |
| 449 wait_for_plugins_ = false; | 447 wait_for_plugins_ = false; |
| 450 if (!request_) | 448 if (!request_) |
| 451 return; | 449 return; |
| 452 | 450 |
| 453 ResourceRequestInfoImpl* info = | 451 ResourceRequestInfoImpl* info = |
| 454 ResourceRequestInfoImpl::ForRequest(request_); | 452 ResourceRequestInfoImpl::ForRequest(request_); |
| 455 int child_id = info->GetChildID(); | 453 int child_id = info->GetChildID(); |
| 456 int request_id = info->GetRequestID(); | 454 int request_id = info->GetRequestID(); |
| 457 | 455 |
| 458 host_->PauseRequest(child_id, request_id, false); | 456 bool defer = false; |
| 459 if (!CompleteResponseStarted(request_id)) | 457 if (!CompleteResponseStarted(request_id, &defer)) { |
| 460 host_->CancelRequest(child_id, request_id, false); | 458 host_->CancelRequest(child_id, request_id, false); |
| 459 } else if (!defer && needs_resume) { |
| 460 host_->ResumeDeferredRequest(child_id, request_id); |
| 461 } |
| 461 } | 462 } |
| 462 | 463 |
| 463 } // namespace content | 464 } // namespace content |
| OLD | NEW |