#include <string>
#include <vector>
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/scoped_temp_dir.h"
#include "base/run_loop.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/stringprintf.h"
#include "base/test/bind.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/test_future.h"
#include "base/threading/thread_restrictions.h"
#include "build/build_config.h"
#include "content/browser/loader/prefetch_browsertest_base.h"
#include "content/browser/web_package/mock_signed_exchange_handler.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/web_contents.h"
#include "content/public/test/browser_test.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/content_browser_test.h"
#include "content/public/test/content_browser_test_utils.h"
#include "content/public/test/fenced_frame_test_util.h"
#include "content/public/test/test_frame_navigation_observer.h"
#include "content/public/test/url_loader_monitor.h"
#include "content/shell/browser/shell.h"
#include "net/base/features.h"
#include "net/base/filename_util.h"
#include "net/base/isolation_info.h"
#include "net/dns/mock_host_resolver.h"
#include "net/test/embedded_test_server/default_handlers.h"
#include "net/test/scoped_mutually_exclusive_feature_list.h"
#include "services/network/public/cpp/features.h"
#include "services/network/public/cpp/resource_request.h"
#include "third_party/blink/public/common/features.h"
namespace content {
namespace {
enum class SplitCacheTestCase {
kDisabled,
kEnabledTripleKeyed,
kEnabledTriplePlusCredsBool,
};
const struct {
const SplitCacheTestCase test_case;
base::test::FeatureRef feature;
} kTestCaseToFeatureMapping[] = {
{SplitCacheTestCase::kEnabledTriplePlusCredsBool,
net::features::kSplitCacheByIncludeCredentials}};
}
class PrefetchBrowserTest
: public PrefetchBrowserTestBase,
public testing::WithParamInterface<SplitCacheTestCase> {
public:
PrefetchBrowserTest()
: cross_origin_server_(std::make_unique<net::EmbeddedTestServer>()),
split_cache_test_case_(GetParam()),
split_cache_experiment_feature_list_(GetParam(),
kTestCaseToFeatureMapping) {
std::vector<base::test::FeatureRef> enabled_features;
std::vector<base::test::FeatureRef> disabled_features;
if (IsSplitCacheEnabled()) {
enabled_features.emplace_back(
net::features::kSplitCacheByNetworkIsolationKey);
} else {
disabled_features.emplace_back(
net::features::kSplitCacheByNetworkIsolationKey);
}
enabled_features.emplace_back(net::features::kHttpCacheNoVarySearch);
split_cache_enabled_feature_list_.InitWithFeatures(enabled_features,
disabled_features);
}
PrefetchBrowserTest(const PrefetchBrowserTest&) = delete;
PrefetchBrowserTest& operator=(const PrefetchBrowserTest&) = delete;
~PrefetchBrowserTest() override = default;
void SetUpOnMainThread() override {
PrefetchBrowserTestBase::SetUpOnMainThread();
host_resolver()->AddRule("*", "127.0.0.1");
}
bool IsSplitCacheEnabled() const {
return split_cache_test_case_ != SplitCacheTestCase::kDisabled;
}
protected:
std::unique_ptr<net::EmbeddedTestServer> cross_origin_server_;
const SplitCacheTestCase split_cache_test_case_;
private:
net::test::ScopedMutuallyExclusiveFeatureList
split_cache_experiment_feature_list_;
base::test::ScopedFeatureList split_cache_enabled_feature_list_;
};
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginDocumentHasNoSameSiteCookies) {
const char* prefetch_path = "/prefetch.html";
const char* target_path = "/target.html";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title></head>"));
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), target_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' as='document' href='%s'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
URLLoaderMonitor monitor({cross_origin_target_url});
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
monitor.WaitForUrls();
std::optional<network::ResourceRequest> request =
monitor.GetRequestInfo(cross_origin_target_url);
ASSERT_TRUE(request);
ASSERT_TRUE(request->site_for_cookies.IsNull());
ASSERT_TRUE(request->trusted_params);
url::Origin cross_origin = url::Origin::Create(cross_origin_target_url);
EXPECT_TRUE(net::IsolationInfo::Create(
net::IsolationInfo::RequestType::kMainFrame, cross_origin,
cross_origin, net::SiteForCookies())
.IsEqualForTesting(request->trusted_params->isolation_info));
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginDocumentReusedAsNavigation) {
const char* prefetch_path = "/prefetch.html";
const char* target_path = "/target.html";
RegisterResponse(target_path,
ResponseEntry("<head><title>Prefetch Target</title></head>",
""));
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), target_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' as='document' href='%s'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(cross_origin_target_url, "Prefetch Target");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginDocumentFromOpaqueOrigin) {
EXPECT_TRUE(NavigateToURL(
shell(),
GURL("data:text/html,<title>Data URL Prefetch Target</title><link "
"rel=prefetch as=document href=https://google.com>")));
}
IN_PROC_BROWSER_TEST_P(
PrefetchBrowserTest,
DISABLED_CrossOriginDocumentNotReusedAsNestedFrameNavigation) {
if (!IsSplitCacheEnabled()) {
GTEST_SKIP() << "This test is relevant only with SplitCache.";
}
const char* prefetch_path = "/prefetch.html";
const char* host_path = "/host.html";
const char* iframe_path = "/iframe.html";
RegisterResponse(
host_path,
ResponseEntry(base::StringPrintf(
"<head><title>Cross-Origin Host</title></head><body><iframe "
"onload='document.title=\"Host Loaded\"' src='%s'></iframe></body>",
iframe_path)));
RegisterResponse(iframe_path, ResponseEntry("<h1>I am an iframe</h1>"));
base::RunLoop prefetch_waiter;
auto cross_origin_iframe_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), iframe_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_host_url =
cross_origin_server_->GetURL("3p.example", host_path);
const GURL cross_origin_iframe_url =
cross_origin_server_->GetURL("3p.example", iframe_path);
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' as='document' href='%s'></body>",
cross_origin_iframe_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, cross_origin_iframe_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, cross_origin_iframe_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
NavigateToURLAndWaitTitle(cross_origin_host_url, "Host Loaded");
EXPECT_EQ(2, cross_origin_iframe_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginSubresourceNotReused) {
if (!IsSplitCacheEnabled()) {
GTEST_SKIP() << "This test is relevant only with SplitCache.";
}
const char* prefetch_path = "/prefetch.html";
const char* host_path = "/host.html";
const char* subresource_path = "/subresource.js";
RegisterResponse(
host_path,
ResponseEntry(base::StringPrintf(
"<head><title>Cross-Origin Host</title></head><body><script src='%s' "
"onload='document.title=\"Host Loaded\"'></script></body>",
subresource_path)));
RegisterResponse(subresource_path, ResponseEntry("console.log('I loaded')"));
base::RunLoop prefetch_waiter;
auto cross_origin_subresource_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), subresource_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_host_url =
cross_origin_server_->GetURL("3p.example", host_path);
const GURL cross_origin_subresource_url =
cross_origin_server_->GetURL("3p.example", subresource_path);
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>",
cross_origin_subresource_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
NavigateToURLAndWaitTitle(cross_origin_host_url, "Host Loaded");
EXPECT_EQ(2, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginSubresourceReusedByCurrentFrame) {
const char* prefetch_path = "/prefetch.html";
const char* use_prefetch_path = "/use-prefetch.html";
const char* subresource_path = "/subresource.js";
RegisterResponse(subresource_path, ResponseEntry("console.log('I loaded')"));
base::RunLoop prefetch_waiter;
auto cross_origin_subresource_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), subresource_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_subresource_url =
cross_origin_server_->GetURL("3p.example", subresource_path);
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>",
cross_origin_subresource_url.spec().c_str())));
RegisterResponse(use_prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><script src='%s' onload='document.title=\"Use "
"Prefetch Loaded\"'></script></body>",
cross_origin_subresource_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(embedded_test_server()->GetURL(use_prefetch_path),
"Use Prefetch Loaded");
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginSubresourceNotReusedAsNavigation) {
if (!IsSplitCacheEnabled()) {
GTEST_SKIP() << "This test is relevant only with SplitCache.";
}
const char* prefetch_path = "/prefetch.html";
const char* subresource_path = "/subresource.js";
RegisterResponse(subresource_path, ResponseEntry("console.log('I loaded');"));
base::RunLoop prefetch_waiter;
auto cross_origin_subresource_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), subresource_path, &prefetch_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_subresource_url =
cross_origin_server_->GetURL("3p.example", subresource_path);
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>",
cross_origin_subresource_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(NavigateToURL(shell(), cross_origin_subresource_url));
EXPECT_EQ(2, cross_origin_subresource_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, Simple) {
const char* prefetch_path = "/prefetch.html";
const char* target_path = "/target.html";
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>", target_path)));
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title></head>", "text/html",
{{"cache-control", "public, max-age=3600"}}));
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), target_path, &prefetch_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL target_url = embedded_test_server()->GetURL(target_path);
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_url, "Prefetch Target");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, DoublePrefetch) {
const char* prefetch_path = "/prefetch.html";
const char* target_path = "/target.html";
RegisterResponse(prefetch_path, ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'>"
"<link rel='prefetch' href='%s'></body>",
target_path, target_path)));
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title></head>", "text/html",
{{"cache-control", "public, max-age=3600"}}));
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), target_path, &prefetch_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL target_url = embedded_test_server()->GetURL(target_path);
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_url, "Prefetch Target");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, NoCacheAndNoStore) {
const char* prefetch_path = "/prefetch.html";
const char* nocache_path = "/target1.html";
const char* nostore_path = "/target2.html";
RegisterResponse(prefetch_path, ResponseEntry(base::StringPrintf(
"<body>"
"<link rel='prefetch' href='%s'>"
"<link rel='prefetch' href='%s'></body>",
nocache_path, nostore_path)));
RegisterResponse(nocache_path,
ResponseEntry("<head><title>NoCache Target</title></head>",
"text/html", {{"cache-control", "no-cache"}}));
RegisterResponse(nostore_path,
ResponseEntry("<head><title>NoStore Target</title></head>",
"text/html", {{"cache-control", "no-store"}}));
base::RunLoop nocache_waiter;
base::RunLoop nostore_waiter;
auto nocache_request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), nocache_path, &nocache_waiter);
auto nostore_request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), nostore_path, &nostore_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
nocache_waiter.Run();
nostore_waiter.Run();
EXPECT_EQ(1, nocache_request_counter->GetRequestCount());
EXPECT_EQ(1, nostore_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
NavigateToURLAndWaitTitle(embedded_test_server()->GetURL(nocache_path),
"NoCache Target");
EXPECT_EQ(2, nocache_request_counter->GetRequestCount());
NavigateToURLAndWaitTitle(embedded_test_server()->GetURL(nostore_path),
"NoStore Target");
EXPECT_EQ(2, nostore_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, WithPreload) {
const char* prefetch_path = "/prefetch.html";
const char* target_path = "/target.html";
const char* preload_path = "/preload.js";
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>", target_path)));
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title><script "
"src=\"./preload.js\"></script></head>",
"text/html",
{{"link", "</preload.js>;rel=\"preload\";as=\"script\""},
{"cache-control", "public, max-age=600"}}));
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
auto target_request_counter =
RequestCounter::CreateAndMonitor(embedded_test_server(), target_path);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), preload_path, &preload_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL target_url = embedded_test_server()->GetURL(target_path);
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
preload_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
WaitUntilLoaded(embedded_test_server()->GetURL(preload_path));
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_url, "done");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginWithPreloadHasNoSameSiteCookies) {
const char* target_path = "/target.html";
const char* preload_path = "/preload.js";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title><script "
"src=\"./preload.js\"></script></head>",
"text/html",
{{"link", "</preload.js>;rel=\"preload\";as=\"script\""},
{"access-control-allow-origin", "*"}}));
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), target_path);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), preload_path, &preload_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s' as='document' "
"crossorigin='anonymous'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
URLLoaderMonitor monitor({cross_origin_target_url});
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
preload_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
GURL cross_origin_preload_url =
cross_origin_server_->GetURL("3p.example", preload_path);
WaitUntilLoaded(cross_origin_preload_url);
monitor.WaitForUrls();
std::optional<network::ResourceRequest> request =
monitor.GetRequestInfo(cross_origin_target_url);
ASSERT_TRUE(request);
ASSERT_TRUE(request->site_for_cookies.IsNull());
ASSERT_TRUE(request->trusted_params);
url::Origin cross_origin = url::Origin::Create(cross_origin_target_url);
EXPECT_TRUE(net::IsolationInfo::Create(
net::IsolationInfo::RequestType::kMainFrame, cross_origin,
cross_origin, net::SiteForCookies())
.IsEqualForTesting(request->trusted_params->isolation_info));
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginWithPreloadAnonymous) {
const char* target_path = "/target.html";
const char* preload_path = "/preload.js";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title><script "
"src=\"./preload.js\"></script></head>",
"text/html",
{{"link", "</preload.js>;rel=\"preload\";as=\"script\""},
{"access-control-allow-origin", "*"}}));
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), target_path);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), preload_path, &preload_waiter);
RegisterRequestHandler(cross_origin_server_.get());
base::RunLoop preload_waiter_second_request;
auto preload_request_counter_second_request =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), preload_path,
&preload_waiter_second_request);
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s' as='document' "
"crossorigin='anonymous'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
preload_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
GURL cross_origin_preload_url =
cross_origin_server_->GetURL("3p.example", preload_path);
WaitUntilLoaded(cross_origin_preload_url);
if (IsSplitCacheEnabled()) {
const char* reuse_preload_attempt_path = "/reuse.html";
RegisterResponse(
reuse_preload_attempt_path,
ResponseEntry(
base::StringPrintf("<head><title>Other site</title><script "
"src='%s'></script></head>",
cross_origin_preload_url.spec().c_str()),
"text/html",
{{"link",
base::StringPrintf("<%s>;rel=\"preload\";as=\"script\"",
cross_origin_preload_url.spec().c_str())},
{"access-control-allow-origin", "*"}}));
std::unique_ptr<net::EmbeddedTestServer> other_cross_origin_server =
std::make_unique<net::EmbeddedTestServer>();
RegisterRequestHandler(other_cross_origin_server.get());
ASSERT_TRUE(other_cross_origin_server->Start());
EXPECT_TRUE(NavigateToURL(
shell(), other_cross_origin_server->GetURL(
"other3p.example", reuse_preload_attempt_path)));
preload_waiter_second_request.Run();
EXPECT_EQ(2, preload_request_counter_second_request->GetRequestCount());
EXPECT_TRUE(other_cross_origin_server->ShutdownAndWaitUntilComplete());
}
if (split_cache_test_case_ ==
SplitCacheTestCase::kEnabledTriplePlusCredsBool) {
return;
}
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(ExecJs(shell()->web_contents(), "document.title = 'not done';"));
NavigateToURLAndWaitTitle(cross_origin_target_url, "done");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginWithInvalidPreloadAsDocument) {
const char* target_path = "/target.html";
const char* preload_path = "/preload.js";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title><script "
"src=\"./preload.js\"></script></head>",
"text/html",
{{"link", "</preload.js>;rel=\"preload\";as=\"document\""},
{"access-control-allow-origin", "*"}}));
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), target_path);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), preload_path);
RegisterRequestHandler(cross_origin_server_.get());
base::RunLoop preload_waiter_second_request;
auto preload_request_counter_second_request =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), preload_path,
&preload_waiter_second_request);
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s' as='document' "
"crossorigin='anonymous'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
base::test::TestFuture<void> prefetch_future;
RegisterPrefetchLoaderCallback(prefetch_future.GetCallback());
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
EXPECT_TRUE(prefetch_future.Wait());
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_EQ(0, preload_request_counter->GetRequestCount());
NavigateToURLAndWaitTitle(cross_origin_target_url, "done");
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginWithPreloadCredentialled) {
ASSERT_TRUE(embedded_test_server()->InitializeAndListen());
const auto port = embedded_test_server()->port();
const char target_path[] = "/target.html";
const char preload_path[] = "/preload.js";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title><script "
"src=\"./preload.js\"></script></head>",
"text/html",
{
{
"link",
"</preload.js>;rel=\"preload\";as=\"script\"",
},
{
"Access-Control-Allow-Origin",
"http://prefetch.com:" + base::NumberToString(port),
},
{
"Access-Control-Allow-Credentials",
"true",
},
}));
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), target_path);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), preload_path, &preload_waiter);
RegisterRequestHandler(cross_origin_server_.get());
base::RunLoop preload_waiter_second_request;
auto preload_request_counter_second_request =
RequestCounter::CreateAndMonitor(cross_origin_server_.get(), preload_path,
&preload_waiter_second_request);
ASSERT_TRUE(cross_origin_server_->Start());
const GURL cross_origin_target_url =
cross_origin_server_->GetURL("3p.example", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s' as='document' "
"crossorigin='use-credentials'></body>",
cross_origin_target_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
embedded_test_server()->StartAcceptingConnections();
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(NavigateToURL(
shell(), embedded_test_server()->GetURL("prefetch.com", prefetch_path)));
preload_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
GURL cross_origin_preload_url =
cross_origin_server_->GetURL("3p.example", preload_path);
WaitUntilLoaded(cross_origin_preload_url);
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(cross_origin_target_url, "done");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, SignedExchangeWithPreload) {
const char* prefetch_path = "/prefetch.html";
const char* target_sxg_path = "/target.sxg";
const char* target_path = "/target.html";
const char* preload_path_in_sxg = "/preload.js";
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' href='%s'></body>", target_sxg_path)));
RegisterResponse(
target_sxg_path,
ResponseEntry(MockSignedExchangeHandler::kMockSxgPrefix +
"<head><title>Prefetch Target (SXG)</title><script "
"src=\"./preload.js\"></script></head>",
"application/signed-exchange;v=b3",
{{"x-content-type-options", "nosniff"}}));
RegisterResponse(preload_path_in_sxg,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
base::RunLoop prefetch_waiter;
auto target_request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), target_sxg_path, &prefetch_waiter);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), preload_path_in_sxg, &preload_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL preload_url_in_sxg =
embedded_test_server()->GetURL(preload_path_in_sxg);
const GURL target_sxg_url = embedded_test_server()->GetURL(target_sxg_path);
MockSignedExchangeHandlerFactory factory({MockSignedExchangeHandlerParams(
target_sxg_url, SignedExchangeLoadResult::kSuccess, net::OK,
GURL(embedded_test_server()->GetURL(target_path)), "text/html",
{{"Link", base::StringPrintf("<%s>;rel=\"preload\";as=\"script\"",
preload_url_in_sxg.spec().c_str())}},
net::SHA256HashValue({{0x00}}))});
ScopedSignedExchangeHandlerFactory scoped_factory(&factory);
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
preload_waiter.Run();
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_sxg_url, "done");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest,
DISABLED_CrossOriginSignedExchangeWithPreload) {
const char* prefetch_path = "/prefetch.html";
const char* target_sxg_path = "/target.sxg";
const char* target_path = "/target.html";
const char* preload_path_in_sxg = "/preload.js";
RegisterResponse(
target_sxg_path,
ResponseEntry(MockSignedExchangeHandler::kMockSxgPrefix +
"<head><title>Prefetch Target (SXG)</title><script "
"src=\"./preload.js\"></script></head>",
"application/signed-exchange;v=b3",
{{"x-content-type-options", "nosniff"}}));
RegisterResponse(preload_path_in_sxg,
ResponseEntry("document.title=\"done\";", "text/javascript",
{{"cache-control", "public, max-age=600"}}));
base::RunLoop preload_waiter;
base::RunLoop prefetch_waiter;
auto target_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), target_sxg_path, &prefetch_waiter);
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server_.get(), preload_path_in_sxg, &preload_waiter);
RegisterRequestHandler(cross_origin_server_.get());
ASSERT_TRUE(cross_origin_server_->Start());
const GURL target_sxg_url =
cross_origin_server_->GetURL("3p.example", target_sxg_path);
const GURL preload_url_in_sxg =
cross_origin_server_->GetURL("3p.example", preload_path_in_sxg);
RegisterResponse(
prefetch_path,
ResponseEntry(base::StringPrintf(
"<body><link rel='prefetch' as='document' href='%s'></body>",
target_sxg_url.spec().c_str())));
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
MockSignedExchangeHandlerFactory factory({MockSignedExchangeHandlerParams(
target_sxg_url, SignedExchangeLoadResult::kSuccess, net::OK,
GURL(cross_origin_server_->GetURL("3p.example", target_path)),
"text/html",
{{"Link", base::StringPrintf("<%s>;rel=\"preload\";as=\"script\"",
preload_url_in_sxg.spec().c_str())}},
net::SHA256HashValue({{0x00}}))});
ScopedSignedExchangeHandlerFactory scoped_factory(&factory);
EXPECT_TRUE(
NavigateToURL(shell(), embedded_test_server()->GetURL(prefetch_path)));
prefetch_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
preload_waiter.Run();
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
WaitUntilLoaded(preload_url_in_sxg);
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server_->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_sxg_url, "done");
}
IN_PROC_BROWSER_TEST_P(PrefetchBrowserTest, FileToHttp) {
const char* target_path = "/target.html";
RegisterResponse(
target_path,
ResponseEntry("<head><title>Prefetch Target</title></head>",
"",
{{"cache-control", "public, max-age=31536000"}}));
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
embedded_test_server(), target_path, &prefetch_waiter);
RegisterRequestHandler(embedded_test_server());
ASSERT_TRUE(embedded_test_server()->Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL target_url = embedded_test_server()->GetURL(target_path);
{
base::ScopedAllowBlockingForTesting allow_blocking;
base::ScopedTempDir temp_dir;
ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
base::FilePath file_path = temp_dir.GetPath().AppendASCII("test.html");
std::string file_content = base::StringPrintf(
"<body><link rel='prefetch' as='document' href='%s'></body>",
target_url.spec().c_str());
ASSERT_TRUE(base::WriteFile(file_path, file_content));
GURL file_url = net::FilePathToFileURL(file_path);
EXPECT_TRUE(NavigateToURL(shell(), file_url));
prefetch_waiter.Run();
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
}
EXPECT_TRUE(embedded_test_server()->ShutdownAndWaitUntilComplete());
NavigateToURLAndWaitTitle(target_url, "Prefetch Target");
}
class FencedFramePrefetchTest : public PrefetchBrowserTestBase {
public:
FencedFramePrefetchTest()
: cross_origin_server_(net::EmbeddedTestServer::TYPE_HTTPS) {}
void SetUpOnMainThread() override {
PrefetchBrowserTestBase::SetUpOnMainThread();
host_resolver()->AddRule("*", "127.0.0.1");
embedded_https_test_server().SetSSLConfig(
net::EmbeddedTestServer::CERT_TEST_NAMES);
SetupCrossSiteRedirector(&embedded_https_test_server());
net::test_server::RegisterDefaultHandlers(&embedded_https_test_server());
cross_origin_server()->SetSSLConfig(
net::EmbeddedTestServer::CERT_TEST_NAMES);
SetupCrossSiteRedirector(cross_origin_server());
net::test_server::RegisterDefaultHandlers(cross_origin_server());
}
content::test::FencedFrameTestHelper& fenced_frame_test_helper() {
return fenced_frame_test_helper_;
}
net::EmbeddedTestServer* cross_origin_server() {
return &cross_origin_server_;
}
private:
test::FencedFrameTestHelper fenced_frame_test_helper_;
net::EmbeddedTestServer cross_origin_server_;
};
IN_PROC_BROWSER_TEST_F(FencedFramePrefetchTest, BasicPrefetch) {
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
&embedded_https_test_server(), "/image.jpg", &prefetch_waiter);
RegisterRequestHandler(&embedded_https_test_server());
ASSERT_TRUE(embedded_https_test_server().Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
GURL prefetch_url =
embedded_https_test_server().GetURL("a.test", "/image.jpg");
URLLoaderMonitor monitor({prefetch_url});
const GURL main_url =
embedded_https_test_server().GetURL("a.test", "/title1.html");
EXPECT_TRUE(NavigateToURL(shell(), main_url));
const GURL fenced_frame_url = embedded_https_test_server().GetURL(
"a.test", "/fenced_frames/title1.html");
RenderFrameHost* fenced_frame_rfh =
fenced_frame_test_helper().CreateFencedFrame(
shell()->web_contents()->GetPrimaryMainFrame(), fenced_frame_url);
TestFrameNavigationObserver observer(fenced_frame_rfh);
EXPECT_TRUE(ExecJs(shell()->web_contents()->GetPrimaryMainFrame(),
JsReplace(
R"(document.querySelector('fencedframe').config
= new FencedFrameConfig($1);)",
embedded_https_test_server().GetURL(
"a.test", "/link_rel_prefetch.html"))));
observer.WaitForCommit();
prefetch_waiter.Run();
monitor.WaitForUrls();
std::optional<network::ResourceRequest> request =
monitor.GetRequestInfo(prefetch_url);
EXPECT_TRUE(request->load_flags & net::LOAD_PREFETCH);
EXPECT_EQ(1, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_https_test_server().ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_F(FencedFramePrefetchTest, NetworkCutoffDisablesPrefetch) {
base::RunLoop prefetch_waiter;
auto request_counter = RequestCounter::CreateAndMonitor(
&embedded_https_test_server(), "/image.jpg", &prefetch_waiter);
RegisterRequestHandler(&embedded_https_test_server());
ASSERT_TRUE(embedded_https_test_server().Start());
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
GURL prefetch_url =
embedded_https_test_server().GetURL("a.test", "/image.jpg");
URLLoaderMonitor monitor({prefetch_url});
const GURL main_url =
embedded_https_test_server().GetURL("a.test", "/title1.html");
EXPECT_TRUE(NavigateToURL(shell(), main_url));
const GURL fenced_frame_url = embedded_https_test_server().GetURL(
"a.test", "/fenced_frames/title1.html");
RenderFrameHost* fenced_frame_rfh =
fenced_frame_test_helper().CreateFencedFrame(
shell()->web_contents()->GetPrimaryMainFrame(), fenced_frame_url);
TestFrameNavigationObserver observer(fenced_frame_rfh);
EXPECT_TRUE(
ExecJs(shell()->web_contents()->GetPrimaryMainFrame(),
JsReplace(
R"(document.querySelector('fencedframe').config
= new FencedFrameConfig($1);)",
embedded_https_test_server().GetURL(
"a.test", "/link_rel_prefetch_disable_network.html"))));
observer.WaitForCommit();
prefetch_waiter.RunUntilIdle();
EXPECT_EQ(monitor.WaitForRequestCompletion(prefetch_url).error_code,
net::ERR_NETWORK_ACCESS_REVOKED);
EXPECT_EQ(0, request_counter->GetRequestCount());
EXPECT_EQ(1, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_https_test_server().ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_F(FencedFramePrefetchTest,
CrossOriginWithPreloadCredentialled) {
ASSERT_TRUE(embedded_https_test_server().InitializeAndListen());
const auto port = embedded_https_test_server().port();
const char target_path[] = "/target.html";
const char preload_path[] = "/preload.js";
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";",
"text/javascript",
{{"cache-control", "public, max-age=600"},
{"Supports-Loading-Mode", "fenced-frame"}}));
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server(), target_path);
base::RunLoop preload_waiter;
auto preload_request_counter = RequestCounter::CreateAndMonitor(
cross_origin_server(), preload_path, &preload_waiter);
RegisterRequestHandler(cross_origin_server());
ASSERT_TRUE(cross_origin_server()->Start());
const GURL cross_origin_target_url =
cross_origin_server()->GetURL("b.test", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(
prefetch_path,
ResponseEntry(JsReplace(
R"(
<body>
<link rel='prefetch' href=$1 as='document'
crossorigin='use-credentials'>
</body>
)",
cross_origin_target_url),
"text/html",
{{"Supports-Loading-Mode", "fenced-frame"}}));
RegisterRequestHandler(&embedded_https_test_server());
embedded_https_test_server().StartAcceptingConnections();
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL preload_url =
cross_origin_server()->GetURL("c.test", preload_path);
RegisterResponse(
target_path,
ResponseEntry(
JsReplace(R"(
<head>
<title>
Prefetch Target
</title>
<script src=$1></script>
</head>
)",
preload_url),
"text/html",
{{
"link",
base::StringPrintf("<%s>;rel=\"preload\";as=\"script\"",
preload_url.spec().c_str()),
},
{
"Access-Control-Allow-Origin",
"https://a.test:" + base::NumberToString(port),
},
{
"Access-Control-Allow-Credentials",
"true",
},
{"Supports-Loading-Mode", "fenced-frame"}}));
const GURL main_url =
embedded_https_test_server().GetURL("a.test", "/title1.html");
EXPECT_TRUE(NavigateToURL(shell(), main_url));
const GURL fenced_frame_url = embedded_https_test_server().GetURL(
"a.test", "/fenced_frames/title1.html");
RenderFrameHost* fenced_frame_rfh =
fenced_frame_test_helper().CreateFencedFrame(
shell()->web_contents()->GetPrimaryMainFrame(), fenced_frame_url);
TestFrameNavigationObserver observer(fenced_frame_rfh);
EXPECT_TRUE(ExecJs(
shell()->web_contents()->GetPrimaryMainFrame(),
JsReplace(
R"(document.querySelector('fencedframe').config
= new FencedFrameConfig($1);)",
embedded_https_test_server().GetURL("a.test", prefetch_path))));
observer.WaitForCommit();
preload_waiter.Run();
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(1, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_https_test_server().ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server()->ShutdownAndWaitUntilComplete());
}
IN_PROC_BROWSER_TEST_F(FencedFramePrefetchTest,
DISABLED_NetworkCutoffDisablesRecursivePrefetch) {
ASSERT_TRUE(embedded_https_test_server().InitializeAndListen());
const auto port = embedded_https_test_server().port();
const char target_path[] = "/target.html";
const char preload_path[] = "/preload.js";
RegisterResponse(preload_path,
ResponseEntry("document.title=\"done\";",
"text/javascript",
{{"cache-control", "public, max-age=600"},
{"Supports-Loading-Mode", "fenced-frame"}}));
auto target_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server(), target_path);
auto preload_request_counter =
RequestCounter::CreateAndMonitor(cross_origin_server(), preload_path);
RegisterRequestHandler(cross_origin_server());
ASSERT_TRUE(cross_origin_server()->Start());
const GURL cross_origin_target_url =
cross_origin_server()->GetURL("b.test", target_path);
const char* prefetch_path = "/prefetch.html";
RegisterResponse(
prefetch_path,
ResponseEntry(JsReplace(
R"(
<body>
<link rel='prefetch' href=$1 as='document'
crossorigin='use-credentials'>
</body>
)",
cross_origin_target_url),
"text/html",
{{
"Access-Control-Allow-Origin",
"https://a.test:" + base::NumberToString(port),
},
{"Supports-Loading-Mode", "fenced-frame"}}));
RegisterRequestHandler(&embedded_https_test_server());
embedded_https_test_server().StartAcceptingConnections();
EXPECT_EQ(0, GetPrefetchURLLoaderCallCount());
const GURL preload_url =
cross_origin_server()->GetURL("c.test", preload_path);
RegisterResponse(
target_path,
ResponseEntry(
JsReplace(R"(
<head>
<title>
Prefetch Target
</title>
<script src=$1></script>
</head>
)",
preload_url),
"text/html",
{{
"link",
base::StringPrintf("<%s>;rel=\"preload\";as=\"script\"",
preload_url.spec().c_str()),
},
{
"Access-Control-Allow-Origin",
"https://a.test:" + base::NumberToString(port),
},
{
"Access-Control-Allow-Credentials",
"true",
},
{"Supports-Loading-Mode", "fenced-frame"}}));
const GURL main_url =
embedded_https_test_server().GetURL("a.test", "/title1.html");
EXPECT_TRUE(NavigateToURL(shell(), main_url));
const GURL fenced_frame_url = embedded_https_test_server().GetURL(
"a.test", "/fenced_frames/title1.html");
RenderFrameHost* fenced_frame_rfh =
fenced_frame_test_helper().CreateFencedFrame(
shell()->web_contents()->GetPrimaryMainFrame(), fenced_frame_url);
RegisterPrefetchLoaderCallback(base::BindLambdaForTesting([&]() {
RenderFrameHost* rfh =
test::FencedFrameTestHelper::GetMostRecentlyAddedFencedFrame(
shell()->web_contents()->GetPrimaryMainFrame());
test::ExemptUrlsFromFencedFrameNetworkRevocation(rfh,
{cross_origin_target_url});
EXPECT_TRUE(ExecJs(rfh, R"(
(async () => {
return window.fence.disableUntrustedNetwork();
})();
)"));
}));
URLLoaderMonitor monitor({preload_url});
TestFrameNavigationObserver observer(fenced_frame_rfh);
EXPECT_TRUE(ExecJs(
shell()->web_contents()->GetPrimaryMainFrame(),
JsReplace(
R"(document.querySelector('fencedframe').config
= new FencedFrameConfig($1);)",
embedded_https_test_server().GetURL("a.test", prefetch_path))));
observer.WaitForCommit();
EXPECT_EQ(monitor.WaitForRequestCompletion(preload_url).error_code,
net::ERR_NETWORK_ACCESS_REVOKED);
EXPECT_EQ(1, target_request_counter->GetRequestCount());
EXPECT_EQ(0, preload_request_counter->GetRequestCount());
EXPECT_EQ(2, GetPrefetchURLLoaderCallCount());
EXPECT_TRUE(embedded_https_test_server().ShutdownAndWaitUntilComplete());
EXPECT_TRUE(cross_origin_server()->ShutdownAndWaitUntilComplete());
}
INSTANTIATE_TEST_SUITE_P(
All,
PrefetchBrowserTest,
testing::ValuesIn({SplitCacheTestCase::kDisabled,
SplitCacheTestCase::kEnabledTripleKeyed,
SplitCacheTestCase::kEnabledTriplePlusCredsBool}),
[](const testing::TestParamInfo<SplitCacheTestCase>& info) {
switch (info.param) {
case SplitCacheTestCase::kDisabled:
return "SplitCacheDisabled";
case SplitCacheTestCase::kEnabledTripleKeyed:
return "SplitCacheEnabledTripleKeyed";
case SplitCacheTestCase::kEnabledTriplePlusCredsBool:
return "SplitCacheEnabledTriplePlusCredsBool";
}
});
}