1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "net/tools/quic/quic_spdy_server_stream.h"
7 #include "base/strings/string_number_conversions.h"
8 #include "base/strings/string_piece.h"
9 #include "net/quic/quic_connection.h"
10 #include "net/quic/quic_protocol.h"
11 #include "net/quic/quic_utils.h"
12 #include "net/quic/test_tools/quic_test_utils.h"
13 #include "net/tools/epoll_server/epoll_server.h"
14 #include "net/tools/quic/quic_in_memory_cache.h"
15 #include "net/tools/quic/spdy_utils.h"
16 #include "net/tools/quic/test_tools/quic_in_memory_cache_peer.h"
17 #include "net/tools/quic/test_tools/quic_test_utils.h"
18 #include "testing/gmock/include/gmock/gmock.h"
19 #include "testing/gtest/include/gtest/gtest.h"
21 using base::StringPiece
;
22 using net::test::MockSession
;
23 using net::test::SupportedVersions
;
24 using net::tools::test::MockConnection
;
27 using testing::AnyNumber
;
28 using testing::Invoke
;
29 using testing::InvokeArgument
;
30 using testing::InSequence
;
31 using testing::Return
;
32 using testing::StrictMock
;
33 using testing::WithArgs
;
39 class QuicSpdyServerStreamPeer
: public QuicSpdyServerStream
{
41 QuicSpdyServerStreamPeer(QuicStreamId stream_id
, QuicSession
* session
)
42 : QuicSpdyServerStream(stream_id
, session
) {
45 using QuicSpdyServerStream::SendResponse
;
46 using QuicSpdyServerStream::SendErrorResponse
;
48 BalsaHeaders
* mutable_headers() {
52 static void SendResponse(QuicSpdyServerStream
* stream
) {
53 stream
->SendResponse();
56 static void SendErrorResponse(QuicSpdyServerStream
* stream
) {
57 stream
->SendResponse();
60 static const string
& body(QuicSpdyServerStream
* stream
) {
64 static const BalsaHeaders
& headers(QuicSpdyServerStream
* stream
) {
65 return stream
->headers_
;
71 class QuicSpdyServerStreamTest
: public ::testing::TestWithParam
<QuicVersion
> {
73 QuicSpdyServerStreamTest()
74 : connection_(new StrictMock
<MockConnection
>(
75 true, SupportedVersions(GetParam()))),
76 session_(connection_
),
77 body_("hello world") {
78 BalsaHeaders request_headers
;
79 request_headers
.SetRequestFirstlineFromStringPieces(
80 "POST", "https://www.google.com/", "HTTP/1.1");
81 request_headers
.ReplaceOrAppendHeader("content-length", "11");
83 headers_string_
= SpdyUtils::SerializeRequestHeaders(request_headers
);
85 // New streams rely on having the peer's flow control receive window
86 // negotiated in the config.
87 session_
.config()->SetInitialFlowControlWindowToSend(
88 kInitialSessionFlowControlWindowForTest
);
89 session_
.config()->SetInitialStreamFlowControlWindowToSend(
90 kInitialStreamFlowControlWindowForTest
);
91 session_
.config()->SetInitialSessionFlowControlWindowToSend(
92 kInitialSessionFlowControlWindowForTest
);
93 stream_
.reset(new QuicSpdyServerStreamPeer(3, &session_
));
96 static void SetUpTestCase() {
97 QuicInMemoryCachePeer::ResetForTests();
100 virtual void SetUp() override
{
101 QuicInMemoryCache
* cache
= QuicInMemoryCache::GetInstance();
103 BalsaHeaders request_headers
, response_headers
;
104 StringPiece
body("Yum");
105 request_headers
.SetRequestFirstlineFromStringPieces(
107 "https://www.google.com/foo",
109 response_headers
.SetRequestFirstlineFromStringPieces("HTTP/1.1",
112 response_headers
.AppendHeader("content-length",
113 base::IntToString(body
.length()));
115 // Check if response already exists and matches.
116 const QuicInMemoryCache::Response
* cached_response
=
117 cache
->GetResponse(request_headers
);
118 if (cached_response
!= nullptr) {
119 string cached_response_headers_str
, response_headers_str
;
120 cached_response
->headers().DumpToString(&cached_response_headers_str
);
121 response_headers
.DumpToString(&response_headers_str
);
122 CHECK_EQ(cached_response_headers_str
, response_headers_str
);
123 CHECK_EQ(cached_response
->body(), body
);
127 cache
->AddResponse(request_headers
, response_headers
, body
);
130 const string
& StreamBody() {
131 return QuicSpdyServerStreamPeer::body(stream_
.get());
134 const BalsaHeaders
& StreamHeaders() {
135 return QuicSpdyServerStreamPeer::headers(stream_
.get());
138 BalsaHeaders response_headers_
;
140 StrictMock
<MockConnection
>* connection_
;
141 StrictMock
<MockSession
> session_
;
142 scoped_ptr
<QuicSpdyServerStreamPeer
> stream_
;
143 string headers_string_
;
147 QuicConsumedData
ConsumeAllData(
149 const IOVector
& data
,
150 QuicStreamOffset offset
,
152 FecProtection
/*fec_protection_*/,
153 QuicAckNotifier::DelegateInterface
* /*ack_notifier_delegate*/) {
154 return QuicConsumedData(data
.TotalBufferSize(), fin
);
157 INSTANTIATE_TEST_CASE_P(Tests
, QuicSpdyServerStreamTest
,
158 ::testing::ValuesIn(QuicSupportedVersions()));
160 TEST_P(QuicSpdyServerStreamTest
, TestFraming
) {
161 EXPECT_CALL(session_
, WritevData(_
, _
, _
, _
, _
, _
)).Times(AnyNumber()).
162 WillRepeatedly(Invoke(ConsumeAllData
));
164 EXPECT_EQ(headers_string_
.size(), stream_
->ProcessData(
165 headers_string_
.c_str(), headers_string_
.size()));
166 EXPECT_EQ(body_
.size(), stream_
->ProcessData(body_
.c_str(), body_
.size()));
167 EXPECT_EQ(11u, StreamHeaders().content_length());
168 EXPECT_EQ("https://www.google.com/", StreamHeaders().request_uri());
169 EXPECT_EQ("POST", StreamHeaders().request_method());
170 EXPECT_EQ(body_
, StreamBody());
173 TEST_P(QuicSpdyServerStreamTest
, TestFramingOnePacket
) {
174 EXPECT_CALL(session_
, WritevData(_
, _
, _
, _
, _
, _
)).Times(AnyNumber()).
175 WillRepeatedly(Invoke(ConsumeAllData
));
177 string message
= headers_string_
+ body_
;
179 EXPECT_EQ(message
.size(), stream_
->ProcessData(
180 message
.c_str(), message
.size()));
181 EXPECT_EQ(11u, StreamHeaders().content_length());
182 EXPECT_EQ("https://www.google.com/", StreamHeaders().request_uri());
183 EXPECT_EQ("POST", StreamHeaders().request_method());
184 EXPECT_EQ(body_
, StreamBody());
187 TEST_P(QuicSpdyServerStreamTest
, TestFramingExtraData
) {
188 string large_body
= "hello world!!!!!!";
190 // We'll automatically write out an error (headers + body)
191 EXPECT_CALL(session_
, WritevData(_
, _
, _
, _
, _
, _
)).Times(AnyNumber()).
192 WillRepeatedly(Invoke(ConsumeAllData
));
194 EXPECT_EQ(headers_string_
.size(), stream_
->ProcessData(
195 headers_string_
.c_str(), headers_string_
.size()));
196 // Content length is still 11. This will register as an error and we won't
198 stream_
->ProcessData(large_body
.c_str(), large_body
.size());
199 EXPECT_EQ(11u, StreamHeaders().content_length());
200 EXPECT_EQ("https://www.google.com/", StreamHeaders().request_uri());
201 EXPECT_EQ("POST", StreamHeaders().request_method());
204 TEST_P(QuicSpdyServerStreamTest
, TestSendResponse
) {
205 BalsaHeaders
* request_headers
= stream_
->mutable_headers();
206 request_headers
->SetRequestFirstlineFromStringPieces(
208 "https://www.google.com/foo",
211 response_headers_
.SetResponseFirstlineFromStringPieces(
212 "HTTP/1.1", "200", "OK");
213 response_headers_
.ReplaceOrAppendHeader("content-length", "3");
216 EXPECT_CALL(session_
, WritevData(kHeadersStreamId
, _
, 0, false, _
, nullptr));
217 EXPECT_CALL(session_
, WritevData(_
, _
, _
, _
, _
, _
)).Times(1).
218 WillOnce(Return(QuicConsumedData(3, true)));
220 QuicSpdyServerStreamPeer::SendResponse(stream_
.get());
221 EXPECT_TRUE(stream_
->read_side_closed());
222 EXPECT_TRUE(stream_
->write_side_closed());
225 TEST_P(QuicSpdyServerStreamTest
, TestSendErrorResponse
) {
226 response_headers_
.SetResponseFirstlineFromStringPieces(
227 "HTTP/1.1", "500", "Server Error");
228 response_headers_
.ReplaceOrAppendHeader("content-length", "3");
231 EXPECT_CALL(session_
, WritevData(kHeadersStreamId
, _
, 0, false, _
, nullptr));
232 EXPECT_CALL(session_
, WritevData(_
, _
, _
, _
, _
, _
)).Times(1).
233 WillOnce(Return(QuicConsumedData(3, true)));
235 QuicSpdyServerStreamPeer::SendErrorResponse(stream_
.get());
236 EXPECT_TRUE(stream_
->read_side_closed());
237 EXPECT_TRUE(stream_
->write_side_closed());
240 TEST_P(QuicSpdyServerStreamTest
, InvalidHeadersWithFin
) {
242 0x3a, 0x68, 0x6f, 0x73, // :hos
243 0x74, 0x00, 0x00, 0x00, // t...
244 0x00, 0x00, 0x00, 0x00, // ....
245 0x07, 0x3a, 0x6d, 0x65, // .:me
246 0x74, 0x68, 0x6f, 0x64, // thod
247 0x00, 0x00, 0x00, 0x03, // ....
248 0x47, 0x45, 0x54, 0x00, // GET.
249 0x00, 0x00, 0x05, 0x3a, // ...:
250 0x70, 0x61, 0x74, 0x68, // path
251 0x00, 0x00, 0x00, 0x04, // ....
252 0x2f, 0x66, 0x6f, 0x6f, // /foo
253 0x00, 0x00, 0x00, 0x07, // ....
254 0x3a, 0x73, 0x63, 0x68, // :sch
255 0x65, 0x6d, 0x65, 0x00, // eme.
256 0x00, 0x00, 0x00, 0x00, // ....
257 0x00, 0x00, 0x08, 0x3a, // ...:
258 0x76, 0x65, 0x72, 0x73, // vers
259 '\x96', 0x6f, 0x6e, 0x00, // <i(69)>on.
260 0x00, 0x00, 0x08, 0x48, // ...H
261 0x54, 0x54, 0x50, 0x2f, // TTP/
262 0x31, 0x2e, 0x31, // 1.1
264 StringPiece
data(arr
, arraysize(arr
));
265 QuicStreamFrame
frame(stream_
->id(), true, 0, MakeIOVector(data
));
266 // Verify that we don't crash when we get a invalid headers in stream frame.
267 stream_
->OnStreamFrame(frame
);