Updating trunk VERSION from 2139.0 to 2140.0
[chromium-blink-merge.git] / net / tools / quic / quic_spdy_server_stream_test.cc
blobe01569ede7913214ba117ae9fd8f1948e0448bbf
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;
25 using std::string;
26 using testing::_;
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;
35 namespace net {
36 namespace tools {
37 namespace test {
39 class QuicSpdyServerStreamPeer : public QuicSpdyServerStream {
40 public:
41 QuicSpdyServerStreamPeer(QuicStreamId stream_id, QuicSession* session)
42 : QuicSpdyServerStream(stream_id, session) {
45 using QuicSpdyServerStream::SendResponse;
46 using QuicSpdyServerStream::SendErrorResponse;
48 BalsaHeaders* mutable_headers() {
49 return &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) {
61 return stream->body_;
64 static const BalsaHeaders& headers(QuicSpdyServerStream* stream) {
65 return stream->headers_;
69 namespace {
71 class QuicSpdyServerStreamTest : public ::testing::TestWithParam<QuicVersion> {
72 public:
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(
106 "GET",
107 "https://www.google.com/foo",
108 "HTTP/1.1");
109 response_headers.SetRequestFirstlineFromStringPieces("HTTP/1.1",
110 "200",
111 "OK");
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 != NULL) {
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);
124 return;
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_;
139 EpollServer eps_;
140 StrictMock<MockConnection>* connection_;
141 StrictMock<MockSession> session_;
142 scoped_ptr<QuicSpdyServerStreamPeer> stream_;
143 string headers_string_;
144 string body_;
147 QuicConsumedData ConsumeAllData(
148 QuicStreamId id,
149 const IOVector& data,
150 QuicStreamOffset offset,
151 bool fin,
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
197 // accept the bytes.
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(
207 "GET",
208 "https://www.google.com/foo",
209 "HTTP/1.1");
211 response_headers_.SetResponseFirstlineFromStringPieces(
212 "HTTP/1.1", "200", "OK");
213 response_headers_.ReplaceOrAppendHeader("content-length", "3");
215 InSequence s;
216 EXPECT_CALL(session_,
217 WritevData(kHeadersStreamId, _, 0, false, _, NULL));
218 EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(1).
219 WillOnce(Return(QuicConsumedData(3, true)));
221 QuicSpdyServerStreamPeer::SendResponse(stream_.get());
222 EXPECT_TRUE(stream_->read_side_closed());
223 EXPECT_TRUE(stream_->write_side_closed());
226 TEST_P(QuicSpdyServerStreamTest, TestSendErrorResponse) {
227 response_headers_.SetResponseFirstlineFromStringPieces(
228 "HTTP/1.1", "500", "Server Error");
229 response_headers_.ReplaceOrAppendHeader("content-length", "3");
231 InSequence s;
232 EXPECT_CALL(session_,
233 WritevData(kHeadersStreamId, _, 0, false, _, NULL));
234 EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(1).
235 WillOnce(Return(QuicConsumedData(3, true)));
237 QuicSpdyServerStreamPeer::SendErrorResponse(stream_.get());
238 EXPECT_TRUE(stream_->read_side_closed());
239 EXPECT_TRUE(stream_->write_side_closed());
242 TEST_P(QuicSpdyServerStreamTest, InvalidHeadersWithFin) {
243 char arr[] = {
244 0x3a, 0x68, 0x6f, 0x73, // :hos
245 0x74, 0x00, 0x00, 0x00, // t...
246 0x00, 0x00, 0x00, 0x00, // ....
247 0x07, 0x3a, 0x6d, 0x65, // .:me
248 0x74, 0x68, 0x6f, 0x64, // thod
249 0x00, 0x00, 0x00, 0x03, // ....
250 0x47, 0x45, 0x54, 0x00, // GET.
251 0x00, 0x00, 0x05, 0x3a, // ...:
252 0x70, 0x61, 0x74, 0x68, // path
253 0x00, 0x00, 0x00, 0x04, // ....
254 0x2f, 0x66, 0x6f, 0x6f, // /foo
255 0x00, 0x00, 0x00, 0x07, // ....
256 0x3a, 0x73, 0x63, 0x68, // :sch
257 0x65, 0x6d, 0x65, 0x00, // eme.
258 0x00, 0x00, 0x00, 0x00, // ....
259 0x00, 0x00, 0x08, 0x3a, // ...:
260 0x76, 0x65, 0x72, 0x73, // vers
261 '\x96', 0x6f, 0x6e, 0x00, // <i(69)>on.
262 0x00, 0x00, 0x08, 0x48, // ...H
263 0x54, 0x54, 0x50, 0x2f, // TTP/
264 0x31, 0x2e, 0x31, // 1.1
266 StringPiece data(arr, arraysize(arr));
267 QuicStreamFrame frame(stream_->id(), true, 0, MakeIOVector(data));
268 // Verify that we don't crash when we get a invalid headers in stream frame.
269 stream_->OnStreamFrame(frame);
272 } // namespace
273 } // namespace test
274 } // namespace tools
275 } // namespace net