blob: 746967e009f2cb15943b517eecd9092b28ae3164 [file] [log] [blame]
// Copyright 2018 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 "chrome/renderer/url_loader_throttle_provider_impl.h"
#include <utility>
#include "base/feature_list.h"
#include "base/message_loop/message_loop.h"
#include "chrome/common/prerender.mojom.h"
#include "chrome/common/prerender_url_loader_throttle.h"
#include "chrome/renderer/chrome_content_renderer_client.h"
#include "chrome/renderer/prerender/prerender_dispatcher.h"
#include "chrome/renderer/prerender/prerender_helper.h"
#include "components/safe_browsing/features.h"
#include "components/safe_browsing/renderer/renderer_url_loader_throttle.h"
#include "components/subresource_filter/content/renderer/ad_delay_renderer_metadata_provider.h"
#include "content/public/common/content_features.h"
#include "content/public/common/service_names.mojom.h"
#include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_thread.h"
#include "content/public/renderer/render_view.h"
#include "services/network/public/cpp/features.h"
#include "services/service_manager/public/cpp/connector.h"
#include "services/service_manager/public/cpp/interface_provider.h"
#if BUILDFLAG(ENABLE_EXTENSIONS)
#include "extensions/renderer/guest_view/mime_handler_view/mime_handler_view_container.h"
#endif
namespace {
chrome::mojom::PrerenderCanceler* GetPrerenderCanceller(int render_frame_id) {
content::RenderFrame* render_frame =
content::RenderFrame::FromRoutingID(render_frame_id);
if (!render_frame)
return nullptr;
prerender::PrerenderHelper* helper =
prerender::PrerenderHelper::Get(render_frame);
if (!helper)
return nullptr;
auto* canceler = new chrome::mojom::PrerenderCancelerPtr;
render_frame->GetRemoteInterfaces()->GetInterface(canceler);
base::MessageLoop::current()->task_runner()->DeleteSoon(FROM_HERE, canceler);
return canceler->get();
}
} // namespace
URLLoaderThrottleProviderImpl::URLLoaderThrottleProviderImpl(
content::URLLoaderThrottleProviderType type,
ChromeContentRendererClient* chrome_content_renderer_client)
: type_(type),
chrome_content_renderer_client_(chrome_content_renderer_client) {
DETACH_FROM_THREAD(thread_checker_);
if (base::FeatureList::IsEnabled(network::features::kNetworkService) ||
base::FeatureList::IsEnabled(safe_browsing::kCheckByURLLoaderThrottle)) {
content::RenderThread::Get()->GetConnector()->BindInterface(
content::mojom::kBrowserServiceName,
mojo::MakeRequest(&safe_browsing_info_));
}
}
URLLoaderThrottleProviderImpl::~URLLoaderThrottleProviderImpl() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
}
URLLoaderThrottleProviderImpl::URLLoaderThrottleProviderImpl(
const URLLoaderThrottleProviderImpl& other)
: type_(other.type_),
chrome_content_renderer_client_(other.chrome_content_renderer_client_) {
DETACH_FROM_THREAD(thread_checker_);
if (other.safe_browsing_)
other.safe_browsing_->Clone(mojo::MakeRequest(&safe_browsing_info_));
// An ad_delay_factory_ is created, rather than cloning the existing one.
}
std::unique_ptr<content::URLLoaderThrottleProvider>
URLLoaderThrottleProviderImpl::Clone() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
if (safe_browsing_info_)
safe_browsing_.Bind(std::move(safe_browsing_info_));
return base::WrapUnique(new URLLoaderThrottleProviderImpl(*this));
}
std::vector<std::unique_ptr<content::URLLoaderThrottle>>
URLLoaderThrottleProviderImpl::CreateThrottles(
int render_frame_id,
const blink::WebURLRequest& request,
content::ResourceType resource_type) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
std::vector<std::unique_ptr<content::URLLoaderThrottle>> throttles;
bool network_service_enabled =
base::FeatureList::IsEnabled(network::features::kNetworkService);
// Some throttles have already been added in the browser for frame resources.
// Don't add them for frame requests.
bool is_frame_resource = content::IsResourceTypeFrame(resource_type);
DCHECK(!is_frame_resource ||
type_ == content::URLLoaderThrottleProviderType::kFrame);
if ((network_service_enabled ||
base::FeatureList::IsEnabled(
safe_browsing::kCheckByURLLoaderThrottle)) &&
!is_frame_resource) {
if (safe_browsing_info_)
safe_browsing_.Bind(std::move(safe_browsing_info_));
throttles.push_back(
std::make_unique<safe_browsing::RendererURLLoaderThrottle>(
safe_browsing_.get(), render_frame_id));
}
if (type_ == content::URLLoaderThrottleProviderType::kFrame &&
!is_frame_resource) {
content::RenderFrame* render_frame =
content::RenderFrame::FromRoutingID(render_frame_id);
auto* prerender_helper =
render_frame ? prerender::PrerenderHelper::Get(
render_frame->GetRenderView()->GetMainRenderFrame())
: nullptr;
if (prerender_helper) {
auto throttle = std::make_unique<prerender::PrerenderURLLoaderThrottle>(
prerender_helper->prerender_mode(),
prerender_helper->histogram_prefix(),
base::BindOnce(GetPrerenderCanceller, render_frame_id),
base::MessageLoop::current()->task_runner());
prerender_helper->AddThrottle(throttle->AsWeakPtr());
if (prerender_helper->prerender_mode() == prerender::PREFETCH_ONLY) {
auto* prerender_dispatcher =
chrome_content_renderer_client_->prerender_dispatcher();
prerender_dispatcher->IncrementPrefetchCount();
throttle->set_destruction_closure(base::BindOnce(
&prerender::PrerenderDispatcher::DecrementPrefetchCount,
base::Unretained(prerender_dispatcher)));
}
throttles.push_back(std::move(throttle));
}
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
if (network_service_enabled &&
type_ == content::URLLoaderThrottleProviderType::kFrame &&
resource_type == content::RESOURCE_TYPE_OBJECT) {
content::RenderFrame* render_frame =
content::RenderFrame::FromRoutingID(render_frame_id);
auto mime_handlers =
extensions::MimeHandlerViewContainer::FromRenderFrame(render_frame);
GURL gurl(request.Url());
for (auto* handler : mime_handlers) {
auto throttle = handler->MaybeCreatePluginThrottle(gurl);
if (throttle) {
throttles.push_back(std::move(throttle));
break;
}
}
}
#endif
// Initialize the factory here rather than in the constructor, since metrics
// does not support registering field trials (as opposed to Features) before
// Blink is initialized (after this class).
if (!ad_delay_factory_) {
ad_delay_factory_ =
std::make_unique<subresource_filter::AdDelayThrottle::Factory>();
}
if (auto ad_throttle = ad_delay_factory_->MaybeCreate(
std::make_unique<subresource_filter::AdDelayRendererMetadataProvider>(
request, type_, render_frame_id))) {
throttles.push_back(std::move(ad_throttle));
}
return throttles;
}