#include "content/browser/preloading/prerender/prerender_host_registry.h"
#include <cstdint>
#include "base/strings/string_number_conversions.h"
#include "base/test/bind.h"
#include "base/test/scoped_feature_list.h"
#include "content/browser/preloading/preload_pipeline_info_impl.h"
#include "content/browser/preloading/preloading.h"
#include "content/browser/preloading/preloading_confidence.h"
#include "content/browser/preloading/preloading_config.h"
#include "content/browser/preloading/prerender/prerender_features.h"
#include "content/browser/preloading/prerender/prerender_final_status.h"
#include "content/browser/preloading/prerender/prerender_host.h"
#include "content/browser/preloading/prerender/prerender_metrics.h"
#include "content/browser/preloading/speculation_rules/speculation_host_impl.h"
#include "content/browser/renderer_host/render_frame_host_impl.h"
#include "content/browser/site_instance_impl.h"
#include "content/public/browser/preload_pipeline_info.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/test/preloading_test_util.h"
#include "content/public/test/prerender_test_util.h"
#include "content/public/test/test_browser_context.h"
#include "content/test/mock_commit_deferring_condition.h"
#include "content/test/navigation_simulator_impl.h"
#include "content/test/test_render_view_host.h"
#include "content/test/test_web_contents.h"
#include "net/base/load_flags.h"
#include "third_party/blink/public/common/features.h"
#include "third_party/blink/public/mojom/loader/mixed_content.mojom.h"
#include "third_party/blink/public/mojom/speculation_rules/speculation_rules.mojom.h"
namespace content {
namespace {
blink::mojom::SpeculationCandidatePtr CreatePrerenderCandidate(
const GURL& url) {
auto candidate = blink::mojom::SpeculationCandidate::New();
candidate->action = blink::mojom::SpeculationAction::kPrerender;
candidate->url = url;
candidate->referrer = blink::mojom::Referrer::New();
candidate->eagerness = blink::mojom::SpeculationEagerness::kImmediate;
candidate->tags = {std::nullopt};
return candidate;
}
void SendCandidates(const std::vector<GURL>& urls,
mojo::Remote<blink::mojom::SpeculationHost>& remote) {
std::vector<blink::mojom::SpeculationCandidatePtr> candidates;
candidates.resize(urls.size());
std::ranges::transform(urls, candidates.begin(), &CreatePrerenderCandidate);
remote->UpdateSpeculationCandidates(std::move(candidates), false);
remote.FlushForTesting();
}
void SendCandidate(const GURL& url,
mojo::Remote<blink::mojom::SpeculationHost>& remote) {
SendCandidates({url}, remote);
}
std::unique_ptr<NavigationSimulatorImpl> CreateActivation(
const GURL& prerendering_url,
WebContentsImpl& web_contents) {
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(
prerendering_url, web_contents.GetPrimaryMainFrame());
navigation->SetReferrer(blink::mojom::Referrer::New(
web_contents.GetPrimaryMainFrame()->GetLastCommittedURL(),
network::mojom::ReferrerPolicy::kStrictOriginWhenCrossOrigin));
return navigation;
}
void CommitPrerenderNavigation(PrerenderHost& host) {
FrameTreeNode* ftn = FrameTreeNode::From(host.GetPrerenderedMainFrameHost());
std::unique_ptr<NavigationSimulator> sim =
NavigationSimulatorImpl::CreateFromPendingInFrame(ftn);
sim->Commit();
EXPECT_TRUE(host.is_ready_for_activation());
}
class PrerenderHostRegistryTest : public RenderViewHostImplTestHarness {
public:
PrerenderHostRegistryTest() = default;
~PrerenderHostRegistryTest() override = default;
void SetUp() override {
RenderViewHostImplTestHarness::SetUp();
web_contents_delegate_ =
std::make_unique<test::ScopedPrerenderWebContentsDelegate>(*contents());
contents()->NavigateAndCommit(GURL("https://example.com/"));
}
RenderFrameHostImpl* NavigatePrimaryPage(TestWebContents* web_contents,
const GURL& dest_url) {
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(
dest_url, web_contents->GetPrimaryMainFrame());
navigation->SetTransition(ui::PAGE_TRANSITION_LINK);
navigation->Start();
navigation->Commit();
RenderFrameHostImpl* render_frame_host =
web_contents->GetPrimaryMainFrame();
EXPECT_EQ(render_frame_host->GetLastCommittedURL(), dest_url);
return render_frame_host;
}
[[nodiscard]] bool CheckIsActivatedForParams(
base::OnceCallback<void(NavigationSimulatorImpl*)> setup_callback) {
RenderFrameHostImpl* render_frame_host = contents()->GetPrimaryMainFrame();
const GURL kPrerenderingUrl("https://example.com/next");
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate, render_frame_host));
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl,
render_frame_host);
navigation->SetReferrer(blink::mojom::Referrer::New(
contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
network::mojom::ReferrerPolicy::kStrictOriginWhenCrossOrigin));
std::move(setup_callback).Run(navigation.get());
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
return navigation_request
->is_running_potential_prerender_activation_checks();
}
void SetupPrerenderAndCommit(
base::OnceCallback<void(NavigationSimulatorImpl*)> setup_callback) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindNonReservedHostById(prerender_frame_tree_node_id);
FrameTreeNode* ftn =
FrameTreeNode::From(prerender_host->GetPrerenderedMainFrameHost());
std::unique_ptr<NavigationSimulatorImpl> sim =
NavigationSimulatorImpl::CreateFromPendingInFrame(ftn);
std::move(setup_callback).Run(sim.get());
sim->Commit();
EXPECT_TRUE(prerender_host->is_ready_for_activation());
contents()->ActivatePrerenderedPage(kPrerenderingUrl);
}
PrerenderAttributes GeneratePrerenderAttributes(
const GURL& url,
PreloadingTriggerType trigger_type,
const std::string& embedder_histogram_suffix,
std::optional<blink::mojom::SpeculationEagerness> eagerness,
RenderFrameHostImpl* rfh) {
switch (trigger_type) {
case PreloadingTriggerType::kSpeculationRule:
case PreloadingTriggerType::kSpeculationRuleFromIsolatedWorld:
case PreloadingTriggerType::kSpeculationRuleFromAutoSpeculationRules:
return PrerenderAttributes(
url, trigger_type, embedder_histogram_suffix,
std::make_optional(SpeculationRulesParams(
blink::mojom::SpeculationTargetHint::kNoHint,
eagerness.value_or(
blink::mojom::SpeculationEagerness::kImmediate),
SpeculationRulesTags())),
Referrer(),
std::nullopt, rfh, contents()->GetWeakPtr(),
ui::PAGE_TRANSITION_LINK,
false,
false,
blink::mojom::SpeculationAction::kPrerender,
{},
{},
PreloadPipelineInfoImpl::Create(
PreloadingType::kPrerender),
false,
false);
case PreloadingTriggerType::kEmbedder:
return PrerenderAttributes(
url, trigger_type, embedder_histogram_suffix,
std::nullopt, Referrer(),
std::nullopt,
nullptr, contents()->GetWeakPtr(),
ui::PageTransitionFromInt(ui::PAGE_TRANSITION_TYPED |
ui::PAGE_TRANSITION_FROM_ADDRESS_BAR),
false,
false,
blink::mojom::SpeculationAction::kPrerender,
{},
{},
PreloadPipelineInfoImpl::Create(
PreloadingType::kPrerender),
false,
false);
}
}
void ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus status,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectUniqueSample(
"Prerender.Experimental.PrerenderHostFinalStatus.SpeculationRule",
status, count);
}
void ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus status,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectBucketCount(
"Prerender.Experimental.PrerenderHostFinalStatus.SpeculationRule",
status, count);
}
void ExpectUniqueSampleOfEmbedderFinalStatus(
PrerenderFinalStatus status,
const std::string& embedder_histogram_suffix,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectUniqueSample(
"Prerender.Experimental.PrerenderHostFinalStatus.Embedder_" +
embedder_histogram_suffix,
status, count);
}
void ExpectBucketCountOfEmbedderFinalStatus(
PrerenderFinalStatus status,
const std::string& embedder_histogram_suffix,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectBucketCount(
"Prerender.Experimental.PrerenderHostFinalStatus.Embedder_" +
embedder_histogram_suffix,
status, count);
}
void ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch result,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectUniqueSample(
"Prerender.Experimental.ActivationNavigationParamsMatch."
"SpeculationRule",
result, count);
}
void ExpectBucketCountOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch result,
base::HistogramBase::Count32 count = 1) {
histogram_tester_.ExpectBucketCount(
"Prerender.Experimental.ActivationNavigationParamsMatch."
"SpeculationRule",
result, count);
}
PrerenderHostRegistry& registry() {
return *contents()->GetPrerenderHostRegistry();
}
base::HistogramTester& histogram_tester() { return histogram_tester_; }
private:
test::ScopedPrerenderFeatureList scoped_feature_list_;
base::HistogramTester histogram_tester_;
std::unique_ptr<test::ScopedPrerenderWebContentsDelegate>
web_contents_delegate_;
};
TEST_F(PrerenderHostRegistryTest, CreateAndStartHost_SpeculationRule) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
contents()->ActivatePrerenderedPage(kPrerenderingUrl);
histogram_tester().ExpectTotalCount(
"Navigation.TimeToActivatePrerender.SpeculationRule", 1u);
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHost_Embedder_DirectURLInput) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kEmbedder, "DirectURLInput",
std::nullopt, contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
contents()->ActivatePrerenderedPageFromAddressBar(kPrerenderingUrl);
histogram_tester().ExpectTotalCount(
"Navigation.TimeToActivatePrerender.Embedder_DirectURLInput", 1u);
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHost_PreloadingConfigHoldback) {
content::test::PreloadingConfigOverride preloading_config_override;
preloading_config_override.SetHoldback(
PreloadingType::kPrerender,
content_preloading_predictor::kSpeculationRules, true);
const GURL kPrerenderingUrl("https://example.com/next");
auto* preloading_data = PreloadingData::GetOrCreateForWebContents(contents());
PreloadingURLMatchCallback same_url_matcher =
PreloadingData::GetSameURLMatcher(kPrerenderingUrl);
PreloadingAttempt* preloading_attempt = preloading_data->AddPreloadingAttempt(
content_preloading_predictor::kSpeculationRules,
PreloadingType::kPrerender, std::move(same_url_matcher),
contents()->GetPrimaryMainFrame()->GetPageUkmSourceId());
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(
GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()),
preloading_attempt);
EXPECT_TRUE(prerender_frame_tree_node_id.is_null());
}
TEST_F(PrerenderHostRegistryTest,
CreateAndStartHost_HoldbackOverride_Holdback) {
const GURL kPrerenderingUrl("https://example.com/next");
auto* preloading_data = PreloadingData::GetOrCreateForWebContents(contents());
PreloadingURLMatchCallback same_url_matcher =
PreloadingData::GetSameURLMatcher(kPrerenderingUrl);
PreloadingAttempt* preloading_attempt = preloading_data->AddPreloadingAttempt(
content_preloading_predictor::kSpeculationRules,
PreloadingType::kPrerender, std::move(same_url_matcher),
contents()->GetPrimaryMainFrame()->GetPageUkmSourceId());
auto attributes = GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame());
attributes.holdback_status_override = PreloadingHoldbackStatus::kHoldback;
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(attributes, preloading_attempt);
EXPECT_TRUE(prerender_frame_tree_node_id.is_null());
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHost_HoldbackOverride_Allowed) {
content::test::PreloadingConfigOverride preloading_config_override;
preloading_config_override.SetHoldback(
PreloadingType::kPrerender,
content_preloading_predictor::kSpeculationRules, true);
const GURL kPrerenderingUrl("https://example.com/next");
auto* preloading_data = PreloadingData::GetOrCreateForWebContents(contents());
PreloadingURLMatchCallback same_url_matcher =
PreloadingData::GetSameURLMatcher(kPrerenderingUrl);
PreloadingAttempt* preloading_attempt = preloading_data->AddPreloadingAttempt(
content_preloading_predictor::kSpeculationRules,
PreloadingType::kPrerender, std::move(same_url_matcher),
contents()->GetPrimaryMainFrame()->GetPageUkmSourceId());
auto attributes = GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame());
attributes.holdback_status_override = PreloadingHoldbackStatus::kAllowed;
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(attributes, preloading_attempt);
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
contents()->ActivatePrerenderedPage(kPrerenderingUrl);
histogram_tester().ExpectTotalCount(
"Navigation.TimeToActivatePrerender.SpeculationRule", 1u);
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHostForSameURL) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId frame_tree_node_id1 =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_TRUE(frame_tree_node_id1);
PrerenderHost* prerender_host1 =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
const FrameTreeNodeId frame_tree_node_id2 =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_TRUE(frame_tree_node_id2.is_null());
EXPECT_EQ(registry().FindHostByUrlForTesting(kPrerenderingUrl),
prerender_host1);
CommitPrerenderNavigation(*prerender_host1);
contents()->ActivatePrerenderedPage(kPrerenderingUrl);
}
TEST_F(PrerenderHostRegistryTest, NumberLimit_Activation) {
std::vector<FrameTreeNodeId> frame_tree_node_ids;
std::vector<GURL> prerendering_urls;
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders +
1;
i++) {
const GURL prerendering_url("https://example.com/next" +
base::NumberToString(i));
FrameTreeNodeId frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
prerendering_url, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
frame_tree_node_ids.push_back(frame_tree_node_id);
prerendering_urls.push_back(prerendering_url);
}
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders;
i++) {
EXPECT_TRUE(frame_tree_node_ids[i]);
}
EXPECT_TRUE(
frame_tree_node_ids[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders]
.is_null());
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(prerendering_urls[0]);
CommitPrerenderNavigation(*prerender_host);
contents()->ActivatePrerenderedPage(prerendering_urls[0]);
FrameTreeNodeId frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
prerendering_urls[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders],
PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_TRUE(frame_tree_node_id);
ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
}
TEST_F(PrerenderHostRegistryTest, NumberLimit_SameOriginNavigateAway) {
RenderFrameHostImpl* render_frame_host = contents()->GetPrimaryMainFrame();
ASSERT_TRUE(render_frame_host);
mojo::Remote<blink::mojom::SpeculationHost> remote1;
SpeculationHostImpl::Bind(render_frame_host,
remote1.BindNewPipeAndPassReceiver());
ASSERT_TRUE(remote1.is_connected());
std::vector<GURL> prerendering_urls;
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders +
1;
i++) {
prerendering_urls.emplace_back("https://example.com/next" +
base::NumberToString(i));
}
SendCandidates(prerendering_urls, remote1);
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders;
i++) {
ASSERT_NE(registry().FindHostByUrlForTesting(prerendering_urls[i]),
nullptr);
}
ASSERT_EQ(registry().FindHostByUrlForTesting(
prerendering_urls
[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders]),
nullptr);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
render_frame_host =
NavigatePrimaryPage(contents(), GURL("https://example.com/elsewhere"));
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders +
1;
i++) {
EXPECT_EQ(registry().FindHostByUrlForTesting(prerendering_urls[i]),
nullptr);
}
mojo::Remote<blink::mojom::SpeculationHost> remote2;
SpeculationHostImpl::Bind(render_frame_host,
remote2.BindNewPipeAndPassReceiver());
SendCandidate(
prerendering_urls[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders],
remote2);
EXPECT_NE(registry().FindHostByUrlForTesting(
prerendering_urls
[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders]),
nullptr);
ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
}
TEST_F(PrerenderHostRegistryTest, NumberLimit_CrossOriginNavigateAway) {
RenderFrameHostImpl* render_frame_host = contents()->GetPrimaryMainFrame();
ASSERT_TRUE(render_frame_host);
mojo::Remote<blink::mojom::SpeculationHost> remote1;
SpeculationHostImpl::Bind(render_frame_host,
remote1.BindNewPipeAndPassReceiver());
ASSERT_TRUE(remote1.is_connected());
std::vector<GURL> prerendering_urls;
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders +
1;
i++) {
prerendering_urls.emplace_back("https://example.com/next" +
base::NumberToString(i));
}
SendCandidates(prerendering_urls, remote1);
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders;
i++) {
ASSERT_NE(registry().FindHostByUrlForTesting(prerendering_urls[i]),
nullptr);
}
ASSERT_EQ(registry().FindHostByUrlForTesting(
prerendering_urls
[PrerenderHostRegistry::
kMaxRunningSpeculationRulesImmediatePrerenders]),
nullptr);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
render_frame_host =
NavigatePrimaryPage(contents(), GURL("https://example.org/"));
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders +
1;
i++) {
EXPECT_EQ(registry().FindHostByUrlForTesting(prerendering_urls[i]),
nullptr);
}
mojo::Remote<blink::mojom::SpeculationHost> remote2;
SpeculationHostImpl::Bind(render_frame_host,
remote2.BindNewPipeAndPassReceiver());
const GURL prerendering_url("https://example.org/next");
SendCandidate(prerendering_url, remote2);
EXPECT_NE(registry().FindHostByUrlForTesting(prerendering_url), nullptr);
ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded);
}
class PrerenderHostRegistryLimitGroupTest
: public PrerenderHostRegistryTest,
public testing::WithParamInterface<bool> {
public:
using PrerenderLimitGroup = PrerenderHostRegistry::PrerenderLimitGroup;
const std::string embedder_histogram_suffix = "EmbedderSuffixForTest";
bool IsNewTabTrigger(PrerenderLimitGroup limit_group) {
return GetParam() && limit_group != PrerenderLimitGroup::kEmbedder;
}
FrameTreeNodeId CreateAndStartHostByLimitGroup(
PrerenderLimitGroup limit_group) {
static int unique_id = 0;
const GURL prerendering_url("https://example.com/next_" +
base::NumberToString(unique_id));
unique_id++;
auto prerender_attributes = [&] {
switch (limit_group) {
case PrerenderLimitGroup::kSpeculationRulesImmediate:
return GeneratePrerenderAttributes(
prerendering_url, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame());
case PrerenderLimitGroup::kSpeculationRulesNonImmediate:
return GeneratePrerenderAttributes(
prerendering_url, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kModerate,
contents()->GetPrimaryMainFrame());
case PrerenderLimitGroup::kEmbedder:
return GeneratePrerenderAttributes(
prerendering_url, PreloadingTriggerType::kEmbedder,
embedder_histogram_suffix, std::nullopt, nullptr);
}
}();
PreloadingPredictor embedder_predictor(100, "Embedder");
PreloadingPredictor creating_predictor = [&] {
switch (limit_group) {
case PrerenderLimitGroup::kSpeculationRulesImmediate:
case PrerenderLimitGroup::kSpeculationRulesNonImmediate:
return content_preloading_predictor::kSpeculationRules;
case PrerenderLimitGroup::kEmbedder:
return embedder_predictor;
}
}();
PreloadingPredictor enacting_predictor = [&] {
switch (limit_group) {
case PrerenderLimitGroup::kSpeculationRulesImmediate:
return content_preloading_predictor::kSpeculationRules;
case PrerenderLimitGroup::kSpeculationRulesNonImmediate:
return preloading_predictor::kUrlPointerDownOnAnchor;
case PrerenderLimitGroup::kEmbedder:
return embedder_predictor;
}
}();
return IsNewTabTrigger(limit_group)
? registry().CreateAndStartHostForNewTab(
prerender_attributes, creating_predictor,
enacting_predictor, PreloadingConfidence{100})
: registry().CreateAndStartHost(prerender_attributes);
}
};
INSTANTIATE_TEST_SUITE_P(All,
PrerenderHostRegistryLimitGroupTest,
testing::Bool());
TEST_P(PrerenderHostRegistryLimitGroupTest, Immediate) {
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesImmediatePrerenders;
i++) {
FrameTreeNodeId frame_tree_node_id = CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesImmediate);
EXPECT_TRUE(frame_tree_node_id);
}
FrameTreeNodeId frame_tree_node_id_immediate_exceeded =
CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesImmediate);
EXPECT_TRUE(frame_tree_node_id_immediate_exceeded.is_null());
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded, 1);
FrameTreeNodeId frame_tree_node_id_non_immediate =
CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesNonImmediate);
FrameTreeNodeId frame_tree_node_id_embedder =
CreateAndStartHostByLimitGroup(PrerenderLimitGroup::kEmbedder);
EXPECT_TRUE(frame_tree_node_id_non_immediate);
EXPECT_TRUE(frame_tree_node_id_embedder);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded, 1);
ExpectUniqueSampleOfEmbedderFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningEmbedderPrerendersExceeded,
embedder_histogram_suffix, 0);
}
TEST_P(PrerenderHostRegistryLimitGroupTest, NonImmediate) {
std::vector<FrameTreeNodeId> started_prerender_ids;
for (int i = 0;
i <
PrerenderHostRegistry::kMaxRunningSpeculationRulesNonImmediatePrerenders;
i++) {
FrameTreeNodeId frame_tree_node_id = CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesNonImmediate);
started_prerender_ids.push_back(frame_tree_node_id);
EXPECT_TRUE(frame_tree_node_id);
}
FrameTreeNodeId frame_tree_node_id_non_immediate_exceeded =
CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesNonImmediate);
ASSERT_TRUE(frame_tree_node_id_non_immediate_exceeded);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningNonImmediatePrerendersExceeded, 1);
for (auto id : started_prerender_ids) {
auto* web_contents_impl =
static_cast<WebContentsImpl*>(WebContents::FromFrameTreeNodeId(id));
PrerenderHost* prerender_host = nullptr;
if (web_contents_impl) {
prerender_host = web_contents_impl->GetPrerenderHostRegistry()
->FindNonReservedHostById(id);
}
if (id == started_prerender_ids[0]) {
EXPECT_EQ(prerender_host, nullptr);
} else {
EXPECT_NE(prerender_host, nullptr);
}
}
FrameTreeNodeId frame_tree_node_id_immediate = CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesImmediate);
FrameTreeNodeId frame_tree_node_id_embedder =
CreateAndStartHostByLimitGroup(PrerenderLimitGroup::kEmbedder);
EXPECT_TRUE(frame_tree_node_id_immediate);
EXPECT_TRUE(frame_tree_node_id_embedder);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningNonImmediatePrerendersExceeded, 1);
ExpectUniqueSampleOfEmbedderFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningEmbedderPrerendersExceeded,
embedder_histogram_suffix, 0);
}
TEST_P(PrerenderHostRegistryLimitGroupTest, Embedder) {
const int max_embedder_prerenders =
web_contents()->GetDelegate()->AllowedPrerenderingCount(*web_contents());
for (int i = 0; i < max_embedder_prerenders; i++) {
FrameTreeNodeId frame_tree_node_id =
CreateAndStartHostByLimitGroup(PrerenderLimitGroup::kEmbedder);
EXPECT_TRUE(frame_tree_node_id);
}
FrameTreeNodeId frame_tree_node_id_embedder_exceeded =
CreateAndStartHostByLimitGroup(PrerenderLimitGroup::kEmbedder);
EXPECT_TRUE(frame_tree_node_id_embedder_exceeded.is_null());
ExpectUniqueSampleOfEmbedderFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningEmbedderPrerendersExceeded,
embedder_histogram_suffix, 1);
FrameTreeNodeId frame_tree_node_id_immediate = CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesImmediate);
FrameTreeNodeId frame_tree_node_id_non_immediate =
CreateAndStartHostByLimitGroup(
PrerenderLimitGroup::kSpeculationRulesNonImmediate);
EXPECT_TRUE(frame_tree_node_id_immediate);
EXPECT_TRUE(frame_tree_node_id_non_immediate);
ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningImmediatePrerendersExceeded, 0);
ExpectBucketCountOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningNonImmediatePrerendersExceeded, 0);
ExpectUniqueSampleOfEmbedderFinalStatus(
PrerenderFinalStatus::kMaxNumOfRunningEmbedderPrerendersExceeded,
embedder_histogram_suffix, 1);
}
TEST_F(PrerenderHostRegistryTest,
ReserveHostToActivateBeforeReadyForActivation) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
FrameTreeNode* ftn =
FrameTreeNode::From(prerender_host->GetPrerenderedMainFrameHost());
std::unique_ptr<NavigationSimulatorImpl> sim =
NavigationSimulatorImpl::CreateFromPendingInFrame(ftn);
sim->SetAutoAdvance(false);
EXPECT_FALSE(prerender_host->is_ready_for_activation());
test::PrerenderHostObserver prerender_host_observer(*contents(),
kPrerenderingUrl);
std::unique_ptr<NavigationSimulatorImpl> navigation =
CreateActivation(kPrerenderingUrl, *contents());
navigation->Start();
base::RunLoop().RunUntilIdle();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
EXPECT_TRUE(
navigation_request->IsCommitDeferringConditionDeferredForTesting());
EXPECT_FALSE(prerender_host_observer.was_activated());
EXPECT_EQ(contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
original_url);
sim->Commit();
prerender_host_observer.WaitForDestroyed();
EXPECT_TRUE(prerender_host_observer.was_activated());
EXPECT_EQ(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
EXPECT_EQ(contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
kPrerenderingUrl);
}
TEST_F(PrerenderHostRegistryTest, CancelHost) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_NE(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
registry().CancelHost(prerender_frame_tree_node_id,
PrerenderFinalStatus::kDestroyed);
EXPECT_EQ(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
TEST_F(PrerenderHostRegistryTest,
CancelHostWhileCommitDeferringConditionIsRunning) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
test::PrerenderHostObserver prerender_host_observer(*contents(),
kPrerenderingUrl);
std::unique_ptr<NavigationSimulatorImpl> navigation;
{
MockCommitDeferringConditionInstaller installer(
kPrerenderingUrl, CommitDeferringCondition::Result::kDefer);
navigation = CreateActivation(kPrerenderingUrl, *contents());
navigation->Start();
installer.WaitUntilInstalled();
installer.condition().WaitUntilInvoked();
auto* navigation_request =
static_cast<NavigationRequest*>(navigation->GetNavigationHandle());
EXPECT_TRUE(
navigation_request->IsCommitDeferringConditionDeferredForTesting());
EXPECT_EQ(contents()->GetLastCommittedURL(), original_url);
registry().CancelHost(prerender_frame_tree_node_id,
PrerenderFinalStatus::kDestroyed);
prerender_host_observer.WaitForDestroyed();
EXPECT_FALSE(prerender_host_observer.was_activated());
EXPECT_EQ(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
installer.condition().CallResumeClosure();
}
navigation->Commit();
EXPECT_EQ(contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
kPrerenderingUrl);
}
TEST_F(PrerenderHostRegistryTest,
CancelAndStartHostWhileCommitDeferringConditionIsRunning) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
test::PrerenderHostObserver prerender_host_observer(*contents(),
kPrerenderingUrl);
std::unique_ptr<NavigationSimulatorImpl> navigation;
base::OnceClosure resume_navigation;
{
MockCommitDeferringConditionInstaller installer(
kPrerenderingUrl, CommitDeferringCondition::Result::kDefer);
navigation = CreateActivation(kPrerenderingUrl, *contents());
navigation->Start();
installer.WaitUntilInstalled();
installer.condition().WaitUntilInvoked();
resume_navigation = installer.condition().TakeResumeClosure();
auto* navigation_request =
static_cast<NavigationRequest*>(navigation->GetNavigationHandle());
EXPECT_TRUE(
navigation_request->IsCommitDeferringConditionDeferredForTesting());
EXPECT_EQ(contents()->GetLastCommittedURL(), original_url);
registry().CancelHost(prerender_frame_tree_node_id,
PrerenderFinalStatus::kDestroyed);
prerender_host_observer.WaitForDestroyed();
EXPECT_FALSE(prerender_host_observer.was_activated());
EXPECT_EQ(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
{
const FrameTreeNodeId prerender_frame_tree_node_id2 =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id2);
PrerenderHost* prerender_host2 =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host2);
EXPECT_NE(prerender_frame_tree_node_id, prerender_frame_tree_node_id2);
}
std::move(resume_navigation).Run();
navigation->Commit();
EXPECT_EQ(contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
kPrerenderingUrl);
EXPECT_NE(registry().FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
TEST_F(PrerenderHostRegistryTest,
DontStartPrerenderWhenEmbedderTriggerIsAlreadyHidden) {
contents()->WasHidden();
const GURL kPrerenderingUrl = GURL("https://example.com/empty.html");
RenderFrameHostImpl* initiator_rfh = contents()->GetPrimaryMainFrame();
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kEmbedder, "DirectURLInput",
std::nullopt, initiator_rfh));
EXPECT_TRUE(prerender_frame_tree_node_id.is_null());
PrerenderHost* prerender_host =
registry().FindNonReservedHostById(prerender_frame_tree_node_id);
EXPECT_EQ(prerender_host, nullptr);
histogram_tester().ExpectUniqueSample(
"Prerender.Experimental.PrerenderHostFinalStatus.Embedder_DirectURLInput",
PrerenderFinalStatus::kTriggerBackgrounded, 1u);
}
TEST_F(PrerenderHostRegistryTest, SameInitialAndActivationParams) {
EXPECT_TRUE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kOk);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_InitiatorFrameToken) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
const GURL kOriginalUrl("https://example.com/");
navigation->SetInitiatorFrame(nullptr);
navigation->set_initiator_origin(url::Origin::Create(kOriginalUrl));
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kInitiatorFrameToken);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_Headers) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_request_headers("User-Agent: Test");
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kHttpRequestHeader);
}
TEST_F(PrerenderHostRegistryTest, PurposeHeaderIsIgnoredForParamMatching) {
EXPECT_TRUE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_request_headers("Purpose: Test");
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kOk);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_LoadFlags) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_load_flags(net::LOAD_ONLY_FROM_CACHE);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kLoadFlags);
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_load_flags(net::LOAD_VALIDATE_CACHE);
})));
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_load_flags(net::LOAD_BYPASS_CACHE);
})));
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_load_flags(net::LOAD_DISABLE_CACHE);
})));
ExpectBucketCountOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kCacheLoadFlags, 3);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_SkipServiceWorker) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_skip_service_worker(true);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kSkipServiceWorker);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_MixedContentContextType) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_mixed_content_context_type(
blink::mojom::MixedContentContextType::kNotMixedContent);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kMixedContentContextType);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_IsFormSubmission) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->SetIsFormSubmission(true);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kIsFormSubmission);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_SearchableFormUrl) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
const GURL kOriginalUrl("https://example.com/");
navigation->set_searchable_form_url(kOriginalUrl);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kSearchableFormUrl);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationBeginParams_SearchableFormEncoding) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_searchable_form_encoding("Test encoding");
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kSearchableFormEncoding);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_InitiatorOrigin) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_initiator_origin(url::Origin());
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kInitiatorOrigin);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_ShouldNotCheckMainWorldCSP) {
EXPECT_TRUE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_should_check_main_world_csp(
network::mojom::CSPDisposition::DO_NOT_CHECK);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kOk);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_Method) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->SetMethod("POST");
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kHttpRequestHeader);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_HrefTranslate) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_href_translate("test");
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kHrefTranslate);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_Transition) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->SetTransition(ui::PAGE_TRANSITION_FORM_SUBMIT);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kTransition);
histogram_tester().ExpectUniqueSample(
"Prerender.Experimental.ActivationTransitionMismatch.SpeculationRule",
ui::PAGE_TRANSITION_FORM_SUBMIT, 1);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_RequestContextType) {
EXPECT_FALSE(CheckIsActivatedForParams(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_request_context_type(
blink::mojom::RequestContextType::AUDIO);
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kRequestContextType);
}
TEST_F(PrerenderHostRegistryTest,
CompareInitialAndActivationCommonParams_ReferrerPolicy) {
EXPECT_TRUE(CheckIsActivatedForParams(
base::BindLambdaForTesting([&](NavigationSimulatorImpl* navigation) {
navigation->SetReferrer(blink::mojom::Referrer::New(
contents()->GetPrimaryMainFrame()->GetLastCommittedURL(),
network::mojom::ReferrerPolicy::kAlways));
})));
ExpectUniqueSampleOfActivationNavigationParamsMatch(
PrerenderHost::ActivationNavigationParamsMatch::kOk);
}
TEST_F(PrerenderHostRegistryTest, InsecureRequestPolicyIsSetWhilePrerendering) {
SetupPrerenderAndCommit(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_insecure_request_policy(
blink::mojom::InsecureRequestPolicy::kBlockAllMixedContent);
}));
EXPECT_EQ(contents()
->GetPrimaryMainFrame()
->frame_tree_node()
->current_replication_state()
.insecure_request_policy,
blink::mojom::InsecureRequestPolicy::kBlockAllMixedContent);
}
TEST_F(PrerenderHostRegistryTest,
InsecureNavigationsSetIsSetWhilePrerendering) {
SetupPrerenderAndCommit(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
const std::vector<uint32_t> insecure_navigations = {1, 2};
navigation->set_insecure_navigations_set(insecure_navigations);
}));
const std::vector<uint32_t> insecure_navigations = {1, 2};
EXPECT_EQ(contents()
->GetPrimaryMainFrame()
->frame_tree_node()
->current_replication_state()
.insecure_navigations_set,
insecure_navigations);
}
TEST_F(PrerenderHostRegistryTest,
HasPotentiallyTrustworthyUniqueOriginIsSetWhilePrerendering) {
SetupPrerenderAndCommit(
base::BindLambdaForTesting([](NavigationSimulatorImpl* navigation) {
navigation->set_has_potentially_trustworthy_unique_origin(true);
}));
EXPECT_TRUE(contents()
->GetPrimaryMainFrame()
->frame_tree_node()
->current_replication_state()
.has_potentially_trustworthy_unique_origin);
}
TEST_F(PrerenderHostRegistryTest, OneTaskToDeleteAllHosts) {
std::vector<FrameTreeNodeId> frame_tree_node_ids;
std::vector<std::unique_ptr<test::PrerenderHostObserver>>
prerender_host_observers;
for (int i = 0; i < 2; i++) {
const GURL prerendering_url("https://example.com/next" +
base::NumberToString(i));
FrameTreeNodeId frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
prerendering_url, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
prerender_host_observers.emplace_back(
std::make_unique<test::PrerenderHostObserver>(*contents(),
frame_tree_node_id));
frame_tree_node_ids.push_back(frame_tree_node_id);
}
int pending_task_before_posting_abandon_task =
task_environment()->GetPendingMainThreadTaskCount();
registry().CancelHosts(
frame_tree_node_ids,
PrerenderCancellationReason(PrerenderFinalStatus::kDestroyed));
int pending_task_after_posting_abandon_task =
task_environment()->GetPendingMainThreadTaskCount();
EXPECT_EQ(pending_task_before_posting_abandon_task + 1,
pending_task_after_posting_abandon_task);
for (auto& observer : prerender_host_observers) {
observer->WaitForDestroyed();
}
}
TEST_F(PrerenderHostRegistryTest, DisallowPageHavingEffectiveUrl_TriggerUrl) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kModifiedSiteUrl("custom-scheme://custom");
EffectiveURLContentBrowserClient modified_client(
original_url, kModifiedSiteUrl,
false);
ContentBrowserClient* old_client =
SetBrowserClientForTesting(&modified_client);
const GURL kPrerenderingUrl("https://example.com/empty.html");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_TRUE(prerender_frame_tree_node_id.is_null());
PrerenderHost* prerender_host =
registry().FindNonReservedHostById(prerender_frame_tree_node_id);
EXPECT_EQ(prerender_host, nullptr);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kTriggerUrlHasEffectiveUrl);
SetBrowserClientForTesting(old_client);
}
TEST_F(PrerenderHostRegistryTest,
DisallowPageHavingEffectiveUrl_PrerenderingUrl) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kPrerenderingUrl("https://example.com/empty.html");
const GURL kModifiedSiteUrl("custom-scheme://custom");
EffectiveURLContentBrowserClient modified_client(
kPrerenderingUrl, kModifiedSiteUrl,
false);
ContentBrowserClient* old_client =
SetBrowserClientForTesting(&modified_client);
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
EXPECT_TRUE(prerender_frame_tree_node_id.is_null());
PrerenderHost* prerender_host =
registry().FindNonReservedHostById(prerender_frame_tree_node_id);
EXPECT_EQ(prerender_host, nullptr);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kPrerenderingUrlHasEffectiveUrl);
SetBrowserClientForTesting(old_client);
}
TEST_F(PrerenderHostRegistryTest,
DisallowPageHavingEffectiveUrl_ActivationUrl) {
const GURL original_url = contents()->GetLastCommittedURL();
const GURL kPrerenderingUrl("https://example.com/empty.html");
const GURL kModifiedSiteUrl("custom-scheme://custom");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
EffectiveURLContentBrowserClient modified_client(
kPrerenderingUrl, kModifiedSiteUrl,
false);
ContentBrowserClient* old_client =
SetBrowserClientForTesting(&modified_client);
contents()->NavigateAndCommit(kPrerenderingUrl);
ExpectUniqueSampleOfSpeculationRuleFinalStatus(
PrerenderFinalStatus::kActivationUrlHasEffectiveUrl);
SetBrowserClientForTesting(old_client);
}
TEST_F(PrerenderHostRegistryTest, PotentialPrerenderProcessReuseUMA) {
const GURL kPrerenderingUrl("https://example.com/next");
const FrameTreeNodeId prerender_frame_tree_node_id =
registry().CreateAndStartHost(GeneratePrerenderAttributes(
kPrerenderingUrl, PreloadingTriggerType::kSpeculationRule, "",
blink::mojom::SpeculationEagerness::kImmediate,
contents()->GetPrimaryMainFrame()));
ASSERT_TRUE(prerender_frame_tree_node_id);
PrerenderHost* prerender_host =
registry().FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
{
base::HistogramTester histogram_tester;
auto navigation = CreateActivation(kPrerenderingUrl, *contents());
navigation->Start();
histogram_tester.ExpectUniqueSample(
"Prerender.Experimental.PrerenderProcessReuseAvailability",
PrerenderHostRegistry::PrerenderProcessReuseAvailability::
kHasMatchableHosts,
1);
}
{
base::HistogramTester histogram_tester;
const GURL kSameSiteUrl("https://example.com/other");
auto navigation = CreateActivation(kSameSiteUrl, *contents());
navigation->Start();
histogram_tester.ExpectUniqueSample(
"Prerender.Experimental.PrerenderProcessReuseAvailability",
PrerenderHostRegistry::PrerenderProcessReuseAvailability::
kHasSameOriginHosts,
1);
}
{
base::HistogramTester histogram_tester;
const GURL kSameSiteUrl("https://www.example.com:8000/other");
auto navigation = CreateActivation(kSameSiteUrl, *contents());
navigation->Start();
histogram_tester.ExpectUniqueSample(
"Prerender.Experimental.PrerenderProcessReuseAvailability",
PrerenderHostRegistry::PrerenderProcessReuseAvailability::
kHasSameSiteHosts,
1);
}
{
base::HistogramTester histogram_tester;
const GURL kNotSameSiteUrl("https://another.com/other");
auto navigation = CreateActivation(kNotSameSiteUrl, *contents());
navigation->Start();
histogram_tester.ExpectUniqueSample(
"Prerender.Experimental.PrerenderProcessReuseAvailability",
PrerenderHostRegistry::PrerenderProcessReuseAvailability::
kNoSameOriginOrSiteHosts,
1);
}
}
}
}