diff options
author | asanka@chromium.org <asanka@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-05-13 14:41:38 +0000 |
---|---|---|
committer | asanka@chromium.org <asanka@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-05-13 14:41:38 +0000 |
commit | 4afa2c4afda77b5d68e6f5ba00fa6ba3e5a0f610 (patch) | |
tree | 6317240fd8b7832d66a962a94c21fe272ffc6c29 /net | |
parent | 05fce3e98c2d81bac184c456988d8d382f744f58 (diff) | |
download | chromium_src-4afa2c4afda77b5d68e6f5ba00fa6ba3e5a0f610.zip chromium_src-4afa2c4afda77b5d68e6f5ba00fa6ba3e5a0f610.tar.gz chromium_src-4afa2c4afda77b5d68e6f5ba00fa6ba3e5a0f610.tar.bz2 |
Merge 84257 - Re-connect if Keep-Alive connection has been closed by the time we get around to reusing it.
BUG=81437
TEST=net_unittests --gtest_filter=HttpNetworkTransactionTest.BasicAuthKeepAliveImpatientServer:*.BasicAuthKeepAliveNoBody:*.BasicAuthKeepAliveLargeBody && unit_tests --gtest_filter=*TransportClientSocketTest.IsConnected*
Review URL: http://codereview.chromium.org/6902165
TBR=asanka@chromium.org
Review URL: http://codereview.chromium.org/7012038
git-svn-id: svn://svn.chromium.org/chrome/branches/742/src@85266 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'net')
-rw-r--r-- | net/http/http_basic_stream.cc | 6 | ||||
-rw-r--r-- | net/http/http_basic_stream.h | 35 | ||||
-rw-r--r-- | net/http/http_network_transaction.cc | 7 | ||||
-rw-r--r-- | net/http/http_network_transaction_unittest.cc | 31 | ||||
-rw-r--r-- | net/http/http_response_body_drainer_unittest.cc | 36 | ||||
-rw-r--r-- | net/http/http_stream.h | 6 | ||||
-rw-r--r-- | net/http/http_stream_parser.cc | 6 | ||||
-rw-r--r-- | net/http/http_stream_parser.h | 4 | ||||
-rw-r--r-- | net/socket/socket_test_util.cc | 8 | ||||
-rw-r--r-- | net/socket/socket_test_util.h | 1 | ||||
-rw-r--r-- | net/socket/transport_client_socket_unittest.cc | 171 | ||||
-rw-r--r-- | net/spdy/spdy_http_stream.cc | 7 | ||||
-rw-r--r-- | net/spdy/spdy_http_stream.h | 48 |
13 files changed, 245 insertions, 121 deletions
diff --git a/net/http/http_basic_stream.cc b/net/http/http_basic_stream.cc index ef4d777..6501a59 100644 --- a/net/http/http_basic_stream.cc +++ b/net/http/http_basic_stream.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2006-2009 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. @@ -102,6 +102,10 @@ void HttpBasicStream::SetConnectionReused() { parser_->SetConnectionReused(); } +bool HttpBasicStream::IsConnectionReusable() const { + return parser_->IsConnectionReusable(); +} + void HttpBasicStream::GetSSLInfo(SSLInfo* ssl_info) { parser_->GetSSLInfo(ssl_info); } diff --git a/net/http/http_basic_stream.h b/net/http/http_basic_stream.h index 2c26315..267c7c1 100644 --- a/net/http/http_basic_stream.h +++ b/net/http/http_basic_stream.h @@ -42,41 +42,44 @@ class HttpBasicStream : public HttpStream { // HttpStream methods: virtual int InitializeStream(const HttpRequestInfo* request_info, const BoundNetLog& net_log, - CompletionCallback* callback); + CompletionCallback* callback) OVERRIDE; virtual int SendRequest(const HttpRequestHeaders& headers, UploadDataStream* request_body, HttpResponseInfo* response, - CompletionCallback* callback); + CompletionCallback* callback) OVERRIDE; - virtual uint64 GetUploadProgress() const; + virtual uint64 GetUploadProgress() const OVERRIDE; - virtual int ReadResponseHeaders(CompletionCallback* callback); + virtual int ReadResponseHeaders(CompletionCallback* callback) OVERRIDE; - virtual const HttpResponseInfo* GetResponseInfo() const; + virtual const HttpResponseInfo* GetResponseInfo() const OVERRIDE; virtual int ReadResponseBody(IOBuffer* buf, int buf_len, - CompletionCallback* callback); + CompletionCallback* callback) OVERRIDE; - virtual void Close(bool not_reusable); + virtual void Close(bool not_reusable) OVERRIDE; - virtual HttpStream* RenewStreamForAuth(); + virtual HttpStream* RenewStreamForAuth() OVERRIDE; - virtual bool IsResponseBodyComplete() const; + virtual bool IsResponseBodyComplete() const OVERRIDE; - virtual bool CanFindEndOfResponse() const; + virtual bool CanFindEndOfResponse() const OVERRIDE; - virtual bool IsMoreDataBuffered() const; + virtual bool IsMoreDataBuffered() const OVERRIDE; - virtual bool IsConnectionReused() const; + virtual bool IsConnectionReused() const OVERRIDE; - virtual void SetConnectionReused(); + virtual void SetConnectionReused() OVERRIDE; - virtual void GetSSLInfo(SSLInfo* ssl_info); + virtual bool IsConnectionReusable() const OVERRIDE; - virtual void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info); + virtual void GetSSLInfo(SSLInfo* ssl_info) OVERRIDE; - virtual bool IsSpdyHttpStream() const; + virtual void GetSSLCertRequestInfo( + SSLCertRequestInfo* cert_request_info) OVERRIDE; + + virtual bool IsSpdyHttpStream() const OVERRIDE; private: scoped_refptr<GrowableIOBuffer> read_buf_; diff --git a/net/http/http_network_transaction.cc b/net/http/http_network_transaction.cc index 5651a41..55330e2 100644 --- a/net/http/http_network_transaction.cc +++ b/net/http/http_network_transaction.cc @@ -273,7 +273,7 @@ void HttpNetworkTransaction::DidDrainBodyForAuthRestart(bool keep_alive) { if (stream_.get()) { HttpStream* new_stream = NULL; - if (keep_alive) { + if (keep_alive && stream_->IsConnectionReusable()) { // We should call connection_->set_idle_time(), but this doesn't occur // often enough to be worth the trouble. stream_->SetConnectionReused(); @@ -281,7 +281,10 @@ void HttpNetworkTransaction::DidDrainBodyForAuthRestart(bool keep_alive) { } if (!new_stream) { - stream_->Close(!keep_alive); + // Close the stream and mark it as not_reusable. Even in the + // keep_alive case, we've determined that the stream_ is not + // reusable if new_stream is NULL. + stream_->Close(true); next_state_ = STATE_CREATE_STREAM; } else { next_state_ = STATE_INIT_STREAM; diff --git a/net/http/http_network_transaction_unittest.cc b/net/http/http_network_transaction_unittest.cc index db490c1..b614ec8 100644 --- a/net/http/http_network_transaction_unittest.cc +++ b/net/http/http_network_transaction_unittest.cc @@ -1192,9 +1192,18 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAlive) { MockRead("Hello"), }; + // If there is a regression where we disconnect a Keep-Alive + // connection during an auth roundtrip, we'll end up reading this. + MockRead data_reads2[] = { + MockRead(false, ERR_FAILED), + }; + StaticSocketDataProvider data1(data_reads1, arraysize(data_reads1), data_writes1, arraysize(data_writes1)); + StaticSocketDataProvider data2(data_reads2, arraysize(data_reads2), + NULL, 0); session_deps.socket_factory.AddSocketDataProvider(&data1); + session_deps.socket_factory.AddSocketDataProvider(&data2); TestCompletionCallback callback1; @@ -1224,7 +1233,7 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAlive) { EXPECT_EQ(OK, rv); response = trans->GetResponseInfo(); - EXPECT_FALSE(response == NULL); + ASSERT_FALSE(response == NULL); EXPECT_TRUE(response->auth_challenge.get() == NULL); EXPECT_EQ(5, response->headers->GetContentLength()); } @@ -1265,9 +1274,17 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAliveNoBody) { MockRead("hello"), }; + // An incorrect reconnect would cause this to be read. + MockRead data_reads2[] = { + MockRead(false, ERR_FAILED), + }; + StaticSocketDataProvider data1(data_reads1, arraysize(data_reads1), data_writes1, arraysize(data_writes1)); + StaticSocketDataProvider data2(data_reads2, arraysize(data_reads2), + NULL, 0); session_deps.socket_factory.AddSocketDataProvider(&data1); + session_deps.socket_factory.AddSocketDataProvider(&data2); TestCompletionCallback callback1; @@ -1297,7 +1314,7 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAliveNoBody) { EXPECT_EQ(OK, rv); response = trans->GetResponseInfo(); - EXPECT_FALSE(response == NULL); + ASSERT_FALSE(response == NULL); EXPECT_TRUE(response->auth_challenge.get() == NULL); EXPECT_EQ(5, response->headers->GetContentLength()); } @@ -1346,9 +1363,17 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAliveLargeBody) { MockRead("hello"), }; + // An incorrect reconnect would cause this to be read. + MockRead data_reads2[] = { + MockRead(false, ERR_FAILED), + }; + StaticSocketDataProvider data1(data_reads1, arraysize(data_reads1), data_writes1, arraysize(data_writes1)); + StaticSocketDataProvider data2(data_reads2, arraysize(data_reads2), + NULL, 0); session_deps.socket_factory.AddSocketDataProvider(&data1); + session_deps.socket_factory.AddSocketDataProvider(&data2); TestCompletionCallback callback1; @@ -1378,7 +1403,7 @@ TEST_F(HttpNetworkTransactionTest, BasicAuthKeepAliveLargeBody) { EXPECT_EQ(OK, rv); response = trans->GetResponseInfo(); - EXPECT_FALSE(response == NULL); + ASSERT_FALSE(response == NULL); EXPECT_TRUE(response->auth_challenge.get() == NULL); EXPECT_EQ(5, response->headers->GetContentLength()); } diff --git a/net/http/http_response_body_drainer_unittest.cc b/net/http/http_response_body_drainer_unittest.cc index ea9d299..1bdecea 100644 --- a/net/http/http_response_body_drainer_unittest.cc +++ b/net/http/http_response_body_drainer_unittest.cc @@ -74,44 +74,48 @@ class MockHttpStream : public HttpStream { // HttpStream implementation: virtual int InitializeStream(const HttpRequestInfo* request_info, const BoundNetLog& net_log, - CompletionCallback* callback) { + CompletionCallback* callback) OVERRIDE { return ERR_UNEXPECTED; } virtual int SendRequest(const HttpRequestHeaders& request_headers, UploadDataStream* request_body, HttpResponseInfo* response, - CompletionCallback* callback) { + CompletionCallback* callback) OVERRIDE { return ERR_UNEXPECTED; } - virtual uint64 GetUploadProgress() const { return 0; } - virtual int ReadResponseHeaders(CompletionCallback* callback) { + virtual uint64 GetUploadProgress() const OVERRIDE { return 0; } + virtual int ReadResponseHeaders(CompletionCallback* callback) OVERRIDE { return ERR_UNEXPECTED; } - virtual const HttpResponseInfo* GetResponseInfo() const { return NULL; } + virtual const HttpResponseInfo* GetResponseInfo() const OVERRIDE { + return NULL; + } - virtual bool CanFindEndOfResponse() const { return true; } - virtual bool IsMoreDataBuffered() const { return false; } - virtual bool IsConnectionReused() const { return false; } - virtual void SetConnectionReused() {} - virtual void GetSSLInfo(SSLInfo* ssl_info) {} - virtual void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info) {} + virtual bool CanFindEndOfResponse() const OVERRIDE { return true; } + virtual bool IsMoreDataBuffered() const OVERRIDE { return false; } + virtual bool IsConnectionReused() const OVERRIDE { return false; } + virtual void SetConnectionReused() OVERRIDE {} + virtual bool IsConnectionReusable() const OVERRIDE { return false; } + virtual void GetSSLInfo(SSLInfo* ssl_info) OVERRIDE {} + virtual void GetSSLCertRequestInfo( + SSLCertRequestInfo* cert_request_info) OVERRIDE {} // Mocked API virtual int ReadResponseBody(IOBuffer* buf, int buf_len, - CompletionCallback* callback); - virtual void Close(bool not_reusable) { + CompletionCallback* callback) OVERRIDE; + virtual void Close(bool not_reusable) OVERRIDE { DCHECK(!closed_); closed_ = true; result_waiter_->set_result(not_reusable); } - virtual HttpStream* RenewStreamForAuth() { + virtual HttpStream* RenewStreamForAuth() OVERRIDE { return NULL; } - virtual bool IsResponseBodyComplete() const { return is_complete_; } + virtual bool IsResponseBodyComplete() const OVERRIDE { return is_complete_; } - virtual bool IsSpdyHttpStream() const { return false; } + virtual bool IsSpdyHttpStream() const OVERRIDE { return false; } // Methods to tweak/observer mock behavior: void StallReadsForever() { stall_reads_forever_ = true; } diff --git a/net/http/http_stream.h b/net/http/http_stream.h index e262038..fed64655 100644 --- a/net/http/http_stream.h +++ b/net/http/http_stream.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. // @@ -116,6 +116,10 @@ class HttpStream { virtual bool IsConnectionReused() const = 0; virtual void SetConnectionReused() = 0; + // Checks whether the current state of the underlying connection + // allows it to be reused. + virtual bool IsConnectionReusable() const = 0; + // Get the SSLInfo associated with this stream's connection. This should // only be called for streams over SSL sockets, otherwise the behavior is // undefined. diff --git a/net/http/http_stream_parser.cc b/net/http/http_stream_parser.cc index eb1ed35..0649bce 100644 --- a/net/http/http_stream_parser.cc +++ b/net/http/http_stream_parser.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. @@ -700,6 +700,10 @@ void HttpStreamParser::SetConnectionReused() { connection_->set_is_reused(true); } +bool HttpStreamParser::IsConnectionReusable() const { + return connection_->socket() && connection_->socket()->IsConnectedAndIdle(); +} + void HttpStreamParser::GetSSLInfo(SSLInfo* ssl_info) { if (request_->url.SchemeIs("https") && connection_->socket()) { SSLClientSocket* ssl_socket = diff --git a/net/http/http_stream_parser.h b/net/http/http_stream_parser.h index d9241a6..2192eff 100644 --- a/net/http/http_stream_parser.h +++ b/net/http/http_stream_parser.h @@ -1,4 +1,4 @@ -// Copyright (c) 2009 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. @@ -67,6 +67,8 @@ class HttpStreamParser : public ChunkCallback { void SetConnectionReused(); + bool IsConnectionReusable() const; + void GetSSLInfo(SSLInfo* ssl_info); void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info); diff --git a/net/socket/socket_test_util.cc b/net/socket/socket_test_util.cc index 577c7d2..ab10549 100644 --- a/net/socket/socket_test_util.cc +++ b/net/socket/socket_test_util.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. @@ -786,6 +786,12 @@ bool MockTCPClientSocket::IsConnectedAndIdle() const { return IsConnected(); } +int MockTCPClientSocket::GetPeerAddress(AddressList* address) const { + if (!IsConnected()) + return ERR_SOCKET_NOT_CONNECTED; + return MockClientSocket::GetPeerAddress(address); +} + bool MockTCPClientSocket::WasEverUsed() const { return was_used_to_convey_data_; } diff --git a/net/socket/socket_test_util.h b/net/socket/socket_test_util.h index 823c3b8..42d4a46 100644 --- a/net/socket/socket_test_util.h +++ b/net/socket/socket_test_util.h @@ -622,6 +622,7 @@ class MockTCPClientSocket : public MockClientSocket { virtual void Disconnect(); virtual bool IsConnected() const; virtual bool IsConnectedAndIdle() const; + virtual int GetPeerAddress(AddressList* address) const; virtual bool WasEverUsed() const; virtual bool UsingTCPFastOpen() const; diff --git a/net/socket/transport_client_socket_unittest.cc b/net/socket/transport_client_socket_unittest.cc index 26e0f14..95e7acd 100644 --- a/net/socket/transport_client_socket_unittest.cc +++ b/net/socket/transport_client_socket_unittest.cc @@ -36,7 +36,8 @@ class TransportClientSocketTest TransportClientSocketTest() : listen_port_(0), net_log_(CapturingNetLog::kUnbounded), - socket_factory_(ClientSocketFactory::GetDefaultFactory()) { + socket_factory_(ClientSocketFactory::GetDefaultFactory()), + close_server_socket_on_next_send_(false) { } ~TransportClientSocketTest() { @@ -48,9 +49,10 @@ class TransportClientSocketTest } virtual void DidRead(ListenSocket*, const char* str, int len) { // TODO(dkegel): this might not be long enough to tickle some bugs. - connected_sock_->Send(kServerReply, - arraysize(kServerReply) - 1, - false /* don't append line feed */); + connected_sock_->Send(kServerReply, arraysize(kServerReply) - 1, + false /* Don't append line feed */); + if (close_server_socket_on_next_send_) + CloseServerSocket(); } virtual void DidClose(ListenSocket* sock) {} @@ -70,6 +72,17 @@ class TransportClientSocketTest connected_sock_->ResumeReads(); } + int DrainClientSocket(IOBuffer* buf, + uint32 buf_len, + uint32 bytes_to_read, + TestCompletionCallback* callback); + + void SendClientRequest(); + + void set_close_server_socket_on_next_send(bool close) { + close_server_socket_on_next_send_ = close; + } + protected: int listen_port_; CapturingNetLog net_log_; @@ -79,6 +92,7 @@ class TransportClientSocketTest private: scoped_refptr<ListenSocket> listen_sock_; scoped_refptr<ListenSocket> connected_sock_; + bool close_server_socket_on_next_send_; }; void TransportClientSocketTest::SetUp() { @@ -115,6 +129,43 @@ void TransportClientSocketTest::SetUp() { NetLog::Source())); } +int TransportClientSocketTest::DrainClientSocket( + IOBuffer* buf, uint32 buf_len, + uint32 bytes_to_read, TestCompletionCallback* callback) { + int rv = OK; + uint32 bytes_read = 0; + + while (bytes_read < bytes_to_read) { + rv = sock_->Read(buf, buf_len, callback); + EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); + + if (rv == ERR_IO_PENDING) + rv = callback->WaitForResult(); + + EXPECT_GE(rv, 0); + bytes_read += rv; + } + + return static_cast<int>(bytes_read); +} + +void TransportClientSocketTest::SendClientRequest() { + const char request_text[] = "GET / HTTP/1.0\r\n\r\n"; + scoped_refptr<IOBuffer> request_buffer( + new IOBuffer(arraysize(request_text) - 1)); + TestCompletionCallback callback; + int rv; + + memcpy(request_buffer->data(), request_text, arraysize(request_text) - 1); + rv = sock_->Write(request_buffer, arraysize(request_text) - 1, &callback); + EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); + + if (rv == ERR_IO_PENDING) { + rv = callback.WaitForResult(); + EXPECT_EQ(rv, static_cast<int>(arraysize(request_text) - 1)); + } +} + // TODO(leighton): Add SCTP to this list when it is ready. INSTANTIATE_TEST_CASE_P(ClientSocket, TransportClientSocketTest, @@ -147,45 +198,79 @@ TEST_P(TransportClientSocketTest, Connect) { EXPECT_FALSE(sock_->IsConnected()); } -// TODO(wtc): Add unit tests for IsConnectedAndIdle: -// - Server closes a connection. -// - Server sends data unexpectedly. - -TEST_P(TransportClientSocketTest, Read) { +TEST_P(TransportClientSocketTest, IsConnected) { + scoped_refptr<IOBuffer> buf(new IOBuffer(4096)); TestCompletionCallback callback; + uint32 bytes_read; + + EXPECT_FALSE(sock_->IsConnected()); + EXPECT_FALSE(sock_->IsConnectedAndIdle()); int rv = sock_->Connect(&callback); if (rv != OK) { ASSERT_EQ(rv, ERR_IO_PENDING); - rv = callback.WaitForResult(); EXPECT_EQ(rv, OK); } + EXPECT_TRUE(sock_->IsConnected()); + EXPECT_TRUE(sock_->IsConnectedAndIdle()); - const char request_text[] = "GET / HTTP/1.0\r\n\r\n"; - scoped_refptr<IOBuffer> request_buffer( - new IOBuffer(arraysize(request_text) - 1)); - memcpy(request_buffer->data(), request_text, arraysize(request_text) - 1); + // Send the request and wait for the server to respond. + SendClientRequest(); - rv = sock_->Write(request_buffer, arraysize(request_text) - 1, &callback); - EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); + // Drain a single byte so we know we've received some data. + bytes_read = DrainClientSocket(buf, 1, 1, &callback); + ASSERT_EQ(bytes_read, 1u); - if (rv == ERR_IO_PENDING) { - rv = callback.WaitForResult(); - EXPECT_EQ(rv, static_cast<int>(arraysize(request_text) - 1)); - } + // Socket should be considered connected, but not idle, due to + // pending data. + EXPECT_TRUE(sock_->IsConnected()); + EXPECT_FALSE(sock_->IsConnectedAndIdle()); - scoped_refptr<IOBuffer> buf(new IOBuffer(4096)); - uint32 bytes_read = 0; - while (bytes_read < arraysize(kServerReply) - 1) { - rv = sock_->Read(buf, 4096, &callback); - EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); + bytes_read = DrainClientSocket(buf, 4096, arraysize(kServerReply) - 2, + &callback); + ASSERT_EQ(bytes_read, arraysize(kServerReply) - 2); - if (rv == ERR_IO_PENDING) - rv = callback.WaitForResult(); + // After draining the data, the socket should be back to connected + // and idle. + EXPECT_TRUE(sock_->IsConnected()); + EXPECT_TRUE(sock_->IsConnectedAndIdle()); - ASSERT_GE(rv, 0); - bytes_read += rv; + // This time close the server socket immediately after the server response. + set_close_server_socket_on_next_send(true); + SendClientRequest(); + + bytes_read = DrainClientSocket(buf, 1, 1, &callback); + ASSERT_EQ(bytes_read, 1u); + + // As above because of data. + EXPECT_TRUE(sock_->IsConnected()); + EXPECT_FALSE(sock_->IsConnectedAndIdle()); + + bytes_read = DrainClientSocket(buf, 4096, arraysize(kServerReply) - 2, + &callback); + ASSERT_EQ(bytes_read, arraysize(kServerReply) - 2); + + // Once the data is drained, the socket should now be seen as + // closed. + EXPECT_FALSE(sock_->IsConnected()); + EXPECT_FALSE(sock_->IsConnectedAndIdle()); +} + +TEST_P(TransportClientSocketTest, Read) { + TestCompletionCallback callback; + int rv = sock_->Connect(&callback); + if (rv != OK) { + ASSERT_EQ(rv, ERR_IO_PENDING); + + rv = callback.WaitForResult(); + EXPECT_EQ(rv, OK); } + SendClientRequest(); + + scoped_refptr<IOBuffer> buf(new IOBuffer(4096)); + uint32 bytes_read = DrainClientSocket(buf, 4096, arraysize(kServerReply) - 1, + &callback); + ASSERT_EQ(bytes_read, arraysize(kServerReply) - 1); // All data has been read now. Read once more to force an ERR_IO_PENDING, and // then close the server socket, and note the close. @@ -205,19 +290,7 @@ TEST_P(TransportClientSocketTest, Read_SmallChunks) { rv = callback.WaitForResult(); EXPECT_EQ(rv, OK); } - - const char request_text[] = "GET / HTTP/1.0\r\n\r\n"; - scoped_refptr<IOBuffer> request_buffer( - new IOBuffer(arraysize(request_text) - 1)); - memcpy(request_buffer->data(), request_text, arraysize(request_text) - 1); - - rv = sock_->Write(request_buffer, arraysize(request_text) - 1, &callback); - EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); - - if (rv == ERR_IO_PENDING) { - rv = callback.WaitForResult(); - EXPECT_EQ(rv, static_cast<int>(arraysize(request_text) - 1)); - } + SendClientRequest(); scoped_refptr<IOBuffer> buf(new IOBuffer(1)); uint32 bytes_read = 0; @@ -250,19 +323,7 @@ TEST_P(TransportClientSocketTest, Read_Interrupted) { rv = callback.WaitForResult(); EXPECT_EQ(rv, OK); } - - const char request_text[] = "GET / HTTP/1.0\r\n\r\n"; - scoped_refptr<IOBuffer> request_buffer( - new IOBuffer(arraysize(request_text) - 1)); - memcpy(request_buffer->data(), request_text, arraysize(request_text) - 1); - - rv = sock_->Write(request_buffer, arraysize(request_text) - 1, &callback); - EXPECT_TRUE(rv >= 0 || rv == ERR_IO_PENDING); - - if (rv == ERR_IO_PENDING) { - rv = callback.WaitForResult(); - EXPECT_EQ(rv, static_cast<int>(arraysize(request_text) - 1)); - } + SendClientRequest(); // Do a partial read and then exit. This test should not crash! scoped_refptr<IOBuffer> buf(new IOBuffer(16)); diff --git a/net/spdy/spdy_http_stream.cc b/net/spdy/spdy_http_stream.cc index 130a818..ac0c4ce 100644 --- a/net/spdy/spdy_http_stream.cc +++ b/net/spdy/spdy_http_stream.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Copyright (c) 2011 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. @@ -177,6 +177,11 @@ void SpdyHttpStream::SetConnectionReused() { // SPDY doesn't need an indicator here. } +bool SpdyHttpStream::IsConnectionReusable() const { + // SPDY streams aren't considered reusable. + return false; +} + void SpdyHttpStream::set_chunk_callback(ChunkCallback* callback) { if (request_body_stream_ != NULL) request_body_stream_->set_chunk_callback(callback); diff --git a/net/spdy/spdy_http_stream.h b/net/spdy/spdy_http_stream.h index 6bc5053..31f62f6 100644 --- a/net/spdy/spdy_http_stream.h +++ b/net/spdy/spdy_http_stream.h @@ -45,39 +45,41 @@ class SpdyHttpStream : public SpdyStream::Delegate, public HttpStream { // HttpStream methods: virtual int InitializeStream(const HttpRequestInfo* request_info, const BoundNetLog& net_log, - CompletionCallback* callback); + CompletionCallback* callback) OVERRIDE; virtual int SendRequest(const HttpRequestHeaders& headers, UploadDataStream* request_body, HttpResponseInfo* response, - CompletionCallback* callback); - virtual uint64 GetUploadProgress() const; - virtual int ReadResponseHeaders(CompletionCallback* callback); + CompletionCallback* callback) OVERRIDE; + virtual uint64 GetUploadProgress() const OVERRIDE; + virtual int ReadResponseHeaders(CompletionCallback* callback) OVERRIDE; virtual const HttpResponseInfo* GetResponseInfo() const; virtual int ReadResponseBody(IOBuffer* buf, int buf_len, - CompletionCallback* callback); - virtual void Close(bool not_reusable); - virtual HttpStream* RenewStreamForAuth(); - virtual bool IsResponseBodyComplete() const; - virtual bool CanFindEndOfResponse() const; - virtual bool IsMoreDataBuffered() const; - virtual bool IsConnectionReused() const; - virtual void SetConnectionReused(); - virtual void GetSSLInfo(SSLInfo* ssl_info); - virtual void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info); - virtual bool IsSpdyHttpStream() const; + CompletionCallback* callback) OVERRIDE; + virtual void Close(bool not_reusable) OVERRIDE; + virtual HttpStream* RenewStreamForAuth() OVERRIDE; + virtual bool IsResponseBodyComplete() const OVERRIDE; + virtual bool CanFindEndOfResponse() const OVERRIDE; + virtual bool IsMoreDataBuffered() const OVERRIDE; + virtual bool IsConnectionReused() const OVERRIDE; + virtual void SetConnectionReused() OVERRIDE; + virtual bool IsConnectionReusable() const OVERRIDE; + virtual void GetSSLInfo(SSLInfo* ssl_info) OVERRIDE; + virtual void GetSSLCertRequestInfo( + SSLCertRequestInfo* cert_request_info) OVERRIDE; + virtual bool IsSpdyHttpStream() const OVERRIDE; // SpdyStream::Delegate methods: - virtual bool OnSendHeadersComplete(int status); - virtual int OnSendBody(); - virtual int OnSendBodyComplete(int status, bool* eof); + virtual bool OnSendHeadersComplete(int status) OVERRIDE; + virtual int OnSendBody() OVERRIDE; + virtual int OnSendBodyComplete(int status, bool* eof) OVERRIDE; virtual int OnResponseReceived(const spdy::SpdyHeaderBlock& response, base::Time response_time, - int status); - virtual void OnDataReceived(const char* buffer, int bytes); - virtual void OnDataSent(int length); - virtual void OnClose(int status); - virtual void set_chunk_callback(ChunkCallback* callback); + int status) OVERRIDE; + virtual void OnDataReceived(const char* buffer, int bytes) OVERRIDE; + virtual void OnDataSent(int length) OVERRIDE; + virtual void OnClose(int status) OVERRIDE; + virtual void set_chunk_callback(ChunkCallback* callback) OVERRIDE; private: FRIEND_TEST_ALL_PREFIXES(SpdyNetworkTransactionTest, FlowControlStallResume); |