blob: 20979035a9a76290474402fd0552bc07ab6732f3 [file] [log] [blame]
// 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/url_request/url_fetcher_core.h"
#include "base/bind.h"
#include "base/file_util_proxy.h"
#include "base/logging.h"
#include "base/single_thread_task_runner.h"
#include "base/metrics/histogram.h"
#include "base/stl_util.h"
#include "base/thread_task_runner_handle.h"
#include "base/tracked_objects.h"
#include "net/base/io_buffer.h"
#include "net/base/load_flags.h"
#include "net/base/net_errors.h"
#include "net/base/upload_bytes_element_reader.h"
#include "net/base/upload_data_stream.h"
#include "net/http/http_response_headers.h"
#include "net/url_request/url_fetcher_delegate.h"
#include "net/url_request/url_request_context.h"
#include "net/url_request/url_request_context_getter.h"
#include "net/url_request/url_request_throttler_manager.h"
namespace {
const int kBufferSize = 16384;
const int kDownloadCacheSize = 65536;
const int kUploadProgressTimerInterval = 100;
bool g_interception_enabled = false;
bool g_ignore_certificate_requests = false;
} // namespace
namespace net {
// URLFetcherCore::Registry ---------------------------------------------------
URLFetcherCore::Registry::Registry() {}
URLFetcherCore::Registry::~Registry() {}
void URLFetcherCore::Registry::AddURLFetcherCore(URLFetcherCore* core) {
DCHECK(!ContainsKey(fetchers_, core));
fetchers_.insert(core);
}
void URLFetcherCore::Registry::RemoveURLFetcherCore(URLFetcherCore* core) {
DCHECK(ContainsKey(fetchers_, core));
fetchers_.erase(core);
}
void URLFetcherCore::Registry::CancelAll() {
while (!fetchers_.empty())
(*fetchers_.begin())->CancelURLRequest();
}
// URLFetcherCore::FileWriter -------------------------------------------------
URLFetcherCore::FileWriter::FileWriter(
URLFetcherCore* core,
scoped_refptr<base::TaskRunner> file_task_runner)
: core_(core),
error_code_(base::PLATFORM_FILE_OK),
ALLOW_THIS_IN_INITIALIZER_LIST(weak_factory_(this)),
file_task_runner_(file_task_runner),
file_handle_(base::kInvalidPlatformFileValue) {
}
URLFetcherCore::FileWriter::~FileWriter() {
CloseAndDeleteFile();
}
void URLFetcherCore::FileWriter::CreateFileAtPath(
const FilePath& file_path) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
DCHECK(file_task_runner_.get());
base::FileUtilProxy::CreateOrOpen(
file_task_runner_,
file_path,
base::PLATFORM_FILE_CREATE_ALWAYS | base::PLATFORM_FILE_WRITE,
base::Bind(&URLFetcherCore::FileWriter::DidCreateFile,
weak_factory_.GetWeakPtr(),
file_path));
}
void URLFetcherCore::FileWriter::CreateTempFile() {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
DCHECK(file_task_runner_.get());
base::FileUtilProxy::CreateTemporary(
file_task_runner_,
0, // No additional file flags.
base::Bind(&URLFetcherCore::FileWriter::DidCreateTempFile,
weak_factory_.GetWeakPtr()));
}
void URLFetcherCore::FileWriter::WriteBuffer(int num_bytes) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
// Start writing to the file by setting the initial state
// of |pending_bytes_| and |buffer_offset_| to indicate that the
// entire buffer has not yet been written.
pending_bytes_ = num_bytes;
buffer_offset_ = 0;
ContinueWrite(base::PLATFORM_FILE_OK, 0);
}
void URLFetcherCore::FileWriter::ContinueWrite(
base::PlatformFileError error_code,
int bytes_written) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (file_handle_ == base::kInvalidPlatformFileValue) {
// While a write was being done on the file thread, a request
// to close or disown the file occured on the IO thread. At
// this point a request to close the file is pending on the
// file thread.
return;
}
// Every code path that resets |core_->request_| should reset
// |core->file_writer_| or cause the file writer to disown the file. In the
// former case, this callback can not be called, because the weak pointer to
// |this| will be NULL. In the latter case, the check of |file_handle_| at the
// start of this method ensures that we can not reach this point.
CHECK(core_->request_.get());
if (base::PLATFORM_FILE_OK != error_code) {
error_code_ = error_code;
CloseAndDeleteFile();
core_->delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::InformDelegateFetchIsComplete, core_));
return;
}
total_bytes_written_ += bytes_written;
buffer_offset_ += bytes_written;
pending_bytes_ -= bytes_written;
if (pending_bytes_ > 0) {
base::FileUtilProxy::Write(
file_task_runner_, file_handle_,
total_bytes_written_, // Append to the end
(core_->buffer_->data() + buffer_offset_), pending_bytes_,
base::Bind(&URLFetcherCore::FileWriter::ContinueWrite,
weak_factory_.GetWeakPtr()));
} else {
// Finished writing core_->buffer_ to the file. Read some more.
core_->ReadResponse();
}
}
void URLFetcherCore::FileWriter::DisownFile() {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
// Disowning is done by the delegate's OnURLFetchComplete method.
// The file should be closed by the time that method is called.
DCHECK(file_handle_ == base::kInvalidPlatformFileValue);
// Forget about any file by reseting the path.
file_path_.clear();
}
void URLFetcherCore::FileWriter::CloseFileAndCompleteRequest() {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (file_handle_ != base::kInvalidPlatformFileValue) {
base::FileUtilProxy::Close(
file_task_runner_, file_handle_,
base::Bind(&URLFetcherCore::FileWriter::DidCloseFile,
weak_factory_.GetWeakPtr()));
file_handle_ = base::kInvalidPlatformFileValue;
}
}
void URLFetcherCore::FileWriter::CloseAndDeleteFile() {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (file_handle_ == base::kInvalidPlatformFileValue) {
DeleteFile(base::PLATFORM_FILE_OK);
return;
}
// Close the file if it is open.
base::FileUtilProxy::Close(
file_task_runner_, file_handle_,
base::Bind(&URLFetcherCore::FileWriter::DeleteFile,
weak_factory_.GetWeakPtr()));
file_handle_ = base::kInvalidPlatformFileValue;
}
void URLFetcherCore::FileWriter::DeleteFile(
base::PlatformFileError error_code) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (file_path_.empty())
return;
base::FileUtilProxy::Delete(
file_task_runner_, file_path_,
false, // No need to recurse, as the path is to a file.
base::FileUtilProxy::StatusCallback());
DisownFile();
}
void URLFetcherCore::FileWriter::DidCreateFile(
const FilePath& file_path,
base::PlatformFileError error_code,
base::PassPlatformFile file_handle,
bool created) {
DidCreateFileInternal(file_path, error_code, file_handle);
}
void URLFetcherCore::FileWriter::DidCreateTempFile(
base::PlatformFileError error_code,
base::PassPlatformFile file_handle,
const FilePath& file_path) {
DidCreateFileInternal(file_path, error_code, file_handle);
}
void URLFetcherCore::FileWriter::DidCreateFileInternal(
const FilePath& file_path,
base::PlatformFileError error_code,
base::PassPlatformFile file_handle) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (base::PLATFORM_FILE_OK != error_code) {
error_code_ = error_code;
CloseAndDeleteFile();
core_->delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::InformDelegateFetchIsComplete, core_));
return;
}
file_path_ = file_path;
file_handle_ = file_handle.ReleaseValue();
total_bytes_written_ = 0;
core_->network_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::StartURLRequestWhenAppropriate, core_));
}
void URLFetcherCore::FileWriter::DidCloseFile(
base::PlatformFileError error_code) {
DCHECK(core_->network_task_runner_->BelongsToCurrentThread());
if (base::PLATFORM_FILE_OK != error_code) {
error_code_ = error_code;
CloseAndDeleteFile();
core_->delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::InformDelegateFetchIsComplete, core_));
return;
}
// If the file was successfully closed, then the URL request is complete.
core_->RetryOrCompleteUrlFetch();
}
// URLFetcherCore -------------------------------------------------------------
// static
base::LazyInstance<URLFetcherCore::Registry>
URLFetcherCore::g_registry = LAZY_INSTANCE_INITIALIZER;
URLFetcherCore::URLFetcherCore(URLFetcher* fetcher,
const GURL& original_url,
URLFetcher::RequestType request_type,
URLFetcherDelegate* d)
: fetcher_(fetcher),
original_url_(original_url),
request_type_(request_type),
delegate_(d),
delegate_task_runner_(
base::ThreadTaskRunnerHandle::Get()),
request_(NULL),
load_flags_(LOAD_NORMAL),
response_code_(URLFetcher::RESPONSE_CODE_INVALID),
buffer_(new IOBuffer(kBufferSize)),
url_request_data_key_(NULL),
was_fetched_via_proxy_(false),
is_chunked_upload_(false),
was_cancelled_(false),
response_destination_(STRING),
stop_on_redirect_(false),
stopped_on_redirect_(false),
automatically_retry_on_5xx_(true),
num_retries_on_5xx_(0),
max_retries_on_5xx_(0),
num_retries_on_network_changes_(0),
max_retries_on_network_changes_(0),
current_upload_bytes_(-1),
current_response_bytes_(0),
total_response_bytes_(-1) {
CHECK(original_url_.is_valid());
}
void URLFetcherCore::Start() {
DCHECK(delegate_task_runner_);
DCHECK(request_context_getter_) << "We need an URLRequestContext!";
if (network_task_runner_) {
DCHECK_EQ(network_task_runner_,
request_context_getter_->GetNetworkTaskRunner());
} else {
network_task_runner_ = request_context_getter_->GetNetworkTaskRunner();
}
DCHECK(network_task_runner_.get()) << "We need an IO task runner";
network_task_runner_->PostTask(
FROM_HERE, base::Bind(&URLFetcherCore::StartOnIOThread, this));
}
void URLFetcherCore::Stop() {
if (delegate_task_runner_) // May be NULL in tests.
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
delegate_ = NULL;
fetcher_ = NULL;
if (!network_task_runner_.get())
return;
if (network_task_runner_->RunsTasksOnCurrentThread()) {
CancelURLRequest();
} else {
network_task_runner_->PostTask(
FROM_HERE, base::Bind(&URLFetcherCore::CancelURLRequest, this));
}
}
void URLFetcherCore::SetUploadData(const std::string& upload_content_type,
const std::string& upload_content) {
DCHECK(!is_chunked_upload_);
upload_content_type_ = upload_content_type;
upload_content_ = upload_content;
}
void URLFetcherCore::SetChunkedUpload(const std::string& content_type) {
DCHECK(is_chunked_upload_ ||
(upload_content_type_.empty() &&
upload_content_.empty()));
upload_content_type_ = content_type;
upload_content_.clear();
is_chunked_upload_ = true;
}
void URLFetcherCore::AppendChunkToUpload(const std::string& content,
bool is_last_chunk) {
DCHECK(delegate_task_runner_);
DCHECK(network_task_runner_.get());
network_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::CompleteAddingUploadDataChunk, this, content,
is_last_chunk));
}
void URLFetcherCore::SetLoadFlags(int load_flags) {
load_flags_ = load_flags;
}
int URLFetcherCore::GetLoadFlags() const {
return load_flags_;
}
void URLFetcherCore::SetReferrer(const std::string& referrer) {
referrer_ = referrer;
}
void URLFetcherCore::SetExtraRequestHeaders(
const std::string& extra_request_headers) {
extra_request_headers_.Clear();
extra_request_headers_.AddHeadersFromString(extra_request_headers);
}
void URLFetcherCore::AddExtraRequestHeader(const std::string& header_line) {
extra_request_headers_.AddHeaderFromString(header_line);
}
void URLFetcherCore::GetExtraRequestHeaders(
HttpRequestHeaders* headers) const {
headers->CopyFrom(extra_request_headers_);
}
void URLFetcherCore::SetRequestContext(
URLRequestContextGetter* request_context_getter) {
DCHECK(!request_context_getter_);
DCHECK(request_context_getter);
request_context_getter_ = request_context_getter;
}
void URLFetcherCore::SetFirstPartyForCookies(
const GURL& first_party_for_cookies) {
DCHECK(first_party_for_cookies_.is_empty());
first_party_for_cookies_ = first_party_for_cookies;
}
void URLFetcherCore::SetURLRequestUserData(
const void* key,
const URLFetcher::CreateDataCallback& create_data_callback) {
DCHECK(key);
DCHECK(!create_data_callback.is_null());
url_request_data_key_ = key;
url_request_create_data_callback_ = create_data_callback;
}
void URLFetcherCore::SetStopOnRedirect(bool stop_on_redirect) {
stop_on_redirect_ = stop_on_redirect;
}
void URLFetcherCore::SetAutomaticallyRetryOn5xx(bool retry) {
automatically_retry_on_5xx_ = retry;
}
void URLFetcherCore::SetMaxRetriesOn5xx(int max_retries) {
max_retries_on_5xx_ = max_retries;
}
int URLFetcherCore::GetMaxRetriesOn5xx() const {
return max_retries_on_5xx_;
}
base::TimeDelta URLFetcherCore::GetBackoffDelay() const {
return backoff_delay_;
}
void URLFetcherCore::SetAutomaticallyRetryOnNetworkChanges(int max_retries) {
max_retries_on_network_changes_ = max_retries;
}
void URLFetcherCore::SaveResponseToFileAtPath(
const FilePath& file_path,
scoped_refptr<base::TaskRunner> file_task_runner) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
file_task_runner_ = file_task_runner;
response_destination_ = URLFetcherCore::PERMANENT_FILE;
response_destination_file_path_ = file_path;
}
void URLFetcherCore::SaveResponseToTemporaryFile(
scoped_refptr<base::TaskRunner> file_task_runner) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
file_task_runner_ = file_task_runner;
response_destination_ = URLFetcherCore::TEMP_FILE;
}
#if defined(COBALT)
void URLFetcherCore::DiscardResponse() {
response_destination_ = URLFetcherCore::DISCARD;
}
#endif
HttpResponseHeaders* URLFetcherCore::GetResponseHeaders() const {
return response_headers_;
}
// TODO(panayiotis): socket_address_ is written in the IO thread,
// if this is accessed in the UI thread, this could result in a race.
// Same for response_headers_ above and was_fetched_via_proxy_ below.
HostPortPair URLFetcherCore::GetSocketAddress() const {
return socket_address_;
}
bool URLFetcherCore::WasFetchedViaProxy() const {
return was_fetched_via_proxy_;
}
const GURL& URLFetcherCore::GetOriginalURL() const {
return original_url_;
}
const GURL& URLFetcherCore::GetURL() const {
return url_;
}
const URLRequestStatus& URLFetcherCore::GetStatus() const {
return status_;
}
int URLFetcherCore::GetResponseCode() const {
return response_code_;
}
const ResponseCookies& URLFetcherCore::GetCookies() const {
return cookies_;
}
bool URLFetcherCore::FileErrorOccurred(
base::PlatformFileError* out_error_code) const {
// Can't have a file error if no file is being created or written to.
if (!file_writer_.get())
return false;
base::PlatformFileError error_code = file_writer_->error_code();
if (error_code == base::PLATFORM_FILE_OK)
return false;
*out_error_code = error_code;
return true;
}
void URLFetcherCore::ReceivedContentWasMalformed() {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
if (network_task_runner_.get()) {
network_task_runner_->PostTask(
FROM_HERE, base::Bind(&URLFetcherCore::NotifyMalformedContent, this));
}
}
bool URLFetcherCore::GetResponseAsString(
std::string* out_response_string) const {
if (response_destination_ != URLFetcherCore::STRING)
return false;
*out_response_string = data_;
UMA_HISTOGRAM_MEMORY_KB("UrlFetcher.StringResponseSize",
(data_.length() / 1024));
return true;
}
bool URLFetcherCore::GetResponseAsFilePath(bool take_ownership,
FilePath* out_response_path) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
const bool destination_is_file =
response_destination_ == URLFetcherCore::TEMP_FILE ||
response_destination_ == URLFetcherCore::PERMANENT_FILE;
if (!destination_is_file || !file_writer_.get())
return false;
*out_response_path = file_writer_->file_path();
if (take_ownership) {
network_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::DisownFile, this));
}
return true;
}
void URLFetcherCore::OnReceivedRedirect(URLRequest* request,
const GURL& new_url,
bool* defer_redirect) {
DCHECK_EQ(request, request_.get());
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (stop_on_redirect_) {
stopped_on_redirect_ = true;
url_ = new_url;
response_code_ = request_->GetResponseCode();
was_fetched_via_proxy_ = request_->was_fetched_via_proxy();
request->Cancel();
OnReadCompleted(request, 0);
}
}
void URLFetcherCore::OnResponseStarted(URLRequest* request) {
DCHECK_EQ(request, request_.get());
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (request_->status().is_success()) {
response_code_ = request_->GetResponseCode();
response_headers_ = request_->response_headers();
socket_address_ = request_->GetSocketAddress();
was_fetched_via_proxy_ = request_->was_fetched_via_proxy();
total_response_bytes_ = request_->GetExpectedContentSize();
#if defined(COBALT)
// We update this earlier than OnReadCompleted(), so that the delegate
// can know about it if they call GetURL() in any callback.
if (!stopped_on_redirect_) {
url_ = request_->url();
}
InformDelegateResponseStarted();
#endif // defined(COBALT)
}
ReadResponse();
}
void URLFetcherCore::OnCertificateRequested(
URLRequest* request,
SSLCertRequestInfo* cert_request_info) {
DCHECK_EQ(request, request_.get());
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (g_ignore_certificate_requests) {
request->ContinueWithCertificate(NULL);
} else {
request->Cancel();
}
}
void URLFetcherCore::OnReadCompleted(URLRequest* request,
int bytes_read) {
DCHECK(request == request_);
DCHECK(network_task_runner_->BelongsToCurrentThread());
#if !defined(COBALT)
if (!stopped_on_redirect_) {
url_ = request->url();
}
#endif
URLRequestThrottlerManager* throttler_manager =
request->context()->throttler_manager();
if (throttler_manager) {
url_throttler_entry_ = throttler_manager->RegisterRequestUrl(url_);
}
download_data_cache_.reset();
bool waiting_on_write = false;
do {
if (!request_->status().is_success() || bytes_read <= 0)
break;
current_response_bytes_ += bytes_read;
InformDelegateDownloadDataIfNecessary(bytes_read);
if (!WriteBuffer(bytes_read)) {
// If WriteBuffer() returns false, we have a pending write to
// wait on before reading further.
waiting_on_write = true;
break;
}
} while (request_->Read(buffer_, kBufferSize, &bytes_read));
InformDelegateDownloadData();
const URLRequestStatus status = request_->status();
if (status.is_success())
request_->GetResponseCookies(&cookies_);
// See comments re: HEAD requests in ReadResponse().
if ((!status.is_io_pending() && !waiting_on_write) ||
(request_type_ == URLFetcher::HEAD)) {
status_ = status;
ReleaseRequest();
// If a file is open, close it.
if (file_writer_.get()) {
// If the file is open, close it. After closing the file,
// RetryOrCompleteUrlFetch() will be called.
file_writer_->CloseFileAndCompleteRequest();
} else {
// Otherwise, complete or retry the URL request directly.
RetryOrCompleteUrlFetch();
}
}
}
void URLFetcherCore::CancelAll() {
g_registry.Get().CancelAll();
}
int URLFetcherCore::GetNumFetcherCores() {
return g_registry.Get().size();
}
void URLFetcherCore::SetEnableInterceptionForTests(bool enabled) {
g_interception_enabled = enabled;
}
void URLFetcherCore::SetIgnoreCertificateRequests(bool ignored) {
g_ignore_certificate_requests = ignored;
}
URLFetcherCore::~URLFetcherCore() {
// |request_| should be NULL. If not, it's unsafe to delete it here since we
// may not be on the IO thread.
DCHECK(!request_.get());
}
void URLFetcherCore::StartOnIOThread() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
switch (response_destination_) {
case STRING:
StartURLRequestWhenAppropriate();
break;
case PERMANENT_FILE:
case TEMP_FILE:
DCHECK(file_task_runner_.get())
<< "Need to set the file task runner.";
file_writer_.reset(new FileWriter(this, file_task_runner_));
// If the file is successfully created,
// URLFetcherCore::StartURLRequestWhenAppropriate() will be called.
switch (response_destination_) {
case PERMANENT_FILE:
file_writer_->CreateFileAtPath(response_destination_file_path_);
break;
case TEMP_FILE:
file_writer_->CreateTempFile();
break;
default:
NOTREACHED();
}
break;
#if defined(COBALT)
case DISCARD:
StartURLRequestWhenAppropriate();
break;
#endif
default:
NOTREACHED();
}
}
void URLFetcherCore::StartURLRequest() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (was_cancelled_) {
// Since StartURLRequest() is posted as a *delayed* task, it may
// run after the URLFetcher was already stopped.
return;
}
DCHECK(request_context_getter_);
DCHECK(!request_.get());
g_registry.Get().AddURLFetcherCore(this);
current_response_bytes_ = 0;
request_.reset(request_context_getter_->GetURLRequestContext()->CreateRequest(
original_url_, this));
request_->set_stack_trace(stack_trace_);
int flags = request_->load_flags() | load_flags_;
if (!g_interception_enabled)
flags = flags | LOAD_DISABLE_INTERCEPT;
if (is_chunked_upload_)
request_->EnableChunkedUpload();
request_->set_load_flags(flags);
request_->set_referrer(referrer_);
request_->set_first_party_for_cookies(first_party_for_cookies_.is_empty() ?
original_url_ : first_party_for_cookies_);
if (url_request_data_key_ && !url_request_create_data_callback_.is_null()) {
request_->SetUserData(url_request_data_key_,
url_request_create_data_callback_.Run());
}
switch (request_type_) {
case URLFetcher::GET:
break;
case URLFetcher::POST:
case URLFetcher::PUT:
#if !defined(COBALT)
// Allow not specifying a content type when uploading ArrayBuffer data.
DCHECK(!upload_content_type_.empty());
#endif
request_->set_method(
request_type_ == URLFetcher::POST ? "POST" : "PUT");
if (!upload_content_type_.empty()) {
extra_request_headers_.SetHeader(HttpRequestHeaders::kContentType,
upload_content_type_);
}
if (!upload_content_.empty()) {
scoped_ptr<UploadElementReader> reader(new UploadBytesElementReader(
upload_content_.data(), upload_content_.size()));
request_->set_upload(make_scoped_ptr(
UploadDataStream::CreateWithReader(reader.Pass(), 0)));
}
current_upload_bytes_ = -1;
// TODO(kinaba): http://crbug.com/118103. Implement upload callback in the
// layer and avoid using timer here.
upload_progress_checker_timer_.reset(
new base::RepeatingTimer<URLFetcherCore>());
upload_progress_checker_timer_->Start(
FROM_HERE,
base::TimeDelta::FromMilliseconds(kUploadProgressTimerInterval),
this,
&URLFetcherCore::InformDelegateUploadProgress);
break;
case URLFetcher::HEAD:
request_->set_method("HEAD");
break;
case URLFetcher::DELETE_REQUEST:
request_->set_method("DELETE");
break;
default:
NOTREACHED();
}
if (!extra_request_headers_.IsEmpty())
request_->SetExtraRequestHeaders(extra_request_headers_);
// There might be data left over from a previous request attempt.
data_.clear();
// If we are writing the response to a file, the only caller
// of this function should have created it and not written yet.
DCHECK(!file_writer_.get() || file_writer_->total_bytes_written() == 0);
request_->Start();
}
void URLFetcherCore::StartURLRequestWhenAppropriate() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (was_cancelled_)
return;
DCHECK(request_context_getter_);
int64 delay = 0LL;
if (original_url_throttler_entry_ == NULL) {
URLRequestThrottlerManager* manager =
request_context_getter_->GetURLRequestContext()->throttler_manager();
if (manager) {
original_url_throttler_entry_ =
manager->RegisterRequestUrl(original_url_);
}
}
if (original_url_throttler_entry_ != NULL) {
delay = original_url_throttler_entry_->ReserveSendingTimeForNextRequest(
GetBackoffReleaseTime());
}
if (delay == 0) {
StartURLRequest();
} else {
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
FROM_HERE, base::Bind(&URLFetcherCore::StartURLRequest, this),
base::TimeDelta::FromMilliseconds(delay));
}
}
void URLFetcherCore::CancelURLRequest() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (request_.get()) {
request_->Cancel();
ReleaseRequest();
}
// Release the reference to the request context. There could be multiple
// references to URLFetcher::Core at this point so it may take a while to
// delete the object, but we cannot delay the destruction of the request
// context.
request_context_getter_ = NULL;
first_party_for_cookies_ = GURL();
url_request_data_key_ = NULL;
url_request_create_data_callback_.Reset();
was_cancelled_ = true;
file_writer_.reset();
}
void URLFetcherCore::OnCompletedURLRequest(
base::TimeDelta backoff_delay) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
// Save the status and backoff_delay so that delegates can read it.
if (delegate_) {
backoff_delay_ = backoff_delay;
InformDelegateFetchIsComplete();
}
}
void URLFetcherCore::InformDelegateFetchIsComplete() {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
if (delegate_)
delegate_->OnURLFetchComplete(fetcher_);
}
void URLFetcherCore::NotifyMalformedContent() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (url_throttler_entry_ != NULL) {
int status_code = response_code_;
if (status_code == URLFetcher::RESPONSE_CODE_INVALID) {
// The status code will generally be known by the time clients
// call the |ReceivedContentWasMalformed()| function (which ends up
// calling the current function) but if it's not, we need to assume
// the response was successful so that the total failure count
// used to calculate exponential back-off goes up.
status_code = 200;
}
url_throttler_entry_->ReceivedContentWasMalformed(status_code);
}
}
void URLFetcherCore::RetryOrCompleteUrlFetch() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
base::TimeDelta backoff_delay;
// Checks the response from server.
if (response_code_ >= 500 ||
status_.error() == ERR_TEMPORARILY_THROTTLED) {
// When encountering a server error, we will send the request again
// after backoff time.
++num_retries_on_5xx_;
// Note that backoff_delay may be 0 because (a) the
// URLRequestThrottlerManager and related code does not
// necessarily back off on the first error, (b) it only backs off
// on some of the 5xx status codes, (c) not all URLRequestContexts
// have a throttler manager.
base::TimeTicks backoff_release_time = GetBackoffReleaseTime();
backoff_delay = backoff_release_time - base::TimeTicks::Now();
if (backoff_delay < base::TimeDelta())
backoff_delay = base::TimeDelta();
if (automatically_retry_on_5xx_ &&
num_retries_on_5xx_ <= max_retries_on_5xx_) {
StartOnIOThread();
return;
}
} else {
backoff_delay = base::TimeDelta();
}
// Retry if the request failed due to network changes.
if (status_.error() == ERR_NETWORK_CHANGED &&
num_retries_on_network_changes_ < max_retries_on_network_changes_) {
++num_retries_on_network_changes_;
// Retry soon, after flushing all the current tasks which may include
// further network change observers.
network_task_runner_->PostTask(
FROM_HERE, base::Bind(&URLFetcherCore::StartOnIOThread, this));
return;
}
request_context_getter_ = NULL;
first_party_for_cookies_ = GURL();
url_request_data_key_ = NULL;
url_request_create_data_callback_.Reset();
bool posted = delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::OnCompletedURLRequest, this, backoff_delay));
// If the delegate message loop does not exist any more, then the delegate
// should be gone too.
DCHECK(posted || !delegate_);
}
void URLFetcherCore::ReleaseRequest() {
#if defined(COBALT)
if (upload_progress_checker_timer_) {
// The request may have completed too quickly, before the upload
// progress checker had a chance to run. Force it to run here.
InformDelegateUploadProgress();
}
#endif
upload_progress_checker_timer_.reset();
request_.reset();
g_registry.Get().RemoveURLFetcherCore(this);
}
base::TimeTicks URLFetcherCore::GetBackoffReleaseTime() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (original_url_throttler_entry_) {
base::TimeTicks original_url_backoff =
original_url_throttler_entry_->GetExponentialBackoffReleaseTime();
base::TimeTicks destination_url_backoff;
if (url_throttler_entry_ != NULL &&
original_url_throttler_entry_ != url_throttler_entry_) {
destination_url_backoff =
url_throttler_entry_->GetExponentialBackoffReleaseTime();
}
return original_url_backoff > destination_url_backoff ?
original_url_backoff : destination_url_backoff;
} else {
return base::TimeTicks();
}
}
void URLFetcherCore::CompleteAddingUploadDataChunk(
const std::string& content, bool is_last_chunk) {
if (was_cancelled_) {
// Since CompleteAddingUploadDataChunk() is posted as a *delayed* task, it
// may run after the URLFetcher was already stopped.
return;
}
DCHECK(is_chunked_upload_);
DCHECK(request_.get());
DCHECK(!content.empty());
request_->AppendChunkToUpload(content.data(),
static_cast<int>(content.length()),
is_last_chunk);
}
// Return true if the write was done and reading may continue.
// Return false if the write is pending, and the next read will
// be done later.
bool URLFetcherCore::WriteBuffer(int num_bytes) {
bool write_complete = false;
switch (response_destination_) {
case STRING:
data_.append(buffer_->data(), num_bytes);
write_complete = true;
break;
case PERMANENT_FILE:
case TEMP_FILE:
file_writer_->WriteBuffer(num_bytes);
// WriteBuffer() sends a request the file thread.
// The write is not done yet.
write_complete = false;
break;
#if defined(COBALT)
case DISCARD:
write_complete = true;
break;
#endif
default:
NOTREACHED();
}
return write_complete;
}
void URLFetcherCore::ReadResponse() {
// Some servers may treat HEAD requests as GET requests. To free up the
// network connection as soon as possible, signal that the request has
// completed immediately, without trying to read any data back (all we care
// about is the response code and headers, which we already have).
int bytes_read = 0;
if (request_->status().is_success() &&
(request_type_ != URLFetcher::HEAD))
request_->Read(buffer_, kBufferSize, &bytes_read);
OnReadCompleted(request_.get(), bytes_read);
}
void URLFetcherCore::DisownFile() {
file_writer_->DisownFile();
}
#if defined(COBALT)
void URLFetcherCore::InformDelegateResponseStarted() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
DCHECK(request_);
delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(
&URLFetcherCore::InformDelegateResponseStartedInDelegateThread,
this));
}
void URLFetcherCore::InformDelegateResponseStartedInDelegateThread() {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
if (delegate_) {
delegate_->OnURLFetchResponseStarted(fetcher_);
}
}
#endif // defined(COBALT)
void URLFetcherCore::InformDelegateUploadProgress() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (request_.get()) {
int64 current = request_->GetUploadProgress().position();
if (current_upload_bytes_ != current) {
current_upload_bytes_ = current;
int64 total = -1;
if (!is_chunked_upload_)
total = static_cast<int64>(upload_content_.size());
delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(
&URLFetcherCore::InformDelegateUploadProgressInDelegateThread,
this, current, total));
}
}
}
void URLFetcherCore::InformDelegateUploadProgressInDelegateThread(
int64 current, int64 total) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
if (delegate_)
delegate_->OnURLFetchUploadProgress(fetcher_, current, total);
}
void URLFetcherCore::InformDelegateDownloadDataIfNecessary(int bytes_read) {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (delegate_ && delegate_->ShouldSendDownloadData() && bytes_read != 0) {
if (!download_data_cache_) {
download_data_cache_.reset(new std::string);
download_data_cache_->reserve(kDownloadCacheSize);
}
download_data_cache_->resize(download_data_cache_->size() + bytes_read);
memcpy(&(*download_data_cache_)[download_data_cache_->size() - bytes_read],
buffer_->data(), bytes_read);
if (download_data_cache_->size() >= kDownloadCacheSize) {
delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(
&URLFetcherCore::InformDelegateDownloadDataInDelegateThread, this,
base::Passed(&download_data_cache_)));
}
}
}
void URLFetcherCore::InformDelegateDownloadData() {
DCHECK(network_task_runner_->BelongsToCurrentThread());
if (delegate_ && delegate_->ShouldSendDownloadData() &&
download_data_cache_) {
delegate_task_runner_->PostTask(
FROM_HERE,
base::Bind(&URLFetcherCore::InformDelegateDownloadDataInDelegateThread,
this, base::Passed(&download_data_cache_)));
}
}
void URLFetcherCore::InformDelegateDownloadDataInDelegateThread(
scoped_ptr<std::string> download_data) {
DCHECK(delegate_task_runner_->BelongsToCurrentThread());
if (delegate_) {
delegate_->OnURLFetchDownloadData(fetcher_, download_data.Pass());
}
}
} // namespace net