// 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/url_request/url_request_http_job.h" #include #include "base/compiler_specific.h" #include "base/memory/ref_counted.h" #include "base/run_loop.h" #include "net/base/auth.h" #include "net/base/request_priority.h" #include "net/http/http_transaction_factory.h" #include "net/http/http_transaction_test_util.h" #include "net/socket/socket_test_util.h" #include "net/url_request/url_request_status.h" #include "net/url_request/url_request_test_util.h" #include "net/websockets/websocket_handshake_stream_base.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" #include "url/gurl.h" namespace net { namespace { using ::testing::Return; // Inherit from URLRequestHttpJob to expose the priority and some // other hidden functions. class TestURLRequestHttpJob : public URLRequestHttpJob { public: explicit TestURLRequestHttpJob(URLRequest* request) : URLRequestHttpJob(request, request->context()->network_delegate(), request->context()->http_user_agent_settings()) {} using URLRequestHttpJob::SetPriority; using URLRequestHttpJob::Start; using URLRequestHttpJob::Kill; using URLRequestHttpJob::priority; protected: virtual ~TestURLRequestHttpJob() {} }; class URLRequestHttpJobTest : public ::testing::Test { protected: URLRequestHttpJobTest() : req_(GURL("http://www.example.com"), DEFAULT_PRIORITY, &delegate_, &context_) { context_.set_http_transaction_factory(&network_layer_); } MockNetworkLayer network_layer_; TestURLRequestContext context_; TestDelegate delegate_; TestURLRequest req_; }; // Make sure that SetPriority actually sets the URLRequestHttpJob's // priority, both before and after start. TEST_F(URLRequestHttpJobTest, SetPriorityBasic) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); EXPECT_EQ(DEFAULT_PRIORITY, job->priority()); job->SetPriority(LOWEST); EXPECT_EQ(LOWEST, job->priority()); job->SetPriority(LOW); EXPECT_EQ(LOW, job->priority()); job->Start(); EXPECT_EQ(LOW, job->priority()); job->SetPriority(MEDIUM); EXPECT_EQ(MEDIUM, job->priority()); } // Make sure that URLRequestHttpJob passes on its priority to its // transaction on start. TEST_F(URLRequestHttpJobTest, SetTransactionPriorityOnStart) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); job->SetPriority(LOW); EXPECT_FALSE(network_layer_.last_transaction()); job->Start(); ASSERT_TRUE(network_layer_.last_transaction()); EXPECT_EQ(LOW, network_layer_.last_transaction()->priority()); } // Make sure that URLRequestHttpJob passes on its priority updates to // its transaction. TEST_F(URLRequestHttpJobTest, SetTransactionPriority) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); job->SetPriority(LOW); job->Start(); ASSERT_TRUE(network_layer_.last_transaction()); EXPECT_EQ(LOW, network_layer_.last_transaction()->priority()); job->SetPriority(HIGHEST); EXPECT_EQ(HIGHEST, network_layer_.last_transaction()->priority()); } // Make sure that URLRequestHttpJob passes on its priority updates to // newly-created transactions after the first one. TEST_F(URLRequestHttpJobTest, SetSubsequentTransactionPriority) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); job->Start(); job->SetPriority(LOW); ASSERT_TRUE(network_layer_.last_transaction()); EXPECT_EQ(LOW, network_layer_.last_transaction()->priority()); job->Kill(); network_layer_.ClearLastTransaction(); // Creates a second transaction. job->Start(); ASSERT_TRUE(network_layer_.last_transaction()); EXPECT_EQ(LOW, network_layer_.last_transaction()->priority()); } // This base class just serves to set up some things before the TestURLRequest // constructor is called. class URLRequestHttpJobWebSocketTestBase : public ::testing::Test { protected: URLRequestHttpJobWebSocketTestBase() : socket_data_(NULL, 0, NULL, 0), context_(true) { // A Network Delegate is required for the WebSocketHandshakeStreamBase // object to be passed on to the HttpNetworkTransaction. context_.set_network_delegate(&network_delegate_); // Attempting to create real ClientSocketHandles is not going to work out so // well. Set up a fake socket factory. socket_factory_.AddSocketDataProvider(&socket_data_); context_.set_client_socket_factory(&socket_factory_); context_.Init(); } StaticSocketDataProvider socket_data_; TestNetworkDelegate network_delegate_; MockClientSocketFactory socket_factory_; TestURLRequestContext context_; }; class URLRequestHttpJobWebSocketTest : public URLRequestHttpJobWebSocketTestBase { protected: URLRequestHttpJobWebSocketTest() : req_(GURL("ws://www.example.com"), DEFAULT_PRIORITY, &delegate_, &context_) { // The TestNetworkDelegate expects a call to NotifyBeforeURLRequest before // anything else happens. GURL url("ws://localhost/"); TestCompletionCallback dummy; network_delegate_.NotifyBeforeURLRequest(&req_, dummy.callback(), &url); } TestDelegate delegate_; TestURLRequest req_; }; class MockCreateHelper : public WebSocketHandshakeStreamBase::CreateHelper { public: // GoogleMock does not appear to play nicely with move-only types like // scoped_ptr, so this forwarding method acts as a workaround. virtual WebSocketHandshakeStreamBase* CreateBasicStream( scoped_ptr connection, bool using_proxy) OVERRIDE { // Discard the arguments since we don't need them anyway. return CreateBasicStreamMock(); } MOCK_METHOD0(CreateBasicStreamMock, WebSocketHandshakeStreamBase*()); MOCK_METHOD2(CreateSpdyStream, WebSocketHandshakeStreamBase*(const base::WeakPtr&, bool)); }; class FakeWebSocketHandshakeStream : public WebSocketHandshakeStreamBase { public: FakeWebSocketHandshakeStream() : initialize_stream_was_called_(false) {} bool initialize_stream_was_called() const { return initialize_stream_was_called_; } // Fake implementation of HttpStreamBase methods. virtual int InitializeStream(const HttpRequestInfo* request_info, RequestPriority priority, const BoundNetLog& net_log, const CompletionCallback& callback) OVERRIDE { initialize_stream_was_called_ = true; return ERR_IO_PENDING; } virtual int SendRequest(const HttpRequestHeaders& request_headers, HttpResponseInfo* response, const CompletionCallback& callback) OVERRIDE { return ERR_IO_PENDING; } virtual int ReadResponseHeaders(const CompletionCallback& callback) OVERRIDE { return ERR_IO_PENDING; } virtual const HttpResponseInfo* GetResponseInfo() const OVERRIDE { return NULL; } virtual int ReadResponseBody(IOBuffer* buf, int buf_len, const CompletionCallback& callback) OVERRIDE { return ERR_IO_PENDING; } virtual void Close(bool not_reusable) OVERRIDE {} virtual bool IsResponseBodyComplete() const OVERRIDE { return false; } virtual bool CanFindEndOfResponse() const OVERRIDE { return false; } virtual bool IsConnectionReused() const OVERRIDE { return false; } virtual void SetConnectionReused() OVERRIDE {} virtual bool IsConnectionReusable() const OVERRIDE { return false; } virtual int64 GetTotalReceivedBytes() const OVERRIDE { return 0; } virtual bool GetLoadTimingInfo(LoadTimingInfo* load_timing_info) const OVERRIDE { return false; } virtual void GetSSLInfo(SSLInfo* ssl_info) OVERRIDE {} virtual void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info) OVERRIDE {} virtual bool IsSpdyHttpStream() const OVERRIDE { return false; } virtual void Drain(HttpNetworkSession* session) OVERRIDE {} virtual void SetPriority(RequestPriority priority) OVERRIDE {} // Fake implementation of WebSocketHandshakeStreamBase method(s) virtual scoped_ptr Upgrade() OVERRIDE { return scoped_ptr(); } virtual std::string GetFailureMessage() const OVERRIDE { return std::string(); } private: bool initialize_stream_was_called_; }; TEST_F(URLRequestHttpJobWebSocketTest, RejectedWithoutCreateHelper) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); job->Start(); base::RunLoop().RunUntilIdle(); EXPECT_EQ(URLRequestStatus::FAILED, req_.status().status()); EXPECT_EQ(ERR_DISALLOWED_URL_SCHEME, req_.status().error()); } TEST_F(URLRequestHttpJobWebSocketTest, CreateHelperPassedThrough) { scoped_refptr job(new TestURLRequestHttpJob(&req_)); scoped_ptr create_helper( new ::testing::StrictMock()); FakeWebSocketHandshakeStream* fake_handshake_stream( new FakeWebSocketHandshakeStream); // Ownership of fake_handshake_stream is transferred when CreateBasicStream() // is called. EXPECT_CALL(*create_helper, CreateBasicStreamMock()) .WillOnce(Return(fake_handshake_stream)); req_.SetUserData(WebSocketHandshakeStreamBase::CreateHelper::DataKey(), create_helper.release()); req_.SetLoadFlags(LOAD_DISABLE_CACHE); job->Start(); base::RunLoop().RunUntilIdle(); EXPECT_EQ(URLRequestStatus::IO_PENDING, req_.status().status()); EXPECT_TRUE(fake_handshake_stream->initialize_stream_was_called()); } } // namespace } // namespace net