| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 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/child/web_url_loader_impl.h" | 5 #include "content/child/web_url_loader_impl.h" |
| 6 | 6 |
| 7 #include <stdint.h> | 7 #include <stdint.h> |
| 8 #include <algorithm> | 8 #include <algorithm> |
| 9 #include <string> | 9 #include <string> |
| 10 #include <utility> | 10 #include <utility> |
| 11 #include <vector> | 11 #include <vector> |
| 12 | 12 |
| 13 #include "base/bind.h" | 13 #include "base/bind.h" |
| 14 #include "base/files/file_path.h" | 14 #include "base/files/file_path.h" |
| 15 #include "base/logging.h" | 15 #include "base/logging.h" |
| 16 #include "base/memory/scoped_ptr.h" | 16 #include "base/memory/scoped_ptr.h" |
| 17 #include "base/single_thread_task_runner.h" | 17 #include "base/single_thread_task_runner.h" |
| 18 #include "base/strings/string_util.h" | 18 #include "base/strings/string_util.h" |
| 19 #include "base/time/time.h" | 19 #include "base/time/time.h" |
| 20 #include "build/build_config.h" | 20 #include "build/build_config.h" |
| 21 #include "components/mime_util/mime_util.h" | 21 #include "components/mime_util/mime_util.h" |
| 22 #include "components/scheduler/child/web_task_runner_impl.h" | 22 #include "components/scheduler/child/web_task_runner_impl.h" |
| 23 #include "content/child/child_thread_impl.h" | 23 #include "content/child/child_thread_impl.h" |
| 24 #include "content/child/ftp_directory_listing_response_delegate.h" | 24 #include "content/child/ftp_directory_listing_response_delegate.h" |
| 25 #include "content/child/multipart_response_delegate.h" | |
| 26 #include "content/child/request_extra_data.h" | 25 #include "content/child/request_extra_data.h" |
| 27 #include "content/child/request_info.h" | 26 #include "content/child/request_info.h" |
| 28 #include "content/child/resource_dispatcher.h" | 27 #include "content/child/resource_dispatcher.h" |
| 29 #include "content/child/shared_memory_data_consumer_handle.h" | 28 #include "content/child/shared_memory_data_consumer_handle.h" |
| 30 #include "content/child/sync_load_response.h" | 29 #include "content/child/sync_load_response.h" |
| 31 #include "content/child/web_url_request_util.h" | 30 #include "content/child/web_url_request_util.h" |
| 32 #include "content/child/weburlresponse_extradata_impl.h" | 31 #include "content/child/weburlresponse_extradata_impl.h" |
| 33 #include "content/common/resource_messages.h" | 32 #include "content/common/resource_messages.h" |
| 34 #include "content/common/resource_request_body.h" | 33 #include "content/common/resource_request_body.h" |
| 35 #include "content/common/service_worker/service_worker_types.h" | 34 #include "content/common/service_worker/service_worker_types.h" |
| (...skipping 313 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 349 bool CanHandleDataURLRequestLocally() const; | 348 bool CanHandleDataURLRequestLocally() const; |
| 350 void HandleDataURL(); | 349 void HandleDataURL(); |
| 351 | 350 |
| 352 WebURLLoaderImpl* loader_; | 351 WebURLLoaderImpl* loader_; |
| 353 WebURLRequest request_; | 352 WebURLRequest request_; |
| 354 WebURLLoaderClient* client_; | 353 WebURLLoaderClient* client_; |
| 355 ResourceDispatcher* resource_dispatcher_; | 354 ResourceDispatcher* resource_dispatcher_; |
| 356 scoped_ptr<blink::WebTaskRunner> web_task_runner_; | 355 scoped_ptr<blink::WebTaskRunner> web_task_runner_; |
| 357 WebReferrerPolicy referrer_policy_; | 356 WebReferrerPolicy referrer_policy_; |
| 358 scoped_ptr<FtpDirectoryListingResponseDelegate> ftp_listing_delegate_; | 357 scoped_ptr<FtpDirectoryListingResponseDelegate> ftp_listing_delegate_; |
| 359 scoped_ptr<MultipartResponseDelegate> multipart_delegate_; | |
| 360 scoped_ptr<StreamOverrideParameters> stream_override_; | 358 scoped_ptr<StreamOverrideParameters> stream_override_; |
| 361 scoped_ptr<SharedMemoryDataConsumerHandle::Writer> body_stream_writer_; | 359 scoped_ptr<SharedMemoryDataConsumerHandle::Writer> body_stream_writer_; |
| 362 enum DeferState {NOT_DEFERRING, SHOULD_DEFER, DEFERRED_DATA}; | 360 enum DeferState {NOT_DEFERRING, SHOULD_DEFER, DEFERRED_DATA}; |
| 363 DeferState defers_loading_; | 361 DeferState defers_loading_; |
| 364 int request_id_; | 362 int request_id_; |
| 365 }; | 363 }; |
| 366 | 364 |
| 367 // A thin wrapper class for Context to ensure its lifetime while it is | 365 // A thin wrapper class for Context to ensure its lifetime while it is |
| 368 // handling IPC messages coming from ResourceDispatcher. Owns one ref to | 366 // handling IPC messages coming from ResourceDispatcher. Owns one ref to |
| 369 // Context and held by ResourceDispatcher. | 367 // Context and held by ResourceDispatcher. |
| (...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 408 void WebURLLoaderImpl::Context::Cancel() { | 406 void WebURLLoaderImpl::Context::Cancel() { |
| 409 if (resource_dispatcher_ && // NULL in unittest. | 407 if (resource_dispatcher_ && // NULL in unittest. |
| 410 request_id_ != -1) { | 408 request_id_ != -1) { |
| 411 resource_dispatcher_->Cancel(request_id_); | 409 resource_dispatcher_->Cancel(request_id_); |
| 412 request_id_ = -1; | 410 request_id_ = -1; |
| 413 } | 411 } |
| 414 | 412 |
| 415 if (body_stream_writer_) | 413 if (body_stream_writer_) |
| 416 body_stream_writer_->Fail(); | 414 body_stream_writer_->Fail(); |
| 417 | 415 |
| 418 // Ensure that we do not notify the multipart delegate anymore as it has | 416 // Ensure that we do not notify the delegate anymore as it has |
| 419 // its own pointer to the client. | 417 // its own pointer to the client. |
| 420 if (multipart_delegate_) | |
| 421 multipart_delegate_->Cancel(); | |
| 422 // Ditto for the ftp delegate. | |
| 423 if (ftp_listing_delegate_) | 418 if (ftp_listing_delegate_) |
| 424 ftp_listing_delegate_->Cancel(); | 419 ftp_listing_delegate_->Cancel(); |
| 425 | 420 |
| 426 // Do not make any further calls to the client. | 421 // Do not make any further calls to the client. |
| 427 client_ = NULL; | 422 client_ = NULL; |
| 428 loader_ = NULL; | 423 loader_ = NULL; |
| 429 } | 424 } |
| 430 | 425 |
| 431 void WebURLLoaderImpl::Context::SetDefersLoading(bool value) { | 426 void WebURLLoaderImpl::Context::SetDefersLoading(bool value) { |
| 432 if (request_id_ != -1) | 427 if (request_id_ != -1) |
| (...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 638 | 633 |
| 639 if (info.mime_type == "text/vnd.chromium.ftp-dir") { | 634 if (info.mime_type == "text/vnd.chromium.ftp-dir") { |
| 640 if (show_raw_listing) { | 635 if (show_raw_listing) { |
| 641 // Set the MIME type to plain text to prevent any active content. | 636 // Set the MIME type to plain text to prevent any active content. |
| 642 response.setMIMEType("text/plain"); | 637 response.setMIMEType("text/plain"); |
| 643 } else { | 638 } else { |
| 644 // We're going to produce a parsed listing in HTML. | 639 // We're going to produce a parsed listing in HTML. |
| 645 response.setMIMEType("text/html"); | 640 response.setMIMEType("text/html"); |
| 646 } | 641 } |
| 647 } | 642 } |
| 643 if (info.headers.get() && info.mime_type == "multipart/x-mixed-replace") { |
| 644 std::string content_type; |
| 645 info.headers->EnumerateHeader(NULL, "content-type", &content_type); |
| 646 |
| 647 std::string mime_type; |
| 648 std::string charset; |
| 649 bool had_charset = false; |
| 650 std::string boundary; |
| 651 net::HttpUtil::ParseContentType(content_type, &mime_type, &charset, |
| 652 &had_charset, &boundary); |
| 653 base::TrimString(boundary, " \"", &boundary); |
| 654 response.setMultipartBoundary(boundary.data(), boundary.size()); |
| 655 } |
| 648 | 656 |
| 649 if (request_.useStreamOnResponse()) { | 657 if (request_.useStreamOnResponse()) { |
| 650 SharedMemoryDataConsumerHandle::BackpressureMode mode = | 658 SharedMemoryDataConsumerHandle::BackpressureMode mode = |
| 651 SharedMemoryDataConsumerHandle::kDoNotApplyBackpressure; | 659 SharedMemoryDataConsumerHandle::kDoNotApplyBackpressure; |
| 652 if (info.headers && | 660 if (info.headers && |
| 653 info.headers->HasHeaderValue("Cache-Control", "no-store")) { | 661 info.headers->HasHeaderValue("Cache-Control", "no-store")) { |
| 654 mode = SharedMemoryDataConsumerHandle::kApplyBackpressure; | 662 mode = SharedMemoryDataConsumerHandle::kApplyBackpressure; |
| 655 } | 663 } |
| 656 | 664 |
| 657 auto read_handle = make_scoped_ptr(new SharedMemoryDataConsumerHandle( | 665 auto read_handle = make_scoped_ptr(new SharedMemoryDataConsumerHandle( |
| (...skipping 13 matching lines...) Expand all Loading... |
| 671 } else { | 679 } else { |
| 672 client_->didReceiveResponse(loader_, response); | 680 client_->didReceiveResponse(loader_, response); |
| 673 } | 681 } |
| 674 | 682 |
| 675 // We may have been cancelled after didReceiveResponse, which would leave us | 683 // We may have been cancelled after didReceiveResponse, which would leave us |
| 676 // without a client and therefore without much need to do further handling. | 684 // without a client and therefore without much need to do further handling. |
| 677 if (!client_) | 685 if (!client_) |
| 678 return; | 686 return; |
| 679 | 687 |
| 680 DCHECK(!ftp_listing_delegate_.get()); | 688 DCHECK(!ftp_listing_delegate_.get()); |
| 681 DCHECK(!multipart_delegate_.get()); | 689 if (info.mime_type == "text/vnd.chromium.ftp-dir" && !show_raw_listing) { |
| 682 if (info.headers.get() && info.mime_type == "multipart/x-mixed-replace") { | |
| 683 std::string content_type; | |
| 684 info.headers->EnumerateHeader(NULL, "content-type", &content_type); | |
| 685 | |
| 686 std::string mime_type; | |
| 687 std::string charset; | |
| 688 bool had_charset = false; | |
| 689 std::string boundary; | |
| 690 net::HttpUtil::ParseContentType(content_type, &mime_type, &charset, | |
| 691 &had_charset, &boundary); | |
| 692 base::TrimString(boundary, " \"", &boundary); | |
| 693 | |
| 694 // If there's no boundary, just handle the request normally. In the gecko | |
| 695 // code, nsMultiMixedConv::OnStartRequest throws an exception. | |
| 696 if (!boundary.empty()) { | |
| 697 multipart_delegate_.reset( | |
| 698 new MultipartResponseDelegate(client_, loader_, response, boundary)); | |
| 699 } | |
| 700 } else if (info.mime_type == "text/vnd.chromium.ftp-dir" && | |
| 701 !show_raw_listing) { | |
| 702 ftp_listing_delegate_.reset( | 690 ftp_listing_delegate_.reset( |
| 703 new FtpDirectoryListingResponseDelegate(client_, loader_, response)); | 691 new FtpDirectoryListingResponseDelegate(client_, loader_, response)); |
| 704 } | 692 } |
| 705 } | 693 } |
| 706 | 694 |
| 707 void WebURLLoaderImpl::Context::OnDownloadedData(int len, | 695 void WebURLLoaderImpl::Context::OnDownloadedData(int len, |
| 708 int encoded_data_length) { | 696 int encoded_data_length) { |
| 709 if (client_) | 697 if (client_) |
| 710 client_->didDownloadData(loader_, len, encoded_data_length); | 698 client_->didDownloadData(loader_, len, encoded_data_length); |
| 711 } | 699 } |
| 712 | 700 |
| 713 void WebURLLoaderImpl::Context::OnReceivedData(scoped_ptr<ReceivedData> data) { | 701 void WebURLLoaderImpl::Context::OnReceivedData(scoped_ptr<ReceivedData> data) { |
| 714 const char* payload = data->payload(); | 702 const char* payload = data->payload(); |
| 715 int data_length = data->length(); | 703 int data_length = data->length(); |
| 716 int encoded_data_length = data->encoded_length(); | 704 int encoded_data_length = data->encoded_length(); |
| 717 if (!client_) | 705 if (!client_) |
| 718 return; | 706 return; |
| 719 | 707 |
| 720 if (ftp_listing_delegate_) { | 708 if (ftp_listing_delegate_) { |
| 721 // The FTP listing delegate will make the appropriate calls to | 709 // The FTP listing delegate will make the appropriate calls to |
| 722 // client_->didReceiveData and client_->didReceiveResponse. | 710 // client_->didReceiveData and client_->didReceiveResponse. |
| 723 ftp_listing_delegate_->OnReceivedData(payload, data_length); | 711 ftp_listing_delegate_->OnReceivedData(payload, data_length); |
| 724 } else if (multipart_delegate_) { | |
| 725 // The multipart delegate will make the appropriate calls to | |
| 726 // client_->didReceiveData and client_->didReceiveResponse. | |
| 727 multipart_delegate_->OnReceivedData(payload, data_length, | |
| 728 encoded_data_length); | |
| 729 } else { | 712 } else { |
| 730 // We dispatch the data even when |useStreamOnResponse()| is set, in order | 713 // We dispatch the data even when |useStreamOnResponse()| is set, in order |
| 731 // to make Devtools work. | 714 // to make Devtools work. |
| 732 client_->didReceiveData(loader_, payload, data_length, encoded_data_length); | 715 client_->didReceiveData(loader_, payload, data_length, encoded_data_length); |
| 733 | 716 |
| 734 if (request_.useStreamOnResponse()) { | 717 if (request_.useStreamOnResponse()) { |
| 735 // We don't support ftp_listening_delegate_ and multipart_delegate_ for | 718 // We don't support ftp_listening_delegate_ for now. |
| 736 // now. | 719 // TODO(yhirano): Support ftp listening. |
| 737 // TODO(yhirano): Support ftp listening and multipart. | |
| 738 body_stream_writer_->AddData(std::move(data)); | 720 body_stream_writer_->AddData(std::move(data)); |
| 739 } | 721 } |
| 740 } | 722 } |
| 741 } | 723 } |
| 742 | 724 |
| 743 void WebURLLoaderImpl::Context::OnReceivedCachedMetadata( | 725 void WebURLLoaderImpl::Context::OnReceivedCachedMetadata( |
| 744 const char* data, int len) { | 726 const char* data, int len) { |
| 745 if (client_) | 727 if (client_) |
| 746 client_->didReceiveCachedMetadata(loader_, data, len); | 728 client_->didReceiveCachedMetadata(loader_, data, len); |
| 747 } | 729 } |
| 748 | 730 |
| 749 void WebURLLoaderImpl::Context::OnCompletedRequest( | 731 void WebURLLoaderImpl::Context::OnCompletedRequest( |
| 750 int error_code, | 732 int error_code, |
| 751 bool was_ignored_by_handler, | 733 bool was_ignored_by_handler, |
| 752 bool stale_copy_in_cache, | 734 bool stale_copy_in_cache, |
| 753 const std::string& security_info, | 735 const std::string& security_info, |
| 754 const base::TimeTicks& completion_time, | 736 const base::TimeTicks& completion_time, |
| 755 int64_t total_transfer_size) { | 737 int64_t total_transfer_size) { |
| 756 if (ftp_listing_delegate_) { | 738 if (ftp_listing_delegate_) { |
| 757 ftp_listing_delegate_->OnCompletedRequest(); | 739 ftp_listing_delegate_->OnCompletedRequest(); |
| 758 ftp_listing_delegate_.reset(NULL); | 740 ftp_listing_delegate_.reset(NULL); |
| 759 } else if (multipart_delegate_) { | |
| 760 multipart_delegate_->OnCompletedRequest(); | |
| 761 multipart_delegate_.reset(NULL); | |
| 762 } | 741 } |
| 763 | 742 |
| 764 if (body_stream_writer_ && error_code != net::OK) | 743 if (body_stream_writer_ && error_code != net::OK) |
| 765 body_stream_writer_->Fail(); | 744 body_stream_writer_->Fail(); |
| 766 body_stream_writer_.reset(); | 745 body_stream_writer_.reset(); |
| 767 | 746 |
| 768 if (client_) { | 747 if (client_) { |
| 769 if (error_code != net::OK) { | 748 if (error_code != net::OK) { |
| 770 client_->didFail( | 749 client_->didFail( |
| 771 loader_, | 750 loader_, |
| (...skipping 12 matching lines...) Expand all Loading... |
| 784 DCHECK_LT(request_id_, 0); | 763 DCHECK_LT(request_id_, 0); |
| 785 } | 764 } |
| 786 | 765 |
| 787 void WebURLLoaderImpl::Context::CancelBodyStreaming() { | 766 void WebURLLoaderImpl::Context::CancelBodyStreaming() { |
| 788 scoped_refptr<Context> protect(this); | 767 scoped_refptr<Context> protect(this); |
| 789 | 768 |
| 790 // Notify renderer clients that the request is canceled. | 769 // Notify renderer clients that the request is canceled. |
| 791 if (ftp_listing_delegate_) { | 770 if (ftp_listing_delegate_) { |
| 792 ftp_listing_delegate_->OnCompletedRequest(); | 771 ftp_listing_delegate_->OnCompletedRequest(); |
| 793 ftp_listing_delegate_.reset(NULL); | 772 ftp_listing_delegate_.reset(NULL); |
| 794 } else if (multipart_delegate_) { | |
| 795 multipart_delegate_->OnCompletedRequest(); | |
| 796 multipart_delegate_.reset(NULL); | |
| 797 } | 773 } |
| 798 | 774 |
| 799 if (body_stream_writer_) { | 775 if (body_stream_writer_) { |
| 800 body_stream_writer_->Fail(); | 776 body_stream_writer_->Fail(); |
| 801 body_stream_writer_.reset(); | 777 body_stream_writer_.reset(); |
| 802 } | 778 } |
| 803 if (client_) { | 779 if (client_) { |
| 804 // TODO(yhirano): Set |stale_copy_in_cache| appropriately if possible. | 780 // TODO(yhirano): Set |stale_copy_in_cache| appropriately if possible. |
| 805 client_->didFail( | 781 client_->didFail( |
| 806 loader_, CreateWebURLError(request_.url(), false, net::ERR_ABORTED)); | 782 loader_, CreateWebURLError(request_.url(), false, net::ERR_ABORTED)); |
| (...skipping 389 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1196 response->clearHTTPHeaderField(webStringName); | 1172 response->clearHTTPHeaderField(webStringName); |
| 1197 while (response_headers->EnumerateHeader(&iterator, name, &value)) { | 1173 while (response_headers->EnumerateHeader(&iterator, name, &value)) { |
| 1198 response->addHTTPHeaderField(webStringName, | 1174 response->addHTTPHeaderField(webStringName, |
| 1199 WebString::fromLatin1(value)); | 1175 WebString::fromLatin1(value)); |
| 1200 } | 1176 } |
| 1201 } | 1177 } |
| 1202 return true; | 1178 return true; |
| 1203 } | 1179 } |
| 1204 | 1180 |
| 1205 } // namespace content | 1181 } // namespace content |
| OLD | NEW |