blob: cc27aef4cacfd18f6c2e6c621c73cb7fe177ef4d [file] [log] [blame]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "third_party/blink/public/common/loader/throttling_url_loader.h"
#include <string_view>
#include <vector>
#include "base/containers/contains.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/metrics/histogram_functions.h"
#include "base/strings/strcat.h"
#include "base/strings/stringprintf.h"
#include "base/task/sequenced_task_runner.h"
#include "base/task/single_thread_task_runner.h"
#include "base/timer/elapsed_timer.h"
#include "base/trace_event/trace_event.h"
#include "net/http/http_response_headers.h"
#include "net/http/http_status_code.h"
#include "net/http/http_util.h"
#include "net/url_request/redirect_info.h"
#include "net/url_request/redirect_util.h"
#include "services/network/public/cpp/cors/cors.h"
#include "services/network/public/cpp/features.h"
#include "services/network/public/cpp/record_ontransfersizeupdate_utils.h"
#include "services/network/public/mojom/early_hints.mojom.h"
#include "services/network/public/mojom/url_response_head.mojom.h"
#include "third_party/blink/public/mojom/origin_trials/origin_trial_feature.mojom-shared.h"
namespace blink {
namespace {
void RemoveModifiedHeadersBeforeMerge(
net::HttpRequestHeaders* modified_headers) {
DCHECK(modified_headers);
modified_headers->RemoveHeader(net::HttpRequestHeaders::kAcceptLanguage);
}
// Merges |removed_headers_B| into |removed_headers_A|.
void MergeRemovedHeaders(std::vector<std::string>* removed_headers_A,
const std::vector<std::string>& removed_headers_B) {
for (auto& header : removed_headers_B) {
if (!base::Contains(*removed_headers_A, header))
removed_headers_A->emplace_back(std::move(header));
}
}
#if DCHECK_IS_ON()
void CheckThrottleWillNotCauseCorsPreflight(
const std::set<std::string>& initial_headers,
const std::set<std::string>& initial_cors_exempt_headers,
const net::HttpRequestHeaders& headers,
const net::HttpRequestHeaders& cors_exempt_headers,
const std::vector<std::string> cors_exempt_header_list) {
// There are many ways for the renderer to cache the list, e.g. for workers,
// and it might have been cached before the renderer receives a message with
// the list. This isn't guaranteed because the caching paths aren't triggered
// by mojo calls that are associated with the method that receives the list.
// Since the renderer just checks to help catch develper bugs, if the list
// isn't received don't DCHECK. Most of the time it will which is all we need
// on bots.
if (cors_exempt_header_list.empty())
return;
base::flat_set<std::string> cors_exempt_header_flat_set(
cors_exempt_header_list);
for (auto& header : headers.GetHeaderVector()) {
if (!base::Contains(initial_headers, header.key) &&
!network::cors::IsCorsSafelistedHeader(header.key, header.value) &&
net::HttpUtil::IsSafeHeader(header.key, header.value)) {
bool is_cors_exempt = cors_exempt_header_flat_set.count(header.key);
NOTREACHED()
<< "Throttle added cors unsafe header " << header.key
<< (is_cors_exempt
? " . Header is cors exempt so should have "
"been added to RequestHeaders::cors_exempt_headers "
"instead of "
"of RequestHeaders::cors_exempt_headers."
: "");
}
}
for (auto& header : cors_exempt_headers.GetHeaderVector()) {
if (cors_exempt_header_flat_set.count(header.key) == 0 &&
!base::Contains(initial_cors_exempt_headers, header.key)) {
NOTREACHED()
<< "Throttle added cors exempt header " << header.key
<< " but it wasn't configured as cors exempt by the browser. See "
"content::StoragePartitionImpl::InitNetworkContext() and "
"content::ContentBrowserClient::ConfigureNetworkContextParams().";
}
}
}
#endif
void RecordHistogram(const std::string& stage,
base::Time start,
const std::string& metric_type) {
base::TimeDelta delta = base::Time::Now() - start;
base::UmaHistogramTimes(
base::StrCat({"Net.URLLoaderThrottle", metric_type, ".", stage}), delta);
}
void RecordDeferTimeHistogram(const std::string& stage,
base::Time start,
const char* throttle_name) {
constexpr char kMetricType[] = "DeferTime";
RecordHistogram(stage, start, kMetricType);
if (throttle_name != nullptr) {
RecordHistogram(base::StrCat({stage, ".", throttle_name}), start,
kMetricType);
}
}
void RecordExecutionTimeHistogram(const std::string& stage, base::Time start) {
RecordHistogram(stage, start, "ExecutionTime");
}
} // namespace
const char ThrottlingURLLoader::kFollowRedirectReason[] = "FollowRedirect";
class ThrottlingURLLoader::ForwardingThrottleDelegate
: public URLLoaderThrottle::Delegate {
public:
ForwardingThrottleDelegate(ThrottlingURLLoader* loader,
URLLoaderThrottle* throttle)
: loader_(loader), throttle_(throttle) {}
ForwardingThrottleDelegate(const ForwardingThrottleDelegate&) = delete;
ForwardingThrottleDelegate& operator=(const ForwardingThrottleDelegate&) =
delete;
~ForwardingThrottleDelegate() override = default;
// URLLoaderThrottle::Delegate:
void CancelWithError(int error_code,
std::string_view custom_reason) override {
CancelWithExtendedError(error_code, 0, custom_reason);
}
void CancelWithExtendedError(int error_code,
int extended_reason_code,
std::string_view custom_reason) override {
if (!loader_)
return;
ScopedDelegateCall scoped_delegate_call(this);
loader_->CancelWithExtendedError(error_code, extended_reason_code,
custom_reason);
}
void Resume() override {
if (!loader_)
return;
ScopedDelegateCall scoped_delegate_call(this);
loader_->StopDeferringForThrottle(throttle_);
}
void UpdateDeferredResponseHead(
network::mojom::URLResponseHeadPtr new_response_head,
mojo::ScopedDataPipeConsumerHandle body) override {
if (!loader_)
return;
ScopedDelegateCall scoped_delegate_call(this);
loader_->UpdateDeferredResponseHead(std::move(new_response_head),
std::move(body));
}
void InterceptResponse(
mojo::PendingRemote<network::mojom::URLLoader> new_loader,
mojo::PendingReceiver<network::mojom::URLLoaderClient>
new_client_receiver,
mojo::PendingRemote<network::mojom::URLLoader>* original_loader,
mojo::PendingReceiver<network::mojom::URLLoaderClient>*
original_client_receiver,
mojo::ScopedDataPipeConsumerHandle* body) override {
if (!loader_)
return;
ScopedDelegateCall scoped_delegate_call(this);
loader_->InterceptResponse(std::move(new_loader),
std::move(new_client_receiver), original_loader,
original_client_receiver, body);
}
void Detach() { loader_ = nullptr; }
void DidRestartForCriticalClientHint() override {
loader_->DidRestartForCriticalClientHint();
}
private:
// This class helps ThrottlingURLLoader to keep track of whether it is being
// called by its throttles.
// If ThrottlingURLLoader is destoyed while any of the throttles is calling
// into it, it delays destruction of the throttles. That way throttles don't
// need to worry about any delegate calls may destory them synchronously.
class ScopedDelegateCall {
public:
explicit ScopedDelegateCall(ForwardingThrottleDelegate* owner)
: owner_(owner) {
DCHECK(owner_->loader_);
owner_->loader_->inside_delegate_calls_++;
}
ScopedDelegateCall(const ScopedDelegateCall&) = delete;
ScopedDelegateCall& operator=(const ScopedDelegateCall&) = delete;
~ScopedDelegateCall() {
// The loader may have been detached and destroyed.
if (owner_->loader_)
owner_->loader_->inside_delegate_calls_--;
}
private:
const raw_ptr<ForwardingThrottleDelegate> owner_;
};
raw_ptr<ThrottlingURLLoader, DanglingUntriaged> loader_;
const raw_ptr<URLLoaderThrottle> throttle_;
};
ThrottlingURLLoader::StartInfo::StartInfo(
scoped_refptr<network::SharedURLLoaderFactory> in_url_loader_factory,
int32_t in_request_id,
uint32_t in_options,
network::ResourceRequest* in_url_request,
scoped_refptr<base::SequencedTaskRunner> in_task_runner,
std::optional<std::vector<std::string>> in_cors_exempt_header_list)
: url_loader_factory(std::move(in_url_loader_factory)),
request_id(in_request_id),
options(in_options),
url_request(*in_url_request),
task_runner(std::move(in_task_runner)),
cors_exempt_header_list(std::move(in_cors_exempt_header_list)) {}
ThrottlingURLLoader::StartInfo::~StartInfo() = default;
ThrottlingURLLoader::ResponseInfo::ResponseInfo(
network::mojom::URLResponseHeadPtr in_response_head)
: response_head(std::move(in_response_head)) {}
ThrottlingURLLoader::ResponseInfo::~ResponseInfo() = default;
ThrottlingURLLoader::RedirectInfo::RedirectInfo(
const net::RedirectInfo& in_redirect_info,
network::mojom::URLResponseHeadPtr in_response_head)
: redirect_info(in_redirect_info),
response_head(std::move(in_response_head)) {}
ThrottlingURLLoader::RedirectInfo::~RedirectInfo() = default;
ThrottlingURLLoader::PriorityInfo::PriorityInfo(
net::RequestPriority in_priority,
int32_t in_intra_priority_value)
: priority(in_priority), intra_priority_value(in_intra_priority_value) {}
// static
std::unique_ptr<ThrottlingURLLoader> ThrottlingURLLoader::CreateLoader(
std::vector<std::unique_ptr<URLLoaderThrottle>> throttles,
network::mojom::URLLoaderClient* client,
const net::NetworkTrafficAnnotationTag& traffic_annotation,
ClientReceiverDelegate* client_receiver_delegate) {
return std::unique_ptr<ThrottlingURLLoader>(
new ThrottlingURLLoader(std::move(throttles), client, traffic_annotation,
client_receiver_delegate));
}
// static
std::unique_ptr<ThrottlingURLLoader> ThrottlingURLLoader::CreateLoaderAndStart(
scoped_refptr<network::SharedURLLoaderFactory> factory,
std::vector<std::unique_ptr<URLLoaderThrottle>> throttles,
int32_t request_id,
uint32_t options,
network::ResourceRequest* url_request,
network::mojom::URLLoaderClient* client,
const net::NetworkTrafficAnnotationTag& traffic_annotation,
scoped_refptr<base::SequencedTaskRunner> task_runner,
std::optional<std::vector<std::string>> cors_exempt_header_list,
ClientReceiverDelegate* client_receiver_delegate,
const std::vector<int>* initiator_origin_trial_features) {
DCHECK(url_request);
std::unique_ptr<ThrottlingURLLoader> loader =
CreateLoader(std::move(throttles), client, traffic_annotation,
client_receiver_delegate);
loader->Start(std::move(factory), request_id, options, url_request,
std::move(task_runner), std::move(cors_exempt_header_list),
initiator_origin_trial_features);
return loader;
}
ThrottlingURLLoader::~ThrottlingURLLoader() {
TRACE_EVENT_WITH_FLOW0("loading", "ThrottlingURLLoader::~ThrottlingURLLoader",
TRACE_ID_LOCAL(this), TRACE_EVENT_FLAG_FLOW_IN);
if (inside_delegate_calls_ > 0) {
// A throttle is calling into this object. In this case, delay destruction
// of the throttles, so that throttles don't need to worry about any
// delegate calls may destroy them synchronously.
for (auto& entry : throttles_)
entry.delegate->Detach();
auto throttles =
std::make_unique<std::vector<ThrottleEntry>>(std::move(throttles_));
base::SingleThreadTaskRunner::GetCurrentDefault()->DeleteSoon(
FROM_HERE, std::move(throttles));
}
}
void ThrottlingURLLoader::FollowRedirectForcingRestart() {
url_loader_.ResetWithReason(
network::mojom::URLLoader::kClientDisconnectReason,
kFollowRedirectReason);
client_receiver_.reset();
CHECK(throttle_will_redirect_redirect_url_.is_empty());
UpdateRequestHeaders(start_info_->url_request);
removed_headers_.clear();
modified_headers_.Clear();
modified_cors_exempt_headers_.Clear();
StartNow();
}
void ThrottlingURLLoader::ResetForFollowRedirect(
network::ResourceRequest& resource_request,
const std::vector<std::string>& removed_headers,
const net::HttpRequestHeaders& modified_headers,
const net::HttpRequestHeaders& modified_cors_exempt_headers) {
MergeRemovedHeaders(&removed_headers_, removed_headers);
RemoveModifiedHeadersBeforeMerge(&modified_headers_);
modified_headers_.MergeFrom(modified_headers);
modified_cors_exempt_headers_.MergeFrom(modified_cors_exempt_headers);
// Call UpdateRequestHeaders() after headers are merged.
UpdateRequestHeaders(resource_request);
url_loader_.ResetWithReason(
network::mojom::URLLoader::kClientDisconnectReason,
kFollowRedirectReason);
}
void ThrottlingURLLoader::FollowRedirect(
const std::vector<std::string>& removed_headers,
const net::HttpRequestHeaders& modified_headers,
const net::HttpRequestHeaders& modified_cors_exempt_headers) {
MergeRemovedHeaders(&removed_headers_, removed_headers);
RemoveModifiedHeadersBeforeMerge(&modified_headers_);
modified_headers_.MergeFrom(modified_headers);
modified_cors_exempt_headers_.MergeFrom(modified_cors_exempt_headers);
if (!throttle_will_start_redirect_url_.is_empty()) {
throttle_will_start_redirect_url_ = GURL();
// This is a synthesized redirect, so no need to tell the URLLoader.
UpdateRequestHeaders(start_info_->url_request);
StartNow();
return;
}
if (url_loader_) {
std::optional<GURL> new_url;
if (!throttle_will_redirect_redirect_url_.is_empty())
new_url = throttle_will_redirect_redirect_url_;
url_loader_->FollowRedirect(removed_headers_, modified_headers_,
modified_cors_exempt_headers_, new_url);
throttle_will_redirect_redirect_url_ = GURL();
}
removed_headers_.clear();
modified_headers_.Clear();
modified_cors_exempt_headers_.Clear();
}
void ThrottlingURLLoader::SetPriority(net::RequestPriority priority,
int32_t intra_priority_value) {
if (!url_loader_) {
if (!loader_completed_) {
// Only check |deferred_stage_| if this resource has not been redirected
// by a throttle.
if (throttle_will_start_redirect_url_.is_empty() &&
throttle_will_redirect_redirect_url_.is_empty()) {
DCHECK_EQ(DEFERRED_START, deferred_stage_);
}
priority_info_ =
std::make_unique<PriorityInfo>(priority, intra_priority_value);
}
return;
}
url_loader_->SetPriority(priority, intra_priority_value);
}
network::mojom::URLLoaderClientEndpointsPtr ThrottlingURLLoader::Unbind() {
return network::mojom::URLLoaderClientEndpoints::New(
url_loader_.Unbind(), client_receiver_.Unbind());
}
ThrottlingURLLoader::ThrottlingURLLoader(
std::vector<std::unique_ptr<URLLoaderThrottle>> throttles,
network::mojom::URLLoaderClient* client,
const net::NetworkTrafficAnnotationTag& traffic_annotation,
ClientReceiverDelegate* client_receiver_delegate)
: forwarding_client_(client),
client_receiver_delegate_(std::move(client_receiver_delegate)),
traffic_annotation_(traffic_annotation) {
TRACE_EVENT_WITH_FLOW0("loading", "ThrottlingURLLoader::ThrottlingURLLoader",
TRACE_ID_LOCAL(this), TRACE_EVENT_FLAG_FLOW_OUT);
throttles_.reserve(throttles.size());
for (auto& throttle : throttles)
throttles_.emplace_back(this, std::move(throttle));
}
void ThrottlingURLLoader::Start(
scoped_refptr<network::SharedURLLoaderFactory> factory,
int32_t request_id,
uint32_t options,
network::ResourceRequest* url_request,
scoped_refptr<base::SequencedTaskRunner> task_runner,
std::optional<std::vector<std::string>> cors_exempt_header_list,
const std::vector<int>* initiator_origin_trial_features) {
TRACE_EVENT_WITH_FLOW1("loading", "ThrottlingURLLoader::Start",
TRACE_ID_LOCAL(this),
TRACE_EVENT_FLAG_FLOW_IN | TRACE_EVENT_FLAG_FLOW_OUT,
"request_id", request_id);
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
bool deferred = false;
DCHECK(deferring_throttles_.empty());
if (!throttles_.empty()) {
original_url_ = url_request->url;
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
bool throttle_deferred = false;
#if DCHECK_IS_ON()
std::set<std::string> initial_headers, initial_cors_exempt_headers;
if (cors_exempt_header_list) {
for (auto& header : url_request->headers.GetHeaderVector())
initial_headers.insert(header.key);
for (auto& header : url_request->cors_exempt_headers.GetHeaderVector())
initial_cors_exempt_headers.insert(header.key);
}
#endif
base::Time start = base::Time::Now();
auto weak_ptr = weak_factory_.GetWeakPtr();
throttle->WillStartRequest(url_request, &throttle_deferred);
if (!weak_ptr) {
return;
}
RecordExecutionTimeHistogram(GetStageNameForHistogram(DEFERRED_START),
start);
#if DCHECK_IS_ON()
if (cors_exempt_header_list) {
CheckThrottleWillNotCauseCorsPreflight(
initial_headers, initial_cors_exempt_headers, url_request->headers,
url_request->cors_exempt_headers, *cors_exempt_header_list);
}
#endif
if (original_url_ != url_request->url) {
DCHECK(throttle_will_start_redirect_url_.is_empty())
<< "ThrottlingURLLoader doesn't support multiple throttles "
"changing the URL.";
if (original_url_.SchemeIsHTTPOrHTTPS() &&
!url_request->url.SchemeIsHTTPOrHTTPS()) {
NOTREACHED() << "A URLLoaderThrottle can't redirect from http(s) to "
<< "a non http(s) scheme.";
} else {
throttle_will_start_redirect_url_ = url_request->url;
}
// Restore the original URL so that all throttles see the same original
// URL.
url_request->url = original_url_;
}
if (!HandleThrottleResult(throttle, throttle_deferred, &deferred))
return;
}
}
if (initiator_origin_trial_features &&
(base::Contains(
*initiator_origin_trial_features,
static_cast<int>(
mojom::OriginTrialFeature::kDeviceBoundSessionCredentials)) ||
base::Contains(
*initiator_origin_trial_features,
static_cast<int>(
mojom::OriginTrialFeature::kDeviceBoundSessionCredentials2)))) {
url_request->allows_device_bound_session_registration = true;
}
start_info_ = std::make_unique<StartInfo>(factory, request_id, options,
url_request, std::move(task_runner),
std::move(cors_exempt_header_list));
if (deferred)
deferred_stage_ = DEFERRED_START;
else
StartNow();
}
void ThrottlingURLLoader::StartNow() {
TRACE_EVENT_WITH_FLOW0("loading", "ThrottlingURLLoader::StartNow",
TRACE_ID_LOCAL(this),
TRACE_EVENT_FLAG_FLOW_IN | TRACE_EVENT_FLAG_FLOW_OUT);
DCHECK(start_info_);
if (!throttle_will_start_redirect_url_.is_empty()) {
auto first_party_url_policy =
start_info_->url_request.update_first_party_url_on_redirect
? net::RedirectInfo::FirstPartyURLPolicy::UPDATE_URL_ON_REDIRECT
: net::RedirectInfo::FirstPartyURLPolicy::NEVER_CHANGE_URL;
net::RedirectInfo redirect_info = net::RedirectInfo::ComputeRedirectInfo(
start_info_->url_request.method, start_info_->url_request.url,
start_info_->url_request.site_for_cookies, first_party_url_policy,
start_info_->url_request.referrer_policy,
start_info_->url_request.referrer.spec(),
start_info_->url_request.request_initiator,
// Use status code 307 to preserve the method, so POST requests work.
net::HTTP_TEMPORARY_REDIRECT, throttle_will_start_redirect_url_,
std::nullopt, false, false, false);
// Set Critical-CH restart info and clear for next redirect.
redirect_info.critical_ch_restart_time = critical_ch_restart_time_;
critical_ch_restart_time_ = base::TimeTicks();
bool should_clear_upload = false;
net::RedirectUtil::UpdateHttpRequest(
start_info_->url_request.url, start_info_->url_request.method,
redirect_info, std::nullopt, std::nullopt,
&start_info_->url_request.headers, &should_clear_upload);
if (should_clear_upload) {
start_info_->url_request.request_body = nullptr;
}
// Set the new URL in the ResourceRequest struct so that it is the URL
// that's requested.
start_info_->url_request.url = throttle_will_start_redirect_url_;
auto response_head = network::mojom::URLResponseHead::New();
std::string header_string = base::StringPrintf(
"HTTP/1.1 %i Internal Redirect\n"
"Location: %s",
net::HTTP_TEMPORARY_REDIRECT,
throttle_will_start_redirect_url_.spec().c_str());
response_head->headers = base::MakeRefCounted<net::HttpResponseHeaders>(
net::HttpUtil::AssembleRawHeaders(header_string));
response_head->encoded_data_length = header_string.size();
start_info_->task_runner->PostTask(
FROM_HERE,
base::BindOnce(&ThrottlingURLLoader::OnReceiveRedirect,
weak_factory_.GetWeakPtr(), std::move(redirect_info),
std::move(response_head)));
return;
}
if (start_info_->url_request.keepalive) {
base::UmaHistogramBoolean("FetchKeepAlive.Renderer.Total.Started", true);
}
DCHECK(start_info_->url_loader_factory);
start_info_->url_loader_factory->CreateLoaderAndStart(
url_loader_.BindNewPipeAndPassReceiver(start_info_->task_runner),
start_info_->request_id, start_info_->options, start_info_->url_request,
client_receiver_.BindNewPipeAndPassRemote(start_info_->task_runner),
net::MutableNetworkTrafficAnnotationTag(traffic_annotation_));
// TODO(https://crbug.com/919736): Remove this call.
client_receiver_.internal_state()->EnableBatchDispatch();
client_receiver_.set_disconnect_handler(base::BindOnce(
&ThrottlingURLLoader::OnClientConnectionError, base::Unretained(this)));
if (priority_info_) {
auto priority_info = std::move(priority_info_);
url_loader_->SetPriority(priority_info->priority,
priority_info->intra_priority_value);
}
// Initialize with the request URL, may be updated when on redirects
response_url_ = start_info_->url_request.url;
}
void ThrottlingURLLoader::RestartWithURLResetNow() {
url_loader_.reset();
client_receiver_.reset();
throttle_will_start_redirect_url_ = original_url_;
StartNow();
}
bool ThrottlingURLLoader::HandleThrottleResult(URLLoaderThrottle* throttle,
bool throttle_deferred,
bool* should_defer) {
DCHECK(!deferring_throttles_.count(throttle));
if (loader_completed_)
return false;
if (throttle_deferred) {
*should_defer = true;
deferring_throttles_.insert({throttle, base::Time::Now()});
}
return true;
}
void ThrottlingURLLoader::StopDeferringForThrottle(
URLLoaderThrottle* throttle) {
auto iter = deferring_throttles_.find(throttle);
if (iter == deferring_throttles_.end())
return;
if (deferred_stage_ != DEFERRED_NONE) {
const char* name = nullptr;
if (deferred_stage_ == DEFERRED_START) {
name = throttle->NameForLoggingWillStartRequest();
} else if (deferred_stage_ == DEFERRED_RESPONSE) {
name = throttle->NameForLoggingWillProcessResponse();
}
RecordDeferTimeHistogram(GetStageNameForHistogram(deferred_stage_),
iter->second, name);
}
deferring_throttles_.erase(iter);
if (deferring_throttles_.empty() && !loader_completed_)
Resume();
}
void ThrottlingURLLoader::OnReceiveEarlyHints(
network::mojom::EarlyHintsPtr early_hints) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
forwarding_client_->OnReceiveEarlyHints(std::move(early_hints));
}
void ThrottlingURLLoader::OnReceiveResponse(
network::mojom::URLResponseHeadPtr response_head,
mojo::ScopedDataPipeConsumerHandle body,
std::optional<mojo_base::BigBuffer> cached_metadata) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
DCHECK(deferring_throttles_.empty());
TRACE_EVENT_WITH_FLOW1("loading", "ThrottlingURLLoader::OnReceiveResponse",
TRACE_ID_LOCAL(this),
TRACE_EVENT_FLAG_FLOW_IN | TRACE_EVENT_FLAG_FLOW_OUT,
"url", response_url_.possibly_invalid_spec());
if (client_receiver_delegate_) {
client_receiver_delegate_->OnReceiveResponse(
std::move(response_head), std::move(body), std::move(cached_metadata));
return;
}
if (start_info_ && start_info_->url_request.keepalive) {
base::UmaHistogramBoolean("FetchKeepAlive.Renderer.Total.ReceivedResponse",
true);
}
base::ElapsedTimer timer;
did_receive_response_ = true;
body_ = std::move(body);
cached_metadata_ = std::move(cached_metadata);
// Dispatch BeforeWillProcessResponse().
if (!throttles_.empty()) {
URLLoaderThrottle::RestartWithURLReset has_pending_restart(false);
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
base::Time start = base::Time::Now();
auto weak_ptr = weak_factory_.GetWeakPtr();
throttle->BeforeWillProcessResponse(response_url_, *response_head,
&has_pending_restart);
if (!weak_ptr) {
return;
}
RecordExecutionTimeHistogram("BeforeWillProcessResponse", start);
if (!HandleThrottleResult(throttle)) {
return;
}
}
if (has_pending_restart) {
RestartWithURLResetNow();
return;
}
}
// Dispatch WillProcessResponse().
if (!throttles_.empty()) {
bool deferred = false;
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
bool throttle_deferred = false;
base::Time start = base::Time::Now();
auto weak_ptr = weak_factory_.GetWeakPtr();
throttle->WillProcessResponse(response_url_, response_head.get(),
&throttle_deferred);
if (!weak_ptr) {
return;
}
RecordExecutionTimeHistogram(GetStageNameForHistogram(DEFERRED_RESPONSE),
start);
if (!HandleThrottleResult(throttle, throttle_deferred, &deferred))
return;
}
if (deferred) {
deferred_stage_ = DEFERRED_RESPONSE;
response_info_ = std::make_unique<ResponseInfo>(std::move(response_head));
client_receiver_.Pause();
return;
}
}
forwarding_client_->OnReceiveResponse(
std::move(response_head), std::move(body_), std::move(cached_metadata_));
base::UmaHistogramTimes("Net.URLLoaderThrottle.OnReceiveResponseTime",
timer.Elapsed());
}
void ThrottlingURLLoader::OnReceiveRedirect(
const net::RedirectInfo& redirect_info,
network::mojom::URLResponseHeadPtr response_head) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
DCHECK(deferring_throttles_.empty());
if (start_info_ && start_info_->url_request.keepalive) {
base::UmaHistogramBoolean("FetchKeepAlive.Renderer.Total.Redirected", true);
}
if (!throttles_.empty()) {
URLLoaderThrottle::RestartWithURLReset has_pending_restart(false);
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
auto weak_ptr = weak_factory_.GetWeakPtr();
std::vector<std::string> removed_headers;
net::HttpRequestHeaders modified_headers;
net::HttpRequestHeaders modified_cors_exempt_headers;
net::RedirectInfo redirect_info_copy = redirect_info;
throttle->BeforeWillRedirectRequest(
&redirect_info_copy, *response_head, &has_pending_restart,
&removed_headers, &modified_headers, &modified_cors_exempt_headers);
if (!weak_ptr)
return;
}
if (has_pending_restart) {
RestartWithURLResetNow();
return;
}
bool deferred = false;
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
bool throttle_deferred = false;
auto weak_ptr = weak_factory_.GetWeakPtr();
std::vector<std::string> removed_headers;
net::HttpRequestHeaders modified_headers;
net::HttpRequestHeaders modified_cors_exempt_headers;
net::RedirectInfo redirect_info_copy = redirect_info;
base::Time start = base::Time::Now();
throttle->WillRedirectRequest(
&redirect_info_copy, *response_head, &throttle_deferred,
&removed_headers, &modified_headers, &modified_cors_exempt_headers);
if (!weak_ptr)
return;
RecordExecutionTimeHistogram(GetStageNameForHistogram(DEFERRED_REDIRECT),
start);
#if DCHECK_IS_ON()
if (start_info_->cors_exempt_header_list) {
CheckThrottleWillNotCauseCorsPreflight(
std::set<std::string>(), std::set<std::string>(), modified_headers,
modified_cors_exempt_headers,
*start_info_->cors_exempt_header_list);
}
#endif
if (redirect_info_copy.new_url != redirect_info.new_url) {
DCHECK(throttle_will_redirect_redirect_url_.is_empty())
<< "ThrottlingURLLoader doesn't support multiple throttles "
"changing the URL.";
throttle_will_redirect_redirect_url_ = redirect_info_copy.new_url;
}
if (!HandleThrottleResult(throttle, throttle_deferred, &deferred))
return;
MergeRemovedHeaders(&removed_headers_, removed_headers);
RemoveModifiedHeadersBeforeMerge(&modified_headers_);
modified_headers_.MergeFrom(modified_headers);
modified_cors_exempt_headers_.MergeFrom(modified_cors_exempt_headers);
}
if (deferred) {
deferred_stage_ = DEFERRED_REDIRECT;
redirect_info_ = std::make_unique<RedirectInfo>(redirect_info,
std::move(response_head));
// |client_receiver_| can be unbound if the redirect came from a
// throttle.
if (client_receiver_.is_bound())
client_receiver_.Pause();
return;
}
}
// Update the request in case |FollowRedirectForcingRestart()| is called, and
// needs to use the request updated for the redirect.
network::ResourceRequest& request = start_info_->url_request;
request.url = redirect_info.new_url;
request.method = redirect_info.new_method;
request.site_for_cookies = redirect_info.new_site_for_cookies;
request.referrer = GURL(redirect_info.new_referrer);
request.referrer_policy = redirect_info.new_referrer_policy;
if (request.trusted_params) {
request.trusted_params->isolation_info =
request.trusted_params->isolation_info.CreateForRedirect(
url::Origin::Create(request.url));
}
// TODO(dhausknecht) at this point we do not actually know if we commit to the
// redirect or if it will be cancelled. FollowRedirect would be a more
// suitable place to set this URL but there we do not have the data.
response_url_ = redirect_info.new_url;
if (client_receiver_delegate_) {
client_receiver_delegate_->EndReceiveRedirect(redirect_info,
std::move(response_head));
return;
}
forwarding_client_->OnReceiveRedirect(redirect_info,
std::move(response_head));
}
void ThrottlingURLLoader::OnUploadProgress(
int64_t current_position,
int64_t total_size,
OnUploadProgressCallback ack_callback) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
forwarding_client_->OnUploadProgress(current_position, total_size,
std::move(ack_callback));
}
void ThrottlingURLLoader::OnTransferSizeUpdated(int32_t transfer_size_diff) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
network::RecordOnTransferSizeUpdatedUMA(
network::OnTransferSizeUpdatedFrom::kThrottlingURLLoader);
forwarding_client_->OnTransferSizeUpdated(transfer_size_diff);
}
void ThrottlingURLLoader::OnComplete(
const network::URLLoaderCompletionStatus& status) {
DCHECK_EQ(DEFERRED_NONE, deferred_stage_);
DCHECK(!loader_completed_);
if (client_receiver_delegate_) {
client_receiver_delegate_->OnComplete(status);
return;
}
// Only dispatch WillOnCompleteWithError() if status is not OK.
if (!throttles_.empty() && status.error_code != net::OK) {
for (auto& entry : throttles_) {
auto* throttle = entry.throttle.get();
base::Time start = base::Time::Now();
auto weak_ptr = weak_factory_.GetWeakPtr();
throttle->WillOnCompleteWithError(status);
if (!weak_ptr) {
return;
}
RecordExecutionTimeHistogram("WillOnCompleteWithError", start);
if (!HandleThrottleResult(throttle)) {
return;
}
}
}
// This is the last expected message. Pipe closure before this is an error
// (see OnClientConnectionError). After this it is expected and should be
// ignored. The owner of |this| is expected to destroy |this| when
// OnComplete() and all data has been read. Destruction of |this| will
// destroy |url_loader_| appropriately.
loader_completed_ = true;
forwarding_client_->OnComplete(status);
}
void ThrottlingURLLoader::OnClientConnectionError() {
CancelWithError(net::ERR_ABORTED, "");
}
void ThrottlingURLLoader::CancelWithError(int error_code,
std::string_view custom_reason) {
CancelWithExtendedError(error_code, 0, custom_reason);
}
void ThrottlingURLLoader::CancelWithExtendedError(
int error_code,
int extended_reason_code,
std::string_view custom_reason) {
if (loader_completed_)
return;
network::URLLoaderCompletionStatus status;
status.error_code = error_code;
status.completion_time = base::TimeTicks::Now();
status.extended_error_code = extended_reason_code;
deferred_stage_ = DEFERRED_NONE;
DisconnectClient(custom_reason);
if (client_receiver_delegate_) {
client_receiver_delegate_->CancelWithStatus(status);
return;
}
forwarding_client_->OnComplete(status);
}
void ThrottlingURLLoader::Resume() {
if (loader_completed_ || deferred_stage_ == DEFERRED_NONE)
return;
auto prev_deferred_stage = deferred_stage_;
deferred_stage_ = DEFERRED_NONE;
switch (prev_deferred_stage) {
case DEFERRED_START: {
StartNow();
break;
}
case DEFERRED_REDIRECT: {
// |client_receiver_| can be unbound if the redirect came from a
// throttle.
if (client_receiver_.is_bound())
client_receiver_.Resume();
// TODO(dhausknecht) at this point we do not actually know if we commit to
// the redirect or if it will be cancelled. FollowRedirect would be a more
// suitable place to set this URL but there we do not have the data.
response_url_ = redirect_info_->redirect_info.new_url;
forwarding_client_->OnReceiveRedirect(
redirect_info_->redirect_info,
std::move(redirect_info_->response_head));
// Note: |this| may be deleted here.
break;
}
case DEFERRED_RESPONSE: {
client_receiver_.Resume();
forwarding_client_->OnReceiveResponse(
std::move(response_info_->response_head), std::move(body_),
std::move(cached_metadata_));
// Note: |this| may be deleted here.
break;
}
case DEFERRED_NONE:
NOTREACHED();
}
}
void ThrottlingURLLoader::SetPriority(net::RequestPriority priority) {
if (url_loader_)
url_loader_->SetPriority(priority, -1);
}
void ThrottlingURLLoader::UpdateRequestHeaders(
network::ResourceRequest& resource_request) {
for (const std::string& header : removed_headers_) {
resource_request.headers.RemoveHeader(header);
resource_request.cors_exempt_headers.RemoveHeader(header);
}
resource_request.headers.MergeFrom(modified_headers_);
resource_request.cors_exempt_headers.MergeFrom(modified_cors_exempt_headers_);
}
void ThrottlingURLLoader::UpdateDeferredResponseHead(
network::mojom::URLResponseHeadPtr new_response_head,
mojo::ScopedDataPipeConsumerHandle body) {
DCHECK(response_info_);
DCHECK(!body_);
DCHECK_EQ(DEFERRED_RESPONSE, deferred_stage_);
response_info_->response_head = std::move(new_response_head);
body_ = std::move(body);
}
void ThrottlingURLLoader::InterceptResponse(
mojo::PendingRemote<network::mojom::URLLoader> new_loader,
mojo::PendingReceiver<network::mojom::URLLoaderClient> new_client_receiver,
mojo::PendingRemote<network::mojom::URLLoader>* original_loader,
mojo::PendingReceiver<network::mojom::URLLoaderClient>*
original_client_receiver,
mojo::ScopedDataPipeConsumerHandle* body) {
response_intercepted_ = true;
body->swap(body_);
if (original_loader) {
*original_loader = url_loader_.Unbind();
}
url_loader_.Bind(std::move(new_loader));
if (original_client_receiver)
*original_client_receiver = client_receiver_.Unbind();
client_receiver_.Bind(std::move(new_client_receiver),
start_info_->task_runner);
client_receiver_.set_disconnect_handler(base::BindOnce(
&ThrottlingURLLoader::OnClientConnectionError, base::Unretained(this)));
}
void ThrottlingURLLoader::DisconnectClient(std::string_view custom_reason) {
client_receiver_.reset();
if (!custom_reason.empty()) {
url_loader_.ResetWithReason(
network::mojom::URLLoader::kClientDisconnectReason,
std::string(custom_reason));
} else {
url_loader_.reset();
}
loader_completed_ = true;
}
const char* ThrottlingURLLoader::GetStageNameForHistogram(DeferredStage stage) {
switch (stage) {
case DEFERRED_START:
return "WillStartRequest";
case DEFERRED_REDIRECT:
return "WillRedirectRequest";
case DEFERRED_RESPONSE:
return "WillProcessResponse";
case DEFERRED_NONE:
NOTREACHED();
}
}
ThrottlingURLLoader::ThrottleEntry::ThrottleEntry(
ThrottlingURLLoader* loader,
std::unique_ptr<URLLoaderThrottle> the_throttle)
: throttle(std::move(the_throttle)),
delegate(std::make_unique<ForwardingThrottleDelegate>(loader,
throttle.get())) {
throttle->set_delegate(delegate.get());
}
ThrottlingURLLoader::ThrottleEntry::ThrottleEntry(ThrottleEntry&& other) =
default;
ThrottlingURLLoader::ThrottleEntry::~ThrottleEntry() {
// `delegate` is destroyed before `throttle`; clear the pointer so the
// throttle cannot inadvertently use-after-free the delegate.
throttle->set_delegate(nullptr);
}
ThrottlingURLLoader::ThrottleEntry& ThrottlingURLLoader::ThrottleEntry::
operator=(ThrottleEntry&& other) = default;
} // namespace blink