blob: 58825741d92d2028d3cd4cfc2461139a53de6539 [file] [log] [blame]
// 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_impl_request.h"
#include "base/callback.h"
#include "base/logging.h"
#include "base/stl_util.h"
#include "net/http/http_stream_factory_impl_job.h"
#include "net/spdy/spdy_http_stream.h"
#include "net/spdy/spdy_session.h"
namespace net {
HttpStreamFactoryImpl::Request::Request(const GURL& url,
HttpStreamFactoryImpl* factory,
HttpStreamRequest::Delegate* delegate,
const BoundNetLog& net_log)
: url_(url),
factory_(factory),
delegate_(delegate),
net_log_(net_log),
completed_(false),
was_npn_negotiated_(false),
protocol_negotiated_(kProtoUnknown),
using_spdy_(false) {
DCHECK(factory_);
DCHECK(delegate_);
net_log_.BeginEvent(NetLog::TYPE_HTTP_STREAM_REQUEST);
}
HttpStreamFactoryImpl::Request::~Request() {
if (bound_job_.get())
DCHECK(jobs_.empty());
else
DCHECK(!jobs_.empty());
net_log_.EndEvent(NetLog::TYPE_HTTP_STREAM_REQUEST);
for (std::set<Job*>::iterator it = jobs_.begin(); it != jobs_.end(); ++it)
factory_->request_map_.erase(*it);
RemoveRequestFromSpdySessionRequestMap();
RemoveRequestFromHttpPipeliningRequestMap();
STLDeleteElements(&jobs_);
}
void HttpStreamFactoryImpl::Request::SetSpdySessionKey(
const HostPortProxyPair& spdy_session_key) {
CHECK(!spdy_session_key_.get());
#if defined(COBALT_DISABLE_SPDY)
NOTREACHED();
#else
spdy_session_key_.reset(new HostPortProxyPair(spdy_session_key));
RequestSet& request_set =
factory_->spdy_session_request_map_[spdy_session_key];
DCHECK(!ContainsKey(request_set, this));
request_set.insert(this);
#endif // defined(COBALT_DISABLE_SPDY)
}
bool HttpStreamFactoryImpl::Request::SetHttpPipeliningKey(
const HttpPipelinedHost::Key& http_pipelining_key) {
#if defined(COBALT_DISABLE_SPDY)
NOTREACHED();
return false;
#else
CHECK(!http_pipelining_key_.get());
http_pipelining_key_.reset(new HttpPipelinedHost::Key(http_pipelining_key));
bool was_new_key = !ContainsKey(factory_->http_pipelining_request_map_,
http_pipelining_key);
RequestVector& request_vector =
factory_->http_pipelining_request_map_[http_pipelining_key];
request_vector.push_back(this);
return was_new_key;
#endif // defined(COBALT_DISABLE_SPDY)
}
void HttpStreamFactoryImpl::Request::AttachJob(Job* job) {
DCHECK(job);
jobs_.insert(job);
factory_->request_map_[job] = this;
}
void HttpStreamFactoryImpl::Request::Complete(
bool was_npn_negotiated,
NextProto protocol_negotiated,
bool using_spdy,
const BoundNetLog& job_net_log) {
DCHECK(!completed_);
completed_ = true;
was_npn_negotiated_ = was_npn_negotiated;
protocol_negotiated_ = protocol_negotiated;
using_spdy_ = using_spdy;
net_log_.AddEvent(
NetLog::TYPE_HTTP_STREAM_REQUEST_BOUND_TO_JOB,
job_net_log.source().ToEventParametersCallback());
job_net_log.AddEvent(
NetLog::TYPE_HTTP_STREAM_JOB_BOUND_TO_REQUEST,
net_log_.source().ToEventParametersCallback());
}
void HttpStreamFactoryImpl::Request::OnStreamReady(
Job* job,
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpStreamBase* stream) {
DCHECK(stream);
DCHECK(completed_);
// |job| should only be NULL if we're being serviced by a late bound
// SpdySession or HttpPipelinedConnection (one that was not created by a job
// in our |jobs_| set).
if (!job) {
DCHECK(!bound_job_.get());
DCHECK(!jobs_.empty());
// NOTE(willchan): We do *NOT* call OrphanJobs() here. The reason is because
// we *WANT* to cancel the unnecessary Jobs from other requests if another
// Job completes first.
// TODO(mbelshe): Revisit this when we implement ip connection pooling of
// SpdySessions. Do we want to orphan the jobs for a different hostname so
// they complete? Or do we want to prevent connecting a new SpdySession if
// we've already got one available for a different hostname where the ip
// address matches up?
} else if (!bound_job_.get()) {
// We may have other jobs in |jobs_|. For example, if we start multiple jobs
// for Alternate-Protocol.
OrphanJobsExcept(job);
} else {
DCHECK(jobs_.empty());
}
delegate_->OnStreamReady(used_ssl_config, used_proxy_info, stream);
}
void HttpStreamFactoryImpl::Request::OnStreamFailed(
Job* job,
int status,
const SSLConfig& used_ssl_config) {
DCHECK_NE(OK, status);
// |job| should only be NULL if we're being canceled by a late bound
// HttpPipelinedConnection (one that was not created by a job in our |jobs_|
// set).
if (!job) {
DCHECK(!bound_job_.get());
DCHECK(!jobs_.empty());
// NOTE(willchan): We do *NOT* call OrphanJobs() here. The reason is because
// we *WANT* to cancel the unnecessary Jobs from other requests if another
// Job completes first.
} else if (!bound_job_.get()) {
// Hey, we've got other jobs! Maybe one of them will succeed, let's just
// ignore this failure.
if (jobs_.size() > 1) {
jobs_.erase(job);
factory_->request_map_.erase(job);
delete job;
return;
} else {
bound_job_.reset(job);
jobs_.erase(job);
DCHECK(jobs_.empty());
factory_->request_map_.erase(job);
}
} else {
DCHECK(jobs_.empty());
}
delegate_->OnStreamFailed(status, used_ssl_config);
}
void HttpStreamFactoryImpl::Request::OnCertificateError(
Job* job,
int status,
const SSLConfig& used_ssl_config,
const SSLInfo& ssl_info) {
DCHECK_NE(OK, status);
if (!bound_job_.get())
OrphanJobsExcept(job);
else
DCHECK(jobs_.empty());
delegate_->OnCertificateError(status, used_ssl_config, ssl_info);
}
void HttpStreamFactoryImpl::Request::OnNeedsProxyAuth(
Job* job,
const HttpResponseInfo& proxy_response,
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpAuthController* auth_controller) {
if (!bound_job_.get())
OrphanJobsExcept(job);
else
DCHECK(jobs_.empty());
delegate_->OnNeedsProxyAuth(
proxy_response, used_ssl_config, used_proxy_info, auth_controller);
}
void HttpStreamFactoryImpl::Request::OnNeedsClientAuth(
Job* job,
const SSLConfig& used_ssl_config,
SSLCertRequestInfo* cert_info) {
if (!bound_job_.get())
OrphanJobsExcept(job);
else
DCHECK(jobs_.empty());
delegate_->OnNeedsClientAuth(used_ssl_config, cert_info);
}
void HttpStreamFactoryImpl::Request::OnHttpsProxyTunnelResponse(
Job *job,
const HttpResponseInfo& response_info,
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpStreamBase* stream) {
if (!bound_job_.get())
OrphanJobsExcept(job);
else
DCHECK(jobs_.empty());
delegate_->OnHttpsProxyTunnelResponse(
response_info, used_ssl_config, used_proxy_info, stream);
}
int HttpStreamFactoryImpl::Request::RestartTunnelWithProxyAuth(
const AuthCredentials& credentials) {
DCHECK(bound_job_.get());
return bound_job_->RestartTunnelWithProxyAuth(credentials);
}
LoadState HttpStreamFactoryImpl::Request::GetLoadState() const {
if (bound_job_.get())
return bound_job_->GetLoadState();
DCHECK(!jobs_.empty());
// Just pick the first one.
return (*jobs_.begin())->GetLoadState();
}
bool HttpStreamFactoryImpl::Request::was_npn_negotiated() const {
DCHECK(completed_);
return was_npn_negotiated_;
}
NextProto HttpStreamFactoryImpl::Request::protocol_negotiated()
const {
DCHECK(completed_);
return protocol_negotiated_;
}
bool HttpStreamFactoryImpl::Request::using_spdy() const {
DCHECK(completed_);
return using_spdy_;
}
void
HttpStreamFactoryImpl::Request::RemoveRequestFromSpdySessionRequestMap() {
#if !defined(COBALT_DISABLE_SPDY)
if (spdy_session_key_.get()) {
SpdySessionRequestMap& spdy_session_request_map =
factory_->spdy_session_request_map_;
DCHECK(ContainsKey(spdy_session_request_map, *spdy_session_key_));
RequestSet& request_set =
spdy_session_request_map[*spdy_session_key_];
DCHECK(ContainsKey(request_set, this));
request_set.erase(this);
if (request_set.empty())
spdy_session_request_map.erase(*spdy_session_key_);
spdy_session_key_.reset();
}
#endif
}
void
HttpStreamFactoryImpl::Request::RemoveRequestFromHttpPipeliningRequestMap() {
#if !defined(COBALT_DISABLE_SPDY)
if (http_pipelining_key_.get()) {
HttpPipeliningRequestMap& http_pipelining_request_map =
factory_->http_pipelining_request_map_;
DCHECK(ContainsKey(http_pipelining_request_map, *http_pipelining_key_));
RequestVector& request_vector =
http_pipelining_request_map[*http_pipelining_key_];
for (RequestVector::iterator it = request_vector.begin();
it != request_vector.end(); ++it) {
if (*it == this) {
request_vector.erase(it);
break;
}
}
if (request_vector.empty())
http_pipelining_request_map.erase(*http_pipelining_key_);
http_pipelining_key_.reset();
}
#endif
}
bool HttpStreamFactoryImpl::Request::HasSpdySessionKey() const {
return spdy_session_key_.get() != NULL;
}
void HttpStreamFactoryImpl::Request::OnSpdySessionReady(
Job* job,
scoped_refptr<SpdySession> spdy_session,
bool direct) {
#if defined(COBALT_DISABLE_SPDY)
NOTREACHED();
#else
DCHECK(job);
DCHECK(job->using_spdy());
// The first case is the usual case.
if (!bound_job_.get()) {
OrphanJobsExcept(job);
} else { // This is the case for HTTPS proxy tunneling.
DCHECK_EQ(bound_job_.get(), job);
DCHECK(jobs_.empty());
}
// Cache these values in case the job gets deleted.
const SSLConfig used_ssl_config = job->server_ssl_config();
const ProxyInfo used_proxy_info = job->proxy_info();
const bool was_npn_negotiated = job->was_npn_negotiated();
const NextProto protocol_negotiated =
job->protocol_negotiated();
const bool using_spdy = job->using_spdy();
const BoundNetLog net_log = job->net_log();
Complete(was_npn_negotiated, protocol_negotiated, using_spdy, net_log);
// Cache this so we can still use it if the request is deleted.
HttpStreamFactoryImpl* factory = factory_;
bool use_relative_url = direct || url().SchemeIs("https");
delegate_->OnStreamReady(
job->server_ssl_config(),
job->proxy_info(),
new SpdyHttpStream(spdy_session, use_relative_url));
// |this| may be deleted after this point.
factory->OnSpdySessionReady(
spdy_session, direct, used_ssl_config, used_proxy_info,
was_npn_negotiated, protocol_negotiated, using_spdy, net_log);
#endif // defined(COBALT_DISABLE_SPDY)
}
void HttpStreamFactoryImpl::Request::OrphanJobsExcept(Job* job) {
DCHECK(job);
DCHECK(!bound_job_.get());
DCHECK(ContainsKey(jobs_, job));
bound_job_.reset(job);
jobs_.erase(job);
factory_->request_map_.erase(job);
OrphanJobs();
}
void HttpStreamFactoryImpl::Request::OrphanJobs() {
RemoveRequestFromSpdySessionRequestMap();
RemoveRequestFromHttpPipeliningRequestMap();
std::set<Job*> tmp;
tmp.swap(jobs_);
for (std::set<Job*>::iterator it = tmp.begin(); it != tmp.end(); ++it)
factory_->OrphanJob(*it, this);
}
} // namespace net