| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/test/url_request/url_request_failed_job.h" |
| |
| #include "base/bind.h" |
| #include "base/location.h" |
| #include "base/logging.h" |
| #include "base/macros.h" |
| #include "base/single_thread_task_runner.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/threading/thread_task_runner_handle.h" |
| #include "net/base/net_errors.h" |
| #include "net/base/url_util.h" |
| #include "net/http/http_response_headers.h" |
| #include "net/url_request/url_request.h" |
| #include "net/url_request/url_request_filter.h" |
| #include "net/url_request/url_request_interceptor.h" |
| |
| namespace net { |
| |
| namespace { |
| |
| const char kMockHostname[] = "mock.failed.request"; |
| |
| // String names of failure phases matching FailurePhase enum. |
| const char* kFailurePhase[]{ |
| "start", // START |
| "readsync", // READ_SYNC |
| "readasync", // READ_ASYNC |
| }; |
| |
| static_assert(arraysize(kFailurePhase) == |
| URLRequestFailedJob::FailurePhase::MAX_FAILURE_PHASE, |
| "kFailurePhase must match FailurePhase enum"); |
| |
| class MockJobInterceptor : public URLRequestInterceptor { |
| public: |
| MockJobInterceptor() = default; |
| ~MockJobInterceptor() override = default; |
| |
| // URLRequestJobFactory::ProtocolHandler implementation: |
| URLRequestJob* MaybeInterceptRequest( |
| URLRequest* request, |
| NetworkDelegate* network_delegate) const override { |
| int net_error = OK; |
| URLRequestFailedJob::FailurePhase phase = |
| URLRequestFailedJob::FailurePhase::MAX_FAILURE_PHASE; |
| for (size_t i = 0; i < arraysize(kFailurePhase); i++) { |
| std::string phase_error_string; |
| if (GetValueForKeyInQuery(request->url(), kFailurePhase[i], |
| &phase_error_string)) { |
| if (base::StringToInt(phase_error_string, &net_error)) { |
| phase = static_cast<URLRequestFailedJob::FailurePhase>(i); |
| break; |
| } |
| } |
| } |
| return new URLRequestFailedJob(request, network_delegate, phase, net_error); |
| } |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(MockJobInterceptor); |
| }; |
| |
| GURL GetMockUrl(const std::string& scheme, |
| const std::string& hostname, |
| URLRequestFailedJob::FailurePhase phase, |
| int net_error) { |
| CHECK_GE(phase, URLRequestFailedJob::FailurePhase::START); |
| CHECK_LE(phase, URLRequestFailedJob::FailurePhase::READ_ASYNC); |
| CHECK_LT(net_error, OK); |
| return GURL(scheme + "://" + hostname + "/error?" + kFailurePhase[phase] + |
| "=" + base::IntToString(net_error)); |
| } |
| |
| } // namespace |
| |
| URLRequestFailedJob::URLRequestFailedJob(URLRequest* request, |
| NetworkDelegate* network_delegate, |
| FailurePhase phase, |
| int net_error) |
| : URLRequestJob(request, network_delegate), |
| phase_(phase), |
| net_error_(net_error), |
| total_received_bytes_(0), |
| weak_factory_(this) { |
| CHECK_GE(phase, URLRequestFailedJob::FailurePhase::START); |
| CHECK_LE(phase, URLRequestFailedJob::FailurePhase::READ_ASYNC); |
| CHECK_LT(net_error, OK); |
| } |
| |
| URLRequestFailedJob::URLRequestFailedJob(URLRequest* request, |
| NetworkDelegate* network_delegate, |
| int net_error) |
| : URLRequestFailedJob(request, network_delegate, START, net_error) { |
| } |
| |
| void URLRequestFailedJob::Start() { |
| base::ThreadTaskRunnerHandle::Get()->PostTask( |
| FROM_HERE, |
| base::Bind(&URLRequestFailedJob::StartAsync, weak_factory_.GetWeakPtr())); |
| } |
| |
| int URLRequestFailedJob::ReadRawData(IOBuffer* buf, int buf_size) { |
| CHECK(phase_ == READ_SYNC || phase_ == READ_ASYNC); |
| if (net_error_ == ERR_IO_PENDING || phase_ == READ_SYNC) |
| return net_error_; |
| |
| base::ThreadTaskRunnerHandle::Get()->PostTask( |
| FROM_HERE, base::Bind(&URLRequestFailedJob::ReadRawDataComplete, |
| weak_factory_.GetWeakPtr(), net_error_)); |
| return ERR_IO_PENDING; |
| } |
| |
| void URLRequestFailedJob::GetResponseInfo(HttpResponseInfo* info) { |
| *info = response_info_; |
| } |
| |
| void URLRequestFailedJob::PopulateNetErrorDetails( |
| NetErrorDetails* details) const { |
| if (net_error_ == ERR_QUIC_PROTOCOL_ERROR) { |
| details->quic_connection_error = quic::QUIC_INTERNAL_ERROR; |
| } else if (net_error_ == ERR_NETWORK_CHANGED) { |
| details->quic_connection_error = |
| quic::QUIC_CONNECTION_MIGRATION_NO_NEW_NETWORK; |
| } |
| } |
| |
| int64_t URLRequestFailedJob::GetTotalReceivedBytes() const { |
| return total_received_bytes_; |
| } |
| |
| // static |
| void URLRequestFailedJob::AddUrlHandler() { |
| return AddUrlHandlerForHostname(kMockHostname); |
| } |
| |
| // static |
| void URLRequestFailedJob::AddUrlHandlerForHostname( |
| const std::string& hostname) { |
| URLRequestFilter* filter = URLRequestFilter::GetInstance(); |
| // Add |hostname| to URLRequestFilter for HTTP and HTTPS. |
| filter->AddHostnameInterceptor( |
| "http", hostname, |
| std::unique_ptr<URLRequestInterceptor>(new MockJobInterceptor())); |
| filter->AddHostnameInterceptor( |
| "https", hostname, |
| std::unique_ptr<URLRequestInterceptor>(new MockJobInterceptor())); |
| } |
| |
| // static |
| GURL URLRequestFailedJob::GetMockHttpUrl(int net_error) { |
| return GetMockHttpUrlForHostname(net_error, kMockHostname); |
| } |
| |
| // static |
| GURL URLRequestFailedJob::GetMockHttpsUrl(int net_error) { |
| return GetMockHttpsUrlForHostname(net_error, kMockHostname); |
| } |
| |
| // static |
| GURL URLRequestFailedJob::GetMockHttpUrlWithFailurePhase(FailurePhase phase, |
| int net_error) { |
| return GetMockUrl("http", kMockHostname, phase, net_error); |
| } |
| |
| // static |
| GURL URLRequestFailedJob::GetMockHttpUrlForHostname( |
| int net_error, |
| const std::string& hostname) { |
| return GetMockUrl("http", hostname, START, net_error); |
| } |
| |
| // static |
| GURL URLRequestFailedJob::GetMockHttpsUrlForHostname( |
| int net_error, |
| const std::string& hostname) { |
| return GetMockUrl("https", hostname, START, net_error); |
| } |
| |
| URLRequestFailedJob::~URLRequestFailedJob() = default; |
| |
| void URLRequestFailedJob::StartAsync() { |
| if (phase_ == START) { |
| if (net_error_ != ERR_IO_PENDING) { |
| NotifyStartError(URLRequestStatus(URLRequestStatus::FAILED, net_error_)); |
| return; |
| } |
| return; |
| } |
| const std::string headers = "HTTP/1.1 200 OK"; |
| response_info_.headers = new net::HttpResponseHeaders(headers); |
| total_received_bytes_ = headers.size(); |
| NotifyHeadersComplete(); |
| } |
| |
| } // namespace net |