15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "jingle/glue/proxy_resolving_client_socket.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/basictypes.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h"
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind_helpers.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/compiler_specific.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/base/io_buffer.h"
13116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch#include "net/base/load_flags.h"
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/base/net_errors.h"
151320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "net/http/http_auth_controller.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/http/http_network_session.h"
171320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci#include "net/http/proxy_client_socket.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/socket/client_socket_handle.h"
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/socket/client_socket_pool_manager.h"
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/url_request/url_request_context.h"
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "net/url_request/url_request_context_getter.h"
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace jingle_glue {
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProxyResolvingClientSocket::ProxyResolvingClientSocket(
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    net::ClientSocketFactory* socket_factory,
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const scoped_refptr<net::URLRequestContextGetter>& request_context_getter,
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const net::SSLConfig& ssl_config,
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const net::HostPortPair& dest_host_port_pair)
30c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        : proxy_resolve_callback_(
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              base::Bind(&ProxyResolvingClientSocket::ProcessProxyResolveDone,
32c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                         base::Unretained(this))),
33c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)          connect_callback_(
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              base::Bind(&ProxyResolvingClientSocket::ProcessConnectDone,
35c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                         base::Unretained(this))),
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          ssl_config_(ssl_config),
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          pac_request_(NULL),
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          dest_host_port_pair_(dest_host_port_pair),
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          // Assume that we intend to do TLS on this socket; all
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          // current use cases do.
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          proxy_url_("https://" + dest_host_port_pair_.ToString()),
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          tried_direct_connect_fallback_(false),
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          bound_net_log_(
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)              net::BoundNetLog::Make(
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                  request_context_getter->GetURLRequestContext()->net_log(),
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                  net::NetLog::SOURCE_SOCKET)),
47c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)          weak_factory_(this) {
48868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  DCHECK(request_context_getter.get());
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  net::URLRequestContext* request_context =
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      request_context_getter->GetURLRequestContext();
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(request_context);
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!dest_host_port_pair_.host().empty());
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_GT(dest_host_port_pair_.port(), 0);
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(proxy_url_.is_valid());
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  net::HttpNetworkSession::Params session_params;
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.client_socket_factory = socket_factory;
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.host_resolver = request_context->host_resolver();
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.cert_verifier = request_context->cert_verifier();
607d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  session_params.transport_security_state =
617d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)      request_context->transport_security_state();
625f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  // TODO(rkn): This is NULL because ChannelIDService is not thread safe.
635f1c94371a64b3196d4be9466099bb892df9b88eTorne (Richard Coles)  session_params.channel_id_service = NULL;
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.proxy_service = request_context->proxy_service();
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.ssl_config_service = request_context->ssl_config_service();
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.http_auth_handler_factory =
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      request_context->http_auth_handler_factory();
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.network_delegate = request_context->network_delegate();
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.http_server_properties =
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      request_context->http_server_properties();
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  session_params.net_log = request_context->net_log();
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const net::HttpNetworkSession::Params* reference_params =
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      request_context->GetNetworkSessionParams();
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (reference_params) {
76cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    // TODO(mmenke):  Just copying specific parameters seems highly regression
77cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    // prone.  Should have a better way to do this.
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    session_params.host_mapping_rules = reference_params->host_mapping_rules;
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    session_params.ignore_certificate_errors =
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        reference_params->ignore_certificate_errors;
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    session_params.testing_fixed_http_port =
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        reference_params->testing_fixed_http_port;
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    session_params.testing_fixed_https_port =
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        reference_params->testing_fixed_https_port;
85cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    session_params.next_protos = reference_params->next_protos;
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    session_params.trusted_spdy_proxy = reference_params->trusted_spdy_proxy;
87cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    session_params.force_spdy_over_ssl = reference_params->force_spdy_over_ssl;
88cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    session_params.force_spdy_always = reference_params->force_spdy_always;
89cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    session_params.forced_spdy_exclusions =
90cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        reference_params->forced_spdy_exclusions;
91cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    session_params.use_alternate_protocols =
92cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)        reference_params->use_alternate_protocols;
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  network_session_ = new net::HttpNetworkSession(session_params);
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)ProxyResolvingClientSocket::~ProxyResolvingClientSocket() {
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Disconnect();
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::Read(net::IOBuffer* buf, int buf_len,
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                     const net::CompletionCallback& callback) {
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->Read(buf, buf_len, callback);
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::ERR_SOCKET_NOT_CONNECTED;
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::Write(
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    net::IOBuffer* buf,
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    int buf_len,
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const net::CompletionCallback& callback) {
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->Write(buf, buf_len, callback);
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::ERR_SOCKET_NOT_CONNECTED;
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
120c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdochint ProxyResolvingClientSocket::SetReceiveBufferSize(int32 size) {
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->SetReceiveBufferSize(size);
1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
124c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch  return net::ERR_SOCKET_NOT_CONNECTED;
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
127c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdochint ProxyResolvingClientSocket::SetSendBufferSize(int32 size) {
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->SetSendBufferSize(size);
1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
131c5cede9ae108bb15f6b7a8aea21c7e1fefa2834cBen Murdoch  return net::ERR_SOCKET_NOT_CONNECTED;
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::Connect(
1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const net::CompletionCallback& callback) {
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(user_connect_callback_.is_null());
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  tried_direct_connect_fallback_ = false;
1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // First we try and resolve the proxy.
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int status = network_session_->proxy_service()->ResolveProxy(
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      proxy_url_,
143116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      net::LOAD_NORMAL,
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      &proxy_info_,
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      proxy_resolve_callback_,
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      &pac_request_,
147116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      NULL,
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      bound_net_log_);
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (status != net::ERR_IO_PENDING) {
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // We defer execution of ProcessProxyResolveDone instead of calling it
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // directly here for simplicity. From the caller's point of view,
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // the connect always happens asynchronously.
153c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    base::MessageLoop* message_loop = base::MessageLoop::current();
1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    CHECK(message_loop);
1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    message_loop->PostTask(
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        FROM_HERE,
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        base::Bind(&ProxyResolvingClientSocket::ProcessProxyResolveDone,
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                   weak_factory_.GetWeakPtr(), status));
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  user_connect_callback_ = callback;
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::ERR_IO_PENDING;
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::RunUserConnectCallback(int status) {
1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_LE(status, net::OK);
1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  net::CompletionCallback user_connect_callback = user_connect_callback_;
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  user_connect_callback_.Reset();
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  user_connect_callback.Run(status);
1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Always runs asynchronously.
1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::ProcessProxyResolveDone(int status) {
1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  pac_request_ = NULL;
1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_NE(status, net::ERR_IO_PENDING);
1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (status == net::OK) {
1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Remove unsupported proxies from the list.
1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    proxy_info_.RemoveProxiesWithoutScheme(
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        net::ProxyServer::SCHEME_DIRECT |
1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        net::ProxyServer::SCHEME_HTTP | net::ProxyServer::SCHEME_HTTPS |
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        net::ProxyServer::SCHEME_SOCKS4 | net::ProxyServer::SCHEME_SOCKS5);
1825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (proxy_info_.is_empty()) {
1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // No proxies/direct to choose from. This happens when we don't support
1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // any of the proxies in the returned list.
1865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      status = net::ERR_NO_SUPPORTED_PROXIES;
1875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
1885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Since we are faking the URL, it is possible that no proxies match our URL.
1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Try falling back to a direct connection if we have not tried that before.
1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (status != net::OK) {
1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!tried_direct_connect_fallback_) {
1945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      tried_direct_connect_fallback_ = true;
1955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      proxy_info_.UseDirect();
1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    } else {
1975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      CloseTransportSocket();
1985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      RunUserConnectCallback(status);
1995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return;
2005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
2015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  transport_.reset(new net::ClientSocketHandle);
2045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Now that we have resolved the proxy, we need to connect.
2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  status = net::InitSocketHandleForRawConnect(
2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      dest_host_port_pair_, network_session_.get(), proxy_info_, ssl_config_,
207e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      ssl_config_, net::PRIVACY_MODE_DISABLED, bound_net_log_, transport_.get(),
20890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      connect_callback_);
2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (status != net::ERR_IO_PENDING) {
2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Since this method is always called asynchronously. it is OK to call
2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // ProcessConnectDone synchronously.
2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ProcessConnectDone(status);
2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::ProcessConnectDone(int status) {
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (status != net::OK) {
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If the connection fails, try another proxy.
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    status = ReconsiderProxyAfterError(status);
2205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // ReconsiderProxyAfterError either returns an error (in which case it is
2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // not reconsidering a proxy) or returns ERR_IO_PENDING if it is considering
2225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // another proxy.
2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    DCHECK_NE(status, net::OK);
2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (status == net::ERR_IO_PENDING)
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // Proxy reconsideration pending. Return.
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return;
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    CloseTransportSocket();
2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
2295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ReportSuccessfulProxyConnection();
2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  RunUserConnectCallback(status);
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// TODO(sanjeevr): This has largely been copied from
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// HttpStreamFactoryImpl::Job::ReconsiderProxyAfterError. This should be
2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// refactored into some common place.
2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// This method reconsiders the proxy on certain errors. If it does reconsider
2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// a proxy it always returns ERR_IO_PENDING and posts a call to
2395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// ProcessProxyResolveDone with the result of the reconsideration.
2405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::ReconsiderProxyAfterError(int error) {
2415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!pac_request_);
2425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_NE(error, net::OK);
2435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_NE(error, net::ERR_IO_PENDING);
2445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // A failure to resolve the hostname or any error related to establishing a
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TCP connection could be grounds for trying a new proxy configuration.
2465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Why do this when a hostname cannot be resolved?  Some URLs only make sense
2485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // to proxy servers.  The hostname in those URLs might fail to resolve if we
2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // are still using a non-proxy config.  We need to check if a proxy config
2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // now exists that corresponds to a proxy server that could load the URL.
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
2525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  switch (error) {
2535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_PROXY_CONNECTION_FAILED:
2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_NAME_NOT_RESOLVED:
2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_INTERNET_DISCONNECTED:
2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_ADDRESS_UNREACHABLE:
2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_CONNECTION_CLOSED:
2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_CONNECTION_RESET:
2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_CONNECTION_REFUSED:
2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_CONNECTION_ABORTED:
2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_TIMED_OUT:
2625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_TUNNEL_CONNECTION_FAILED:
2635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_SOCKS_CONNECTION_FAILED:
2645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      break;
2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    case net::ERR_SOCKS_CONNECTION_HOST_UNREACHABLE:
2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // Remap the SOCKS-specific "host unreachable" error to a more
2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // generic error code (this way consumers like the link doctor
2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // know to substitute their error page).
2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      //
2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // Note that if the host resolving was done by the SOCSK5 proxy, we can't
2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // differentiate between a proxy-side "host not found" versus a proxy-side
2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // "address unreachable" error, and will report both of these failures as
2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // ERR_ADDRESS_UNREACHABLE.
2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return net::ERR_ADDRESS_UNREACHABLE;
2751320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    case net::ERR_PROXY_AUTH_REQUESTED: {
2761320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      net::ProxyClientSocket* proxy_socket =
2771320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci          static_cast<net::ProxyClientSocket*>(transport_->socket());
2781320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci
2791320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      if (proxy_socket->GetAuthController()->HaveAuth())
2801320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci        return proxy_socket->RestartWithAuth(connect_callback_);
2811320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci
2821320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      return error;
2831320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    }
2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    default:
2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return error;
2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (proxy_info_.is_https() && ssl_config_.send_client_cert) {
2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    network_session_->ssl_client_auth_cache()->Remove(
2905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        proxy_info_.proxy_server().host_port_pair());
2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int rv = network_session_->proxy_service()->ReconsiderProxyAfterError(
294116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      proxy_url_, net::LOAD_NORMAL, error, &proxy_info_,
295116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch      proxy_resolve_callback_, &pac_request_, NULL, bound_net_log_);
2965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (rv == net::OK || rv == net::ERR_IO_PENDING) {
2975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    CloseTransportSocket();
2985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
2995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If ReconsiderProxyAfterError() failed synchronously, it means
3005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // there was nothing left to fall-back to, so fail the transaction
3015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // with the last connection error we got.
3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    rv = error;
3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We either have new proxy info or there was an error in falling back.
3065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // In both cases we want to post ProcessProxyResolveDone (in the error case
3075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // we might still want to fall back a direct connection).
3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (rv != net::ERR_IO_PENDING) {
309c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    base::MessageLoop* message_loop = base::MessageLoop::current();
3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    CHECK(message_loop);
3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    message_loop->PostTask(
3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        FROM_HERE,
3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        base::Bind(&ProxyResolvingClientSocket::ProcessProxyResolveDone,
3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                   weak_factory_.GetWeakPtr(), rv));
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Since we potentially have another try to go (trying the direct connect)
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // set the return code code to ERR_IO_PENDING.
3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    rv = net::ERR_IO_PENDING;
3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return rv;
3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::ReportSuccessfulProxyConnection() {
32303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  network_session_->proxy_service()->ReportSuccess(proxy_info_, NULL);
3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::Disconnect() {
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  CloseTransportSocket();
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (pac_request_) {
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    network_session_->proxy_service()->CancelPacRequest(pac_request_);
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    pac_request_ = NULL;
3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  user_connect_callback_.Reset();
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::IsConnected() const {
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!transport_.get() || !transport_->socket())
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return transport_->socket()->IsConnected();
3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::IsConnectedAndIdle() const {
3425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!transport_.get() || !transport_->socket())
3435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
3445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return transport_->socket()->IsConnectedAndIdle();
3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::GetPeerAddress(
3485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    net::IPEndPoint* address) const {
3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->GetPeerAddress(address);
3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::ERR_SOCKET_NOT_CONNECTED;
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int ProxyResolvingClientSocket::GetLocalAddress(
3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    net::IPEndPoint* address) const {
3575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->GetLocalAddress(address);
3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::ERR_SOCKET_NOT_CONNECTED;
3615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const net::BoundNetLog& ProxyResolvingClientSocket::NetLog() const {
3645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->NetLog();
3665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
3675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return bound_net_log_;
3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::SetSubresourceSpeculation() {
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    transport_->socket()->SetSubresourceSpeculation();
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  else
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NOTREACHED();
3755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::SetOmniboxSpeculation() {
3785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    transport_->socket()->SetOmniboxSpeculation();
3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  else
3815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NOTREACHED();
3825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::WasEverUsed() const {
3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->WasEverUsed();
3875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
3885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::UsingTCPFastOpen() const {
3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->UsingTCPFastOpen();
3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
3965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::WasNpnNegotiated() const {
3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)net::NextProto ProxyResolvingClientSocket::GetNegotiatedProtocol() const {
4035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return transport_->socket()->GetNegotiatedProtocol();
4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NOTREACHED();
4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return net::kProtoUnknown;
4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool ProxyResolvingClientSocket::GetSSLInfo(net::SSLInfo* ssl_info) {
4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return false;
4115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void ProxyResolvingClientSocket::CloseTransportSocket() {
4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (transport_.get() && transport_->socket())
4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    transport_->socket()->Disconnect();
4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  transport_.reset();
4175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace jingle_glue
420