1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "components/precache/core/precache_fetcher.h"
11 #include "base/basictypes.h"
12 #include "base/bind.h"
13 #include "base/callback.h"
14 #include "base/command_line.h"
15 #include "base/compiler_specific.h"
16 #include "base/message_loop/message_loop.h"
17 #include "components/precache/core/precache_switches.h"
18 #include "components/precache/core/proto/precache.pb.h"
19 #include "net/http/http_response_headers.h"
20 #include "net/http/http_status_code.h"
21 #include "net/url_request/test_url_fetcher_factory.h"
22 #include "net/url_request/url_request_status.h"
23 #include "net/url_request/url_request_test_util.h"
24 #include "testing/gtest/include/gtest/gtest.h"
30 class TestURLFetcherCallback
{
32 scoped_ptr
<net::FakeURLFetcher
> CreateURLFetcher(
33 const GURL
& url
, net::URLFetcherDelegate
* delegate
,
34 const std::string
& response_data
, net::HttpStatusCode response_code
,
35 net::URLRequestStatus::Status status
) {
36 scoped_ptr
<net::FakeURLFetcher
> fetcher(new net::FakeURLFetcher(
37 url
, delegate
, response_data
, response_code
, status
));
39 if (response_code
== net::HTTP_OK
) {
40 scoped_refptr
<net::HttpResponseHeaders
> download_headers
=
41 new net::HttpResponseHeaders("");
42 download_headers
->AddHeader("Content-Type: text/html");
43 fetcher
->set_response_headers(download_headers
);
46 requested_urls_
.insert(url
);
47 return fetcher
.Pass();
50 const std::multiset
<GURL
>& requested_urls() const {
51 return requested_urls_
;
55 // Multiset with one entry for each URL requested.
56 std::multiset
<GURL
> requested_urls_
;
59 class TestPrecacheDelegate
: public PrecacheFetcher::PrecacheDelegate
{
61 TestPrecacheDelegate() : was_on_done_called_(false) {}
63 virtual void OnDone() OVERRIDE
{
64 was_on_done_called_
= true;
67 bool was_on_done_called() const {
68 return was_on_done_called_
;
72 bool was_on_done_called_
;
75 class PrecacheFetcherTest
: public testing::Test
{
78 : request_context_(new net::TestURLRequestContextGetter(
79 base::MessageLoopProxy::current())),
80 factory_(NULL
, base::Bind(&TestURLFetcherCallback::CreateURLFetcher
,
81 base::Unretained(&url_callback_
))) {}
84 base::MessageLoopForUI loop_
;
85 scoped_refptr
<net::TestURLRequestContextGetter
> request_context_
;
86 TestURLFetcherCallback url_callback_
;
87 net::FakeURLFetcherFactory factory_
;
88 TestPrecacheDelegate precache_delegate_
;
91 const char kConfigURL
[] = "http://config-url.com";
92 const char kManfiestURLPrefix
[] = "http://manifest-url-prefix.com/";
93 const char kManifestFetchFailureURL
[] =
94 "http://manifest-url-prefix.com/"
95 "http%253A%252F%252Fmanifest-fetch-failure.com%252F";
96 const char kBadManifestURL
[] =
97 "http://manifest-url-prefix.com/http%253A%252F%252Fbad-manifest.com%252F";
98 const char kGoodManifestURL
[] =
99 "http://manifest-url-prefix.com/http%253A%252F%252Fgood-manifest.com%252F";
100 const char kResourceFetchFailureURL
[] = "http://resource-fetch-failure.com";
101 const char kGoodResourceURL
[] = "http://good-resource.com";
102 const char kForcedStartingURLManifestURL
[] =
103 "http://manifest-url-prefix.com/"
104 "http%253A%252F%252Fforced-starting-url.com%252F";
106 TEST_F(PrecacheFetcherTest
, FullPrecache
) {
107 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
108 switches::kPrecacheConfigSettingsURL
, kConfigURL
);
109 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
110 switches::kPrecacheManifestURLPrefix
, kManfiestURLPrefix
);
112 std::list
<GURL
> starting_urls
;
113 starting_urls
.push_back(GURL("http://manifest-fetch-failure.com"));
114 starting_urls
.push_back(GURL("http://bad-manifest.com"));
115 starting_urls
.push_back(GURL("http://good-manifest.com"));
116 starting_urls
.push_back(GURL("http://not-in-top-3.com"));
118 PrecacheConfigurationSettings config
;
119 config
.set_top_sites_count(3);
120 config
.add_forced_starting_url("http://forced-starting-url.com");
121 // Duplicate starting URL, the manifest for this should only be fetched once.
122 config
.add_forced_starting_url("http://good-manifest.com");
124 PrecacheManifest good_manifest
;
125 good_manifest
.add_resource()->set_url(kResourceFetchFailureURL
);
126 good_manifest
.add_resource(); // Resource with no URL, should not be fetched.
127 good_manifest
.add_resource()->set_url(kGoodResourceURL
);
129 factory_
.SetFakeResponse(GURL(kConfigURL
), config
.SerializeAsString(),
130 net::HTTP_OK
, net::URLRequestStatus::SUCCESS
);
131 factory_
.SetFakeResponse(GURL(kManifestFetchFailureURL
), "",
132 net::HTTP_INTERNAL_SERVER_ERROR
,
133 net::URLRequestStatus::FAILED
);
134 factory_
.SetFakeResponse(GURL(kBadManifestURL
), "bad protobuf", net::HTTP_OK
,
135 net::URLRequestStatus::SUCCESS
);
136 factory_
.SetFakeResponse(GURL(kGoodManifestURL
),
137 good_manifest
.SerializeAsString(), net::HTTP_OK
,
138 net::URLRequestStatus::SUCCESS
);
139 factory_
.SetFakeResponse(GURL(kResourceFetchFailureURL
),
140 "", net::HTTP_INTERNAL_SERVER_ERROR
,
141 net::URLRequestStatus::FAILED
);
142 factory_
.SetFakeResponse(GURL(kGoodResourceURL
), "good", net::HTTP_OK
,
143 net::URLRequestStatus::SUCCESS
);
144 factory_
.SetFakeResponse(GURL(kForcedStartingURLManifestURL
),
145 PrecacheManifest().SerializeAsString(), net::HTTP_OK
,
146 net::URLRequestStatus::SUCCESS
);
148 PrecacheFetcher
precache_fetcher(starting_urls
, request_context_
.get(),
149 &precache_delegate_
);
150 precache_fetcher
.Start();
152 base::MessageLoop::current()->RunUntilIdle();
154 std::multiset
<GURL
> expected_requested_urls
;
155 expected_requested_urls
.insert(GURL(kConfigURL
));
156 expected_requested_urls
.insert(GURL(kManifestFetchFailureURL
));
157 expected_requested_urls
.insert(GURL(kBadManifestURL
));
158 expected_requested_urls
.insert(GURL(kGoodManifestURL
));
159 expected_requested_urls
.insert(GURL(kResourceFetchFailureURL
));
160 expected_requested_urls
.insert(GURL(kGoodResourceURL
));
161 expected_requested_urls
.insert(GURL(kForcedStartingURLManifestURL
));
163 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
165 EXPECT_TRUE(precache_delegate_
.was_on_done_called());
168 TEST_F(PrecacheFetcherTest
, ConfigFetchFailure
) {
169 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
170 switches::kPrecacheConfigSettingsURL
, kConfigURL
);
172 std::list
<GURL
> starting_urls(1, GURL("http://starting-url.com"));
174 factory_
.SetFakeResponse(GURL(kConfigURL
), "",
175 net::HTTP_INTERNAL_SERVER_ERROR
,
176 net::URLRequestStatus::FAILED
);
178 PrecacheFetcher
precache_fetcher(starting_urls
, request_context_
.get(),
179 &precache_delegate_
);
180 precache_fetcher
.Start();
182 base::MessageLoop::current()->RunUntilIdle();
184 std::multiset
<GURL
> expected_requested_urls
;
185 expected_requested_urls
.insert(GURL(kConfigURL
));
186 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
188 EXPECT_TRUE(precache_delegate_
.was_on_done_called());
191 TEST_F(PrecacheFetcherTest
, BadConfig
) {
192 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
193 switches::kPrecacheConfigSettingsURL
, kConfigURL
);
195 std::list
<GURL
> starting_urls(1, GURL("http://starting-url.com"));
197 factory_
.SetFakeResponse(GURL(kConfigURL
), "bad protobuf", net::HTTP_OK
,
198 net::URLRequestStatus::SUCCESS
);
200 PrecacheFetcher
precache_fetcher(starting_urls
, request_context_
.get(),
201 &precache_delegate_
);
202 precache_fetcher
.Start();
204 base::MessageLoop::current()->RunUntilIdle();
206 std::multiset
<GURL
> expected_requested_urls
;
207 expected_requested_urls
.insert(GURL(kConfigURL
));
208 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
210 EXPECT_TRUE(precache_delegate_
.was_on_done_called());
213 TEST_F(PrecacheFetcherTest
, Cancel
) {
214 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
215 switches::kPrecacheConfigSettingsURL
, kConfigURL
);
217 std::list
<GURL
> starting_urls(1, GURL("http://starting-url.com"));
219 PrecacheConfigurationSettings config
;
220 config
.set_top_sites_count(1);
222 factory_
.SetFakeResponse(GURL(kConfigURL
), config
.SerializeAsString(),
223 net::HTTP_OK
, net::URLRequestStatus::SUCCESS
);
225 scoped_ptr
<PrecacheFetcher
> precache_fetcher(new PrecacheFetcher(
226 starting_urls
, request_context_
.get(), &precache_delegate_
));
227 precache_fetcher
->Start();
229 // Destroy the PrecacheFetcher to cancel precaching. This should not cause
230 // OnDone to be called on the precache delegate.
231 precache_fetcher
.reset();
233 base::MessageLoop::current()->RunUntilIdle();
235 std::multiset
<GURL
> expected_requested_urls
;
236 expected_requested_urls
.insert(GURL(kConfigURL
));
237 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
239 EXPECT_FALSE(precache_delegate_
.was_on_done_called());
242 #if defined(PRECACHE_CONFIG_SETTINGS_URL)
244 // If the default precache configuration settings URL is defined, then test that
245 // it works with the PrecacheFetcher.
246 TEST_F(PrecacheFetcherTest
, PrecacheUsingDefaultConfigSettingsURL
) {
247 std::list
<GURL
> starting_urls(1, GURL("http://starting-url.com"));
249 PrecacheConfigurationSettings config
;
250 config
.set_top_sites_count(0);
252 factory_
.SetFakeResponse(GURL(PRECACHE_CONFIG_SETTINGS_URL
),
253 config
.SerializeAsString(), net::HTTP_OK
,
254 net::URLRequestStatus::SUCCESS
);
256 PrecacheFetcher
precache_fetcher(starting_urls
, request_context_
.get(),
257 &precache_delegate_
);
258 precache_fetcher
.Start();
260 base::MessageLoop::current()->RunUntilIdle();
262 std::multiset
<GURL
> expected_requested_urls
;
263 expected_requested_urls
.insert(GURL(PRECACHE_CONFIG_SETTINGS_URL
));
264 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
266 EXPECT_TRUE(precache_delegate_
.was_on_done_called());
269 #endif // PRECACHE_CONFIG_SETTINGS_URL
271 #if defined(PRECACHE_MANIFEST_URL_PREFIX)
273 // If the default precache manifest URL prefix is defined, then test that it
274 // works with the PrecacheFetcher.
275 TEST_F(PrecacheFetcherTest
, PrecacheUsingDefaultManifestURLPrefix
) {
276 CommandLine::ForCurrentProcess()->AppendSwitchASCII(
277 switches::kPrecacheConfigSettingsURL
, kConfigURL
);
279 std::list
<GURL
> starting_urls(1, GURL("http://starting-url.com"));
281 PrecacheConfigurationSettings config
;
282 config
.set_top_sites_count(1);
284 GURL
manifest_url(PRECACHE_MANIFEST_URL_PREFIX
285 "http%253A%252F%252Fstarting-url.com%252F");
287 factory_
.SetFakeResponse(GURL(kConfigURL
), config
.SerializeAsString(),
288 net::HTTP_OK
, net::URLRequestStatus::SUCCESS
);
289 factory_
.SetFakeResponse(manifest_url
, PrecacheManifest().SerializeAsString(),
290 net::HTTP_OK
, net::URLRequestStatus::SUCCESS
);
292 PrecacheFetcher
precache_fetcher(starting_urls
, request_context_
.get(),
293 &precache_delegate_
);
294 precache_fetcher
.Start();
296 base::MessageLoop::current()->RunUntilIdle();
298 std::multiset
<GURL
> expected_requested_urls
;
299 expected_requested_urls
.insert(GURL(kConfigURL
));
300 expected_requested_urls
.insert(manifest_url
);
301 EXPECT_EQ(expected_requested_urls
, url_callback_
.requested_urls());
303 EXPECT_TRUE(precache_delegate_
.was_on_done_called());
306 #endif // PRECACHE_MANIFEST_URL_PREFIX
310 } // namespace precache