#include "content/browser/preloading/speculation_rules/speculation_host_impl.h"
#include <functional>
#include "base/feature_list.h"
#include "base/strings/string_util.h"
#include "content/browser/preloading/prefetch/prefetch_document_manager.h"
#include "content/browser/preloading/preloading_decider.h"
#include "content/public/browser/web_contents.h"
#include "content/public/browser/web_contents_delegate.h"
#include "third_party/blink/public/common/features.h"
namespace content {
namespace {
bool CandidatesAreValid(
std::vector<blink::mojom::SpeculationCandidatePtr>& candidates) {
for (const auto& candidate : candidates) {
if (!candidate->url.SchemeIsHTTPOrHTTPS()) {
mojo::ReportBadMessage("SH_NON_HTTP");
return false;
}
if (candidate->action != blink::mojom::SpeculationAction::kPrerender &&
candidate->action !=
blink::mojom::SpeculationAction::kPrerenderUntilScript &&
candidate->target_browsing_context_name_hint !=
blink::mojom::SpeculationTargetHint::kNoHint) {
mojo::ReportBadMessage("SH_TARGET_HINT_ON_PREFETCH");
return false;
}
if (candidate->action != blink::mojom::SpeculationAction::kPrefetch &&
candidate->requires_anonymous_client_ip_when_cross_origin) {
mojo::ReportBadMessage(
"SH_INVALID_REQUIRES_ANONYMOUS_CLIENT_IP_WHEN_CROSS_ORIGIN");
return false;
}
if (candidate->tags.empty()) {
mojo::ReportBadMessage("SH_EMPTY_TAGS");
return false;
}
for (auto& tag : candidate->tags) {
if (tag.has_value() &&
!std::all_of(tag.value().begin(), tag.value().end(),
base::IsAsciiPrintable<char>)) {
mojo::ReportBadMessage("SH_INVALID_TAG");
return false;
}
}
}
return true;
}
}
void SpeculationHostImpl::Bind(
RenderFrameHost* frame_host,
mojo::PendingReceiver<blink::mojom::SpeculationHost> receiver) {
CHECK(frame_host);
new SpeculationHostImpl(*frame_host, std::move(receiver));
}
SpeculationHostImpl::SpeculationHostImpl(
RenderFrameHost& frame_host,
mojo::PendingReceiver<blink::mojom::SpeculationHost> receiver)
: DocumentService(frame_host, std::move(receiver)) {}
SpeculationHostImpl::~SpeculationHostImpl() = default;
void SpeculationHostImpl::UpdateSpeculationCandidates(
std::vector<blink::mojom::SpeculationCandidatePtr> candidates,
bool enable_cross_origin_prerender_iframes) {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
if (!CandidatesAreValid(candidates))
return;
if (!render_frame_host().IsActive())
return;
if (render_frame_host().GetParent())
return;
auto* preloading_decider =
PreloadingDecider::GetOrCreateForCurrentDocument(&render_frame_host());
preloading_decider->UpdateSpeculationCandidates(
candidates, enable_cross_origin_prerender_iframes);
}
void SpeculationHostImpl::OnLCPPredicted() {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
auto* preloading_decider =
PreloadingDecider::GetOrCreateForCurrentDocument(&render_frame_host());
preloading_decider->OnLCPPredicted();
}
void SpeculationHostImpl::InitiatePreview(const GURL& url) {
if (!base::FeatureList::IsEnabled(blink::features::kLinkPreview)) {
mojo::ReportBadMessage("SH_PREVIEW");
return;
}
if (render_frame_host().IsUntrustedNetworkDisabled()) {
return;
}
WebContents* web_contents =
WebContents::FromRenderFrameHost(&render_frame_host());
CHECK(web_contents);
WebContentsDelegate* delegate = web_contents->GetDelegate();
CHECK(delegate);
delegate->InitiatePreview(*web_contents, url);
}
}