| // Copyright 2022 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/browser/preloading/prerenderer_impl.h" |
| |
| #include "base/test/scoped_feature_list.h" |
| #include "content/browser/preloading/prerender/prerender_host_registry.h" |
| #include "content/public/browser/web_contents_delegate.h" |
| #include "content/public/common/content_client.h" |
| #include "content/public/test/prerender_test_util.h" |
| #include "content/public/test/test_browser_context.h" |
| #include "content/public/test/test_renderer_host.h" |
| #include "content/test/test_content_browser_client.h" |
| #include "content/test/test_web_contents.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace content { |
| namespace { |
| |
| class PrerendererTest : public RenderViewHostTestHarness { |
| public: |
| PrerendererTest() = default; |
| |
| void SetUp() override { |
| RenderViewHostTestHarness::SetUp(); |
| |
| browser_context_ = std::make_unique<TestBrowserContext>(); |
| web_contents_ = TestWebContents::Create( |
| browser_context_.get(), |
| SiteInstanceImpl::Create(browser_context_.get())); |
| web_contents_delegate_ = |
| std::make_unique<test::ScopedPrerenderWebContentsDelegate>( |
| *web_contents_); |
| web_contents_->NavigateAndCommit(GURL("https://example.com")); |
| } |
| |
| void TearDown() override { |
| web_contents_.reset(); |
| browser_context_.reset(); |
| RenderViewHostTestHarness::TearDown(); |
| } |
| |
| RenderFrameHostImpl* GetRenderFrameHost() { |
| return web_contents_->GetPrimaryMainFrame(); |
| } |
| |
| GURL GetSameOriginUrl(const std::string& path) { |
| return GURL("https://example.com" + path); |
| } |
| |
| GURL GetCrossSiteUrl(const std::string& path) { |
| return GURL("https://example2.com" + path); |
| } |
| |
| PrerenderHostRegistry* GetPrerenderHostRegistry() const { |
| return web_contents_->GetPrerenderHostRegistry(); |
| } |
| |
| 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(); |
| return candidate; |
| } |
| |
| private: |
| test::ScopedPrerenderFeatureList prerender_feature_list_; |
| std::unique_ptr<TestBrowserContext> browser_context_; |
| std::unique_ptr<TestWebContents> web_contents_; |
| std::unique_ptr<test::ScopedPrerenderWebContentsDelegate> |
| web_contents_delegate_; |
| }; |
| |
| // Tests that Prerenderer starts prerendering when it receives prerender |
| // speculation candidates. |
| TEST_F(PrerendererTest, StartPrerender) { |
| PrerenderHostRegistry* registry = GetPrerenderHostRegistry(); |
| PrerendererImpl prerenderer(*GetRenderFrameHost()); |
| |
| const GURL kPrerenderingUrl = GetSameOriginUrl("/empty.html"); |
| std::vector<blink::mojom::SpeculationCandidatePtr> candidates; |
| candidates.push_back(CreatePrerenderCandidate(kPrerenderingUrl)); |
| |
| prerenderer.ProcessCandidatesForPrerender(base::UnguessableToken::Create(), |
| std::move(candidates)); |
| EXPECT_TRUE(registry->FindHostByUrlForTesting(kPrerenderingUrl)); |
| } |
| |
| // Tests that Prerenderer will skip a cross-site candidate even if it is the |
| // first prerender candidate in the candidate list. |
| TEST_F(PrerendererTest, ProcessFirstSameOriginPrerenderCandidate) { |
| PrerenderHostRegistry* registry = GetPrerenderHostRegistry(); |
| PrerendererImpl prerenderer(*GetRenderFrameHost()); |
| |
| const GURL kFirstPrerenderingUrlCrossSite = GetCrossSiteUrl("/title.html"); |
| const GURL kSecondPrerenderingUrlSameOrigin = |
| GetSameOriginUrl("/title1.html"); |
| std::vector<blink::mojom::SpeculationCandidatePtr> candidates; |
| candidates.push_back( |
| CreatePrerenderCandidate(kFirstPrerenderingUrlCrossSite)); |
| candidates.push_back( |
| CreatePrerenderCandidate(kSecondPrerenderingUrlSameOrigin)); |
| |
| prerenderer.ProcessCandidatesForPrerender(base::UnguessableToken::Create(), |
| std::move(candidates)); |
| |
| // The first prerender candidate is a cross-site one, so Prerenderer should |
| // not prerender it. |
| EXPECT_FALSE( |
| registry->FindHostByUrlForTesting(kFirstPrerenderingUrlCrossSite)); |
| // The second element in this list is the first same-origin prerender |
| // candidate, so Prerenderer should prerender this candidate. |
| EXPECT_TRUE( |
| registry->FindHostByUrlForTesting(kSecondPrerenderingUrlSameOrigin)); |
| } |
| |
| // Tests that Prerenderer will remove the rendered host, if the url is removed |
| // from candidates list. |
| TEST_F(PrerendererTest, RemoveRendererHostAfterCandidateRemoved) { |
| PrerenderHostRegistry* registry = GetPrerenderHostRegistry(); |
| PrerendererImpl prerenderer(*GetRenderFrameHost()); |
| |
| const GURL urls[]{GetSameOriginUrl("/title1.html"), |
| GetSameOriginUrl("/title2.html")}; |
| std::vector<blink::mojom::SpeculationCandidatePtr> candidates; |
| for (const auto& url : urls) { |
| candidates.push_back(CreatePrerenderCandidate(url)); |
| } |
| auto initiator_devtools_navigation_token = base::UnguessableToken::Create(); |
| prerenderer.ProcessCandidatesForPrerender(initiator_devtools_navigation_token, |
| std::move(candidates)); |
| EXPECT_TRUE(registry->FindHostByUrlForTesting(urls[0])); |
| EXPECT_TRUE(registry->FindHostByUrlForTesting(urls[1])); |
| |
| std::vector<blink::mojom::SpeculationCandidatePtr> new_candidates; |
| new_candidates.push_back(CreatePrerenderCandidate(urls[1])); |
| prerenderer.ProcessCandidatesForPrerender(initiator_devtools_navigation_token, |
| std::move(new_candidates)); |
| EXPECT_FALSE(registry->FindHostByUrlForTesting(urls[0])); |
| EXPECT_TRUE(registry->FindHostByUrlForTesting(urls[1])); |
| |
| prerenderer.ProcessCandidatesForPrerender( |
| initiator_devtools_navigation_token, |
| std::vector<blink::mojom::SpeculationCandidatePtr>{}); |
| EXPECT_FALSE(registry->FindHostByUrlForTesting(urls[0])); |
| EXPECT_FALSE(registry->FindHostByUrlForTesting(urls[1])); |
| } |
| |
| // Tests that it is possible to start a prerender using MaybePrerender and |
| // ShouldWaitForPrerenderResult methods. |
| TEST_F(PrerendererTest, MaybePrerenderAndShouldWaitForPrerenderResult) { |
| PrerenderHostRegistry* registry = GetPrerenderHostRegistry(); |
| PrerendererImpl prerenderer(*GetRenderFrameHost()); |
| |
| const GURL kUrlToCancel = GetSameOriginUrl("/to_cancel.html"); |
| std::vector<blink::mojom::SpeculationCandidatePtr> candidateToCancel; |
| candidateToCancel.push_back(CreatePrerenderCandidate(kUrlToCancel)); |
| |
| // Candidate is not processed yet. So, it should return false. |
| EXPECT_FALSE(prerenderer.ShouldWaitForPrerenderResult(kUrlToCancel)); |
| // Process the candidate. |
| auto initiator_devtools_navigation_token = base::UnguessableToken::Create(); |
| prerenderer.ProcessCandidatesForPrerender(initiator_devtools_navigation_token, |
| std::move(candidateToCancel)); |
| EXPECT_TRUE(prerenderer.ShouldWaitForPrerenderResult(kUrlToCancel)); |
| // Cancel the prerender |
| prerenderer.ProcessCandidatesForPrerender( |
| initiator_devtools_navigation_token, |
| std::vector<blink::mojom::SpeculationCandidatePtr>{}); |
| EXPECT_FALSE(prerenderer.ShouldWaitForPrerenderResult(kUrlToCancel)); |
| |
| const GURL kPrerenderingUrl = GetSameOriginUrl("/empty.html"); |
| const auto candidate = CreatePrerenderCandidate(kPrerenderingUrl); |
| |
| // Candidate is not processed yet. So, it should return false. |
| EXPECT_FALSE(prerenderer.ShouldWaitForPrerenderResult(kPrerenderingUrl)); |
| // MaybePrerender the candidate and check if ShouldWaitForPrerenderResult |
| // returns true. |
| EXPECT_TRUE(prerenderer.MaybePrerender(initiator_devtools_navigation_token, |
| candidate)); |
| EXPECT_TRUE(prerenderer.ShouldWaitForPrerenderResult(kPrerenderingUrl)); |
| EXPECT_TRUE(registry->FindHostByUrlForTesting(kPrerenderingUrl)); |
| } |
| |
| } // namespace |
| } // namespace content |