• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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.
4 
5 #include "base/memory/ref_counted.h"
6 #include "base/memory/scoped_ptr.h"
7 #include "base/message_loop/message_loop.h"
8 #include "chrome/browser/predictors/resource_prefetcher.h"
9 #include "chrome/browser/predictors/resource_prefetcher_manager.h"
10 #include "chrome/test/base/testing_profile.h"
11 #include "content/public/test/test_browser_thread.h"
12 #include "net/url_request/url_request.h"
13 #include "net/url_request/url_request_test_util.h"
14 #include "testing/gmock/include/gmock/gmock.h"
15 #include "testing/gtest/include/gtest/gtest.h"
16 
17 using testing::Eq;
18 using testing::Property;
19 
20 namespace predictors {
21 
22 // Wrapper over the ResourcePrefetcher that stubs out the StartURLRequest call
23 // since we do not want to do network fetches in this unittest.
24 class TestResourcePrefetcher : public ResourcePrefetcher {
25  public:
TestResourcePrefetcher(ResourcePrefetcher::Delegate * delegate,const ResourcePrefetchPredictorConfig & config,const NavigationID & navigation_id,PrefetchKeyType key_type,scoped_ptr<RequestVector> requests)26   TestResourcePrefetcher(ResourcePrefetcher::Delegate* delegate,
27                          const ResourcePrefetchPredictorConfig& config,
28                          const NavigationID& navigation_id,
29                          PrefetchKeyType key_type,
30                          scoped_ptr<RequestVector> requests)
31       : ResourcePrefetcher(delegate, config, navigation_id,
32                            key_type, requests.Pass()) { }
33 
~TestResourcePrefetcher()34   virtual ~TestResourcePrefetcher() { }
35 
36   MOCK_METHOD1(StartURLRequest, void(net::URLRequest* request));
37 
ReadFullResponse(net::URLRequest * request)38   void ReadFullResponse(net::URLRequest* request) OVERRIDE {
39     FinishRequest(request, Request::PREFETCH_STATUS_FROM_CACHE);
40   }
41 
42  private:
43   DISALLOW_COPY_AND_ASSIGN(TestResourcePrefetcher);
44 };
45 
46 
47 // Delegate for ResourcePrefetcher.
48 class TestResourcePrefetcherDelegate : public ResourcePrefetcher::Delegate {
49  public:
TestResourcePrefetcherDelegate(base::MessageLoop * loop)50   explicit TestResourcePrefetcherDelegate(base::MessageLoop* loop)
51       : request_context_getter_(new net::TestURLRequestContextGetter(
52           loop->message_loop_proxy())) { }
~TestResourcePrefetcherDelegate()53   ~TestResourcePrefetcherDelegate() { }
54 
GetURLRequestContext()55   virtual net::URLRequestContext* GetURLRequestContext() OVERRIDE {
56     return request_context_getter_->GetURLRequestContext();
57   }
58 
59   MOCK_METHOD2(ResourcePrefetcherFinished,
60                void(ResourcePrefetcher* prefetcher,
61                     ResourcePrefetcher::RequestVector* requests));
62 
63  private:
64   scoped_refptr<net::TestURLRequestContextGetter> request_context_getter_;
65 
66   DISALLOW_COPY_AND_ASSIGN(TestResourcePrefetcherDelegate);
67 };
68 
69 
70 // The following unittest tests most of the ResourcePrefetcher except for:
71 // 1. Call to ReadFullResponse. There does not seem to be a good way to test the
72 //    function in a unittest, and probably requires a browser_test.
73 // 2. Setting of the Prefetch status for cache vs non cache.
74 class ResourcePrefetcherTest : public testing::Test {
75  public:
76   ResourcePrefetcherTest();
77   virtual ~ResourcePrefetcherTest();
78 
79  protected:
80   typedef ResourcePrefetcher::Request Request;
81 
AddStartUrlRequestExpectation(const std::string & url)82   void AddStartUrlRequestExpectation(const std::string& url) {
83     EXPECT_CALL(*prefetcher_,
84                 StartURLRequest(Property(&net::URLRequest::original_url,
85                                          Eq(GURL(url)))));
86   }
87 
CheckPrefetcherState(size_t inflight,size_t queue,size_t host)88   void CheckPrefetcherState(size_t inflight, size_t queue, size_t host) {
89     EXPECT_EQ(prefetcher_->inflight_requests_.size(), inflight);
90     EXPECT_EQ(prefetcher_->request_queue_.size(), queue);
91     EXPECT_EQ(prefetcher_->host_inflight_counts_.size(), host);
92   }
93 
GetInFlightRequest(const std::string & url_str)94   net::URLRequest* GetInFlightRequest(const std::string& url_str) {
95     GURL url(url_str);
96 
97     for (std::list<Request*>::const_iterator it =
98          prefetcher_->request_queue_.begin();
99          it != prefetcher_->request_queue_.end(); ++it) {
100       EXPECT_NE((*it)->resource_url, url);
101     }
102     for (std::map<net::URLRequest*, Request*>::const_iterator it =
103          prefetcher_->inflight_requests_.begin();
104          it != prefetcher_->inflight_requests_.end(); ++it) {
105       if (it->first->original_url() == url)
106         return it->first;
107     }
108     EXPECT_TRUE(false) << "Infligh request not found: " << url_str;
109     return NULL;
110   }
111 
112 
OnReceivedRedirect(const std::string & url)113   void OnReceivedRedirect(const std::string& url) {
114     prefetcher_->OnReceivedRedirect(
115         GetInFlightRequest(url), GURL(std::string()), NULL);
116   }
OnAuthRequired(const std::string & url)117   void OnAuthRequired(const std::string& url) {
118     prefetcher_->OnAuthRequired(GetInFlightRequest(url), NULL);
119   }
OnCertificateRequested(const std::string & url)120   void OnCertificateRequested(const std::string& url) {
121     prefetcher_->OnCertificateRequested(GetInFlightRequest(url), NULL);
122   }
OnSSLCertificateError(const std::string & url)123   void OnSSLCertificateError(const std::string& url) {
124     prefetcher_->OnSSLCertificateError(GetInFlightRequest(url),
125                                        net::SSLInfo(), false);
126   }
OnResponse(const std::string & url)127   void OnResponse(const std::string& url) {
128     prefetcher_->OnResponseStarted(GetInFlightRequest(url));
129   }
130 
131   base::MessageLoop loop_;
132   content::TestBrowserThread io_thread_;
133   ResourcePrefetchPredictorConfig config_;
134   TestResourcePrefetcherDelegate prefetcher_delegate_;
135   scoped_ptr<TestResourcePrefetcher> prefetcher_;
136 
137  private:
138   DISALLOW_COPY_AND_ASSIGN(ResourcePrefetcherTest);
139 };
140 
ResourcePrefetcherTest()141 ResourcePrefetcherTest::ResourcePrefetcherTest()
142     : loop_(base::MessageLoop::TYPE_IO),
143       io_thread_(content::BrowserThread::IO, &loop_),
144       prefetcher_delegate_(&loop_) {
145   config_.max_prefetches_inflight_per_navigation = 5;
146   config_.max_prefetches_inflight_per_host_per_navigation = 2;
147 }
148 
~ResourcePrefetcherTest()149 ResourcePrefetcherTest::~ResourcePrefetcherTest() {
150 }
151 
TEST_F(ResourcePrefetcherTest,TestPrefetcherFinishes)152 TEST_F(ResourcePrefetcherTest, TestPrefetcherFinishes) {
153   scoped_ptr<ResourcePrefetcher::RequestVector> requests(
154       new ResourcePrefetcher::RequestVector);
155   requests->push_back(new ResourcePrefetcher::Request(GURL(
156       "http://www.google.com/resource1.html")));
157   requests->push_back(new ResourcePrefetcher::Request(GURL(
158       "http://www.google.com/resource2.png")));
159   requests->push_back(new ResourcePrefetcher::Request(GURL(
160       "http://yahoo.com/resource1.png")));
161   requests->push_back(new ResourcePrefetcher::Request(GURL(
162       "http://yahoo.com/resource2.png")));
163   requests->push_back(new ResourcePrefetcher::Request(GURL(
164       "http://yahoo.com/resource3.png")));
165   requests->push_back(new ResourcePrefetcher::Request(GURL(
166       "http://m.google.com/resource1.jpg")));
167   requests->push_back(new ResourcePrefetcher::Request(GURL(
168       "http://www.google.com/resource3.html")));
169   requests->push_back(new ResourcePrefetcher::Request(GURL(
170       "http://m.google.com/resource2.html")));
171   requests->push_back(new ResourcePrefetcher::Request(GURL(
172       "http://m.google.com/resource3.css")));
173   requests->push_back(new ResourcePrefetcher::Request(GURL(
174       "http://m.google.com/resource4.png")));
175   requests->push_back(new ResourcePrefetcher::Request(GURL(
176       "http://yahoo.com/resource4.png")));
177   requests->push_back(new ResourcePrefetcher::Request(GURL(
178       "http://yahoo.com/resource5.png")));
179 
180   NavigationID navigation_id;
181   navigation_id.render_process_id = 1;
182   navigation_id.render_view_id = 2;
183   navigation_id.main_frame_url = GURL("http://www.google.com");
184 
185   // Needed later for comparison.
186   ResourcePrefetcher::RequestVector* requests_ptr = requests.get();
187 
188   prefetcher_.reset(new TestResourcePrefetcher(&prefetcher_delegate_,
189                                                config_,
190                                                navigation_id,
191                                                PREFETCH_KEY_TYPE_URL,
192                                                requests.Pass()));
193 
194   // Starting the prefetcher maxes out the number of possible requests.
195   AddStartUrlRequestExpectation("http://www.google.com/resource1.html");
196   AddStartUrlRequestExpectation("http://www.google.com/resource2.png");
197   AddStartUrlRequestExpectation("http://yahoo.com/resource1.png");
198   AddStartUrlRequestExpectation("http://yahoo.com/resource2.png");
199   AddStartUrlRequestExpectation("http://m.google.com/resource1.jpg");
200 
201   prefetcher_->Start();
202   CheckPrefetcherState(5, 7, 3);
203 
204   AddStartUrlRequestExpectation("http://m.google.com/resource2.html");
205   OnResponse("http://m.google.com/resource1.jpg");
206   CheckPrefetcherState(5, 6, 3);
207 
208   AddStartUrlRequestExpectation("http://www.google.com/resource3.html");
209   OnSSLCertificateError("http://www.google.com/resource1.html");
210   CheckPrefetcherState(5, 5, 3);
211 
212   AddStartUrlRequestExpectation("http://m.google.com/resource3.css");
213   OnResponse("http://m.google.com/resource2.html");
214   CheckPrefetcherState(5, 4, 3);
215 
216   AddStartUrlRequestExpectation("http://m.google.com/resource4.png");
217   OnReceivedRedirect("http://www.google.com/resource3.html");
218   CheckPrefetcherState(5, 3, 3);
219 
220   OnResponse("http://www.google.com/resource2.png");
221   CheckPrefetcherState(4, 3, 2);
222 
223   AddStartUrlRequestExpectation("http://yahoo.com/resource3.png");
224   OnReceivedRedirect("http://yahoo.com/resource2.png");
225   CheckPrefetcherState(4, 2, 2);
226 
227   AddStartUrlRequestExpectation("http://yahoo.com/resource4.png");
228   OnResponse("http://yahoo.com/resource1.png");
229   CheckPrefetcherState(4, 1, 2);
230 
231   AddStartUrlRequestExpectation("http://yahoo.com/resource5.png");
232   OnResponse("http://yahoo.com/resource4.png");
233   CheckPrefetcherState(4, 0, 2);
234 
235   OnResponse("http://yahoo.com/resource5.png");
236   CheckPrefetcherState(3, 0, 2);
237 
238   OnCertificateRequested("http://m.google.com/resource4.png");
239   CheckPrefetcherState(2, 0, 2);
240 
241   OnAuthRequired("http://m.google.com/resource3.css");
242   CheckPrefetcherState(1, 0, 1);
243 
244   // Expect the final call.
245   EXPECT_CALL(prefetcher_delegate_,
246               ResourcePrefetcherFinished(Eq(prefetcher_.get()),
247                                          Eq(requests_ptr)));
248 
249   OnResponse("http://yahoo.com/resource3.png");
250   CheckPrefetcherState(0, 0, 0);
251 
252   // Check the prefetch status.
253   EXPECT_EQ((*requests_ptr)[0]->prefetch_status,
254             Request::PREFETCH_STATUS_CERT_ERROR);
255   EXPECT_EQ((*requests_ptr)[1]->prefetch_status,
256             Request::PREFETCH_STATUS_FROM_CACHE);
257   EXPECT_EQ((*requests_ptr)[2]->prefetch_status,
258             Request::PREFETCH_STATUS_FROM_CACHE);
259   EXPECT_EQ((*requests_ptr)[3]->prefetch_status,
260             Request::PREFETCH_STATUS_REDIRECTED);
261   EXPECT_EQ((*requests_ptr)[4]->prefetch_status,
262             Request::PREFETCH_STATUS_FROM_CACHE);
263   EXPECT_EQ((*requests_ptr)[5]->prefetch_status,
264             Request::PREFETCH_STATUS_FROM_CACHE);
265   EXPECT_EQ((*requests_ptr)[6]->prefetch_status,
266             Request::PREFETCH_STATUS_REDIRECTED);
267   EXPECT_EQ((*requests_ptr)[7]->prefetch_status,
268             Request::PREFETCH_STATUS_FROM_CACHE);
269   EXPECT_EQ((*requests_ptr)[8]->prefetch_status,
270             Request::PREFETCH_STATUS_AUTH_REQUIRED);
271   EXPECT_EQ((*requests_ptr)[9]->prefetch_status,
272             Request::PREFETCH_STATUS_CERT_REQUIRED);
273   EXPECT_EQ((*requests_ptr)[10]->prefetch_status,
274             Request::PREFETCH_STATUS_FROM_CACHE);
275   EXPECT_EQ((*requests_ptr)[11]->prefetch_status,
276             Request::PREFETCH_STATUS_FROM_CACHE);
277 
278   delete requests_ptr;
279 }
280 
TEST_F(ResourcePrefetcherTest,TestPrefetcherStopped)281 TEST_F(ResourcePrefetcherTest, TestPrefetcherStopped) {
282   scoped_ptr<ResourcePrefetcher::RequestVector> requests(
283       new ResourcePrefetcher::RequestVector);
284   requests->push_back(new ResourcePrefetcher::Request(GURL(
285       "http://www.google.com/resource1.html")));
286   requests->push_back(new ResourcePrefetcher::Request(GURL(
287       "http://www.google.com/resource2.png")));
288   requests->push_back(new ResourcePrefetcher::Request(GURL(
289       "http://yahoo.com/resource1.png")));
290   requests->push_back(new ResourcePrefetcher::Request(GURL(
291       "http://yahoo.com/resource2.png")));
292   requests->push_back(new ResourcePrefetcher::Request(GURL(
293       "http://yahoo.com/resource3.png")));
294   requests->push_back(new ResourcePrefetcher::Request(GURL(
295       "http://m.google.com/resource1.jpg")));
296 
297   NavigationID navigation_id;
298   navigation_id.render_process_id = 1;
299   navigation_id.render_view_id = 2;
300   navigation_id.main_frame_url = GURL("http://www.google.com");
301 
302   // Needed later for comparison.
303   ResourcePrefetcher::RequestVector* requests_ptr = requests.get();
304 
305   prefetcher_.reset(new TestResourcePrefetcher(&prefetcher_delegate_,
306                                                config_,
307                                                navigation_id,
308                                                PREFETCH_KEY_TYPE_HOST,
309                                                requests.Pass()));
310 
311   // Starting the prefetcher maxes out the number of possible requests.
312   AddStartUrlRequestExpectation("http://www.google.com/resource1.html");
313   AddStartUrlRequestExpectation("http://www.google.com/resource2.png");
314   AddStartUrlRequestExpectation("http://yahoo.com/resource1.png");
315   AddStartUrlRequestExpectation("http://yahoo.com/resource2.png");
316   AddStartUrlRequestExpectation("http://m.google.com/resource1.jpg");
317 
318   prefetcher_->Start();
319   CheckPrefetcherState(5, 1, 3);
320 
321   OnResponse("http://www.google.com/resource1.html");
322   CheckPrefetcherState(4, 1, 3);
323 
324   prefetcher_->Stop();  // No more queueing.
325 
326   OnResponse("http://www.google.com/resource2.png");
327   CheckPrefetcherState(3, 1, 2);
328 
329   OnResponse("http://yahoo.com/resource1.png");
330   CheckPrefetcherState(2, 1, 2);
331 
332   OnResponse("http://yahoo.com/resource2.png");
333   CheckPrefetcherState(1, 1, 1);
334 
335   // Expect the final call.
336   EXPECT_CALL(prefetcher_delegate_,
337               ResourcePrefetcherFinished(Eq(prefetcher_.get()),
338                                          Eq(requests_ptr)));
339 
340   OnResponse("http://m.google.com/resource1.jpg");
341   CheckPrefetcherState(0, 1, 0);
342 
343   delete requests_ptr;
344 }
345 
346 }  // namespace predictors
347