Update To 11.40.268.0
[platform/framework/web/crosswalk.git] / src / net / tools / quic / quic_spdy_server_stream_test.cc
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.
4
5 #include "net/tools/quic/quic_spdy_server_stream.h"
6
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"
20
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;
34
35 namespace net {
36 namespace tools {
37 namespace test {
38
39 class QuicSpdyServerStreamPeer : public QuicSpdyServerStream {
40  public:
41   QuicSpdyServerStreamPeer(QuicStreamId stream_id, QuicSession* session)
42       : QuicSpdyServerStream(stream_id, session) {
43   }
44
45   using QuicSpdyServerStream::SendResponse;
46   using QuicSpdyServerStream::SendErrorResponse;
47
48   BalsaHeaders* mutable_headers() {
49     return &headers_;
50   }
51
52   static void SendResponse(QuicSpdyServerStream* stream) {
53     stream->SendResponse();
54   }
55
56   static void SendErrorResponse(QuicSpdyServerStream* stream) {
57     stream->SendResponse();
58   }
59
60   static const string& body(QuicSpdyServerStream* stream) {
61     return stream->body_;
62   }
63
64   static const BalsaHeaders& headers(QuicSpdyServerStream* stream) {
65     return stream->headers_;
66   }
67 };
68
69 namespace {
70
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");
82
83     headers_string_ = SpdyUtils::SerializeRequestHeaders(request_headers);
84
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_));
94   }
95
96   static void SetUpTestCase() {
97     QuicInMemoryCachePeer::ResetForTests();
98   }
99
100   virtual void SetUp() override {
101     QuicInMemoryCache* cache = QuicInMemoryCache::GetInstance();
102
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()));
114
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);
124       return;
125     }
126
127     cache->AddResponse(request_headers, response_headers, body);
128   }
129
130   const string& StreamBody() {
131     return QuicSpdyServerStreamPeer::body(stream_.get());
132   }
133
134   const BalsaHeaders& StreamHeaders() {
135     return QuicSpdyServerStreamPeer::headers(stream_.get());
136   }
137
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_;
145 };
146
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);
155 }
156
157 INSTANTIATE_TEST_CASE_P(Tests, QuicSpdyServerStreamTest,
158                         ::testing::ValuesIn(QuicSupportedVersions()));
159
160 TEST_P(QuicSpdyServerStreamTest, TestFraming) {
161   EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(AnyNumber()).
162       WillRepeatedly(Invoke(ConsumeAllData));
163
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());
171 }
172
173 TEST_P(QuicSpdyServerStreamTest, TestFramingOnePacket) {
174   EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(AnyNumber()).
175       WillRepeatedly(Invoke(ConsumeAllData));
176
177   string message = headers_string_ + body_;
178
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());
185 }
186
187 TEST_P(QuicSpdyServerStreamTest, TestFramingExtraData) {
188   string large_body = "hello world!!!!!!";
189
190   // We'll automatically write out an error (headers + body)
191   EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(AnyNumber()).
192       WillRepeatedly(Invoke(ConsumeAllData));
193
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());
202 }
203
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");
210
211   response_headers_.SetResponseFirstlineFromStringPieces(
212       "HTTP/1.1", "200", "OK");
213   response_headers_.ReplaceOrAppendHeader("content-length", "3");
214
215   InSequence s;
216   EXPECT_CALL(session_, WritevData(kHeadersStreamId, _, 0, false, _, nullptr));
217   EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(1).
218       WillOnce(Return(QuicConsumedData(3, true)));
219
220   QuicSpdyServerStreamPeer::SendResponse(stream_.get());
221   EXPECT_TRUE(stream_->read_side_closed());
222   EXPECT_TRUE(stream_->write_side_closed());
223 }
224
225 TEST_P(QuicSpdyServerStreamTest, TestSendErrorResponse) {
226   response_headers_.SetResponseFirstlineFromStringPieces(
227       "HTTP/1.1", "500", "Server Error");
228   response_headers_.ReplaceOrAppendHeader("content-length", "3");
229
230   InSequence s;
231   EXPECT_CALL(session_, WritevData(kHeadersStreamId, _, 0, false, _, nullptr));
232   EXPECT_CALL(session_, WritevData(_, _, _, _, _, _)).Times(1).
233       WillOnce(Return(QuicConsumedData(3, true)));
234
235   QuicSpdyServerStreamPeer::SendErrorResponse(stream_.get());
236   EXPECT_TRUE(stream_->read_side_closed());
237   EXPECT_TRUE(stream_->write_side_closed());
238 }
239
240 TEST_P(QuicSpdyServerStreamTest, InvalidHeadersWithFin) {
241   char arr[] = {
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
263   };
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);
268 }
269
270 }  // namespace
271 }  // namespace test
272 }  // namespace tools
273 }  // namespace net