• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 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 "net/http/http_stream_factory_impl.h"
6 
7 #include <string>
8 
9 #include "base/basictypes.h"
10 #include "net/base/cert_verifier.h"
11 #include "net/base/mock_host_resolver.h"
12 #include "net/base/net_log.h"
13 #include "net/base/ssl_config_service_defaults.h"
14 #include "net/base/test_completion_callback.h"
15 #include "net/http/http_auth_handler_factory.h"
16 #include "net/http/http_network_session.h"
17 #include "net/http/http_network_session_peer.h"
18 #include "net/http/http_request_info.h"
19 #include "net/proxy/proxy_info.h"
20 #include "net/proxy/proxy_service.h"
21 #include "net/socket/socket_test_util.h"
22 #include "net/spdy/spdy_session.h"
23 #include "net/spdy/spdy_session_pool.h"
24 #include "testing/gtest/include/gtest/gtest.h"
25 
26 namespace net {
27 
28 namespace {
29 
30 class MockHttpStreamFactoryImpl : public HttpStreamFactoryImpl {
31  public:
MockHttpStreamFactoryImpl(HttpNetworkSession * session)32   MockHttpStreamFactoryImpl(HttpNetworkSession* session)
33       : HttpStreamFactoryImpl(session),
34         preconnect_done_(false),
35         waiting_for_preconnect_(false) {}
36 
37 
WaitForPreconnects()38   void WaitForPreconnects() {
39     while (!preconnect_done_) {
40       waiting_for_preconnect_ = true;
41       MessageLoop::current()->Run();
42       waiting_for_preconnect_ = false;
43     }
44   }
45 
46  private:
47   // HttpStreamFactoryImpl methods.
OnPreconnectsCompleteInternal()48   virtual void OnPreconnectsCompleteInternal() {
49     preconnect_done_ = true;
50     if (waiting_for_preconnect_)
51       MessageLoop::current()->Quit();
52   }
53 
54   bool preconnect_done_;
55   bool waiting_for_preconnect_;
56 };
57 
58 struct SessionDependencies {
59   // Custom proxy service dependency.
SessionDependenciesnet::__anon645719b50111::SessionDependencies60   explicit SessionDependencies(ProxyService* proxy_service)
61       : host_resolver(new MockHostResolver),
62         cert_verifier(new CertVerifier),
63         proxy_service(proxy_service),
64         ssl_config_service(new SSLConfigServiceDefaults),
65         http_auth_handler_factory(
66             HttpAuthHandlerFactory::CreateDefault(host_resolver.get())),
67         net_log(NULL) {}
68 
69   scoped_ptr<MockHostResolverBase> host_resolver;
70   scoped_ptr<CertVerifier> cert_verifier;
71   scoped_refptr<ProxyService> proxy_service;
72   scoped_refptr<SSLConfigService> ssl_config_service;
73   MockClientSocketFactory socket_factory;
74   scoped_ptr<HttpAuthHandlerFactory> http_auth_handler_factory;
75   NetLog* net_log;
76 };
77 
CreateSession(SessionDependencies * session_deps)78 HttpNetworkSession* CreateSession(SessionDependencies* session_deps) {
79   HttpNetworkSession::Params params;
80   params.host_resolver = session_deps->host_resolver.get();
81   params.cert_verifier = session_deps->cert_verifier.get();
82   params.proxy_service = session_deps->proxy_service;
83   params.ssl_config_service = session_deps->ssl_config_service;
84   params.client_socket_factory = &session_deps->socket_factory;
85   params.http_auth_handler_factory =
86       session_deps->http_auth_handler_factory.get();
87   params.net_log = session_deps->net_log;
88   return new HttpNetworkSession(params);
89 }
90 
91 struct TestCase {
92   int num_streams;
93   bool ssl;
94 };
95 
96 TestCase kTests[] = {
97   { 1, false },
98   { 2, false },
99   { 1, true},
100   { 2, true},
101 };
102 
PreconnectHelper(const TestCase & test,HttpNetworkSession * session)103 void PreconnectHelper(const TestCase& test,
104                       HttpNetworkSession* session) {
105   HttpNetworkSessionPeer peer(session);
106   MockHttpStreamFactoryImpl* mock_factory =
107       new MockHttpStreamFactoryImpl(session);
108   peer.SetHttpStreamFactory(mock_factory);
109   SSLConfig ssl_config;
110   session->ssl_config_service()->GetSSLConfig(&ssl_config);
111 
112   HttpRequestInfo request;
113   request.method = "GET";
114   request.url = test.ssl ?  GURL("https://www.google.com") :
115       GURL("http://www.google.com");
116   request.load_flags = 0;
117 
118   ProxyInfo proxy_info;
119   TestCompletionCallback callback;
120 
121   session->http_stream_factory()->PreconnectStreams(
122       test.num_streams, request, ssl_config, BoundNetLog());
123   mock_factory->WaitForPreconnects();
124 };
125 
126 template<typename ParentPool>
127 class CapturePreconnectsSocketPool : public ParentPool {
128  public:
129   CapturePreconnectsSocketPool(HostResolver* host_resolver,
130                                CertVerifier* cert_verifier);
131 
last_num_streams() const132   int last_num_streams() const {
133     return last_num_streams_;
134   }
135 
RequestSocket(const std::string & group_name,const void * socket_params,RequestPriority priority,ClientSocketHandle * handle,CompletionCallback * callback,const BoundNetLog & net_log)136   virtual int RequestSocket(const std::string& group_name,
137                             const void* socket_params,
138                             RequestPriority priority,
139                             ClientSocketHandle* handle,
140                             CompletionCallback* callback,
141                             const BoundNetLog& net_log) {
142     ADD_FAILURE();
143     return ERR_UNEXPECTED;
144   }
145 
RequestSockets(const std::string & group_name,const void * socket_params,int num_sockets,const BoundNetLog & net_log)146   virtual void RequestSockets(const std::string& group_name,
147                               const void* socket_params,
148                               int num_sockets,
149                               const BoundNetLog& net_log) {
150     last_num_streams_ = num_sockets;
151   }
152 
CancelRequest(const std::string & group_name,ClientSocketHandle * handle)153   virtual void CancelRequest(const std::string& group_name,
154                              ClientSocketHandle* handle) {
155     ADD_FAILURE();
156   }
ReleaseSocket(const std::string & group_name,ClientSocket * socket,int id)157   virtual void ReleaseSocket(const std::string& group_name,
158                              ClientSocket* socket,
159                              int id) {
160     ADD_FAILURE();
161   }
CloseIdleSockets()162   virtual void CloseIdleSockets() {
163     ADD_FAILURE();
164   }
IdleSocketCount() const165   virtual int IdleSocketCount() const {
166     ADD_FAILURE();
167     return 0;
168   }
IdleSocketCountInGroup(const std::string & group_name) const169   virtual int IdleSocketCountInGroup(const std::string& group_name) const {
170     ADD_FAILURE();
171     return 0;
172   }
GetLoadState(const std::string & group_name,const ClientSocketHandle * handle) const173   virtual LoadState GetLoadState(const std::string& group_name,
174                                  const ClientSocketHandle* handle) const {
175     ADD_FAILURE();
176     return LOAD_STATE_IDLE;
177   }
ConnectionTimeout() const178   virtual base::TimeDelta ConnectionTimeout() const {
179     return base::TimeDelta();
180   }
181 
182  private:
183   int last_num_streams_;
184 };
185 
186 typedef CapturePreconnectsSocketPool<TransportClientSocketPool>
187 CapturePreconnectsTransportSocketPool;
188 typedef CapturePreconnectsSocketPool<HttpProxyClientSocketPool>
189 CapturePreconnectsHttpProxySocketPool;
190 typedef CapturePreconnectsSocketPool<SOCKSClientSocketPool>
191 CapturePreconnectsSOCKSSocketPool;
192 typedef CapturePreconnectsSocketPool<SSLClientSocketPool>
193 CapturePreconnectsSSLSocketPool;
194 
195 template<typename ParentPool>
CapturePreconnectsSocketPool(HostResolver * host_resolver,CertVerifier *)196 CapturePreconnectsSocketPool<ParentPool>::CapturePreconnectsSocketPool(
197     HostResolver* host_resolver, CertVerifier* /* cert_verifier */)
198     : ParentPool(0, 0, NULL, host_resolver, NULL, NULL),
199       last_num_streams_(-1) {}
200 
201 template<>
CapturePreconnectsSocketPool(HostResolver * host_resolver,CertVerifier *)202 CapturePreconnectsHttpProxySocketPool::CapturePreconnectsSocketPool(
203     HostResolver* host_resolver, CertVerifier* /* cert_verifier */)
204     : HttpProxyClientSocketPool(0, 0, NULL, host_resolver, NULL, NULL, NULL),
205       last_num_streams_(-1) {}
206 
207 template<>
CapturePreconnectsSocketPool(HostResolver * host_resolver,CertVerifier * cert_verifier)208 CapturePreconnectsSSLSocketPool::CapturePreconnectsSocketPool(
209     HostResolver* host_resolver, CertVerifier* cert_verifier)
210     : SSLClientSocketPool(0, 0, NULL, host_resolver, cert_verifier, NULL, NULL,
211                           NULL, NULL, NULL, NULL, NULL, NULL, NULL),
212       last_num_streams_(-1) {}
213 
TEST(HttpStreamFactoryTest,PreconnectDirect)214 TEST(HttpStreamFactoryTest, PreconnectDirect) {
215   for (size_t i = 0; i < arraysize(kTests); ++i) {
216     SessionDependencies session_deps(ProxyService::CreateDirect());
217     scoped_refptr<HttpNetworkSession> session(CreateSession(&session_deps));
218     HttpNetworkSessionPeer peer(session);
219     CapturePreconnectsTransportSocketPool* transport_conn_pool =
220         new CapturePreconnectsTransportSocketPool(
221             session_deps.host_resolver.get(),
222             session_deps.cert_verifier.get());
223     peer.SetTransportSocketPool(transport_conn_pool);
224     CapturePreconnectsSSLSocketPool* ssl_conn_pool =
225         new CapturePreconnectsSSLSocketPool(
226             session_deps.host_resolver.get(),
227             session_deps.cert_verifier.get());
228     peer.SetSSLSocketPool(ssl_conn_pool);
229     PreconnectHelper(kTests[i], session);
230     if (kTests[i].ssl)
231       EXPECT_EQ(kTests[i].num_streams, ssl_conn_pool->last_num_streams());
232     else
233       EXPECT_EQ(kTests[i].num_streams, transport_conn_pool->last_num_streams());
234   }
235 }
236 
TEST(HttpStreamFactoryTest,PreconnectHttpProxy)237 TEST(HttpStreamFactoryTest, PreconnectHttpProxy) {
238   for (size_t i = 0; i < arraysize(kTests); ++i) {
239     SessionDependencies session_deps(ProxyService::CreateFixed("http_proxy"));
240     scoped_refptr<HttpNetworkSession> session(CreateSession(&session_deps));
241     HttpNetworkSessionPeer peer(session);
242     HostPortPair proxy_host("http_proxy", 80);
243     CapturePreconnectsHttpProxySocketPool* http_proxy_pool =
244         new CapturePreconnectsHttpProxySocketPool(
245             session_deps.host_resolver.get(),
246             session_deps.cert_verifier.get());
247     peer.SetSocketPoolForHTTPProxy(proxy_host, http_proxy_pool);
248     CapturePreconnectsSSLSocketPool* ssl_conn_pool =
249         new CapturePreconnectsSSLSocketPool(
250             session_deps.host_resolver.get(),
251             session_deps.cert_verifier.get());
252     peer.SetSocketPoolForSSLWithProxy(proxy_host, ssl_conn_pool);
253     PreconnectHelper(kTests[i], session);
254     if (kTests[i].ssl)
255       EXPECT_EQ(kTests[i].num_streams, ssl_conn_pool->last_num_streams());
256     else
257       EXPECT_EQ(kTests[i].num_streams, http_proxy_pool->last_num_streams());
258   }
259 }
260 
TEST(HttpStreamFactoryTest,PreconnectSocksProxy)261 TEST(HttpStreamFactoryTest, PreconnectSocksProxy) {
262   for (size_t i = 0; i < arraysize(kTests); ++i) {
263     SessionDependencies session_deps(
264         ProxyService::CreateFixed("socks4://socks_proxy:1080"));
265     scoped_refptr<HttpNetworkSession> session(CreateSession(&session_deps));
266     HttpNetworkSessionPeer peer(session);
267     HostPortPair proxy_host("socks_proxy", 1080);
268     CapturePreconnectsSOCKSSocketPool* socks_proxy_pool =
269         new CapturePreconnectsSOCKSSocketPool(
270             session_deps.host_resolver.get(),
271             session_deps.cert_verifier.get());
272     peer.SetSocketPoolForSOCKSProxy(proxy_host, socks_proxy_pool);
273     CapturePreconnectsSSLSocketPool* ssl_conn_pool =
274         new CapturePreconnectsSSLSocketPool(
275             session_deps.host_resolver.get(),
276             session_deps.cert_verifier.get());
277     peer.SetSocketPoolForSSLWithProxy(proxy_host, ssl_conn_pool);
278     PreconnectHelper(kTests[i], session);
279     if (kTests[i].ssl)
280       EXPECT_EQ(kTests[i].num_streams, ssl_conn_pool->last_num_streams());
281     else
282       EXPECT_EQ(kTests[i].num_streams, socks_proxy_pool->last_num_streams());
283   }
284 }
285 
TEST(HttpStreamFactoryTest,PreconnectDirectWithExistingSpdySession)286 TEST(HttpStreamFactoryTest, PreconnectDirectWithExistingSpdySession) {
287   for (size_t i = 0; i < arraysize(kTests); ++i) {
288     SessionDependencies session_deps(ProxyService::CreateDirect());
289     scoped_refptr<HttpNetworkSession> session(CreateSession(&session_deps));
290     HttpNetworkSessionPeer peer(session);
291 
292     // Set an existing SpdySession in the pool.
293     HostPortPair host_port_pair("www.google.com", 443);
294     HostPortProxyPair pair(host_port_pair, ProxyServer::Direct());
295     scoped_refptr<SpdySession> spdy_session =
296         session->spdy_session_pool()->Get(pair, BoundNetLog());
297 
298     CapturePreconnectsTransportSocketPool* transport_conn_pool =
299         new CapturePreconnectsTransportSocketPool(
300             session_deps.host_resolver.get(),
301             session_deps.cert_verifier.get());
302     peer.SetTransportSocketPool(transport_conn_pool);
303     CapturePreconnectsSSLSocketPool* ssl_conn_pool =
304         new CapturePreconnectsSSLSocketPool(
305             session_deps.host_resolver.get(),
306             session_deps.cert_verifier.get());
307     peer.SetSSLSocketPool(ssl_conn_pool);
308     PreconnectHelper(kTests[i], session);
309     // We shouldn't be preconnecting if we have an existing session, which is
310     // the case for https://www.google.com.
311     if (kTests[i].ssl)
312       EXPECT_EQ(-1, ssl_conn_pool->last_num_streams());
313     else
314       EXPECT_EQ(kTests[i].num_streams, transport_conn_pool->last_num_streams());
315   }
316 }
317 
318 }  // namespace
319 
320 }  // namespace net
321