• 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 #ifndef CHROME_BROWSER_PREDICTORS_RESOURCE_PREFETCHER_H_
6 #define CHROME_BROWSER_PREDICTORS_RESOURCE_PREFETCHER_H_
7 
8 #include <list>
9 #include <map>
10 #include <vector>
11 
12 #include "base/gtest_prod_util.h"
13 #include "base/memory/scoped_ptr.h"
14 #include "base/memory/scoped_vector.h"
15 #include "base/threading/non_thread_safe.h"
16 #include "chrome/browser/predictors/resource_prefetch_common.h"
17 #include "net/url_request/url_request.h"
18 #include "url/gurl.h"
19 
20 namespace net {
21 class URLRequestContext;
22 }
23 
24 namespace predictors {
25 
26 // Responsible for prefetching resources for a single navigation based on the
27 // input list of resources.
28 //  - Limits the max number of resources in flight for any host and also across
29 //    hosts.
30 //  - When stopped, will wait for the pending requests to finish.
31 //  - Lives entirely on the IO thread.
32 class ResourcePrefetcher : public base::NonThreadSafe,
33                            public net::URLRequest::Delegate {
34  public:
35   // Denotes the prefetch request for a single subresource.
36   struct Request {
37     explicit Request(const GURL& i_resource_url);
38     Request(const Request& other);
39 
40     enum PrefetchStatus {
41       PREFETCH_STATUS_NOT_STARTED,
42       PREFETCH_STATUS_STARTED,
43 
44       // Cancellation reasons.
45       PREFETCH_STATUS_REDIRECTED,
46       PREFETCH_STATUS_AUTH_REQUIRED,
47       PREFETCH_STATUS_CERT_REQUIRED,
48       PREFETCH_STATUS_CERT_ERROR,
49       PREFETCH_STATUS_CANCELLED,
50       PREFETCH_STATUS_FAILED,
51 
52       // Successful prefetch states.
53       PREFETCH_STATUS_FROM_CACHE,
54       PREFETCH_STATUS_FROM_NETWORK
55     };
56 
57     enum UsageStatus {
58       USAGE_STATUS_NOT_REQUESTED,
59       USAGE_STATUS_FROM_CACHE,
60       USAGE_STATUS_FROM_NETWORK,
61       USAGE_STATUS_NAVIGATION_ABANDONED
62     };
63 
64     GURL resource_url;
65     PrefetchStatus prefetch_status;
66     UsageStatus usage_status;
67   };
68   typedef ScopedVector<Request> RequestVector;
69 
70   // Used to communicate when the prefetching is done. All methods are invoked
71   // on the IO thread.
72   class Delegate {
73    public:
~Delegate()74     virtual ~Delegate() { }
75 
76     // Called when the ResourcePrefetcher is finished, i.e. there is nothing
77     // pending in flight. Should take ownership of |requests|.
78     virtual void ResourcePrefetcherFinished(
79         ResourcePrefetcher* prefetcher,
80         RequestVector* requests) = 0;
81 
82     virtual net::URLRequestContext* GetURLRequestContext() = 0;
83   };
84 
85   // |delegate| has to outlive the ResourcePrefetcher. The ResourcePrefetcher
86   // takes ownership of |requests|.
87   ResourcePrefetcher(Delegate* delegate,
88                      const ResourcePrefetchPredictorConfig& config,
89                      const NavigationID& navigation_id,
90                      PrefetchKeyType key_type,
91                      scoped_ptr<RequestVector> requests);
92   virtual ~ResourcePrefetcher();
93 
94   void Start();  // Kicks off the prefetching. Can only be called once.
95   void Stop();   // No additional prefetches will be queued after this.
96 
navigation_id()97   const NavigationID& navigation_id() const { return navigation_id_; }
key_type()98   PrefetchKeyType key_type() const { return key_type_; }
99 
100  private:
101   friend class ResourcePrefetcherTest;
102   friend class TestResourcePrefetcher;
103 
104   // Launches new prefetch requests if possible.
105   void TryToLaunchPrefetchRequests();
106 
107   // Starts a net::URLRequest for the input |request|.
108   void SendRequest(Request* request);
109 
110   // Called by |SendRequest| to start the |request|. This is necessary to stub
111   // out the Start() call to net::URLRequest for unittesting.
112   virtual void StartURLRequest(net::URLRequest* request);
113 
114   // Marks the request as finished, with the given status.
115   void FinishRequest(net::URLRequest* request, Request::PrefetchStatus status);
116 
117   // Reads the response data from the response - required for the resource to
118   // be cached correctly. Stubbed out during testing.
119   virtual void ReadFullResponse(net::URLRequest* request);
120 
121   // Returns true if the request has more data that needs to be read. If it
122   // returns false, the request should not be referenced again.
123   bool ShouldContinueReadingRequest(net::URLRequest* request, int bytes_read);
124 
125   // net::URLRequest::Delegate methods.
126   virtual void OnReceivedRedirect(net::URLRequest* request,
127                                   const GURL& new_url,
128                                   bool* defer_redirect) OVERRIDE;
129   virtual void OnAuthRequired(net::URLRequest* request,
130                               net::AuthChallengeInfo* auth_info) OVERRIDE;
131   virtual void OnCertificateRequested(
132       net::URLRequest* request,
133       net::SSLCertRequestInfo* cert_request_info) OVERRIDE;
134   virtual void OnSSLCertificateError(net::URLRequest* request,
135                                      const net::SSLInfo& ssl_info,
136                                      bool fatal) OVERRIDE;
137   virtual void OnResponseStarted(net::URLRequest* request) OVERRIDE;
138   virtual void OnReadCompleted(net::URLRequest* request,
139                                int bytes_read) OVERRIDE;
140 
141   enum PrefetcherState {
142     INITIALIZED = 0,  // Prefetching hasn't started.
143     RUNNING = 1,      // Prefetching started, allowed to add more requests.
144     STOPPED = 2,      // Prefetching started, not allowed to add more requests.
145     FINISHED = 3      // No more inflight request, new requests not possible.
146   };
147 
148   PrefetcherState state_;
149   Delegate* const delegate_;
150   ResourcePrefetchPredictorConfig const config_;
151   NavigationID navigation_id_;
152   PrefetchKeyType key_type_;
153   scoped_ptr<RequestVector> request_vector_;
154 
155   std::map<net::URLRequest*, Request*> inflight_requests_;
156   std::list<Request*> request_queue_;
157   std::map<std::string, int> host_inflight_counts_;
158 
159   DISALLOW_COPY_AND_ASSIGN(ResourcePrefetcher);
160 };
161 
162 }  // namespace predictors
163 
164 #endif  // CHROME_BROWSER_PREDICTORS_RESOURCE_PREFETCHER_H_
165