http_network_session.cc revision 6e8cce623b6e4fe0c9e4af605d675dd9d0338c38
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 "net/http/http_network_session.h"
6
7#include <utility>
8
9#include "base/compiler_specific.h"
10#include "base/debug/stack_trace.h"
11#include "base/logging.h"
12#include "base/stl_util.h"
13#include "base/strings/string_util.h"
14#include "base/values.h"
15#include "net/http/http_auth_handler_factory.h"
16#include "net/http/http_response_body_drainer.h"
17#include "net/http/http_stream_factory_impl.h"
18#include "net/http/url_security_manager.h"
19#include "net/proxy/proxy_service.h"
20#include "net/quic/crypto/quic_random.h"
21#include "net/quic/quic_clock.h"
22#include "net/quic/quic_crypto_client_stream_factory.h"
23#include "net/quic/quic_protocol.h"
24#include "net/quic/quic_stream_factory.h"
25#include "net/socket/client_socket_factory.h"
26#include "net/socket/client_socket_pool_manager_impl.h"
27#include "net/socket/next_proto.h"
28#include "net/spdy/hpack_huffman_aggregator.h"
29#include "net/spdy/spdy_session_pool.h"
30
31namespace {
32
33net::ClientSocketPoolManager* CreateSocketPoolManager(
34    net::HttpNetworkSession::SocketPoolType pool_type,
35    const net::HttpNetworkSession::Params& params) {
36  // TODO(yutak): Differentiate WebSocket pool manager and allow more
37  // simultaneous connections for WebSockets.
38  return new net::ClientSocketPoolManagerImpl(
39      params.net_log,
40      params.client_socket_factory
41          ? params.client_socket_factory
42          : net::ClientSocketFactory::GetDefaultFactory(),
43      params.host_resolver,
44      params.cert_verifier,
45      params.channel_id_service,
46      params.transport_security_state,
47      params.cert_transparency_verifier,
48      params.ssl_session_cache_shard,
49      params.proxy_service,
50      params.ssl_config_service,
51      params.enable_ssl_connect_job_waiting,
52      pool_type);
53}
54
55}  // unnamed namespace
56
57namespace net {
58
59HttpNetworkSession::Params::Params()
60    : client_socket_factory(NULL),
61      host_resolver(NULL),
62      cert_verifier(NULL),
63      channel_id_service(NULL),
64      transport_security_state(NULL),
65      cert_transparency_verifier(NULL),
66      proxy_service(NULL),
67      ssl_config_service(NULL),
68      http_auth_handler_factory(NULL),
69      network_delegate(NULL),
70      net_log(NULL),
71      host_mapping_rules(NULL),
72      enable_ssl_connect_job_waiting(false),
73      ignore_certificate_errors(false),
74      testing_fixed_http_port(0),
75      testing_fixed_https_port(0),
76      force_spdy_single_domain(false),
77      enable_spdy_compression(true),
78      enable_spdy_ping_based_connection_checking(true),
79      spdy_default_protocol(kProtoUnknown),
80      spdy_stream_initial_recv_window_size(0),
81      spdy_initial_max_concurrent_streams(0),
82      spdy_max_concurrent_streams_limit(0),
83      time_func(&base::TimeTicks::Now),
84      force_spdy_over_ssl(true),
85      force_spdy_always(false),
86      use_alternate_protocols(false),
87      alternate_protocol_probability_threshold(1),
88      enable_websocket_over_spdy(false),
89      enable_quic(false),
90      enable_quic_port_selection(true),
91      enable_quic_time_based_loss_detection(false),
92      quic_clock(NULL),
93      quic_random(NULL),
94      quic_max_packet_length(kDefaultMaxPacketSize),
95      enable_user_alternate_protocol_ports(false),
96      quic_crypto_client_stream_factory(NULL) {
97  quic_supported_versions.push_back(QUIC_VERSION_21);
98}
99
100HttpNetworkSession::Params::~Params() {}
101
102// TODO(mbelshe): Move the socket factories into HttpStreamFactory.
103HttpNetworkSession::HttpNetworkSession(const Params& params)
104    : net_log_(params.net_log),
105      network_delegate_(params.network_delegate),
106      http_server_properties_(params.http_server_properties),
107      cert_verifier_(params.cert_verifier),
108      http_auth_handler_factory_(params.http_auth_handler_factory),
109      proxy_service_(params.proxy_service),
110      ssl_config_service_(params.ssl_config_service),
111      normal_socket_pool_manager_(
112          CreateSocketPoolManager(NORMAL_SOCKET_POOL, params)),
113      websocket_socket_pool_manager_(
114          CreateSocketPoolManager(WEBSOCKET_SOCKET_POOL, params)),
115      quic_stream_factory_(params.host_resolver,
116                           params.client_socket_factory ?
117                               params.client_socket_factory :
118                               net::ClientSocketFactory::GetDefaultFactory(),
119                           params.http_server_properties,
120                           params.cert_verifier,
121                           params.channel_id_service,
122                           params.transport_security_state,
123                           params.quic_crypto_client_stream_factory,
124                           params.quic_random ? params.quic_random :
125                               QuicRandom::GetInstance(),
126                           params.quic_clock ? params. quic_clock :
127                               new QuicClock(),
128                           params.quic_max_packet_length,
129                           params.quic_user_agent_id,
130                           params.quic_supported_versions,
131                           params.enable_quic_port_selection,
132                           params.enable_quic_time_based_loss_detection,
133                           params.quic_connection_options),
134      spdy_session_pool_(params.host_resolver,
135                         params.ssl_config_service,
136                         params.http_server_properties,
137                         params.force_spdy_single_domain,
138                         params.enable_spdy_compression,
139                         params.enable_spdy_ping_based_connection_checking,
140                         params.spdy_default_protocol,
141                         params.spdy_stream_initial_recv_window_size,
142                         params.spdy_initial_max_concurrent_streams,
143                         params.spdy_max_concurrent_streams_limit,
144                         params.time_func,
145                         params.trusted_spdy_proxy),
146      http_stream_factory_(new HttpStreamFactoryImpl(this, false)),
147      http_stream_factory_for_websocket_(
148          new HttpStreamFactoryImpl(this, true)),
149      params_(params) {
150  DCHECK(proxy_service_);
151  DCHECK(ssl_config_service_.get());
152  CHECK(http_server_properties_);
153
154  for (int i = ALTERNATE_PROTOCOL_MINIMUM_VALID_VERSION;
155       i <= ALTERNATE_PROTOCOL_MAXIMUM_VALID_VERSION; ++i) {
156    enabled_protocols_[i - ALTERNATE_PROTOCOL_MINIMUM_VALID_VERSION] = false;
157  }
158
159  // TODO(rtenneti): bug 116575 - consider combining the NextProto and
160  // AlternateProtocol.
161  for (std::vector<NextProto>::const_iterator it = params_.next_protos.begin();
162       it != params_.next_protos.end(); ++it) {
163    NextProto proto = *it;
164
165    // Add the protocol to the TLS next protocol list, except for QUIC
166    // since it uses UDP.
167    if (proto != kProtoQUIC1SPDY3) {
168      next_protos_.push_back(SSLClientSocket::NextProtoToString(proto));
169    }
170
171    // Enable the corresponding alternate protocol, except for HTTP
172    // which has not corresponding alternative.
173    if (proto != kProtoHTTP11) {
174      AlternateProtocol alternate = AlternateProtocolFromNextProto(proto);
175      if (!IsAlternateProtocolValid(alternate)) {
176        NOTREACHED() << "Invalid next proto: " << proto;
177        continue;
178      }
179      enabled_protocols_[alternate - ALTERNATE_PROTOCOL_MINIMUM_VALID_VERSION] =
180          true;
181    }
182  }
183
184  if (HpackHuffmanAggregator::UseAggregator()) {
185    huffman_aggregator_.reset(new HpackHuffmanAggregator());
186  }
187
188  http_server_properties_->SetAlternateProtocolProbabilityThreshold(
189      params.alternate_protocol_probability_threshold);
190}
191
192HttpNetworkSession::~HttpNetworkSession() {
193  STLDeleteElements(&response_drainers_);
194  spdy_session_pool_.CloseAllSessions();
195}
196
197void HttpNetworkSession::AddResponseDrainer(HttpResponseBodyDrainer* drainer) {
198  DCHECK(!ContainsKey(response_drainers_, drainer));
199  response_drainers_.insert(drainer);
200}
201
202void HttpNetworkSession::RemoveResponseDrainer(
203    HttpResponseBodyDrainer* drainer) {
204  DCHECK(ContainsKey(response_drainers_, drainer));
205  response_drainers_.erase(drainer);
206}
207
208TransportClientSocketPool* HttpNetworkSession::GetTransportSocketPool(
209    SocketPoolType pool_type) {
210  return GetSocketPoolManager(pool_type)->GetTransportSocketPool();
211}
212
213SSLClientSocketPool* HttpNetworkSession::GetSSLSocketPool(
214    SocketPoolType pool_type) {
215  return GetSocketPoolManager(pool_type)->GetSSLSocketPool();
216}
217
218SOCKSClientSocketPool* HttpNetworkSession::GetSocketPoolForSOCKSProxy(
219    SocketPoolType pool_type,
220    const HostPortPair& socks_proxy) {
221  return GetSocketPoolManager(pool_type)->GetSocketPoolForSOCKSProxy(
222      socks_proxy);
223}
224
225HttpProxyClientSocketPool* HttpNetworkSession::GetSocketPoolForHTTPProxy(
226    SocketPoolType pool_type,
227    const HostPortPair& http_proxy) {
228  return GetSocketPoolManager(pool_type)->GetSocketPoolForHTTPProxy(http_proxy);
229}
230
231SSLClientSocketPool* HttpNetworkSession::GetSocketPoolForSSLWithProxy(
232    SocketPoolType pool_type,
233    const HostPortPair& proxy_server) {
234  return GetSocketPoolManager(pool_type)->GetSocketPoolForSSLWithProxy(
235      proxy_server);
236}
237
238base::Value* HttpNetworkSession::SocketPoolInfoToValue() const {
239  // TODO(yutak): Should merge values from normal pools and WebSocket pools.
240  return normal_socket_pool_manager_->SocketPoolInfoToValue();
241}
242
243base::Value* HttpNetworkSession::SpdySessionPoolInfoToValue() const {
244  return spdy_session_pool_.SpdySessionPoolInfoToValue();
245}
246
247base::Value* HttpNetworkSession::QuicInfoToValue() const {
248  base::DictionaryValue* dict = new base::DictionaryValue();
249  dict->Set("sessions", quic_stream_factory_.QuicStreamFactoryInfoToValue());
250  dict->SetBoolean("quic_enabled", params_.enable_quic);
251  dict->SetBoolean("enable_quic_port_selection",
252                   params_.enable_quic_port_selection);
253  dict->SetBoolean("enable_quic_pacing",
254                   ContainsQuicTag(params_.quic_connection_options, kPACE));
255  dict->SetBoolean("enable_quic_time_based_loss_detection",
256                   params_.enable_quic_time_based_loss_detection);
257  dict->SetString("origin_to_force_quic_on",
258                  params_.origin_to_force_quic_on.ToString());
259  return dict;
260}
261
262void HttpNetworkSession::CloseAllConnections() {
263  normal_socket_pool_manager_->FlushSocketPoolsWithError(ERR_ABORTED);
264  websocket_socket_pool_manager_->FlushSocketPoolsWithError(ERR_ABORTED);
265  spdy_session_pool_.CloseCurrentSessions(ERR_ABORTED);
266  quic_stream_factory_.CloseAllSessions(ERR_ABORTED);
267}
268
269void HttpNetworkSession::CloseIdleConnections() {
270  normal_socket_pool_manager_->CloseIdleSockets();
271  websocket_socket_pool_manager_->CloseIdleSockets();
272  spdy_session_pool_.CloseCurrentIdleSessions();
273}
274
275bool HttpNetworkSession::IsProtocolEnabled(AlternateProtocol protocol) const {
276  DCHECK(IsAlternateProtocolValid(protocol));
277  return enabled_protocols_[
278      protocol - ALTERNATE_PROTOCOL_MINIMUM_VALID_VERSION];
279}
280
281void HttpNetworkSession::GetNextProtos(
282    std::vector<std::string>* next_protos) const {
283  if (HttpStreamFactory::spdy_enabled()) {
284    *next_protos = next_protos_;
285  } else {
286    next_protos->clear();
287  }
288}
289
290bool HttpNetworkSession::HasSpdyExclusion(
291    HostPortPair host_port_pair) const {
292  return params_.forced_spdy_exclusions.find(host_port_pair) !=
293      params_.forced_spdy_exclusions.end();
294}
295
296ClientSocketPoolManager* HttpNetworkSession::GetSocketPoolManager(
297    SocketPoolType pool_type) {
298  switch (pool_type) {
299    case NORMAL_SOCKET_POOL:
300      return normal_socket_pool_manager_.get();
301    case WEBSOCKET_SOCKET_POOL:
302      return websocket_socket_pool_manager_.get();
303    default:
304      NOTREACHED();
305      break;
306  }
307  return NULL;
308}
309
310}  //  namespace net
311