blob: a1cde80d91b12d84adbd23290b0cbd0710013a45 [file] [log] [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/renderer/url_loader_throttle_provider_impl.h"
#include <memory>
#include <utility>
#include "base/command_line.h"
#include "base/feature_list.h"
#include "base/functional/bind.h"
#include "base/memory/ptr_util.h"
#include "base/task/sequenced_task_runner.h"
#include "build/build_config.h"
#include "chrome/common/google_url_loader_throttle.h"
#include "chrome/common/request_header_integrity/buildflags.h"
#include "chrome/renderer/chrome_content_renderer_client.h"
#include "chrome/renderer/chrome_render_frame_observer.h"
#include "chrome/renderer/chrome_render_thread_observer.h"
#include "components/fingerprinting_protection_filter/common/fingerprinting_protection_filter_features.h"
#include "components/fingerprinting_protection_filter/renderer/renderer_agent.h"
#include "components/fingerprinting_protection_filter/renderer/renderer_metrics_url_loader_throttle.h"
#include "components/fingerprinting_protection_filter/renderer/renderer_url_loader_throttle.h"
#include "components/fingerprinting_protection_filter/renderer/unverified_ruleset_dealer.h"
#include "components/no_state_prefetch/renderer/no_state_prefetch_helper.h"
#include "components/safe_browsing/core/common/features.h"
#include "components/signin/public/base/signin_buildflags.h"
#include "components/subresource_filter/core/common/first_party_origin.h"
#include "components/subresource_filter/core/common/memory_mapped_ruleset.h"
#include "content/public/common/content_features.h"
#include "content/public/common/web_identity.h"
#include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_thread.h"
#include "extensions/renderer/extension_localization_throttle.h"
#include "services/network/public/cpp/resource_request.h"
#include "third_party/blink/public/common/loader/resource_type_util.h"
#include "third_party/blink/public/common/thread_safe_browser_interface_broker_proxy.h"
#include "third_party/blink/public/mojom/loader/resource_load_info.mojom-shared.h"
#include "third_party/blink/public/platform/browser_interface_broker_proxy.h"
#include "third_party/blink/public/web/modules/credentialmanagement/throttle_helper.h"
#include "third_party/blink/public/web/web_local_frame.h"
#include "url/gurl.h"
#if BUILDFLAG(ENABLE_EXTENSIONS)
#include "extensions/common/switches.h"
#include "extensions/renderer/extension_throttle_manager.h"
#endif
#if BUILDFLAG(ENABLE_REQUEST_HEADER_INTEGRITY)
#include "chrome/common/request_header_integrity/request_header_integrity_url_loader_throttle.h" // nogncheck crbug.com/1125897
#endif
#if BUILDFLAG(IS_CHROMEOS)
#include "chrome/renderer/ash_merge_session_loader_throttle.h"
#endif // BUILDFLAG(IS_CHROMEOS)
#if BUILDFLAG(SAFE_BROWSING_AVAILABLE)
#include "components/safe_browsing/content/renderer/renderer_url_loader_throttle.h"
#endif
namespace {
#if BUILDFLAG(ENABLE_EXTENSIONS)
std::unique_ptr<extensions::ExtensionThrottleManager>
CreateExtensionThrottleManager() {
if (base::CommandLine::ForCurrentProcess()->HasSwitch(
extensions::switches::kDisableExtensionsHttpThrottling)) {
return nullptr;
}
return std::make_unique<extensions::ExtensionThrottleManager>();
}
void SetExtensionThrottleManagerTestPolicy(
extensions::ExtensionThrottleManager* extension_throttle_manager) {
std::unique_ptr<net::BackoffEntry::Policy> policy(
new net::BackoffEntry::Policy{
// Number of initial errors (in sequence) to ignore before
// applying exponential back-off rules.
1,
// Initial delay for exponential back-off in ms.
10 * 60 * 1000,
// Factor by which the waiting time will be multiplied.
10,
// Fuzzing percentage. ex: 10% will spread requests randomly
// between 90%-100% of the calculated time.
0.1,
// Maximum amount of time we are willing to delay our request in ms.
15 * 60 * 1000,
// Time to keep an entry from being discarded even when it
// has no significant state, -1 to never discard.
-1,
// Don't use initial delay unless the last request was an error.
false,
});
extension_throttle_manager->SetBackoffPolicyForTests(std::move(policy));
}
#endif
} // namespace
// static
std::unique_ptr<blink::URLLoaderThrottleProvider>
URLLoaderThrottleProviderImpl::Create(
blink::URLLoaderThrottleProviderType type,
ChromeContentRendererClient* chrome_content_renderer_client,
blink::ThreadSafeBrowserInterfaceBrokerProxy* broker) {
mojo::PendingRemote<safe_browsing::mojom::SafeBrowsing> pending_safe_browsing;
broker->GetInterface(pending_safe_browsing.InitWithNewPipeAndPassReceiver());
#if BUILDFLAG(ENABLE_EXTENSIONS)
mojo::PendingRemote<safe_browsing::mojom::ExtensionWebRequestReporter>
pending_extension_web_request_reporter;
broker->GetInterface(
pending_extension_web_request_reporter.InitWithNewPipeAndPassReceiver());
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
return std::make_unique<URLLoaderThrottleProviderImpl>(
type, chrome_content_renderer_client, std::move(pending_safe_browsing),
#if BUILDFLAG(ENABLE_EXTENSIONS)
std::move(pending_extension_web_request_reporter),
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
/*main_thread_task_runner=*/
content::RenderThread::IsMainThread()
? base::SequencedTaskRunner::GetCurrentDefault()
: nullptr,
base::PassKey<URLLoaderThrottleProviderImpl>());
}
URLLoaderThrottleProviderImpl::URLLoaderThrottleProviderImpl(
blink::URLLoaderThrottleProviderType type,
ChromeContentRendererClient* chrome_content_renderer_client,
mojo::PendingRemote<safe_browsing::mojom::SafeBrowsing>
pending_safe_browsing,
#if BUILDFLAG(ENABLE_EXTENSIONS)
mojo::PendingRemote<safe_browsing::mojom::ExtensionWebRequestReporter>
pending_extension_web_request_reporter,
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
scoped_refptr<base::SequencedTaskRunner> main_thread_task_runner,
base::PassKey<URLLoaderThrottleProviderImpl>)
: type_(type),
chrome_content_renderer_client_(chrome_content_renderer_client),
pending_safe_browsing_(std::move(pending_safe_browsing)),
#if BUILDFLAG(ENABLE_EXTENSIONS)
pending_extension_web_request_reporter_(
std::move(pending_extension_web_request_reporter)),
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
main_thread_task_runner_(std::move(main_thread_task_runner)) {
if (main_thread_task_runner_ &&
main_thread_task_runner_->RunsTasksInCurrentSequence()) {
// This provider is being created on the main thread.
fingerprinting_protection_ruleset_ =
chrome_content_renderer_client_->GetFingerprintingProtectionRuleset();
}
DETACH_FROM_SEQUENCE(sequence_checker_);
}
URLLoaderThrottleProviderImpl::~URLLoaderThrottleProviderImpl() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
// TODO(https://crbug.com/437115067): Remove this once `MemoryMappedRuleset`
// lifetime guarantees are cleaned up.
if (main_thread_task_runner_ &&
!main_thread_task_runner_->RunsTasksInCurrentSequence()) {
// Ensure the ruleset is released on the correct sequence if it is not the
// current one.
main_thread_task_runner_->ReleaseSoon(
FROM_HERE, std::move(fingerprinting_protection_ruleset_));
}
}
std::unique_ptr<blink::URLLoaderThrottleProvider>
URLLoaderThrottleProviderImpl::Clone() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return std::make_unique<URLLoaderThrottleProviderImpl>(
type_, chrome_content_renderer_client_, CloneSafeBrowsingPendingRemote(),
#if BUILDFLAG(ENABLE_EXTENSIONS)
CloneExtensionWebRequestReporterPendingRemote(),
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
main_thread_task_runner_, base::PassKey<URLLoaderThrottleProviderImpl>());
}
std::vector<std::unique_ptr<blink::URLLoaderThrottle>>
URLLoaderThrottleProviderImpl::CreateThrottles(
base::optional_ref<const blink::LocalFrameToken> local_frame_token,
const network::ResourceRequest& request) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
std::vector<std::unique_ptr<blink::URLLoaderThrottle>> throttles;
// Some throttles have already been added in the browser for frame resources.
// Don't add them for frame requests.
bool is_frame_resource =
blink::IsRequestDestinationFrame(request.destination);
DCHECK(!is_frame_resource ||
type_ == blink::URLLoaderThrottleProviderType::kFrame);
#if BUILDFLAG(SAFE_BROWSING_AVAILABLE)
if (!is_frame_resource) {
if (pending_safe_browsing_) {
safe_browsing_.Bind(std::move(pending_safe_browsing_));
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
auto throttle = std::make_unique<safe_browsing::RendererURLLoaderThrottle>(
safe_browsing_.get(), local_frame_token,
CloneExtensionWebRequestReporterPendingRemote());
#else
auto throttle = std::make_unique<safe_browsing::RendererURLLoaderThrottle>(
safe_browsing_.get(), local_frame_token);
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
throttles.emplace_back(std::move(throttle));
}
#endif
if (chrome_content_renderer_client_
->IsContentBasedFingerprintingProtectionEnabledForMetrics()) {
using fingerprinting_protection_filter::RendererThrottleCreationResult;
auto get_throttle_creation_result =
[&]() -> RendererThrottleCreationResult {
// Restrict the requests that we check as much as possible.
if (!chrome_content_renderer_client_
->IsContentBasedFingerprintingProtectionEnabled()) {
return RendererThrottleCreationResult::
kSkipDisabledForCrossSiteSubframe;
}
if (is_frame_resource) {
return RendererThrottleCreationResult::kSkipFrameResource;
}
if (type_ != blink::URLLoaderThrottleProviderType::kFrame) {
return RendererThrottleCreationResult::kSkipWorkerThrottle;
}
if (std::optional<RendererThrottleCreationResult> result =
fingerprinting_protection_filter::RendererURLLoaderThrottle::
WillIgnoreRequest(request.url, request.destination)) {
return result.value();
}
if (!local_frame_token.has_value()) {
return RendererThrottleCreationResult::kSkipNoFrameToken;
}
if (!fingerprinting_protection_ruleset_) {
return RendererThrottleCreationResult::kSkipNoRuleset;
}
// Uses net::SchemefulSite::IsSameSite to reduce memory performance
// impact.
if (net::SchemefulSite::IsSameSite(url::Origin::Create(request.url),
request.request_initiator.value())) {
return RendererThrottleCreationResult::kSkipSameSite;
}
return RendererThrottleCreationResult::kCreate;
};
RendererThrottleCreationResult creation_result =
get_throttle_creation_result();
throttles.emplace_back(
std::make_unique<
fingerprinting_protection_filter::RendererMetricsURLLoaderThrottle>(
creation_result, request.request_initiator, request.url));
if (creation_result == RendererThrottleCreationResult::kCreate) {
throttles.emplace_back(
std::make_unique<
fingerprinting_protection_filter::RendererURLLoaderThrottle>(
main_thread_task_runner_, local_frame_token.value(),
fingerprinting_protection_ruleset_));
}
}
if (type_ == blink::URLLoaderThrottleProviderType::kFrame &&
!is_frame_resource && local_frame_token.has_value()) {
auto throttle = prerender::NoStatePrefetchHelper::MaybeCreateThrottle(
local_frame_token.value());
if (throttle) {
throttles.emplace_back(std::move(throttle));
}
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
if (!extension_throttle_manager_) {
extension_throttle_manager_ = CreateExtensionThrottleManager();
}
if (extension_throttle_manager_) {
if (base::CommandLine::ForCurrentProcess()->HasSwitch(
extensions::switches::kSetExtensionThrottleTestParams)) {
SetExtensionThrottleManagerTestPolicy(extension_throttle_manager_.get());
}
std::unique_ptr<blink::URLLoaderThrottle> throttle =
extension_throttle_manager_->MaybeCreateURLLoaderThrottle(request);
if (throttle) {
throttles.emplace_back(std::move(throttle));
}
}
std::unique_ptr<blink::URLLoaderThrottle> localization_throttle =
extensions::ExtensionLocalizationThrottle::MaybeCreate(local_frame_token,
request.url);
if (localization_throttle) {
throttles.emplace_back(std::move(localization_throttle));
}
#endif
#if BUILDFLAG(IS_ANDROID)
std::string client_data_header;
if (!is_frame_resource && local_frame_token.has_value()) {
client_data_header = ChromeRenderFrameObserver::GetCCTClientHeader(
local_frame_token.value());
}
#endif
throttles.emplace_back(std::make_unique<GoogleURLLoaderThrottle>(
#if BUILDFLAG(IS_ANDROID)
client_data_header,
#endif
#if BUILDFLAG(ENABLE_BOUND_SESSION_CREDENTIALS)
chrome_content_renderer_client_->GetChromeObserver()
->CreateBoundSessionRequestThrottledHandler(),
#endif
chrome_content_renderer_client_->GetChromeObserver()
->GetDynamicParams()));
#if BUILDFLAG(IS_CHROMEOS)
throttles.emplace_back(std::make_unique<AshMergeSessionLoaderThrottle>(
chrome_content_renderer_client_->GetChromeObserver()
->chromeos_listener()));
#endif // BUILDFLAG(IS_CHROMEOS)
#if BUILDFLAG(ENABLE_REQUEST_HEADER_INTEGRITY)
if (request_header_integrity::RequestHeaderIntegrityURLLoaderThrottle::
IsFeatureEnabled()) {
throttles.push_back(
std::make_unique<request_header_integrity::
RequestHeaderIntegrityURLLoaderThrottle>());
}
#endif
if (local_frame_token.has_value()) {
auto throttle =
content::MaybeCreateIdentityUrlLoaderThrottle(base::BindRepeating(
[](const blink::LocalFrameToken& token,
const scoped_refptr<base::SequencedTaskRunner>
main_thread_task_runner,
const url::Origin& origin,
blink::mojom::IdpSigninStatus status) {
if (content::RenderThread::IsMainThread()) {
blink::SetIdpSigninStatus(token, origin, status);
return;
}
if (main_thread_task_runner) {
main_thread_task_runner->PostTask(
FROM_HERE, base::BindOnce(&blink::SetIdpSigninStatus, token,
origin, status));
}
},
local_frame_token.value(), main_thread_task_runner_));
if (throttle) {
throttles.push_back(std::move(throttle));
}
}
return throttles;
}
void URLLoaderThrottleProviderImpl::SetOnline(bool is_online) {
#if BUILDFLAG(ENABLE_EXTENSIONS)
if (extension_throttle_manager_) {
extension_throttle_manager_->SetOnline(is_online);
}
#endif
}
mojo::PendingRemote<safe_browsing::mojom::SafeBrowsing>
URLLoaderThrottleProviderImpl::CloneSafeBrowsingPendingRemote() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
mojo::PendingRemote<safe_browsing::mojom::SafeBrowsing>
new_pending_safe_browsing;
if (pending_safe_browsing_) {
safe_browsing_.Bind(std::move(pending_safe_browsing_));
}
if (safe_browsing_) {
safe_browsing_->Clone(
new_pending_safe_browsing.InitWithNewPipeAndPassReceiver());
}
return new_pending_safe_browsing;
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
mojo::PendingRemote<safe_browsing::mojom::ExtensionWebRequestReporter>
URLLoaderThrottleProviderImpl::CloneExtensionWebRequestReporterPendingRemote() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
mojo::PendingRemote<safe_browsing::mojom::ExtensionWebRequestReporter>
new_pending_extension_web_request_reporter;
if (pending_extension_web_request_reporter_) {
extension_web_request_reporter_.Bind(
std::move(pending_extension_web_request_reporter_));
}
if (extension_web_request_reporter_) {
extension_web_request_reporter_->Clone(
new_pending_extension_web_request_reporter
.InitWithNewPipeAndPassReceiver());
}
return new_pending_extension_web_request_reporter;
}
#endif // BUILDFLAG(ENABLE_EXTENSIONS)