blob: 16c1a7b3e476d8a5b34c72338fe567d6ceb1f78d [file] [log] [blame]
// Copyright 2020 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 "content/browser/prerender/prerender_host_registry.h"
#include "base/test/scoped_feature_list.h"
#include "content/browser/prerender/prerender_host.h"
#include "content/browser/renderer_host/render_frame_host_impl.h"
#include "content/browser/site_instance_impl.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/test/test_browser_context.h"
#include "content/test/navigation_simulator_impl.h"
#include "content/test/test_render_view_host.h"
#include "content/test/test_web_contents.h"
#include "third_party/blink/public/common/features.h"
namespace content {
namespace {
// This definition is needed because this constant is odr-used in gtest macros.
// https://en.cppreference.com/w/cpp/language/static#Constant_static_members
const int kNoFrameTreeNodeId = RenderFrameHost::kNoFrameTreeNodeId;
class PrerenderHostRegistryTest : public RenderViewHostImplTestHarness {
public:
PrerenderHostRegistryTest() = default;
~PrerenderHostRegistryTest() override = default;
void SetUp() override {
scoped_feature_list_.InitAndEnableFeature(blink::features::kPrerender2);
RenderViewHostImplTestHarness::SetUp();
browser_context_ = std::make_unique<TestBrowserContext>();
}
void TearDown() override {
browser_context_.reset();
RenderViewHostImplTestHarness::TearDown();
}
std::unique_ptr<TestWebContents> CreateWebContents(const GURL& url) {
std::unique_ptr<TestWebContents> web_contents(TestWebContents::Create(
browser_context_.get(),
SiteInstanceImpl::Create(browser_context_.get())));
web_contents->NavigateAndCommit(url);
return web_contents;
}
private:
base::test::ScopedFeatureList scoped_feature_list_;
std::unique_ptr<TestBrowserContext> browser_context_;
};
// Finish a prerendering navigation that was already started with
// CreateAndStartHost().
void CommitPrerenderNavigation(PrerenderHost& host) {
// Normally we could use EmbeddedTestServer to provide a response, but these
// tests use RenderViewHostImplTestHarness so the load goes through a
// TestNavigationURLLoader which we don't have access to in order to
// complete. Use NavigationSimulator to finish the navigation.
FrameTreeNode* ftn = FrameTreeNode::From(host.GetPrerenderedMainFrameHost());
std::unique_ptr<NavigationSimulator> sim =
NavigationSimulatorImpl::CreateFromPendingInFrame(ftn);
sim->ReadyToCommit();
sim->Commit();
EXPECT_TRUE(host.is_ready_for_activation());
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHost) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl("https://example.com/next");
auto attributes = blink::mojom::PrerenderAttributes::New();
attributes->url = kPrerenderingUrl;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int prerender_frame_tree_node_id =
registry->CreateAndStartHost(std::move(attributes), *render_frame_host);
ASSERT_NE(prerender_frame_tree_node_id, kNoFrameTreeNodeId);
PrerenderHost* prerender_host =
registry->FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl,
render_frame_host);
// Implicitly will call ReserveHostToActivate.
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
EXPECT_EQ(navigation_request->prerender_frame_tree_node_id(),
prerender_frame_tree_node_id);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
registry->OnActivationFinished(prerender_frame_tree_node_id);
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHostForSameURL) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl("https://example.com/next");
auto attributes1 = blink::mojom::PrerenderAttributes::New();
attributes1->url = kPrerenderingUrl;
auto attributes2 = blink::mojom::PrerenderAttributes::New();
attributes2->url = kPrerenderingUrl;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int frame_tree_node_id1 =
registry->CreateAndStartHost(std::move(attributes1), *render_frame_host);
PrerenderHost* prerender_host1 =
registry->FindHostByUrlForTesting(kPrerenderingUrl);
// Start the prerender host for the same URL. This second host should be
// ignored, and the first host should still be findable.
const int frame_tree_node_id2 =
registry->CreateAndStartHost(std::move(attributes2), *render_frame_host);
EXPECT_EQ(frame_tree_node_id1, frame_tree_node_id2);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl),
prerender_host1);
CommitPrerenderNavigation(*prerender_host1);
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl,
render_frame_host);
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
EXPECT_EQ(navigation_request->prerender_frame_tree_node_id(),
frame_tree_node_id1);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
registry->OnActivationFinished(frame_tree_node_id1);
}
TEST_F(PrerenderHostRegistryTest, CreateAndStartHostForDifferentURLs) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl1("https://example.com/next1");
auto attributes1 = blink::mojom::PrerenderAttributes::New();
attributes1->url = kPrerenderingUrl1;
const GURL kPrerenderingUrl2("https://example.com/next2");
auto attributes2 = blink::mojom::PrerenderAttributes::New();
attributes2->url = kPrerenderingUrl2;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int frame_tree_node_id1 =
registry->CreateAndStartHost(std::move(attributes1), *render_frame_host);
const int frame_tree_node_id2 =
registry->CreateAndStartHost(std::move(attributes2), *render_frame_host);
EXPECT_NE(frame_tree_node_id1, frame_tree_node_id2);
PrerenderHost* prerender_host1 =
registry->FindHostByUrlForTesting(kPrerenderingUrl1);
PrerenderHost* prerender_host2 =
registry->FindHostByUrlForTesting(kPrerenderingUrl2);
CommitPrerenderNavigation(*prerender_host1);
CommitPrerenderNavigation(*prerender_host2);
// Select the first host.
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl1,
render_frame_host);
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
EXPECT_EQ(navigation_request->prerender_frame_tree_node_id(),
frame_tree_node_id1);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl1), nullptr);
// The second host should still be findable.
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl2),
prerender_host2);
// Select the second host.
std::unique_ptr<NavigationSimulatorImpl> navigation2 =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl2,
render_frame_host);
navigation2->Start();
NavigationRequest* navigation_request2 = navigation2->GetNavigationHandle();
EXPECT_EQ(navigation_request2->prerender_frame_tree_node_id(),
frame_tree_node_id2);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl2), nullptr);
registry->OnActivationFinished(frame_tree_node_id1);
registry->OnActivationFinished(frame_tree_node_id2);
}
TEST_F(PrerenderHostRegistryTest,
ReserveHostToActivateBeforeReadyForActivation) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl("https://example.com/next");
auto attributes = blink::mojom::PrerenderAttributes::New();
attributes->url = kPrerenderingUrl;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int prerender_frame_tree_node_id =
registry->CreateAndStartHost(std::move(attributes), *render_frame_host);
ASSERT_NE(prerender_frame_tree_node_id, kNoFrameTreeNodeId);
PrerenderHost* prerender_host =
registry->FindHostByUrlForTesting(kPrerenderingUrl);
FrameTreeNode* ftn =
FrameTreeNode::From(prerender_host->GetPrerenderedMainFrameHost());
std::unique_ptr<NavigationSimulatorImpl> sim =
NavigationSimulatorImpl::CreateFromPendingInFrame(ftn);
// Ensure that navigation in prerendering frame tree does not commit and
// PrerenderHost doesn't become ready for activation.
sim->SetAutoAdvance(false);
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl,
render_frame_host);
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
// The prerender host is not ready for activation yet, the registry
// should still continue to select the host as the activation is deferred
// until the host is ready for activation.
EXPECT_EQ(navigation_request->prerender_frame_tree_node_id(),
prerender_frame_tree_node_id);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
TEST_F(PrerenderHostRegistryTest, CancelHost) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl("https://example.com/next");
auto attributes = blink::mojom::PrerenderAttributes::New();
attributes->url = kPrerenderingUrl;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int prerender_frame_tree_node_id =
registry->CreateAndStartHost(std::move(attributes), *render_frame_host);
EXPECT_NE(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
registry->CancelHost(prerender_frame_tree_node_id,
PrerenderHost::FinalStatus::kDestroyed);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
TEST_F(PrerenderHostRegistryTest,
CompareLoadAndActivationBeginParams_SkipServiceWorker) {
std::unique_ptr<TestWebContents> web_contents =
CreateWebContents(GURL("https://example.com/"));
RenderFrameHostImpl* render_frame_host = web_contents->GetMainFrame();
ASSERT_TRUE(render_frame_host);
const GURL kPrerenderingUrl("https://example.com/next");
auto attributes = blink::mojom::PrerenderAttributes::New();
attributes->url = kPrerenderingUrl;
PrerenderHostRegistry* registry = web_contents->GetPrerenderHostRegistry();
const int prerender_frame_tree_node_id =
registry->CreateAndStartHost(std::move(attributes), *render_frame_host);
ASSERT_NE(prerender_frame_tree_node_id, kNoFrameTreeNodeId);
PrerenderHost* prerender_host =
registry->FindHostByUrlForTesting(kPrerenderingUrl);
CommitPrerenderNavigation(*prerender_host);
std::unique_ptr<NavigationSimulatorImpl> navigation =
NavigationSimulatorImpl::CreateRendererInitiated(kPrerenderingUrl,
render_frame_host);
// Change a parameter to differentiate the activation request from the
// prerendering request.
navigation->set_skip_service_worker(true);
navigation->Start();
NavigationRequest* navigation_request = navigation->GetNavigationHandle();
EXPECT_EQ(navigation_request->prerender_frame_tree_node_id(),
kNoFrameTreeNodeId);
EXPECT_EQ(registry->FindHostByUrlForTesting(kPrerenderingUrl), nullptr);
}
} // namespace
} // namespace content