mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 14:26:09 +03:00
1522 lines
55 KiB
C++
1522 lines
55 KiB
C++
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "net/http/http_stream_factory_job.h"
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
|
|
#include "base/bind.h"
|
|
#include "base/bind_helpers.h"
|
|
#include "base/feature_list.h"
|
|
#include "base/location.h"
|
|
#include "base/logging.h"
|
|
#include "base/metrics/histogram_macros.h"
|
|
#include "base/metrics/sparse_histogram.h"
|
|
#include "base/single_thread_task_runner.h"
|
|
#include "base/stl_util.h"
|
|
#include "base/strings/string_number_conversions.h"
|
|
#include "base/strings/string_util.h"
|
|
#include "base/threading/thread_task_runner_handle.h"
|
|
#include "base/trace_event/trace_event.h"
|
|
#include "base/values.h"
|
|
#include "build/build_config.h"
|
|
#include "net/base/port_util.h"
|
|
#include "net/base/proxy_delegate.h"
|
|
#include "net/base/trace_constants.h"
|
|
#include "net/cert/cert_verifier.h"
|
|
#include "net/http/bidirectional_stream_impl.h"
|
|
#include "net/http/http_basic_stream.h"
|
|
#include "net/http/http_network_session.h"
|
|
#include "net/http/http_proxy_client_socket.h"
|
|
#include "net/http/http_proxy_client_socket_pool.h"
|
|
#include "net/http/http_request_info.h"
|
|
#include "net/http/http_server_properties.h"
|
|
#include "net/http/http_stream_factory.h"
|
|
#include "net/http/proxy_fallback.h"
|
|
#include "net/log/net_log.h"
|
|
#include "net/log/net_log_capture_mode.h"
|
|
#include "net/log/net_log_event_type.h"
|
|
#include "net/log/net_log_source.h"
|
|
#include "net/log/net_log_source_type.h"
|
|
#include "net/quic/bidirectional_stream_quic_impl.h"
|
|
#include "net/quic/quic_http_stream.h"
|
|
#include "net/socket/client_socket_handle.h"
|
|
#include "net/socket/client_socket_pool.h"
|
|
#include "net/socket/client_socket_pool_manager.h"
|
|
#include "net/socket/socks_client_socket_pool.h"
|
|
#include "net/socket/ssl_client_socket.h"
|
|
#include "net/socket/ssl_client_socket_pool.h"
|
|
#include "net/socket/stream_socket.h"
|
|
#include "net/spdy/bidirectional_stream_spdy_impl.h"
|
|
#include "net/spdy/http2_push_promise_index.h"
|
|
#include "net/spdy/spdy_http_stream.h"
|
|
#include "net/spdy/spdy_session.h"
|
|
#include "net/spdy/spdy_session_pool.h"
|
|
#include "net/ssl/channel_id_service.h"
|
|
#include "net/ssl/ssl_cert_request_info.h"
|
|
#include "net/third_party/spdy/core/spdy_protocol.h"
|
|
#include "url/url_constants.h"
|
|
|
|
namespace net {
|
|
|
|
namespace {
|
|
|
|
// Experiment to preconnect only one connection if HttpServerProperties is
|
|
// not supported or initialized.
|
|
const base::Feature kLimitEarlyPreconnectsExperiment{
|
|
"LimitEarlyPreconnects", base::FEATURE_ENABLED_BY_DEFAULT};
|
|
void RecordChannelIDKeyMatch(StreamSocket* socket,
|
|
ChannelIDService* channel_id_service,
|
|
std::string host) {
|
|
SSLInfo ssl_info;
|
|
socket->GetSSLInfo(&ssl_info);
|
|
if (!ssl_info.channel_id_sent)
|
|
return;
|
|
std::unique_ptr<crypto::ECPrivateKey> request_key;
|
|
ChannelIDService::Request request;
|
|
int result = channel_id_service->GetOrCreateChannelID(
|
|
host, &request_key, base::DoNothing(), &request);
|
|
// GetOrCreateChannelID only returns ERR_IO_PENDING before its first call
|
|
// (over the lifetime of the ChannelIDService) has completed or if it is
|
|
// creating a new key. The key that is being looked up here should already
|
|
// have been looked up before the channel ID was sent on the ssl socket, so
|
|
// the expectation is that this call will return synchronously. If this does
|
|
// return ERR_IO_PENDING, treat that as any other lookup failure and cancel
|
|
// the async request.
|
|
if (result == ERR_IO_PENDING)
|
|
request.Cancel();
|
|
crypto::ECPrivateKey* socket_key = socket->GetChannelIDKey();
|
|
|
|
// This enum is used for an UMA histogram - do not change or re-use values.
|
|
enum {
|
|
NO_KEYS = 0,
|
|
MATCH = 1,
|
|
SOCKET_KEY_MISSING = 2,
|
|
REQUEST_KEY_MISSING = 3,
|
|
KEYS_DIFFER = 4,
|
|
KEY_LOOKUP_ERROR = 5,
|
|
KEY_MATCH_MAX
|
|
} match;
|
|
if (result != OK) {
|
|
match = KEY_LOOKUP_ERROR;
|
|
} else if (!socket_key && !request_key) {
|
|
match = NO_KEYS;
|
|
} else if (!socket_key) {
|
|
match = SOCKET_KEY_MISSING;
|
|
} else if (!request_key) {
|
|
match = REQUEST_KEY_MISSING;
|
|
} else {
|
|
match = KEYS_DIFFER;
|
|
std::string raw_socket_key, raw_request_key;
|
|
if (socket_key->ExportRawPublicKey(&raw_socket_key) &&
|
|
request_key->ExportRawPublicKey(&raw_request_key) &&
|
|
raw_socket_key == raw_request_key) {
|
|
match = MATCH;
|
|
}
|
|
}
|
|
UMA_HISTOGRAM_ENUMERATION("Net.TokenBinding.KeyMatch", match, KEY_MATCH_MAX);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// Returns parameters associated with the start of a HTTP stream job.
|
|
std::unique_ptr<base::Value> NetLogHttpStreamJobCallback(
|
|
const NetLogSource& source,
|
|
const GURL* original_url,
|
|
const GURL* url,
|
|
bool expect_spdy,
|
|
bool using_quic,
|
|
RequestPriority priority,
|
|
NetLogCaptureMode /* capture_mode */) {
|
|
auto dict = std::make_unique<base::DictionaryValue>();
|
|
if (source.IsValid())
|
|
source.AddToEventParameters(dict.get());
|
|
dict->SetString("original_url", original_url->GetOrigin().spec());
|
|
dict->SetString("url", url->GetOrigin().spec());
|
|
dict->SetString("expect_spdy", expect_spdy ? "true" : "false");
|
|
dict->SetString("using_quic", using_quic ? "true" : "false");
|
|
dict->SetString("priority", RequestPriorityToString(priority));
|
|
return std::move(dict);
|
|
}
|
|
|
|
// Returns parameters associated with the Proto (with NPN negotiation) of a HTTP
|
|
// stream.
|
|
std::unique_ptr<base::Value> NetLogHttpStreamProtoCallback(
|
|
NextProto negotiated_protocol,
|
|
NetLogCaptureMode /* capture_mode */) {
|
|
auto dict = std::make_unique<base::DictionaryValue>();
|
|
|
|
dict->SetString("proto", NextProtoToString(negotiated_protocol));
|
|
return std::move(dict);
|
|
}
|
|
|
|
HttpStreamFactory::Job::Job(Delegate* delegate,
|
|
JobType job_type,
|
|
HttpNetworkSession* session,
|
|
const HttpRequestInfo& request_info,
|
|
RequestPriority priority,
|
|
const ProxyInfo& proxy_info,
|
|
const SSLConfig& server_ssl_config,
|
|
const SSLConfig& proxy_ssl_config,
|
|
HostPortPair destination,
|
|
GURL origin_url,
|
|
NextProto alternative_protocol,
|
|
quic::QuicTransportVersion quic_version,
|
|
const ProxyServer& alternative_proxy_server,
|
|
bool is_websocket,
|
|
bool enable_ip_based_pooling,
|
|
NetLog* net_log)
|
|
: request_info_(request_info),
|
|
priority_(priority),
|
|
proxy_info_(proxy_info),
|
|
server_ssl_config_(server_ssl_config),
|
|
proxy_ssl_config_(proxy_ssl_config),
|
|
net_log_(
|
|
NetLogWithSource::Make(net_log, NetLogSourceType::HTTP_STREAM_JOB)),
|
|
io_callback_(
|
|
base::BindRepeating(&Job::OnIOComplete, base::Unretained(this))),
|
|
connection_(new ClientSocketHandle),
|
|
session_(session),
|
|
next_state_(STATE_NONE),
|
|
destination_(destination),
|
|
origin_url_(origin_url),
|
|
alternative_proxy_server_(alternative_proxy_server),
|
|
is_websocket_(is_websocket),
|
|
try_websocket_over_http2_(is_websocket_ &&
|
|
origin_url_.SchemeIs(url::kWssScheme) &&
|
|
proxy_info_.is_direct() &&
|
|
session_->params().enable_websocket_over_http2),
|
|
enable_ip_based_pooling_(enable_ip_based_pooling),
|
|
delegate_(delegate),
|
|
job_type_(job_type),
|
|
using_ssl_(origin_url_.SchemeIs(url::kHttpsScheme) ||
|
|
origin_url_.SchemeIs(url::kWssScheme)),
|
|
using_quic_(
|
|
alternative_protocol == kProtoQUIC ||
|
|
(ShouldForceQuic(session, destination, origin_url, proxy_info) &&
|
|
!(proxy_info.is_quic() && using_ssl_))),
|
|
quic_version_(quic_version),
|
|
expect_spdy_(alternative_protocol == kProtoHTTP2 && !using_quic_),
|
|
using_spdy_(false),
|
|
should_reconsider_proxy_(false),
|
|
quic_request_(session_->quic_stream_factory()),
|
|
expect_on_quic_host_resolution_(false),
|
|
using_existing_quic_session_(false),
|
|
establishing_tunnel_(false),
|
|
was_alpn_negotiated_(false),
|
|
negotiated_protocol_(kProtoUnknown),
|
|
num_streams_(0),
|
|
pushed_stream_id_(kNoPushedStreamFound),
|
|
spdy_session_direct_(
|
|
!(proxy_info.is_https() && origin_url_.SchemeIs(url::kHttpScheme))),
|
|
spdy_session_key_(using_quic_
|
|
? SpdySessionKey()
|
|
: GetSpdySessionKey(spdy_session_direct_,
|
|
proxy_info_.proxy_server(),
|
|
origin_url_,
|
|
request_info_.privacy_mode,
|
|
request_info_.socket_tag)),
|
|
stream_type_(HttpStreamRequest::BIDIRECTIONAL_STREAM),
|
|
init_connection_already_resumed_(false),
|
|
ptr_factory_(this) {
|
|
// The Job is forced to use QUIC without a designated version, try the
|
|
// preferred QUIC version that is supported by default.
|
|
if (quic_version_ == quic::QUIC_VERSION_UNSUPPORTED &&
|
|
ShouldForceQuic(session, destination, origin_url, proxy_info)) {
|
|
quic_version_ = session->params().quic_supported_versions[0];
|
|
}
|
|
|
|
if (using_quic_)
|
|
DCHECK_NE(quic_version_, quic::QUIC_VERSION_UNSUPPORTED);
|
|
|
|
DCHECK(session);
|
|
if (alternative_protocol != kProtoUnknown) {
|
|
// The job cannot have protocol requirements dictated by alternative service
|
|
// and have an alternative proxy server set at the same time, since
|
|
// alternative services are used for requests that are fetched directly,
|
|
// while the alternative proxy server is used for requests that should be
|
|
// fetched using proxy.
|
|
DCHECK(!alternative_proxy_server_.is_valid());
|
|
// If the alternative service protocol is specified, then the job type must
|
|
// be either ALTERNATIVE or PRECONNECT.
|
|
DCHECK(job_type_ == ALTERNATIVE || job_type_ == PRECONNECT);
|
|
}
|
|
// If the alternative proxy server is set, then the job must be ALTERNATIVE.
|
|
if (alternative_proxy_server_.is_valid()) {
|
|
DCHECK(job_type_ == ALTERNATIVE);
|
|
}
|
|
|
|
if (expect_spdy_) {
|
|
DCHECK(origin_url_.SchemeIs(url::kHttpsScheme));
|
|
}
|
|
if (using_quic_) {
|
|
DCHECK(session_->IsQuicEnabled());
|
|
}
|
|
if (job_type_ == PRECONNECT || is_websocket_) {
|
|
DCHECK(request_info_.socket_tag == SocketTag());
|
|
}
|
|
if (is_websocket_) {
|
|
DCHECK(origin_url_.SchemeIsWSOrWSS());
|
|
} else {
|
|
DCHECK(!origin_url_.SchemeIsWSOrWSS());
|
|
}
|
|
}
|
|
|
|
HttpStreamFactory::Job::~Job() {
|
|
net_log_.EndEvent(NetLogEventType::HTTP_STREAM_JOB);
|
|
|
|
// When we're in a partially constructed state, waiting for the user to
|
|
// provide certificate handling information or authentication, we can't reuse
|
|
// this stream at all.
|
|
if (next_state_ == STATE_WAITING_USER_ACTION) {
|
|
connection_->socket()->Disconnect();
|
|
connection_.reset();
|
|
}
|
|
|
|
// The stream could be in a partial state. It is not reusable.
|
|
if (stream_.get() && next_state_ != STATE_DONE)
|
|
stream_->Close(true /* not reusable */);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::Start(HttpStreamRequest::StreamType stream_type) {
|
|
stream_type_ = stream_type;
|
|
StartInternal();
|
|
}
|
|
|
|
int HttpStreamFactory::Job::Preconnect(int num_streams) {
|
|
DCHECK_GT(num_streams, 0);
|
|
HttpServerProperties* http_server_properties =
|
|
session_->http_server_properties();
|
|
DCHECK(http_server_properties);
|
|
// Preconnect one connection if either of the following is true:
|
|
// (1) kLimitEarlyPreconnectsStreamExperiment is turned on,
|
|
// HttpServerProperties is not initialized, and url scheme is cryptographic.
|
|
// (2) The server supports H2 or QUIC.
|
|
bool connect_one_stream =
|
|
base::FeatureList::IsEnabled(kLimitEarlyPreconnectsExperiment) &&
|
|
!http_server_properties->IsInitialized() &&
|
|
request_info_.url.SchemeIsCryptographic();
|
|
if (connect_one_stream || http_server_properties->SupportsRequestPriority(
|
|
url::SchemeHostPort(request_info_.url))) {
|
|
num_streams_ = 1;
|
|
} else {
|
|
num_streams_ = num_streams;
|
|
}
|
|
return StartInternal();
|
|
}
|
|
|
|
int HttpStreamFactory::Job::RestartTunnelWithProxyAuth() {
|
|
DCHECK(establishing_tunnel_);
|
|
next_state_ = STATE_RESTART_TUNNEL_AUTH;
|
|
stream_.reset();
|
|
RunLoop(OK);
|
|
return ERR_IO_PENDING;
|
|
}
|
|
|
|
LoadState HttpStreamFactory::Job::GetLoadState() const {
|
|
switch (next_state_) {
|
|
case STATE_INIT_CONNECTION_COMPLETE:
|
|
case STATE_CREATE_STREAM_COMPLETE:
|
|
return using_quic_ ? LOAD_STATE_CONNECTING : connection_->GetLoadState();
|
|
default:
|
|
return LOAD_STATE_IDLE;
|
|
}
|
|
}
|
|
|
|
void HttpStreamFactory::Job::Resume() {
|
|
DCHECK_EQ(job_type_, MAIN);
|
|
DCHECK_EQ(next_state_, STATE_WAIT_COMPLETE);
|
|
OnIOComplete(OK);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::Orphan() {
|
|
DCHECK_EQ(job_type_, ALTERNATIVE);
|
|
net_log_.AddEvent(NetLogEventType::HTTP_STREAM_JOB_ORPHANED);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::SetPriority(RequestPriority priority) {
|
|
priority_ = priority;
|
|
// Ownership of |connection_| is passed to the newly created stream
|
|
// or H2 session in DoCreateStream(), and the consumer is not
|
|
// notified immediately, so this call may occur when |connection_|
|
|
// is null.
|
|
//
|
|
// Note that streams are created without a priority associated with them,
|
|
// and it is up to the consumer to set their priority via
|
|
// HttpStream::InitializeStream(). So there is no need for this code
|
|
// to propagate priority changes to the newly created stream.
|
|
if (connection_ && connection_->is_initialized())
|
|
connection_->SetPriority(priority);
|
|
// TODO(akalin): Maybe Propagate this to the preconnect state.
|
|
}
|
|
|
|
bool HttpStreamFactory::Job::was_alpn_negotiated() const {
|
|
return was_alpn_negotiated_;
|
|
}
|
|
|
|
NextProto HttpStreamFactory::Job::negotiated_protocol() const {
|
|
return negotiated_protocol_;
|
|
}
|
|
|
|
bool HttpStreamFactory::Job::using_spdy() const {
|
|
return using_spdy_;
|
|
}
|
|
|
|
size_t HttpStreamFactory::Job::EstimateMemoryUsage() const {
|
|
StreamSocket::SocketMemoryStats stats;
|
|
if (connection_)
|
|
connection_->DumpMemoryStats(&stats);
|
|
return stats.total_size;
|
|
}
|
|
|
|
const SSLConfig& HttpStreamFactory::Job::server_ssl_config() const {
|
|
return server_ssl_config_;
|
|
}
|
|
|
|
const SSLConfig& HttpStreamFactory::Job::proxy_ssl_config() const {
|
|
return proxy_ssl_config_;
|
|
}
|
|
|
|
const ProxyInfo& HttpStreamFactory::Job::proxy_info() const {
|
|
return proxy_info_;
|
|
}
|
|
|
|
void HttpStreamFactory::Job::GetSSLInfo(SSLInfo* ssl_info) {
|
|
DCHECK(using_ssl_);
|
|
DCHECK(!establishing_tunnel_);
|
|
DCHECK(connection_.get() && connection_->socket());
|
|
connection_->socket()->GetSSLInfo(ssl_info);
|
|
}
|
|
|
|
// static
|
|
bool HttpStreamFactory::Job::ShouldForceQuic(HttpNetworkSession* session,
|
|
const HostPortPair& destination,
|
|
const GURL& origin_url,
|
|
const ProxyInfo& proxy_info) {
|
|
if (!session->IsQuicEnabled())
|
|
return false;
|
|
if (proxy_info.is_quic())
|
|
return true;
|
|
return (base::ContainsKey(session->params().origins_to_force_quic_on,
|
|
HostPortPair()) ||
|
|
base::ContainsKey(session->params().origins_to_force_quic_on,
|
|
destination)) &&
|
|
proxy_info.is_direct() && origin_url.SchemeIs(url::kHttpsScheme);
|
|
}
|
|
|
|
// static
|
|
SpdySessionKey HttpStreamFactory::Job::GetSpdySessionKey(
|
|
bool spdy_session_direct,
|
|
const ProxyServer& proxy_server,
|
|
const GURL& origin_url,
|
|
PrivacyMode privacy_mode,
|
|
const SocketTag& socket_tag) {
|
|
// In the case that we're using an HTTPS proxy for an HTTP url,
|
|
// we look for a SPDY session *to* the proxy, instead of to the
|
|
// origin server.
|
|
if (!spdy_session_direct) {
|
|
return SpdySessionKey(proxy_server.host_port_pair(), ProxyServer::Direct(),
|
|
PRIVACY_MODE_DISABLED, socket_tag);
|
|
}
|
|
return SpdySessionKey(HostPortPair::FromURL(origin_url), proxy_server,
|
|
privacy_mode, socket_tag);
|
|
}
|
|
|
|
bool HttpStreamFactory::Job::CanUseExistingSpdySession() const {
|
|
DCHECK(!using_quic_);
|
|
|
|
if (proxy_info_.is_direct() &&
|
|
session_->http_server_properties()->RequiresHTTP11(destination_)) {
|
|
return false;
|
|
}
|
|
|
|
// We need to make sure that if a spdy session was created for
|
|
// https://somehost/ then we do not use that session for http://somehost:443/.
|
|
// The only time we can use an existing session is if the request URL is
|
|
// https (the normal case) or if we are connecting to a SPDY proxy.
|
|
// https://crbug.com/133176
|
|
return origin_url_.SchemeIs(url::kHttpsScheme) || try_websocket_over_http2_ ||
|
|
proxy_info_.proxy_server().is_https();
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnStreamReadyCallback() {
|
|
DCHECK(stream_.get());
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
DCHECK(!is_websocket_ || try_websocket_over_http2_);
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnStreamReady(this, server_ssl_config_);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnWebSocketHandshakeStreamReadyCallback() {
|
|
DCHECK(websocket_stream_);
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
DCHECK(is_websocket_);
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnWebSocketHandshakeStreamReady(
|
|
this, server_ssl_config_, proxy_info_, std::move(websocket_stream_));
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnBidirectionalStreamImplReadyCallback() {
|
|
DCHECK(bidirectional_stream_impl_);
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnBidirectionalStreamImplReady(this, server_ssl_config_,
|
|
proxy_info_);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnNewSpdySessionReadyCallback() {
|
|
DCHECK(stream_.get() || bidirectional_stream_impl_.get());
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
DCHECK(using_spdy_);
|
|
// Note: an event loop iteration has passed, so |new_spdy_session_| may be
|
|
// NULL at this point if the SpdySession closed immediately after creation.
|
|
base::WeakPtr<SpdySession> spdy_session = new_spdy_session_;
|
|
new_spdy_session_.reset();
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnNewSpdySessionReady(this, spdy_session);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnStreamFailedCallback(int result) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnStreamFailed(this, result, server_ssl_config_);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnCertificateErrorCallback(
|
|
int result,
|
|
const SSLInfo& ssl_info) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
|
|
MaybeCopyConnectionAttemptsFromSocketOrHandle();
|
|
|
|
delegate_->OnCertificateError(this, result, server_ssl_config_, ssl_info);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnNeedsProxyAuthCallback(
|
|
const HttpResponseInfo& response,
|
|
HttpAuthController* auth_controller) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
|
|
delegate_->OnNeedsProxyAuth(this, response, server_ssl_config_, proxy_info_,
|
|
auth_controller);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnNeedsClientAuthCallback(
|
|
SSLCertRequestInfo* cert_info) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
|
|
delegate_->OnNeedsClientAuth(this, server_ssl_config_, cert_info);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnHttpsProxyTunnelResponseCallback(
|
|
const HttpResponseInfo& response_info,
|
|
std::unique_ptr<HttpStream> stream) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
|
|
delegate_->OnHttpsProxyTunnelResponse(this, response_info, server_ssl_config_,
|
|
proxy_info_, std::move(stream));
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnPreconnectsComplete() {
|
|
DCHECK(!new_spdy_session_);
|
|
|
|
delegate_->OnPreconnectsComplete(this);
|
|
// |this| may be deleted after this call.
|
|
}
|
|
|
|
// static
|
|
int HttpStreamFactory::Job::OnHostResolution(
|
|
SpdySessionPool* spdy_session_pool,
|
|
const SpdySessionKey& spdy_session_key,
|
|
bool enable_ip_based_pooling,
|
|
bool is_websocket,
|
|
const AddressList& addresses,
|
|
const NetLogWithSource& net_log) {
|
|
// It is OK to dereference spdy_session_pool, because the
|
|
// ClientSocketPoolManager will be destroyed in the same callback that
|
|
// destroys the SpdySessionPool.
|
|
return spdy_session_pool->FindAvailableSession(
|
|
spdy_session_key, enable_ip_based_pooling, is_websocket, net_log)
|
|
? ERR_SPDY_SESSION_ALREADY_EXISTS
|
|
: OK;
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnIOComplete(int result) {
|
|
TRACE_EVENT0(kNetTracingCategory, "HttpStreamFactory::Job::OnIOComplete");
|
|
RunLoop(result);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::RunLoop(int result) {
|
|
TRACE_EVENT0(kNetTracingCategory, "HttpStreamFactory::Job::RunLoop");
|
|
result = DoLoop(result);
|
|
|
|
if (result == ERR_IO_PENDING)
|
|
return;
|
|
|
|
if (!using_quic_) {
|
|
// Resume all throttled Jobs with the same SpdySessionKey if there are any,
|
|
// now that this job is done.
|
|
session_->spdy_session_pool()->ResumePendingRequests(spdy_session_key_);
|
|
}
|
|
|
|
if (job_type_ == PRECONNECT) {
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::Bind(&HttpStreamFactory::Job::OnPreconnectsComplete,
|
|
ptr_factory_.GetWeakPtr()));
|
|
return;
|
|
}
|
|
|
|
if (IsCertificateError(result)) {
|
|
// Retrieve SSL information from the socket.
|
|
SSLInfo ssl_info;
|
|
GetSSLInfo(&ssl_info);
|
|
|
|
next_state_ = STATE_WAITING_USER_ACTION;
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(&HttpStreamFactory::Job::OnCertificateErrorCallback,
|
|
ptr_factory_.GetWeakPtr(), result, ssl_info));
|
|
return;
|
|
}
|
|
|
|
switch (result) {
|
|
case ERR_PROXY_AUTH_REQUESTED: {
|
|
UMA_HISTOGRAM_BOOLEAN("Net.ProxyAuthRequested.HasConnection",
|
|
connection_.get() != NULL);
|
|
if (!connection_.get()) {
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(&Job::OnStreamFailedCallback, ptr_factory_.GetWeakPtr(),
|
|
ERR_PROXY_AUTH_REQUESTED_WITH_NO_CONNECTION));
|
|
return;
|
|
}
|
|
CHECK(connection_->socket());
|
|
CHECK(establishing_tunnel_);
|
|
|
|
next_state_ = STATE_WAITING_USER_ACTION;
|
|
ProxyClientSocket* proxy_socket =
|
|
static_cast<ProxyClientSocket*>(connection_->socket());
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(&Job::OnNeedsProxyAuthCallback, ptr_factory_.GetWeakPtr(),
|
|
*proxy_socket->GetConnectResponseInfo(),
|
|
base::RetainedRef(proxy_socket->GetAuthController())));
|
|
return;
|
|
}
|
|
|
|
case ERR_SSL_CLIENT_AUTH_CERT_NEEDED:
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(
|
|
&Job::OnNeedsClientAuthCallback, ptr_factory_.GetWeakPtr(),
|
|
base::RetainedRef(
|
|
connection_->ssl_error_response_info().cert_request_info)));
|
|
return;
|
|
|
|
case ERR_HTTPS_PROXY_TUNNEL_RESPONSE: {
|
|
DCHECK(connection_.get());
|
|
DCHECK(connection_->socket());
|
|
DCHECK(establishing_tunnel_);
|
|
|
|
ProxyClientSocket* proxy_socket =
|
|
static_cast<ProxyClientSocket*>(connection_->socket());
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(
|
|
&Job::OnHttpsProxyTunnelResponseCallback,
|
|
ptr_factory_.GetWeakPtr(),
|
|
*proxy_socket->GetConnectResponseInfo(),
|
|
base::Passed(proxy_socket->CreateConnectResponseStream())));
|
|
return;
|
|
}
|
|
|
|
case OK:
|
|
next_state_ = STATE_DONE;
|
|
if (new_spdy_session_.get()) {
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::Bind(&Job::OnNewSpdySessionReadyCallback,
|
|
ptr_factory_.GetWeakPtr()));
|
|
} else if (is_websocket_) {
|
|
DCHECK(websocket_stream_);
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::Bind(&Job::OnWebSocketHandshakeStreamReadyCallback,
|
|
ptr_factory_.GetWeakPtr()));
|
|
} else if (stream_type_ == HttpStreamRequest::BIDIRECTIONAL_STREAM) {
|
|
if (!bidirectional_stream_impl_) {
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::Bind(&Job::OnStreamFailedCallback,
|
|
ptr_factory_.GetWeakPtr(), ERR_FAILED));
|
|
} else {
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(&Job::OnBidirectionalStreamImplReadyCallback,
|
|
ptr_factory_.GetWeakPtr()));
|
|
}
|
|
} else {
|
|
DCHECK(stream_.get());
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE,
|
|
base::Bind(&Job::OnStreamReadyCallback, ptr_factory_.GetWeakPtr()));
|
|
}
|
|
return;
|
|
|
|
default:
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(
|
|
FROM_HERE, base::Bind(&Job::OnStreamFailedCallback,
|
|
ptr_factory_.GetWeakPtr(), result));
|
|
return;
|
|
}
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoLoop(int result) {
|
|
DCHECK_NE(next_state_, STATE_NONE);
|
|
int rv = result;
|
|
do {
|
|
State state = next_state_;
|
|
next_state_ = STATE_NONE;
|
|
switch (state) {
|
|
case STATE_START:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoStart();
|
|
break;
|
|
case STATE_WAIT:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoWait();
|
|
break;
|
|
case STATE_WAIT_COMPLETE:
|
|
rv = DoWaitComplete(rv);
|
|
break;
|
|
case STATE_EVALUATE_THROTTLE:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoEvaluateThrottle();
|
|
break;
|
|
case STATE_INIT_CONNECTION:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoInitConnection();
|
|
break;
|
|
case STATE_INIT_CONNECTION_COMPLETE:
|
|
rv = DoInitConnectionComplete(rv);
|
|
break;
|
|
case STATE_WAITING_USER_ACTION:
|
|
rv = DoWaitingUserAction(rv);
|
|
break;
|
|
case STATE_RESTART_TUNNEL_AUTH:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoRestartTunnelAuth();
|
|
break;
|
|
case STATE_RESTART_TUNNEL_AUTH_COMPLETE:
|
|
rv = DoRestartTunnelAuthComplete(rv);
|
|
break;
|
|
case STATE_CREATE_STREAM:
|
|
DCHECK_EQ(OK, rv);
|
|
rv = DoCreateStream();
|
|
break;
|
|
case STATE_CREATE_STREAM_COMPLETE:
|
|
rv = DoCreateStreamComplete(rv);
|
|
break;
|
|
default:
|
|
NOTREACHED() << "bad state";
|
|
rv = ERR_FAILED;
|
|
break;
|
|
}
|
|
} while (rv != ERR_IO_PENDING && next_state_ != STATE_NONE);
|
|
return rv;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::StartInternal() {
|
|
CHECK_EQ(STATE_NONE, next_state_);
|
|
next_state_ = STATE_START;
|
|
RunLoop(OK);
|
|
return ERR_IO_PENDING;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoStart() {
|
|
const NetLogWithSource* net_log = delegate_->GetNetLog();
|
|
|
|
if (net_log) {
|
|
net_log_.BeginEvent(
|
|
NetLogEventType::HTTP_STREAM_JOB,
|
|
base::Bind(&NetLogHttpStreamJobCallback, net_log->source(),
|
|
&request_info_.url, &origin_url_, expect_spdy_, using_quic_,
|
|
priority_));
|
|
net_log->AddEvent(NetLogEventType::HTTP_STREAM_REQUEST_STARTED_JOB,
|
|
net_log_.source().ToEventParametersCallback());
|
|
}
|
|
|
|
// Don't connect to restricted ports.
|
|
if (!IsPortAllowedForScheme(destination_.port(),
|
|
request_info_.url.scheme())) {
|
|
return ERR_UNSAFE_PORT;
|
|
}
|
|
|
|
next_state_ = STATE_WAIT;
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoWait() {
|
|
next_state_ = STATE_WAIT_COMPLETE;
|
|
bool should_wait = delegate_->ShouldWait(this);
|
|
net_log_.BeginEvent(NetLogEventType::HTTP_STREAM_JOB_WAITING,
|
|
NetLog::BoolCallback("should_wait", should_wait));
|
|
if (should_wait)
|
|
return ERR_IO_PENDING;
|
|
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoWaitComplete(int result) {
|
|
net_log_.EndEvent(NetLogEventType::HTTP_STREAM_JOB_WAITING);
|
|
DCHECK_EQ(OK, result);
|
|
next_state_ = STATE_EVALUATE_THROTTLE;
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoEvaluateThrottle() {
|
|
next_state_ = STATE_INIT_CONNECTION;
|
|
if (!using_ssl_)
|
|
return OK;
|
|
if (using_quic_)
|
|
return OK;
|
|
// Ask |delegate_delegate_| to update the spdy session key for the request
|
|
// that launched this job.
|
|
delegate_->SetSpdySessionKey(this, spdy_session_key_);
|
|
|
|
// Throttle connect to an HTTP/2 supported server, if there are pending
|
|
// requests with the same SpdySessionKey.
|
|
if (session_->http_server_properties()->RequiresHTTP11(
|
|
spdy_session_key_.host_port_pair())) {
|
|
return OK;
|
|
}
|
|
url::SchemeHostPort scheme_host_port(
|
|
using_ssl_ ? url::kHttpsScheme : url::kHttpScheme,
|
|
spdy_session_key_.host_port_pair().host(),
|
|
spdy_session_key_.host_port_pair().port());
|
|
if (!session_->http_server_properties()->GetSupportsSpdy(scheme_host_port))
|
|
return OK;
|
|
base::Closure callback = base::Bind(
|
|
&HttpStreamFactory::Job::ResumeInitConnection, ptr_factory_.GetWeakPtr());
|
|
if (session_->spdy_session_pool()->StartRequest(spdy_session_key_,
|
|
callback)) {
|
|
return OK;
|
|
}
|
|
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
|
|
FROM_HERE, callback, base::TimeDelta::FromMilliseconds(kHTTP2ThrottleMs));
|
|
net_log_.AddEvent(NetLogEventType::HTTP_STREAM_JOB_THROTTLED);
|
|
return ERR_IO_PENDING;
|
|
}
|
|
|
|
void HttpStreamFactory::Job::ResumeInitConnection() {
|
|
if (init_connection_already_resumed_)
|
|
return;
|
|
DCHECK_EQ(next_state_, STATE_INIT_CONNECTION);
|
|
net_log_.AddEvent(NetLogEventType::HTTP_STREAM_JOB_RESUME_INIT_CONNECTION);
|
|
init_connection_already_resumed_ = true;
|
|
OnIOComplete(OK);
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoInitConnection() {
|
|
net_log_.BeginEvent(NetLogEventType::HTTP_STREAM_JOB_INIT_CONNECTION);
|
|
int result = DoInitConnectionImpl();
|
|
if (result != ERR_SPDY_SESSION_ALREADY_EXISTS &&
|
|
!expect_on_quic_host_resolution_) {
|
|
delegate_->OnConnectionInitialized(this, result);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoInitConnectionImpl() {
|
|
DCHECK(!connection_->is_initialized());
|
|
|
|
if (using_quic_ && !proxy_info_.is_quic() && !proxy_info_.is_direct()) {
|
|
// QUIC can not be spoken to non-QUIC proxies. This error should not be
|
|
// user visible, because the non-alternative Job should be resumed.
|
|
return ERR_NO_SUPPORTED_PROXIES;
|
|
}
|
|
|
|
DCHECK(proxy_info_.proxy_server().is_valid());
|
|
next_state_ = STATE_INIT_CONNECTION_COMPLETE;
|
|
|
|
if (delegate_->OnInitConnection(proxy_info_)) {
|
|
// Return since the connection initialization can be skipped.
|
|
return OK;
|
|
}
|
|
|
|
if (proxy_info_.is_https() || proxy_info_.is_quic()) {
|
|
InitSSLConfig(&proxy_ssl_config_, /*is_proxy=*/true);
|
|
// Disable network fetches for HTTPS proxies, since the network requests
|
|
// are probably going to need to go through the proxy too.
|
|
proxy_ssl_config_.disable_cert_verification_network_fetches = true;
|
|
}
|
|
if (using_ssl_) {
|
|
InitSSLConfig(&server_ssl_config_, /*is_proxy=*/false);
|
|
}
|
|
|
|
if (using_quic_) {
|
|
HostPortPair destination;
|
|
SSLConfig* ssl_config;
|
|
GURL url(request_info_.url);
|
|
if (proxy_info_.is_quic()) {
|
|
// A proxy's certificate is expected to be valid for the proxy hostname.
|
|
destination = proxy_info_.proxy_server().host_port_pair();
|
|
ssl_config = &proxy_ssl_config_;
|
|
GURL::Replacements replacements;
|
|
replacements.SetSchemeStr(url::kHttpsScheme);
|
|
replacements.SetHostStr(destination.host());
|
|
const std::string new_port = base::UintToString(destination.port());
|
|
replacements.SetPortStr(new_port);
|
|
replacements.ClearUsername();
|
|
replacements.ClearPassword();
|
|
replacements.ClearPath();
|
|
replacements.ClearQuery();
|
|
replacements.ClearRef();
|
|
url = url.ReplaceComponents(replacements);
|
|
} else {
|
|
DCHECK(using_ssl_);
|
|
// The certificate of a QUIC alternative server is expected to be valid
|
|
// for the origin of the request (in addition to being valid for the
|
|
// server itself).
|
|
destination = destination_;
|
|
ssl_config = &server_ssl_config_;
|
|
}
|
|
int rv = quic_request_.Request(
|
|
destination, quic_version_, request_info_.privacy_mode, priority_,
|
|
request_info_.socket_tag, ssl_config->GetCertVerifyFlags(), url,
|
|
net_log_, &net_error_details_, io_callback_);
|
|
if (rv == OK) {
|
|
using_existing_quic_session_ = true;
|
|
} else if (rv == ERR_IO_PENDING) {
|
|
// There's no available QUIC session. Inform the delegate how long to
|
|
// delay the main job.
|
|
delegate_->MaybeSetWaitTimeForMainJob(
|
|
quic_request_.GetTimeDelayForWaitingJob());
|
|
expect_on_quic_host_resolution_ =
|
|
quic_request_.WaitForHostResolution(base::BindRepeating(
|
|
&Job::OnQuicHostResolution, base::Unretained(this)));
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
// Check first if there is a pushed stream matching the request, or an HTTP/2
|
|
// connection this request can pool to. If so, then go straight to using
|
|
// that.
|
|
if (CanUseExistingSpdySession()) {
|
|
if (!is_websocket_) {
|
|
session_->spdy_session_pool()->push_promise_index()->ClaimPushedStream(
|
|
spdy_session_key_, origin_url_, request_info_,
|
|
&existing_spdy_session_, &pushed_stream_id_);
|
|
}
|
|
if (!existing_spdy_session_) {
|
|
existing_spdy_session_ =
|
|
session_->spdy_session_pool()->FindAvailableSession(
|
|
spdy_session_key_, enable_ip_based_pooling_,
|
|
try_websocket_over_http2_, net_log_);
|
|
}
|
|
if (existing_spdy_session_) {
|
|
// If we're preconnecting, but we already have a SpdySession, we don't
|
|
// actually need to preconnect any sockets, so we're done.
|
|
if (job_type_ == PRECONNECT)
|
|
return OK;
|
|
using_spdy_ = true;
|
|
next_state_ = STATE_CREATE_STREAM;
|
|
return OK;
|
|
}
|
|
}
|
|
|
|
if (proxy_info_.is_http() || proxy_info_.is_https() || proxy_info_.is_quic())
|
|
establishing_tunnel_ = using_ssl_;
|
|
|
|
HttpServerProperties* http_server_properties =
|
|
session_->http_server_properties();
|
|
if (http_server_properties) {
|
|
http_server_properties->MaybeForceHTTP11(destination_, &server_ssl_config_);
|
|
if (proxy_info_.is_http() || proxy_info_.is_https()) {
|
|
http_server_properties->MaybeForceHTTP11(
|
|
proxy_info_.proxy_server().host_port_pair(), &proxy_ssl_config_);
|
|
}
|
|
}
|
|
|
|
if (job_type_ == PRECONNECT) {
|
|
DCHECK(!is_websocket_);
|
|
DCHECK(request_info_.socket_tag == SocketTag());
|
|
return PreconnectSocketsForHttpRequest(
|
|
GetSocketGroup(), destination_, request_info_.extra_headers,
|
|
request_info_.load_flags, priority_, session_, proxy_info_,
|
|
server_ssl_config_, proxy_ssl_config_, request_info_.privacy_mode,
|
|
net_log_, num_streams_);
|
|
}
|
|
|
|
// If we can't use a SPDY session, don't bother checking for one after
|
|
// the hostname is resolved.
|
|
OnHostResolutionCallback resolution_callback =
|
|
CanUseExistingSpdySession()
|
|
? base::Bind(&Job::OnHostResolution, session_->spdy_session_pool(),
|
|
spdy_session_key_, enable_ip_based_pooling_,
|
|
try_websocket_over_http2_)
|
|
: OnHostResolutionCallback();
|
|
if (is_websocket_) {
|
|
DCHECK(request_info_.socket_tag == SocketTag());
|
|
SSLConfig websocket_server_ssl_config = server_ssl_config_;
|
|
websocket_server_ssl_config.alpn_protos.clear();
|
|
return InitSocketHandleForWebSocketRequest(
|
|
GetSocketGroup(), destination_, request_info_.extra_headers,
|
|
request_info_.load_flags, priority_, session_, proxy_info_,
|
|
websocket_server_ssl_config, proxy_ssl_config_,
|
|
request_info_.privacy_mode, net_log_, connection_.get(),
|
|
resolution_callback, io_callback_);
|
|
}
|
|
|
|
return InitSocketHandleForHttpRequest(
|
|
GetSocketGroup(), destination_, request_info_.extra_headers,
|
|
request_info_.load_flags, priority_, session_, proxy_info_, quic_version_,
|
|
server_ssl_config_, proxy_ssl_config_, request_info_.privacy_mode,
|
|
request_info_.socket_tag, net_log_, connection_.get(),
|
|
resolution_callback, io_callback_);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::OnQuicHostResolution(int result) {
|
|
DCHECK(expect_on_quic_host_resolution_);
|
|
expect_on_quic_host_resolution_ = false;
|
|
delegate_->OnConnectionInitialized(this, result);
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoInitConnectionComplete(int result) {
|
|
net_log_.EndEvent(NetLogEventType::HTTP_STREAM_JOB_INIT_CONNECTION);
|
|
if (job_type_ == PRECONNECT) {
|
|
if (using_quic_)
|
|
return result;
|
|
DCHECK_EQ(OK, result);
|
|
return OK;
|
|
}
|
|
|
|
if (result == ERR_SPDY_SESSION_ALREADY_EXISTS) {
|
|
// We found a SPDY connection after resolving the host. This is
|
|
// probably an IP pooled connection.
|
|
existing_spdy_session_ =
|
|
session_->spdy_session_pool()->FindAvailableSession(
|
|
spdy_session_key_, enable_ip_based_pooling_,
|
|
try_websocket_over_http2_, net_log_);
|
|
if (existing_spdy_session_) {
|
|
using_spdy_ = true;
|
|
next_state_ = STATE_CREATE_STREAM;
|
|
} else {
|
|
// It is possible that the spdy session no longer exists.
|
|
ReturnToStateInitConnection(true /* close connection */);
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
// |result| may be the result of any of the stacked pools. The following
|
|
// logic is used when determining how to interpret an error.
|
|
// If |result| < 0:
|
|
// and connection_->socket() != NULL, then the SSL handshake ran and it
|
|
// is a potentially recoverable error.
|
|
// and connection_->socket == NULL and connection_->is_ssl_error() is true,
|
|
// then the SSL handshake ran with an unrecoverable error.
|
|
// otherwise, the error came from one of the other pools.
|
|
bool ssl_started = using_ssl_ && (result == OK || connection_->socket() ||
|
|
connection_->is_ssl_error());
|
|
|
|
if (ssl_started && (result == OK || IsCertificateError(result))) {
|
|
if (using_quic_ && result == OK) {
|
|
was_alpn_negotiated_ = true;
|
|
negotiated_protocol_ = kProtoQUIC;
|
|
} else {
|
|
if (connection_->socket()->WasAlpnNegotiated()) {
|
|
was_alpn_negotiated_ = true;
|
|
negotiated_protocol_ = connection_->socket()->GetNegotiatedProtocol();
|
|
net_log_.AddEvent(
|
|
NetLogEventType::HTTP_STREAM_REQUEST_PROTO,
|
|
base::Bind(&NetLogHttpStreamProtoCallback, negotiated_protocol_));
|
|
if (negotiated_protocol_ == kProtoHTTP2) {
|
|
if (is_websocket_) {
|
|
// WebSocket is not supported over a fresh HTTP/2 connection.
|
|
return ERR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
using_spdy_ = true;
|
|
}
|
|
}
|
|
}
|
|
} else if (proxy_info_.is_https() && connection_->socket() && result == OK) {
|
|
ProxyClientSocket* proxy_socket =
|
|
static_cast<ProxyClientSocket*>(connection_->socket());
|
|
// http://crbug.com/642354
|
|
if (!proxy_socket->IsConnected())
|
|
return ERR_CONNECTION_CLOSED;
|
|
if (proxy_socket->IsUsingSpdy()) {
|
|
was_alpn_negotiated_ = true;
|
|
negotiated_protocol_ = proxy_socket->GetProxyNegotiatedProtocol();
|
|
using_spdy_ = true;
|
|
}
|
|
}
|
|
|
|
if (result == ERR_PROXY_AUTH_REQUESTED ||
|
|
result == ERR_HTTPS_PROXY_TUNNEL_RESPONSE) {
|
|
DCHECK(!ssl_started);
|
|
// Other state (i.e. |using_ssl_|) suggests that |connection_| will have an
|
|
// SSL socket, but there was an error before that could happen. This
|
|
// puts the in progress HttpProxy socket into |connection_| in order to
|
|
// complete the auth (or read the response body). The tunnel restart code
|
|
// is careful to remove it before returning control to the rest of this
|
|
// class.
|
|
connection_ = connection_->release_pending_http_proxy_connection();
|
|
return result;
|
|
}
|
|
|
|
if (proxy_info_.is_quic() && using_quic_ && result < 0)
|
|
return ReconsiderProxyAfterError(result);
|
|
|
|
if (expect_spdy_ && !using_spdy_)
|
|
return ERR_ALPN_NEGOTIATION_FAILED;
|
|
|
|
if (!ssl_started && result < 0 && (expect_spdy_ || using_quic_))
|
|
return result;
|
|
|
|
if (using_quic_) {
|
|
if (result < 0)
|
|
return result;
|
|
|
|
if (stream_type_ == HttpStreamRequest::BIDIRECTIONAL_STREAM) {
|
|
std::unique_ptr<QuicChromiumClientSession::Handle> session =
|
|
quic_request_.ReleaseSessionHandle();
|
|
if (!session) {
|
|
// Quic session is closed before stream can be created.
|
|
return ERR_CONNECTION_CLOSED;
|
|
}
|
|
bidirectional_stream_impl_.reset(
|
|
new BidirectionalStreamQuicImpl(std::move(session)));
|
|
} else {
|
|
std::unique_ptr<QuicChromiumClientSession::Handle> session =
|
|
quic_request_.ReleaseSessionHandle();
|
|
if (!session) {
|
|
// Quic session is closed before stream can be created.
|
|
return ERR_CONNECTION_CLOSED;
|
|
}
|
|
stream_ = std::make_unique<QuicHttpStream>(std::move(session));
|
|
}
|
|
next_state_ = STATE_NONE;
|
|
return OK;
|
|
}
|
|
|
|
if (result < 0 && !ssl_started)
|
|
return ReconsiderProxyAfterError(result);
|
|
|
|
establishing_tunnel_ = false;
|
|
|
|
// Handle SSL errors below.
|
|
if (using_ssl_) {
|
|
DCHECK(ssl_started);
|
|
if (IsCertificateError(result)) {
|
|
result = HandleCertificateError(result);
|
|
if (result == OK && !connection_->socket()->IsConnectedAndIdle()) {
|
|
ReturnToStateInitConnection(true /* close connection */);
|
|
return result;
|
|
}
|
|
}
|
|
if (result < 0)
|
|
return result;
|
|
}
|
|
|
|
next_state_ = STATE_CREATE_STREAM;
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoWaitingUserAction(int result) {
|
|
// This state indicates that the stream request is in a partially
|
|
// completed state, and we've called back to the delegate for more
|
|
// information.
|
|
|
|
// We're always waiting here for the delegate to call us back.
|
|
return ERR_IO_PENDING;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::SetSpdyHttpStreamOrBidirectionalStreamImpl(
|
|
base::WeakPtr<SpdySession> session) {
|
|
DCHECK(using_spdy_);
|
|
if (is_websocket_) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
DCHECK(delegate_->websocket_handshake_stream_create_helper());
|
|
|
|
if (!try_websocket_over_http2_) {
|
|
// Plaintext WebSocket is not supported over HTTP/2 proxy,
|
|
// see https://crbug.com/684681.
|
|
return ERR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
websocket_stream_ = delegate_->websocket_handshake_stream_create_helper()
|
|
->CreateHttp2Stream(session);
|
|
return OK;
|
|
}
|
|
if (stream_type_ == HttpStreamRequest::BIDIRECTIONAL_STREAM) {
|
|
bidirectional_stream_impl_ = std::make_unique<BidirectionalStreamSpdyImpl>(
|
|
session, net_log_.source());
|
|
return OK;
|
|
}
|
|
|
|
// TODO(willchan): Delete this code, because eventually, the HttpStreamFactory
|
|
// will be creating all the SpdyHttpStreams, since it will know when
|
|
// SpdySessions become available.
|
|
|
|
stream_ = std::make_unique<SpdyHttpStream>(session, pushed_stream_id_,
|
|
net_log_.source());
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoCreateStream() {
|
|
DCHECK(connection_->socket() || existing_spdy_session_.get());
|
|
DCHECK(!using_quic_);
|
|
|
|
next_state_ = STATE_CREATE_STREAM_COMPLETE;
|
|
|
|
if (using_ssl_ && connection_->socket()) {
|
|
RecordChannelIDKeyMatch(connection_->socket(),
|
|
session_->context().channel_id_service,
|
|
destination_.HostForURL());
|
|
}
|
|
|
|
if (!using_spdy_) {
|
|
DCHECK(!expect_spdy_);
|
|
// We may get ftp scheme when fetching ftp resources through proxy.
|
|
bool using_proxy = (proxy_info_.is_http() || proxy_info_.is_https() ||
|
|
proxy_info_.is_quic()) &&
|
|
(request_info_.url.SchemeIs(url::kHttpScheme) ||
|
|
request_info_.url.SchemeIs(url::kFtpScheme));
|
|
if (is_websocket_) {
|
|
DCHECK_NE(job_type_, PRECONNECT);
|
|
DCHECK(delegate_->websocket_handshake_stream_create_helper());
|
|
websocket_stream_ =
|
|
delegate_->websocket_handshake_stream_create_helper()
|
|
->CreateBasicStream(std::move(connection_), using_proxy,
|
|
session_->websocket_endpoint_lock_manager());
|
|
} else {
|
|
stream_ = std::make_unique<HttpBasicStream>(
|
|
std::move(connection_), using_proxy,
|
|
session_->params().http_09_on_non_default_ports_enabled);
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
CHECK(!stream_.get());
|
|
|
|
// It is possible that a pushed stream has been opened by a server since last
|
|
// time Job checked above.
|
|
if (!existing_spdy_session_) {
|
|
// WebSocket over HTTP/2 is only allowed to use existing HTTP/2 connections.
|
|
// Therefore |using_spdy_| could not have been set unless a connection had
|
|
// already been found.
|
|
DCHECK(!try_websocket_over_http2_);
|
|
|
|
session_->spdy_session_pool()->push_promise_index()->ClaimPushedStream(
|
|
spdy_session_key_, origin_url_, request_info_, &existing_spdy_session_,
|
|
&pushed_stream_id_);
|
|
// It is also possible that an HTTP/2 connection has been established since
|
|
// last time Job checked above.
|
|
if (!existing_spdy_session_) {
|
|
existing_spdy_session_ =
|
|
session_->spdy_session_pool()->FindAvailableSession(
|
|
spdy_session_key_, enable_ip_based_pooling_,
|
|
/* is_websocket = */ false, net_log_);
|
|
}
|
|
}
|
|
if (existing_spdy_session_) {
|
|
// We picked up an existing session, so we don't need our socket.
|
|
if (connection_->socket())
|
|
connection_->socket()->Disconnect();
|
|
connection_->Reset();
|
|
|
|
int set_result =
|
|
SetSpdyHttpStreamOrBidirectionalStreamImpl(existing_spdy_session_);
|
|
existing_spdy_session_.reset();
|
|
return set_result;
|
|
}
|
|
|
|
// Close idle sockets in this group, since subsequent requests will go over
|
|
// |spdy_session|.
|
|
if (connection_->socket()->IsConnected())
|
|
connection_->CloseIdleSocketsInGroup();
|
|
|
|
// If |spdy_session_direct_| is false, then |proxy_info_| is guaranteed to
|
|
// have a non-empty proxy list.
|
|
bool is_trusted_proxy =
|
|
!spdy_session_direct_ && proxy_info_.proxy_server().is_trusted_proxy();
|
|
|
|
base::WeakPtr<SpdySession> spdy_session =
|
|
session_->spdy_session_pool()->CreateAvailableSessionFromSocket(
|
|
spdy_session_key_, is_trusted_proxy, std::move(connection_),
|
|
net_log_);
|
|
|
|
if (!spdy_session->HasAcceptableTransportSecurity()) {
|
|
spdy_session->CloseSessionOnError(ERR_SPDY_INADEQUATE_TRANSPORT_SECURITY,
|
|
"");
|
|
return ERR_SPDY_INADEQUATE_TRANSPORT_SECURITY;
|
|
}
|
|
|
|
new_spdy_session_ = spdy_session;
|
|
url::SchemeHostPort scheme_host_port(
|
|
using_ssl_ ? url::kHttpsScheme : url::kHttpScheme,
|
|
spdy_session_key_.host_port_pair().host(),
|
|
spdy_session_key_.host_port_pair().port());
|
|
|
|
HttpServerProperties* http_server_properties =
|
|
session_->http_server_properties();
|
|
if (http_server_properties)
|
|
http_server_properties->SetSupportsSpdy(scheme_host_port, true);
|
|
|
|
// Create a SpdyHttpStream or a BidirectionalStreamImpl attached to the
|
|
// session; OnNewSpdySessionReadyCallback is not called until an event loop
|
|
// iteration later, so if the SpdySession is closed between then, allow
|
|
// reuse state from the underlying socket, sampled by SpdyHttpStream,
|
|
// bubble up to the request.
|
|
return SetSpdyHttpStreamOrBidirectionalStreamImpl(new_spdy_session_);
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoCreateStreamComplete(int result) {
|
|
if (result < 0)
|
|
return result;
|
|
|
|
session_->proxy_resolution_service()->ReportSuccess(
|
|
proxy_info_, session_->context().proxy_delegate);
|
|
next_state_ = STATE_NONE;
|
|
return OK;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoRestartTunnelAuth() {
|
|
next_state_ = STATE_RESTART_TUNNEL_AUTH_COMPLETE;
|
|
ProxyClientSocket* proxy_socket =
|
|
static_cast<ProxyClientSocket*>(connection_->socket());
|
|
return proxy_socket->RestartWithAuth(io_callback_);
|
|
}
|
|
|
|
int HttpStreamFactory::Job::DoRestartTunnelAuthComplete(int result) {
|
|
if (result == ERR_PROXY_AUTH_REQUESTED)
|
|
return result;
|
|
|
|
if (result == OK) {
|
|
// Now that we've got the HttpProxyClientSocket connected. We have
|
|
// to release it as an idle socket into the pool and start the connection
|
|
// process from the beginning. Trying to pass it in with the
|
|
// SSLSocketParams might cause a deadlock since params are dispatched
|
|
// interchangeably. This request won't necessarily get this http proxy
|
|
// socket, but there will be forward progress.
|
|
establishing_tunnel_ = false;
|
|
ReturnToStateInitConnection(false /* do not close connection */);
|
|
return OK;
|
|
}
|
|
|
|
return ReconsiderProxyAfterError(result);
|
|
}
|
|
|
|
void HttpStreamFactory::Job::ReturnToStateInitConnection(
|
|
bool close_connection) {
|
|
if (close_connection && connection_->socket())
|
|
connection_->socket()->Disconnect();
|
|
connection_->Reset();
|
|
|
|
if (!using_quic_)
|
|
delegate_->RemoveRequestFromSpdySessionRequestMapForJob(this);
|
|
|
|
next_state_ = STATE_INIT_CONNECTION;
|
|
}
|
|
|
|
void HttpStreamFactory::Job::InitSSLConfig(SSLConfig* ssl_config,
|
|
bool is_proxy) const {
|
|
if (!is_proxy) {
|
|
// Prior to HTTP/2 and SPDY, some servers use TLS renegotiation to request
|
|
// TLS client authentication after the HTTP request was sent. Allow
|
|
// renegotiation for only those connections.
|
|
//
|
|
// Note that this does NOT implement the provision in
|
|
// https://http2.github.io/http2-spec/#rfc.section.9.2.1 which allows the
|
|
// server to request a renegotiation immediately before sending the
|
|
// connection preface as waiting for the preface would cost the round trip
|
|
// that False Start otherwise saves.
|
|
ssl_config->renego_allowed_default = true;
|
|
ssl_config->renego_allowed_for_protos.push_back(kProtoHTTP11);
|
|
}
|
|
|
|
if (proxy_info_.is_https() && ssl_config->send_client_cert) {
|
|
// When connecting through an HTTPS proxy, disable TLS False Start so
|
|
// that client authentication errors can be distinguished between those
|
|
// originating from the proxy server (ERR_PROXY_CONNECTION_FAILED) and
|
|
// those originating from the endpoint (ERR_SSL_PROTOCOL_ERROR /
|
|
// ERR_BAD_SSL_CLIENT_AUTH_CERT).
|
|
//
|
|
// This assumes the proxy will only request certificates on the initial
|
|
// handshake; renegotiation on the proxy connection is unsupported.
|
|
ssl_config->false_start_enabled = false;
|
|
}
|
|
|
|
// Disable Channel ID if privacy mode is enabled.
|
|
if (request_info_.privacy_mode == PRIVACY_MODE_ENABLED)
|
|
ssl_config->channel_id_enabled = false;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::ReconsiderProxyAfterError(int error) {
|
|
// Check if the error was a proxy failure.
|
|
if (!CanFalloverToNextProxy(proxy_info_.proxy_server(), error, &error))
|
|
return error;
|
|
|
|
// Alternative proxy server job should not use fallback proxies, and instead
|
|
// return. This would resume the main job (if possible) which may try the
|
|
// fallback proxies.
|
|
if (alternative_proxy_server().is_valid()) {
|
|
DCHECK_EQ(STATE_NONE, next_state_);
|
|
return error;
|
|
}
|
|
|
|
should_reconsider_proxy_ = true;
|
|
return error;
|
|
}
|
|
|
|
int HttpStreamFactory::Job::HandleCertificateError(int error) {
|
|
DCHECK(using_ssl_);
|
|
DCHECK(IsCertificateError(error));
|
|
|
|
SSLInfo ssl_info;
|
|
GetSSLInfo(&ssl_info);
|
|
|
|
if (!ssl_info.cert) {
|
|
// If the server's certificate could not be parsed, there is no way
|
|
// to gracefully recover this, so just pass the error up.
|
|
return error;
|
|
}
|
|
|
|
// Add the bad certificate to the set of allowed certificates in the
|
|
// SSL config object. This data structure will be consulted after calling
|
|
// RestartIgnoringLastError(). And the user will be asked interactively
|
|
// before RestartIgnoringLastError() is ever called.
|
|
server_ssl_config_.allowed_bad_certs.emplace_back(ssl_info.cert,
|
|
ssl_info.cert_status);
|
|
|
|
int load_flags = request_info_.load_flags;
|
|
if (session_->params().ignore_certificate_errors)
|
|
load_flags |= LOAD_IGNORE_ALL_CERT_ERRORS;
|
|
if (SSLClientSocket::IgnoreCertError(error, load_flags))
|
|
return OK;
|
|
return error;
|
|
}
|
|
|
|
ClientSocketPoolManager::SocketGroupType
|
|
HttpStreamFactory::Job::GetSocketGroup() const {
|
|
std::string scheme = origin_url_.scheme();
|
|
if (scheme == url::kHttpsScheme || scheme == url::kWssScheme)
|
|
return ClientSocketPoolManager::SSL_GROUP;
|
|
|
|
if (scheme == url::kFtpScheme)
|
|
return ClientSocketPoolManager::FTP_GROUP;
|
|
|
|
return ClientSocketPoolManager::NORMAL_GROUP;
|
|
}
|
|
|
|
// If the connection succeeds, failed connection attempts leading up to the
|
|
// success will be returned via the successfully connected socket. If the
|
|
// connection fails, failed connection attempts will be returned via the
|
|
// ClientSocketHandle. Check whether a socket was returned and copy the
|
|
// connection attempts from the proper place.
|
|
void HttpStreamFactory::Job::MaybeCopyConnectionAttemptsFromSocketOrHandle() {
|
|
if (!connection_)
|
|
return;
|
|
|
|
ConnectionAttempts socket_attempts = connection_->connection_attempts();
|
|
if (connection_->socket()) {
|
|
connection_->socket()->GetConnectionAttempts(&socket_attempts);
|
|
}
|
|
|
|
delegate_->AddConnectionAttemptsToRequest(this, socket_attempts);
|
|
}
|
|
|
|
HttpStreamFactory::JobFactory::JobFactory() = default;
|
|
|
|
HttpStreamFactory::JobFactory::~JobFactory() = default;
|
|
|
|
std::unique_ptr<HttpStreamFactory::Job>
|
|
HttpStreamFactory::JobFactory::CreateMainJob(
|
|
HttpStreamFactory::Job::Delegate* delegate,
|
|
HttpStreamFactory::JobType job_type,
|
|
HttpNetworkSession* session,
|
|
const HttpRequestInfo& request_info,
|
|
RequestPriority priority,
|
|
const ProxyInfo& proxy_info,
|
|
const SSLConfig& server_ssl_config,
|
|
const SSLConfig& proxy_ssl_config,
|
|
HostPortPair destination,
|
|
GURL origin_url,
|
|
bool is_websocket,
|
|
bool enable_ip_based_pooling,
|
|
NetLog* net_log) {
|
|
return std::make_unique<HttpStreamFactory::Job>(
|
|
delegate, job_type, session, request_info, priority, proxy_info,
|
|
server_ssl_config, proxy_ssl_config, destination, origin_url,
|
|
kProtoUnknown, quic::QUIC_VERSION_UNSUPPORTED, ProxyServer(),
|
|
is_websocket, enable_ip_based_pooling, net_log);
|
|
}
|
|
|
|
std::unique_ptr<HttpStreamFactory::Job>
|
|
HttpStreamFactory::JobFactory::CreateAltSvcJob(
|
|
HttpStreamFactory::Job::Delegate* delegate,
|
|
HttpStreamFactory::JobType job_type,
|
|
HttpNetworkSession* session,
|
|
const HttpRequestInfo& request_info,
|
|
RequestPriority priority,
|
|
const ProxyInfo& proxy_info,
|
|
const SSLConfig& server_ssl_config,
|
|
const SSLConfig& proxy_ssl_config,
|
|
HostPortPair destination,
|
|
GURL origin_url,
|
|
NextProto alternative_protocol,
|
|
quic::QuicTransportVersion quic_version,
|
|
bool is_websocket,
|
|
bool enable_ip_based_pooling,
|
|
NetLog* net_log) {
|
|
return std::make_unique<HttpStreamFactory::Job>(
|
|
delegate, job_type, session, request_info, priority, proxy_info,
|
|
server_ssl_config, proxy_ssl_config, destination, origin_url,
|
|
alternative_protocol, quic_version, ProxyServer(), is_websocket,
|
|
enable_ip_based_pooling, net_log);
|
|
}
|
|
|
|
std::unique_ptr<HttpStreamFactory::Job>
|
|
HttpStreamFactory::JobFactory::CreateAltProxyJob(
|
|
HttpStreamFactory::Job::Delegate* delegate,
|
|
HttpStreamFactory::JobType job_type,
|
|
HttpNetworkSession* session,
|
|
const HttpRequestInfo& request_info,
|
|
RequestPriority priority,
|
|
const ProxyInfo& proxy_info,
|
|
const SSLConfig& server_ssl_config,
|
|
const SSLConfig& proxy_ssl_config,
|
|
HostPortPair destination,
|
|
GURL origin_url,
|
|
const ProxyServer& alternative_proxy_server,
|
|
bool is_websocket,
|
|
bool enable_ip_based_pooling,
|
|
NetLog* net_log) {
|
|
return std::make_unique<HttpStreamFactory::Job>(
|
|
delegate, job_type, session, request_info, priority, proxy_info,
|
|
server_ssl_config, proxy_ssl_config, destination, origin_url,
|
|
kProtoUnknown, quic::QUIC_VERSION_UNSUPPORTED, alternative_proxy_server,
|
|
is_websocket, enable_ip_based_pooling, net_log);
|
|
}
|
|
|
|
} // namespace net
|