blob: 89833994829a13a11247477cbb50c5fd047975d7 [file] [log] [blame]
// Copyright (c) 2013 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/memory/scoped_ptr.h"
#include "net/http/http_stream_factory_impl_job.h"
#include "net/proxy/proxy_info.h"
#include "net/proxy/proxy_service.h"
#include "net/spdy/spdy_test_util_common.h"
#include "net/ssl/ssl_failure_state.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace net {
class HttpStreamFactoryImplRequestTest
: public ::testing::Test,
public ::testing::WithParamInterface<NextProto> {};
INSTANTIATE_TEST_CASE_P(NextProto,
HttpStreamFactoryImplRequestTest,
testing::Values(kProtoSPDY31,
kProtoHTTP2));
namespace {
class DoNothingRequestDelegate : public HttpStreamRequest::Delegate {
public:
DoNothingRequestDelegate() {}
~DoNothingRequestDelegate() override {}
// HttpStreamRequest::Delegate
void OnStreamReady(const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpStream* stream) override {}
void OnBidirectionalStreamJobReady(
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
BidirectionalStreamJob* stream_job) override {}
void OnWebSocketHandshakeStreamReady(
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
WebSocketHandshakeStreamBase* stream) override {}
void OnStreamFailed(int status,
const SSLConfig& used_ssl_config,
SSLFailureState ssl_failure_state) override {}
void OnCertificateError(int status,
const SSLConfig& used_ssl_config,
const SSLInfo& ssl_info) override {}
void OnNeedsProxyAuth(const HttpResponseInfo& proxy_response,
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpAuthController* auth_controller) override {}
void OnNeedsClientAuth(const SSLConfig& used_ssl_config,
SSLCertRequestInfo* cert_info) override {}
void OnHttpsProxyTunnelResponse(const HttpResponseInfo& response_info,
const SSLConfig& used_ssl_config,
const ProxyInfo& used_proxy_info,
HttpStream* stream) override {}
void OnQuicBroken() override {}
};
} // namespace
// Make sure that Request passes on its priority updates to its jobs.
TEST_P(HttpStreamFactoryImplRequestTest, SetPriority) {
SpdySessionDependencies session_deps(GetParam(),
ProxyService::CreateDirect());
scoped_ptr<HttpNetworkSession> session =
SpdySessionDependencies::SpdyCreateSession(&session_deps);
HttpStreamFactoryImpl* factory =
static_cast<HttpStreamFactoryImpl*>(session->http_stream_factory());
DoNothingRequestDelegate request_delegate;
HttpStreamFactoryImpl::Request request(
GURL(), factory, &request_delegate, NULL, BoundNetLog(),
HttpStreamFactoryImpl::Request::HTTP_STREAM);
HttpRequestInfo request_info;
HostPortPair server = HostPortPair::FromURL(request_info.url);
GURL original_url = factory->ApplyHostMappingRules(request_info.url, &server);
HttpStreamFactoryImpl::Job* job = new HttpStreamFactoryImpl::Job(
factory, session.get(), request_info, DEFAULT_PRIORITY, SSLConfig(),
SSLConfig(), server, original_url, NULL);
request.AttachJob(job);
EXPECT_EQ(DEFAULT_PRIORITY, job->priority());
request.SetPriority(MEDIUM);
EXPECT_EQ(MEDIUM, job->priority());
// Make |job| the bound job.
request.OnStreamFailed(job, ERR_FAILED, SSLConfig(), SSL_FAILURE_NONE);
request.SetPriority(IDLE);
EXPECT_EQ(IDLE, job->priority());
}
TEST_P(HttpStreamFactoryImplRequestTest, DelayMainJob) {
SpdySessionDependencies session_deps(GetParam(),
ProxyService::CreateDirect());
scoped_ptr<HttpNetworkSession> session =
SpdySessionDependencies::SpdyCreateSession(&session_deps);
StaticSocketDataProvider socket_data;
socket_data.set_connect_data(MockConnect(SYNCHRONOUS, ERR_IO_PENDING));
session_deps.socket_factory->AddSocketDataProvider(&socket_data);
HttpStreamFactoryImpl* factory =
static_cast<HttpStreamFactoryImpl*>(session->http_stream_factory());
DoNothingRequestDelegate request_delegate;
HttpRequestInfo request_info;
request_info.method = "GET";
request_info.url = GURL("http://www.google.com");
HttpStreamFactoryImpl::Request request(
request_info.url, factory, &request_delegate, NULL, BoundNetLog(),
HttpStreamFactoryImpl::Request::HTTP_STREAM);
HostPortPair server = HostPortPair::FromURL(request_info.url);
GURL original_url = factory->ApplyHostMappingRules(request_info.url, &server);
HttpStreamFactoryImpl::Job* job = new HttpStreamFactoryImpl::Job(
factory, session.get(), request_info, DEFAULT_PRIORITY, SSLConfig(),
SSLConfig(), server, original_url, NULL);
request.AttachJob(job);
EXPECT_EQ(DEFAULT_PRIORITY, job->priority());
AlternativeService alternative_service(net::NPN_HTTP_2, server);
HttpStreamFactoryImpl::Job* alternative_job = new HttpStreamFactoryImpl::Job(
factory, session.get(), request_info, DEFAULT_PRIORITY, SSLConfig(),
SSLConfig(), server, original_url, alternative_service, NULL);
request.AttachJob(alternative_job);
job->WaitFor(alternative_job);
EXPECT_EQ(HttpStreamFactoryImpl::Job::STATE_NONE, job->next_state_);
// Test |alternative_job| resuming the |job| after delay.
int wait_time = 1;
base::TimeDelta delay = base::TimeDelta::FromMilliseconds(wait_time);
job->Resume(alternative_job, delay);
// Verify |job| has |wait_time_| and there is no |blocking_job_|
EXPECT_EQ(delay, job->wait_time_);
EXPECT_TRUE(!job->blocking_job_);
// Start the |job| and verify |job|'s |wait_time_| is cleared.
job->Start(&request);
base::PlatformThread::Sleep(base::TimeDelta::FromMilliseconds(wait_time + 1));
base::MessageLoop::current()->RunUntilIdle();
EXPECT_NE(delay, job->wait_time_);
EXPECT_TRUE(job->wait_time_.is_zero());
EXPECT_EQ(HttpStreamFactoryImpl::Job::STATE_INIT_CONNECTION_COMPLETE,
job->next_state_);
}
} // namespace net