#include "android_webview/renderer/aw_content_renderer_client.h"
#include <memory>
#include <string_view>
#include <vector>
#include "android_webview/common/aw_features.h"
#include "android_webview/common/aw_switches.h"
#include "android_webview/common/mojom/frame.mojom.h"
#include "android_webview/common/url_constants.h"
#include "android_webview/renderer/aw_content_settings_client.h"
#include "android_webview/renderer/aw_print_render_frame_helper_delegate.h"
#include "android_webview/renderer/aw_render_frame_ext.h"
#include "android_webview/renderer/aw_render_frame_observer.h"
#include "android_webview/renderer/aw_render_view_ext.h"
#include "android_webview/renderer/aw_url_loader_throttle_provider.h"
#include "android_webview/renderer/browser_exposed_renderer_interfaces.h"
#include "base/android/library_loader/library_prefetcher.h"
#include "base/android/orderfile/orderfile_buildflags.h"
#include "base/command_line.h"
#include "base/feature_list.h"
#include "base/i18n/rtl.h"
#include "base/metrics/histogram_macros.h"
#include "base/strings/string_util.h"
#include "base/task/thread_pool.h"
#include "components/android_system_error_page/error_page_populator.h"
#include "components/autofill/core/common/autofill_features.h"
#include "components/cdm/renderer/key_system_support_update.h"
#include "components/js_injection/renderer/js_communication.h"
#include "components/network_hints/renderer/web_prescient_networking_impl.h"
#include "components/page_load_metrics/renderer/metrics_render_frame_observer.h"
#include "components/printing/renderer/print_render_frame_helper.h"
#include "components/security_interstitials/content/renderer/security_interstitial_page_controller_delegate_impl.h"
#include "components/visitedlink/renderer/visitedlink_reader.h"
#include "content/public/child/child_thread.h"
#include "content/public/common/url_constants.h"
#include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_thread.h"
#include "mojo/public/cpp/bindings/binder_map.h"
#include "third_party/blink/public/common/associated_interfaces/associated_interface_provider.h"
#include "third_party/blink/public/platform/platform.h"
#include "third_party/blink/public/platform/web_runtime_features.h"
#include "third_party/blink/public/platform/web_string.h"
#include "third_party/blink/public/platform/web_url.h"
#include "third_party/blink/public/platform/web_url_request.h"
#include "third_party/blink/public/web/web_frame.h"
#include "third_party/blink/public/web/web_local_frame.h"
#include "third_party/blink/public/web/web_navigation_type.h"
#include "third_party/blink/public/web/web_security_policy.h"
#include "url/gurl.h"
#include "url/url_constants.h"
#if BUILDFLAG(ENABLE_SPELLCHECK)
#include "components/spellcheck/renderer/spellcheck.h"
#include "components/spellcheck/renderer/spellcheck_provider.h"
#endif
using content::RenderThread;
namespace android_webview {
AwContentRendererClient::AwContentRendererClient() = default;
AwContentRendererClient::~AwContentRendererClient() = default;
void AwContentRendererClient::RenderThreadStarted() {
RenderThread* thread = RenderThread::Get();
aw_render_thread_observer_ = std::make_unique<AwRenderThreadObserver>();
thread->AddObserver(aw_render_thread_observer_.get());
visited_link_reader_ = std::make_unique<visitedlink::VisitedLinkReader>();
browser_interface_broker_ =
blink::Platform::Current()->GetBrowserInterfaceBroker();
#if BUILDFLAG(SUPPORTS_CODE_ORDERING)
bool shouldPrefetchNativeLibrary =
base::FeatureList::IsEnabled(features::kWebViewPrefetchNativeLibrary) &&
features::kWebViewPrefetchFromRenderer.Get();
if (base::FeatureList::IsEnabled(
features::kWebViewConfigurableLibraryPrefetch)) {
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
if (command_line->HasSwitch(switches::kWebViewRendererLibraryPrefetch)) {
std::string value = command_line->GetSwitchValueASCII(
switches::kWebViewRendererLibraryPrefetch);
if (value == switches::kWebViewRendererLibraryPrefetchEnabled) {
shouldPrefetchNativeLibrary = true;
} else if (value == switches::kWebViewRendererLibraryPrefetchDisabled) {
shouldPrefetchNativeLibrary = false;
}
}
}
if (shouldPrefetchNativeLibrary) {
base::ThreadPool::PostTask(
FROM_HERE, base::BindOnce([] {
base::android::NativeLibraryPrefetcher::PrefetchNativeLibrary();
}));
}
#endif
#if BUILDFLAG(ENABLE_SPELLCHECK)
if (!spellcheck_)
spellcheck_ = std::make_unique<SpellCheck>(this);
#endif
}
void AwContentRendererClient::ExposeInterfacesToBrowser(
mojo::BinderMap* binders) {
ExposeRendererInterfacesToBrowser(this, binders);
}
bool AwContentRendererClient::HandleNavigation(
content::RenderFrame* render_frame,
blink::WebFrame* frame,
const blink::WebURLRequest& request,
blink::WebNavigationType type,
blink::WebNavigationPolicy default_policy,
bool is_redirect) {
if (!request.HttpMethod().Equals("GET"))
return false;
bool application_initiated = type == blink::kWebNavigationTypeBackForward;
if (application_initiated && !is_redirect)
return false;
bool is_outermost_main_frame = frame->IsOutermostMainFrame();
const GURL& gurl = request.Url();
if (!is_outermost_main_frame &&
(gurl.SchemeIs(url::kHttpScheme) || gurl.SchemeIs(url::kHttpsScheme) ||
gurl.SchemeIs(url::kAboutScheme)))
return false;
AwRenderViewExt* view =
AwRenderViewExt::FromWebView(render_frame->GetWebView());
if (view->created_by_renderer()) {
return false;
}
bool ignore_navigation = false;
std::u16string url = request.Url().GetString().Utf16();
bool has_user_gesture = request.HasUserGesture();
mojo::AssociatedRemote<mojom::FrameHost> frame_host_remote;
render_frame->GetRemoteAssociatedInterfaces()->GetInterface(
&frame_host_remote);
frame_host_remote->ShouldOverrideUrlLoading(
url, has_user_gesture, is_redirect, is_outermost_main_frame,
&ignore_navigation);
return ignore_navigation;
}
void AwContentRendererClient::RenderFrameCreated(
content::RenderFrame* render_frame) {
new AwContentSettingsClient(render_frame);
new printing::PrintRenderFrameHelper(
render_frame, std::make_unique<AwPrintRenderFrameHelperDelegate>());
new AwRenderFrameExt(render_frame);
new js_injection::JsCommunication(render_frame);
new security_interstitials::SecurityInterstitialPageControllerDelegateImpl(
render_frame);
content::RenderFrame* main_frame = render_frame->GetMainRenderFrame();
if (main_frame && main_frame != render_frame) {
GetRenderMessageFilter()->SubFrameCreated(
main_frame->GetWebFrame()->GetLocalFrameToken(),
render_frame->GetWebFrame()->GetLocalFrameToken());
}
#if BUILDFLAG(ENABLE_SPELLCHECK)
new SpellCheckProvider(render_frame, spellcheck_.get());
#endif
new page_load_metrics::MetricsRenderFrameObserver(render_frame);
#if BUILDFLAG(ORDERFILE_INSTRUMENTATION)
new AwRenderFrameObserver(render_frame);
#endif
}
std::unique_ptr<blink::WebPrescientNetworking>
AwContentRendererClient::CreatePrescientNetworking(
content::RenderFrame* render_frame) {
return std::make_unique<network_hints::WebPrescientNetworkingImpl>(
render_frame);
}
void AwContentRendererClient::
SetRuntimeFeaturesDefaultsBeforeBlinkInitialization() {
if (base::FeatureList::IsEnabled(
autofill::features::kAutofillPolicyControlledFeatureAutofill)) {
blink::WebRuntimeFeatures::EnableAutofill(true);
}
if (base::FeatureList::IsEnabled(
autofill::features::kAutofillPolicyControlledFeatureManualText)) {
blink::WebRuntimeFeatures::EnableManualText(true);
}
blink::WebRuntimeFeatures::EnableBlinkExtensionWebView(true);
blink::WebRuntimeFeatures::EnableBlinkExtensionWebViewMediaIntegrity(true);
}
void AwContentRendererClient::WebViewCreated(
blink::WebView* web_view,
bool was_created_by_renderer,
const url::Origin* outermost_origin) {
AwRenderViewExt::WebViewCreated(web_view, was_created_by_renderer);
}
void AwContentRendererClient::PrepareErrorPage(
content::RenderFrame* render_frame,
const blink::WebURLError& error,
const std::string& http_method,
content::mojom::AlternativeErrorPageOverrideInfoPtr
alternative_error_page_info,
std::string* error_html) {
security_interstitials::SecurityInterstitialPageControllerDelegateImpl::Get(
render_frame)
->PrepareForErrorPage();
android_system_error_page::PopulateErrorPageHtml(error, error_html);
}
uint64_t AwContentRendererClient::VisitedLinkHash(
std::string_view canonical_url) {
return visited_link_reader_->ComputeURLFingerprint(canonical_url);
}
uint64_t AwContentRendererClient::PartitionedVisitedLinkFingerprint(
std::string_view canonical_link_url,
const net::SchemefulSite& top_level_site,
const url::Origin& frame_origin) {
return 0;
}
bool AwContentRendererClient::IsLinkVisited(uint64_t link_hash) {
return visited_link_reader_->IsVisited(link_hash);
}
void AwContentRendererClient::AddOrUpdateVisitedLinkSalt(
const url::Origin& origin,
uint64_t salt) {}
void AwContentRendererClient::RunScriptsAtDocumentStart(
content::RenderFrame* render_frame) {
js_injection::JsCommunication* communication =
js_injection::JsCommunication::Get(render_frame);
communication->RunScriptsAtDocumentStart();
}
std::unique_ptr<media::KeySystemSupportRegistration>
AwContentRendererClient::GetSupportedKeySystems(
content::RenderFrame* render_frame,
media::GetSupportedKeySystemsCB cb) {
return cdm::GetSupportedKeySystemsUpdates(
render_frame, true, std::move(cb));
}
std::unique_ptr<blink::URLLoaderThrottleProvider>
AwContentRendererClient::CreateURLLoaderThrottleProvider(
blink::URLLoaderThrottleProviderType provider_type) {
return std::make_unique<AwURLLoaderThrottleProvider>(
browser_interface_broker_.get(), provider_type);
}
void AwContentRendererClient::GetInterface(
const std::string& interface_name,
mojo::ScopedMessagePipeHandle interface_pipe) {
RenderThread::Get()->BindHostReceiver(
mojo::GenericPendingReceiver(interface_name, std::move(interface_pipe)));
}
mojom::RenderMessageFilter* AwContentRendererClient::GetRenderMessageFilter() {
if (!render_message_filter_) {
browser_interface_broker_->GetInterface(
render_message_filter_.BindNewPipeAndPassReceiver());
}
return render_message_filter_.get();
}
}