blob: 7d8e06d0d00e75adf0a8a656cc6d7f3455a52cb7 [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// Copyright 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/third_party/quiche/src/quic/core/http/quic_spdy_stream.h"
6
7#include <memory>
vasilvv872e7a32019-03-12 16:42:44 -07008#include <string>
QUICHE teama6ef0a62019-03-07 20:34:33 -05009#include <utility>
10
11#include "net/third_party/quiche/src/quic/core/http/http_encoder.h"
12#include "net/third_party/quiche/src/quic/core/http/spdy_utils.h"
13#include "net/third_party/quiche/src/quic/core/quic_connection.h"
14#include "net/third_party/quiche/src/quic/core/quic_stream_sequencer_buffer.h"
15#include "net/third_party/quiche/src/quic/core/quic_utils.h"
16#include "net/third_party/quiche/src/quic/core/quic_versions.h"
17#include "net/third_party/quiche/src/quic/core/quic_write_blocked_list.h"
18#include "net/third_party/quiche/src/quic/platform/api/quic_arraysize.h"
19#include "net/third_party/quiche/src/quic/platform/api/quic_expect_bug.h"
20#include "net/third_party/quiche/src/quic/platform/api/quic_map_util.h"
21#include "net/third_party/quiche/src/quic/platform/api/quic_ptr_util.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050022#include "net/third_party/quiche/src/quic/platform/api/quic_string_piece.h"
23#include "net/third_party/quiche/src/quic/platform/api/quic_test.h"
24#include "net/third_party/quiche/src/quic/platform/api/quic_text_utils.h"
25#include "net/third_party/quiche/src/quic/test_tools/quic_flow_controller_peer.h"
26#include "net/third_party/quiche/src/quic/test_tools/quic_session_peer.h"
27#include "net/third_party/quiche/src/quic/test_tools/quic_spdy_session_peer.h"
28#include "net/third_party/quiche/src/quic/test_tools/quic_spdy_stream_peer.h"
29#include "net/third_party/quiche/src/quic/test_tools/quic_stream_peer.h"
30#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
31
32using spdy::kV3HighestPriority;
33using spdy::kV3LowestPriority;
34using spdy::SpdyHeaderBlock;
35using spdy::SpdyPriority;
36using testing::_;
37using testing::AtLeast;
38using testing::Invoke;
39using testing::Return;
40using testing::StrictMock;
41
42namespace quic {
43namespace test {
44namespace {
45
46const bool kShouldProcessData = true;
47
48class TestStream : public QuicSpdyStream {
49 public:
50 TestStream(QuicStreamId id,
51 QuicSpdySession* session,
52 bool should_process_data)
53 : QuicSpdyStream(id, session, BIDIRECTIONAL),
54 should_process_data_(should_process_data) {}
55 ~TestStream() override = default;
56
57 using QuicSpdyStream::set_ack_listener;
58 using QuicStream::CloseWriteSide;
59 using QuicStream::WriteOrBufferData;
60
61 void OnBodyAvailable() override {
62 if (!should_process_data_) {
63 return;
64 }
65 char buffer[2048];
66 struct iovec vec;
67 vec.iov_base = buffer;
68 vec.iov_len = QUIC_ARRAYSIZE(buffer);
69 size_t bytes_read = Readv(&vec, 1);
vasilvvc48c8712019-03-11 13:38:16 -070070 data_ += std::string(buffer, bytes_read);
QUICHE teama6ef0a62019-03-07 20:34:33 -050071 }
72
73 MOCK_METHOD1(WriteHeadersMock, void(bool fin));
74
75 size_t WriteHeadersImpl(spdy::SpdyHeaderBlock header_block,
76 bool fin,
77 QuicReferenceCountedPointer<QuicAckListenerInterface>
78 ack_listener) override {
79 saved_headers_ = std::move(header_block);
80 WriteHeadersMock(fin);
renjietang2abedac2019-05-20 14:04:50 -070081 if (VersionUsesQpack(transport_version())) {
82 // In this case, call QuicSpdyStream::WriteHeadersImpl() that does the
83 // actual work of closing the stream.
84 QuicSpdyStream::WriteHeadersImpl(saved_headers_.Clone(), fin, nullptr);
85 }
QUICHE teama6ef0a62019-03-07 20:34:33 -050086 return 0;
87 }
88
vasilvvc48c8712019-03-11 13:38:16 -070089 const std::string& data() const { return data_; }
QUICHE teama6ef0a62019-03-07 20:34:33 -050090 const spdy::SpdyHeaderBlock& saved_headers() const { return saved_headers_; }
91
92 private:
93 bool should_process_data_;
94 spdy::SpdyHeaderBlock saved_headers_;
vasilvvc48c8712019-03-11 13:38:16 -070095 std::string data_;
QUICHE teama6ef0a62019-03-07 20:34:33 -050096};
97
98class TestMockUpdateStreamSession : public MockQuicSpdySession {
99 public:
100 explicit TestMockUpdateStreamSession(QuicConnection* connection)
101 : MockQuicSpdySession(connection) {}
102
103 void UpdateStreamPriority(QuicStreamId id, SpdyPriority priority) override {
104 EXPECT_EQ(id, expected_stream_->id());
105 EXPECT_EQ(expected_priority_, priority);
106 EXPECT_EQ(expected_priority_, expected_stream_->priority());
107 }
108
109 void SetExpectedStream(QuicSpdyStream* stream) { expected_stream_ = stream; }
110 void SetExpectedPriority(SpdyPriority priority) {
111 expected_priority_ = priority;
112 }
113
114 private:
115 QuicSpdyStream* expected_stream_;
116 SpdyPriority expected_priority_;
117};
118
119class QuicSpdyStreamTest : public QuicTestWithParam<ParsedQuicVersion> {
120 public:
121 QuicSpdyStreamTest() {
122 headers_[":host"] = "www.google.com";
123 headers_[":path"] = "/index.hml";
124 headers_[":scheme"] = "https";
125 headers_["cookie"] =
126 "__utma=208381060.1228362404.1372200928.1372200928.1372200928.1; "
127 "__utmc=160408618; "
128 "GX=DQAAAOEAAACWJYdewdE9rIrW6qw3PtVi2-d729qaa-74KqOsM1NVQblK4VhX"
129 "hoALMsy6HOdDad2Sz0flUByv7etmo3mLMidGrBoljqO9hSVA40SLqpG_iuKKSHX"
130 "RW3Np4bq0F0SDGDNsW0DSmTS9ufMRrlpARJDS7qAI6M3bghqJp4eABKZiRqebHT"
131 "pMU-RXvTI5D5oCF1vYxYofH_l1Kviuiy3oQ1kS1enqWgbhJ2t61_SNdv-1XJIS0"
132 "O3YeHLmVCs62O6zp89QwakfAWK9d3IDQvVSJzCQsvxvNIvaZFa567MawWlXg0Rh"
133 "1zFMi5vzcns38-8_Sns; "
134 "GA=v*2%2Fmem*57968640*47239936%2Fmem*57968640*47114716%2Fno-nm-"
135 "yj*15%2Fno-cc-yj*5%2Fpc-ch*133685%2Fpc-s-cr*133947%2Fpc-s-t*1339"
136 "47%2Fno-nm-yj*4%2Fno-cc-yj*1%2Fceft-as*1%2Fceft-nqas*0%2Fad-ra-c"
137 "v_p%2Fad-nr-cv_p-f*1%2Fad-v-cv_p*859%2Fad-ns-cv_p-f*1%2Ffn-v-ad%"
138 "2Fpc-t*250%2Fpc-cm*461%2Fpc-s-cr*722%2Fpc-s-t*722%2Fau_p*4"
139 "SICAID=AJKiYcHdKgxum7KMXG0ei2t1-W4OD1uW-ecNsCqC0wDuAXiDGIcT_HA2o1"
140 "3Rs1UKCuBAF9g8rWNOFbxt8PSNSHFuIhOo2t6bJAVpCsMU5Laa6lewuTMYI8MzdQP"
141 "ARHKyW-koxuhMZHUnGBJAM1gJODe0cATO_KGoX4pbbFxxJ5IicRxOrWK_5rU3cdy6"
142 "edlR9FsEdH6iujMcHkbE5l18ehJDwTWmBKBzVD87naobhMMrF6VvnDGxQVGp9Ir_b"
143 "Rgj3RWUoPumQVCxtSOBdX0GlJOEcDTNCzQIm9BSfetog_eP_TfYubKudt5eMsXmN6"
144 "QnyXHeGeK2UINUzJ-D30AFcpqYgH9_1BvYSpi7fc7_ydBU8TaD8ZRxvtnzXqj0RfG"
145 "tuHghmv3aD-uzSYJ75XDdzKdizZ86IG6Fbn1XFhYZM-fbHhm3mVEXnyRW4ZuNOLFk"
146 "Fas6LMcVC6Q8QLlHYbXBpdNFuGbuZGUnav5C-2I_-46lL0NGg3GewxGKGHvHEfoyn"
147 "EFFlEYHsBQ98rXImL8ySDycdLEFvBPdtctPmWCfTxwmoSMLHU2SCVDhbqMWU5b0yr"
148 "JBCScs_ejbKaqBDoB7ZGxTvqlrB__2ZmnHHjCr8RgMRtKNtIeuZAo ";
149 }
150
renjietangbd1a0392019-05-31 11:36:24 -0700151 std::string EncodeQpackHeaders(QuicStreamId id, SpdyHeaderBlock* header) {
152 auto qpack_encoder =
153 QuicMakeUnique<QpackEncoder>(session_.get(), session_.get());
154 auto progreesive_encoder = qpack_encoder->EncodeHeaderList(id, header);
155 std::string encoded_headers;
156 while (progreesive_encoder->HasNext()) {
157 progreesive_encoder->Next(std::numeric_limits<size_t>::max(),
158 &encoded_headers);
159 }
160 return encoded_headers;
161 }
162
QUICHE teama6ef0a62019-03-07 20:34:33 -0500163 void Initialize(bool stream_should_process_data) {
164 connection_ = new StrictMock<MockQuicConnection>(
165 &helper_, &alarm_factory_, Perspective::IS_SERVER,
166 SupportedVersions(GetParam()));
167 session_ = QuicMakeUnique<StrictMock<MockQuicSpdySession>>(connection_);
168 session_->Initialize();
169 ON_CALL(*session_, WritevData(_, _, _, _, _))
170 .WillByDefault(Invoke(MockQuicSession::ConsumeData));
171
172 stream_ =
173 new StrictMock<TestStream>(GetNthClientInitiatedBidirectionalId(0),
174 session_.get(), stream_should_process_data);
175 session_->ActivateStream(QuicWrapUnique(stream_));
176 stream2_ =
177 new StrictMock<TestStream>(GetNthClientInitiatedBidirectionalId(1),
178 session_.get(), stream_should_process_data);
179 session_->ActivateStream(QuicWrapUnique(stream2_));
180 }
181
182 QuicHeaderList ProcessHeaders(bool fin, const SpdyHeaderBlock& headers) {
183 QuicHeaderList h = AsHeaderList(headers);
184 stream_->OnStreamHeaderList(fin, h.uncompressed_header_bytes(), h);
185 return h;
186 }
187
188 QuicStreamId GetNthClientInitiatedBidirectionalId(int n) {
189 return GetNthClientInitiatedBidirectionalStreamId(
190 connection_->transport_version(), n);
191 }
192
193 bool HasFrameHeader() const {
bnc677451a2019-06-07 10:13:30 -0700194 return VersionHasDataFrameHeader(GetParam().transport_version);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500195 }
196
197 protected:
198 MockQuicConnectionHelper helper_;
199 MockAlarmFactory alarm_factory_;
200 MockQuicConnection* connection_;
201 std::unique_ptr<MockQuicSpdySession> session_;
202
203 // Owned by the |session_|.
204 TestStream* stream_;
205 TestStream* stream2_;
206
207 SpdyHeaderBlock headers_;
208
209 HttpEncoder encoder_;
210};
211
vasilvvc48c8712019-03-11 13:38:16 -0700212INSTANTIATE_TEST_SUITE_P(Tests,
213 QuicSpdyStreamTest,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500214 ::testing::ValuesIn(AllSupportedVersions()));
215
216TEST_P(QuicSpdyStreamTest, ProcessHeaderList) {
217 Initialize(kShouldProcessData);
218
219 stream_->OnStreamHeadersPriority(kV3HighestPriority);
220 ProcessHeaders(false, headers_);
221 EXPECT_EQ("", stream_->data());
222 EXPECT_FALSE(stream_->header_list().empty());
223 EXPECT_FALSE(stream_->IsDoneReading());
224}
225
226TEST_P(QuicSpdyStreamTest, ProcessTooLargeHeaderList) {
227 Initialize(kShouldProcessData);
228
229 QuicHeaderList headers;
230 stream_->OnStreamHeadersPriority(kV3HighestPriority);
231
renjietang2abedac2019-05-20 14:04:50 -0700232 const bool version_uses_qpack =
bnc677451a2019-06-07 10:13:30 -0700233 VersionUsesQpack(GetParam().transport_version);
renjietang2abedac2019-05-20 14:04:50 -0700234
235 if (version_uses_qpack) {
236 EXPECT_CALL(*connection_,
237 CloseConnection(QUIC_HEADERS_STREAM_DATA_DECOMPRESS_FAILURE,
238 "Too large headers received on stream 4", _));
239 } else {
240 EXPECT_CALL(*session_,
241 SendRstStream(stream_->id(), QUIC_HEADERS_TOO_LARGE, 0));
242 }
243
QUICHE teama6ef0a62019-03-07 20:34:33 -0500244 stream_->OnStreamHeaderList(false, 1 << 20, headers);
renjietang2abedac2019-05-20 14:04:50 -0700245
246 if (!version_uses_qpack) {
247 EXPECT_EQ(QUIC_HEADERS_TOO_LARGE, stream_->stream_error());
248 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500249}
250
251TEST_P(QuicSpdyStreamTest, ProcessHeaderListWithFin) {
252 Initialize(kShouldProcessData);
253
254 size_t total_bytes = 0;
255 QuicHeaderList headers;
256 for (auto p : headers_) {
257 headers.OnHeader(p.first, p.second);
258 total_bytes += p.first.size() + p.second.size();
259 }
260 stream_->OnStreamHeadersPriority(kV3HighestPriority);
261 stream_->OnStreamHeaderList(true, total_bytes, headers);
262 EXPECT_EQ("", stream_->data());
263 EXPECT_FALSE(stream_->header_list().empty());
264 EXPECT_FALSE(stream_->IsDoneReading());
265 EXPECT_TRUE(stream_->HasFinalReceivedByteOffset());
266}
267
bnc677451a2019-06-07 10:13:30 -0700268// A valid status code should be 3-digit integer. The first digit should be in
269// the range of [1, 5]. All the others are invalid.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500270TEST_P(QuicSpdyStreamTest, ParseHeaderStatusCode) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500271 Initialize(kShouldProcessData);
272 int status_code = 0;
273
274 // Valid status codes.
275 headers_[":status"] = "404";
276 EXPECT_TRUE(stream_->ParseHeaderStatusCode(headers_, &status_code));
277 EXPECT_EQ(404, status_code);
278
279 headers_[":status"] = "100";
280 EXPECT_TRUE(stream_->ParseHeaderStatusCode(headers_, &status_code));
281 EXPECT_EQ(100, status_code);
282
283 headers_[":status"] = "599";
284 EXPECT_TRUE(stream_->ParseHeaderStatusCode(headers_, &status_code));
285 EXPECT_EQ(599, status_code);
286
287 // Invalid status codes.
288 headers_[":status"] = "010";
289 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
290
291 headers_[":status"] = "600";
292 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
293
294 headers_[":status"] = "200 ok";
295 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
296
297 headers_[":status"] = "2000";
298 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
299
300 headers_[":status"] = "+200";
301 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
302
303 headers_[":status"] = "+20";
304 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
305
306 headers_[":status"] = "-10";
307 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
308
309 headers_[":status"] = "-100";
310 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
311
312 // Leading or trailing spaces are also invalid.
313 headers_[":status"] = " 200";
314 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
315
316 headers_[":status"] = "200 ";
317 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
318
319 headers_[":status"] = " 200 ";
320 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
321
322 headers_[":status"] = " ";
323 EXPECT_FALSE(stream_->ParseHeaderStatusCode(headers_, &status_code));
324}
325
326TEST_P(QuicSpdyStreamTest, MarkHeadersConsumed) {
327 Initialize(kShouldProcessData);
328
vasilvvc48c8712019-03-11 13:38:16 -0700329 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500330 QuicHeaderList headers = ProcessHeaders(false, headers_);
331 EXPECT_EQ(headers, stream_->header_list());
332
333 stream_->ConsumeHeaderList();
334 EXPECT_EQ(QuicHeaderList(), stream_->header_list());
335}
336
QUICHE team396d1092019-03-20 10:21:07 -0700337TEST_P(QuicSpdyStreamTest, ProcessWrongFramesOnSpdyStream) {
QUICHE team396d1092019-03-20 10:21:07 -0700338 if (!HasFrameHeader()) {
339 return;
340 }
bnc677451a2019-06-07 10:13:30 -0700341
342 testing::InSequence s;
343 Initialize(kShouldProcessData);
QUICHE team396d1092019-03-20 10:21:07 -0700344 connection_->AdvanceTime(QuicTime::Delta::FromSeconds(1));
345 GoAwayFrame goaway;
346 goaway.stream_id = 0x1;
347 std::unique_ptr<char[]> buffer;
348 QuicByteCount header_length = encoder_.SerializeGoAwayFrame(goaway, &buffer);
349 std::string data = std::string(buffer.get(), header_length);
350
351 EXPECT_EQ("", stream_->data());
352 QuicHeaderList headers = ProcessHeaders(false, headers_);
353 EXPECT_EQ(headers, stream_->header_list());
354 stream_->ConsumeHeaderList();
355 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
356 QuicStringPiece(data));
357
358 EXPECT_CALL(*connection_, CloseConnection(QUIC_HTTP_DECODER_ERROR, _, _))
359 .WillOnce(
360 (Invoke([this](QuicErrorCode error, const std::string& error_details,
361 ConnectionCloseBehavior connection_close_behavior) {
362 connection_->ReallyCloseConnection(error, error_details,
363 connection_close_behavior);
364 })));
ianswettdc1e7ab2019-05-03 16:10:44 -0700365 EXPECT_CALL(*connection_, SendConnectionClosePacket(_, _));
QUICHE team396d1092019-03-20 10:21:07 -0700366 EXPECT_CALL(*session_, OnConnectionClosed(_, _, _))
367 .WillOnce(
368 Invoke([this](QuicErrorCode error, const std::string& error_details,
369 ConnectionCloseSource source) {
370 session_->ReallyOnConnectionClosed(error, error_details, source);
371 }));
372 EXPECT_CALL(*session_, SendRstStream(_, _, _));
373 EXPECT_CALL(*session_, SendRstStream(_, _, _));
374
375 stream_->OnStreamFrame(frame);
376}
377
QUICHE teama6ef0a62019-03-07 20:34:33 -0500378TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBody) {
379 Initialize(kShouldProcessData);
380
vasilvvc48c8712019-03-11 13:38:16 -0700381 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500382 std::unique_ptr<char[]> buffer;
383 QuicByteCount header_length =
384 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700385 std::string header = std::string(buffer.get(), header_length);
386 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500387
388 EXPECT_EQ("", stream_->data());
389 QuicHeaderList headers = ProcessHeaders(false, headers_);
390 EXPECT_EQ(headers, stream_->header_list());
391 stream_->ConsumeHeaderList();
392 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
393 QuicStringPiece(data));
394 stream_->OnStreamFrame(frame);
395 EXPECT_EQ(QuicHeaderList(), stream_->header_list());
396 EXPECT_EQ(body, stream_->data());
397}
398
399TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBodyFragments) {
400 Initialize(kShouldProcessData);
vasilvvc48c8712019-03-11 13:38:16 -0700401 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500402 std::unique_ptr<char[]> buffer;
403 QuicByteCount header_length =
404 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700405 std::string header = std::string(buffer.get(), header_length);
406 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500407
408 for (size_t fragment_size = 1; fragment_size < data.size(); ++fragment_size) {
409 Initialize(kShouldProcessData);
410 QuicHeaderList headers = ProcessHeaders(false, headers_);
411 ASSERT_EQ(headers, stream_->header_list());
412 stream_->ConsumeHeaderList();
413 for (size_t offset = 0; offset < data.size(); offset += fragment_size) {
414 size_t remaining_data = data.size() - offset;
415 QuicStringPiece fragment(data.data() + offset,
416 std::min(fragment_size, remaining_data));
417 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false,
418 offset, QuicStringPiece(fragment));
419 stream_->OnStreamFrame(frame);
420 }
421 ASSERT_EQ(body, stream_->data()) << "fragment_size: " << fragment_size;
422 }
423}
424
425TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBodyFragmentsSplit) {
426 Initialize(kShouldProcessData);
vasilvvc48c8712019-03-11 13:38:16 -0700427 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500428 std::unique_ptr<char[]> buffer;
429 QuicByteCount header_length =
430 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700431 std::string header = std::string(buffer.get(), header_length);
432 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500433
434 for (size_t split_point = 1; split_point < data.size() - 1; ++split_point) {
435 Initialize(kShouldProcessData);
436 QuicHeaderList headers = ProcessHeaders(false, headers_);
437 ASSERT_EQ(headers, stream_->header_list());
438 stream_->ConsumeHeaderList();
439
440 QuicStringPiece fragment1(data.data(), split_point);
441 QuicStreamFrame frame1(GetNthClientInitiatedBidirectionalId(0), false, 0,
442 QuicStringPiece(fragment1));
443 stream_->OnStreamFrame(frame1);
444
445 QuicStringPiece fragment2(data.data() + split_point,
446 data.size() - split_point);
447 QuicStreamFrame frame2(GetNthClientInitiatedBidirectionalId(0), false,
448 split_point, QuicStringPiece(fragment2));
449 stream_->OnStreamFrame(frame2);
450
451 ASSERT_EQ(body, stream_->data()) << "split_point: " << split_point;
452 }
453}
454
455TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBodyReadv) {
456 Initialize(!kShouldProcessData);
457
vasilvvc48c8712019-03-11 13:38:16 -0700458 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500459 std::unique_ptr<char[]> buf;
460 QuicByteCount header_length =
461 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700462 std::string header = std::string(buf.get(), header_length);
463 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500464
465 ProcessHeaders(false, headers_);
466 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
467 QuicStringPiece(data));
468 stream_->OnStreamFrame(frame);
469 stream_->ConsumeHeaderList();
470
471 char buffer[2048];
472 ASSERT_LT(data.length(), QUIC_ARRAYSIZE(buffer));
473 struct iovec vec;
474 vec.iov_base = buffer;
475 vec.iov_len = QUIC_ARRAYSIZE(buffer);
476
477 size_t bytes_read = stream_->Readv(&vec, 1);
QUICHE team396d1092019-03-20 10:21:07 -0700478 QuicStreamPeer::CloseReadSide(stream_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500479 EXPECT_EQ(body.length(), bytes_read);
vasilvvc48c8712019-03-11 13:38:16 -0700480 EXPECT_EQ(body, std::string(buffer, bytes_read));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500481}
482
483TEST_P(QuicSpdyStreamTest, ProcessHeadersAndLargeBodySmallReadv) {
484 Initialize(kShouldProcessData);
vasilvvc48c8712019-03-11 13:38:16 -0700485 std::string body(12 * 1024, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500486 std::unique_ptr<char[]> buf;
487 QuicByteCount header_length =
488 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700489 std::string header = std::string(buf.get(), header_length);
490 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500491 ProcessHeaders(false, headers_);
492 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
493 QuicStringPiece(data));
494 stream_->OnStreamFrame(frame);
495 stream_->ConsumeHeaderList();
496 char buffer[2048];
497 char buffer2[2048];
498 struct iovec vec[2];
499 vec[0].iov_base = buffer;
500 vec[0].iov_len = QUIC_ARRAYSIZE(buffer);
501 vec[1].iov_base = buffer2;
502 vec[1].iov_len = QUIC_ARRAYSIZE(buffer2);
503 size_t bytes_read = stream_->Readv(vec, 2);
504 EXPECT_EQ(2048u * 2, bytes_read);
vasilvvc48c8712019-03-11 13:38:16 -0700505 EXPECT_EQ(body.substr(0, 2048), std::string(buffer, 2048));
506 EXPECT_EQ(body.substr(2048, 2048), std::string(buffer2, 2048));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500507}
508
509TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBodyMarkConsumed) {
510 Initialize(!kShouldProcessData);
511
vasilvvc48c8712019-03-11 13:38:16 -0700512 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500513 std::unique_ptr<char[]> buf;
514 QuicByteCount header_length =
515 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700516 std::string header = std::string(buf.get(), header_length);
517 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500518
519 ProcessHeaders(false, headers_);
520 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
521 QuicStringPiece(data));
522 stream_->OnStreamFrame(frame);
523 stream_->ConsumeHeaderList();
524
525 struct iovec vec;
526
527 EXPECT_EQ(1, stream_->GetReadableRegions(&vec, 1));
528 EXPECT_EQ(body.length(), vec.iov_len);
vasilvvc48c8712019-03-11 13:38:16 -0700529 EXPECT_EQ(body, std::string(static_cast<char*>(vec.iov_base), vec.iov_len));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500530
531 stream_->MarkConsumed(body.length());
532 EXPECT_EQ(data.length(), stream_->flow_controller()->bytes_consumed());
533}
534
535TEST_P(QuicSpdyStreamTest, ProcessHeadersAndConsumeMultipleBody) {
536 Initialize(!kShouldProcessData);
vasilvvc48c8712019-03-11 13:38:16 -0700537 std::string body1 = "this is body 1";
538 std::string body2 = "body 2";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500539 std::unique_ptr<char[]> buf;
540 QuicByteCount header_length =
541 encoder_.SerializeDataFrameHeader(body1.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700542 std::string header = std::string(buf.get(), header_length);
543 std::string data1 = HasFrameHeader() ? header + body1 : body1;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500544 header_length = encoder_.SerializeDataFrameHeader(body2.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700545 std::string data2 = HasFrameHeader() ? header + body2 : body2;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500546
547 ProcessHeaders(false, headers_);
548 QuicStreamFrame frame1(GetNthClientInitiatedBidirectionalId(0), false, 0,
549 QuicStringPiece(data1));
550 QuicStreamFrame frame2(GetNthClientInitiatedBidirectionalId(0), false,
551 data1.length(), QuicStringPiece(data2));
552 stream_->OnStreamFrame(frame1);
553 stream_->OnStreamFrame(frame2);
554 stream_->ConsumeHeaderList();
555
556 stream_->MarkConsumed(body1.length() + body2.length());
557 EXPECT_EQ(data1.length() + data2.length(),
558 stream_->flow_controller()->bytes_consumed());
559}
560
561TEST_P(QuicSpdyStreamTest, ProcessHeadersAndBodyIncrementalReadv) {
562 Initialize(!kShouldProcessData);
563
vasilvvc48c8712019-03-11 13:38:16 -0700564 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500565 std::unique_ptr<char[]> buf;
566 QuicByteCount header_length =
567 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700568 std::string header = std::string(buf.get(), header_length);
569 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500570
571 ProcessHeaders(false, headers_);
572 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
573 QuicStringPiece(data));
574 stream_->OnStreamFrame(frame);
575 stream_->ConsumeHeaderList();
576
577 char buffer[1];
578 struct iovec vec;
579 vec.iov_base = buffer;
580 vec.iov_len = QUIC_ARRAYSIZE(buffer);
581
582 for (size_t i = 0; i < body.length(); ++i) {
583 size_t bytes_read = stream_->Readv(&vec, 1);
584 ASSERT_EQ(1u, bytes_read);
585 EXPECT_EQ(body.data()[i], buffer[0]);
586 }
587}
588
589TEST_P(QuicSpdyStreamTest, ProcessHeadersUsingReadvWithMultipleIovecs) {
590 Initialize(!kShouldProcessData);
591
vasilvvc48c8712019-03-11 13:38:16 -0700592 std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500593 std::unique_ptr<char[]> buf;
594 QuicByteCount header_length =
595 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700596 std::string header = std::string(buf.get(), header_length);
597 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500598
599 ProcessHeaders(false, headers_);
600 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
601 QuicStringPiece(data));
602 stream_->OnStreamFrame(frame);
603 stream_->ConsumeHeaderList();
604
605 char buffer1[1];
606 char buffer2[1];
607 struct iovec vec[2];
608 vec[0].iov_base = buffer1;
609 vec[0].iov_len = QUIC_ARRAYSIZE(buffer1);
610 vec[1].iov_base = buffer2;
611 vec[1].iov_len = QUIC_ARRAYSIZE(buffer2);
612
613 for (size_t i = 0; i < body.length(); i += 2) {
614 size_t bytes_read = stream_->Readv(vec, 2);
615 ASSERT_EQ(2u, bytes_read) << i;
616 ASSERT_EQ(body.data()[i], buffer1[0]) << i;
617 ASSERT_EQ(body.data()[i + 1], buffer2[0]) << i;
618 }
619}
620
bnc677451a2019-06-07 10:13:30 -0700621// Tests that we send a BLOCKED frame to the peer when we attempt to write, but
622// are flow control blocked.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500623TEST_P(QuicSpdyStreamTest, StreamFlowControlBlocked) {
nharperf5e68452019-05-29 17:24:18 -0700624 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
625 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
626 // enabled and fix it.
627 return;
628 }
bnc677451a2019-06-07 10:13:30 -0700629
QUICHE teama6ef0a62019-03-07 20:34:33 -0500630 testing::InSequence seq;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500631 Initialize(kShouldProcessData);
632
633 // Set a small flow control limit.
634 const uint64_t kWindow = 36;
635 QuicFlowControllerPeer::SetSendWindowOffset(stream_->flow_controller(),
636 kWindow);
637 EXPECT_EQ(kWindow, QuicFlowControllerPeer::SendWindowOffset(
638 stream_->flow_controller()));
639
640 // Try to send more data than the flow control limit allows.
641 const uint64_t kOverflow = 15;
vasilvvc48c8712019-03-11 13:38:16 -0700642 std::string body(kWindow + kOverflow, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500643
644 const uint64_t kHeaderLength = HasFrameHeader() ? 2 : 0;
645 if (HasFrameHeader()) {
646 EXPECT_CALL(*session_, WritevData(_, _, kHeaderLength, _, NO_FIN));
647 }
648 EXPECT_CALL(*session_, WritevData(_, _, _, _, _))
649 .WillOnce(Return(QuicConsumedData(kWindow - kHeaderLength, true)));
650 EXPECT_CALL(*connection_, SendControlFrame(_));
651 stream_->WriteOrBufferBody(body, false);
652
653 // Should have sent as much as possible, resulting in no send window left.
654 EXPECT_EQ(0u,
655 QuicFlowControllerPeer::SendWindowSize(stream_->flow_controller()));
656
657 // And we should have queued the overflowed data.
bncc7d9e0c2019-04-16 10:22:15 -0700658 EXPECT_EQ(kOverflow + kHeaderLength, stream_->BufferedDataBytes());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500659}
660
bnc677451a2019-06-07 10:13:30 -0700661// The flow control receive window decreases whenever we add new bytes to the
662// sequencer, whether they are consumed immediately or buffered. However we only
663// send WINDOW_UPDATE frames based on increasing number of bytes consumed.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500664TEST_P(QuicSpdyStreamTest, StreamFlowControlNoWindowUpdateIfNotConsumed) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500665 // Don't process data - it will be buffered instead.
666 Initialize(!kShouldProcessData);
667
668 // Expect no WINDOW_UPDATE frames to be sent.
669 EXPECT_CALL(*connection_, SendWindowUpdate(_, _)).Times(0);
670
671 // Set a small flow control receive window.
672 const uint64_t kWindow = 36;
673 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(),
674 kWindow);
675 QuicFlowControllerPeer::SetMaxReceiveWindow(stream_->flow_controller(),
676 kWindow);
677 EXPECT_EQ(kWindow, QuicFlowControllerPeer::ReceiveWindowOffset(
678 stream_->flow_controller()));
679
680 // Stream receives enough data to fill a fraction of the receive window.
vasilvvc48c8712019-03-11 13:38:16 -0700681 std::string body(kWindow / 3, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500682 QuicByteCount header_length = 0;
vasilvvc48c8712019-03-11 13:38:16 -0700683 std::string data;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500684
685 if (HasFrameHeader()) {
686 std::unique_ptr<char[]> buffer;
687 header_length = encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700688 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500689 data = header + body;
690 } else {
691 data = body;
692 }
693
694 ProcessHeaders(false, headers_);
695
696 QuicStreamFrame frame1(GetNthClientInitiatedBidirectionalId(0), false, 0,
697 QuicStringPiece(data));
698 stream_->OnStreamFrame(frame1);
699 EXPECT_EQ(
700 kWindow - (kWindow / 3) - header_length,
701 QuicFlowControllerPeer::ReceiveWindowSize(stream_->flow_controller()));
702
703 // Now receive another frame which results in the receive window being over
704 // half full. This should all be buffered, decreasing the receive window but
705 // not sending WINDOW_UPDATE.
706 QuicStreamFrame frame2(GetNthClientInitiatedBidirectionalId(0), false,
707 kWindow / 3 + header_length, QuicStringPiece(data));
708 stream_->OnStreamFrame(frame2);
709 EXPECT_EQ(
710 kWindow - (2 * kWindow / 3) - 2 * header_length,
711 QuicFlowControllerPeer::ReceiveWindowSize(stream_->flow_controller()));
712}
713
bnc677451a2019-06-07 10:13:30 -0700714// Tests that on receipt of data, the stream updates its receive window offset
715// appropriately, and sends WINDOW_UPDATE frames when its receive window drops
716// too low.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500717TEST_P(QuicSpdyStreamTest, StreamFlowControlWindowUpdate) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500718 Initialize(kShouldProcessData);
719
720 // Set a small flow control limit.
721 const uint64_t kWindow = 36;
722 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(),
723 kWindow);
724 QuicFlowControllerPeer::SetMaxReceiveWindow(stream_->flow_controller(),
725 kWindow);
726 EXPECT_EQ(kWindow, QuicFlowControllerPeer::ReceiveWindowOffset(
727 stream_->flow_controller()));
728
729 // Stream receives enough data to fill a fraction of the receive window.
vasilvvc48c8712019-03-11 13:38:16 -0700730 std::string body(kWindow / 3, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500731 QuicByteCount header_length = 0;
vasilvvc48c8712019-03-11 13:38:16 -0700732 std::string data;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500733
734 if (HasFrameHeader()) {
735 std::unique_ptr<char[]> buffer;
736 header_length = encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700737 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500738 data = header + body;
739 } else {
740 data = body;
741 }
742
743 ProcessHeaders(false, headers_);
744 stream_->ConsumeHeaderList();
745
746 QuicStreamFrame frame1(GetNthClientInitiatedBidirectionalId(0), false, 0,
747 QuicStringPiece(data));
748 stream_->OnStreamFrame(frame1);
749 EXPECT_EQ(
750 kWindow - (kWindow / 3) - header_length,
751 QuicFlowControllerPeer::ReceiveWindowSize(stream_->flow_controller()));
752
753 // Now receive another frame which results in the receive window being over
754 // half full. This will trigger the stream to increase its receive window
755 // offset and send a WINDOW_UPDATE. The result will be again an available
756 // window of kWindow bytes.
757 QuicStreamFrame frame2(GetNthClientInitiatedBidirectionalId(0), false,
758 kWindow / 3 + header_length, QuicStringPiece(data));
759 EXPECT_CALL(*connection_, SendControlFrame(_));
760 stream_->OnStreamFrame(frame2);
761 EXPECT_EQ(kWindow, QuicFlowControllerPeer::ReceiveWindowSize(
762 stream_->flow_controller()));
763}
764
bnc677451a2019-06-07 10:13:30 -0700765// Tests that on receipt of data, the connection updates its receive window
766// offset appropriately, and sends WINDOW_UPDATE frames when its receive window
767// drops too low.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500768TEST_P(QuicSpdyStreamTest, ConnectionFlowControlWindowUpdate) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500769 Initialize(kShouldProcessData);
770
771 // Set a small flow control limit for streams and connection.
772 const uint64_t kWindow = 36;
773 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(),
774 kWindow);
775 QuicFlowControllerPeer::SetMaxReceiveWindow(stream_->flow_controller(),
776 kWindow);
777 QuicFlowControllerPeer::SetReceiveWindowOffset(stream2_->flow_controller(),
778 kWindow);
779 QuicFlowControllerPeer::SetMaxReceiveWindow(stream2_->flow_controller(),
780 kWindow);
781 QuicFlowControllerPeer::SetReceiveWindowOffset(session_->flow_controller(),
782 kWindow);
783 QuicFlowControllerPeer::SetMaxReceiveWindow(session_->flow_controller(),
784 kWindow);
785
786 // Supply headers to both streams so that they are happy to receive data.
787 auto headers = AsHeaderList(headers_);
788 stream_->OnStreamHeaderList(false, headers.uncompressed_header_bytes(),
789 headers);
790 stream_->ConsumeHeaderList();
791 stream2_->OnStreamHeaderList(false, headers.uncompressed_header_bytes(),
792 headers);
793 stream2_->ConsumeHeaderList();
794
795 // Each stream gets a quarter window of data. This should not trigger a
796 // WINDOW_UPDATE for either stream, nor for the connection.
797 QuicByteCount header_length = 0;
vasilvvc48c8712019-03-11 13:38:16 -0700798 std::string body;
799 std::string data;
800 std::string data2;
801 std::string body2(1, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500802
803 if (HasFrameHeader()) {
vasilvvc48c8712019-03-11 13:38:16 -0700804 body = std::string(kWindow / 4 - 2, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500805 std::unique_ptr<char[]> buffer;
806 header_length = encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -0700807 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500808 data = header + body;
809 std::unique_ptr<char[]> buffer2;
810 QuicByteCount header_length2 =
811 encoder_.SerializeDataFrameHeader(body2.length(), &buffer2);
vasilvvc48c8712019-03-11 13:38:16 -0700812 std::string header2 = std::string(buffer2.get(), header_length2);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500813 data2 = header2 + body2;
814 } else {
vasilvvc48c8712019-03-11 13:38:16 -0700815 body = std::string(kWindow / 4, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500816 data = body;
817 data2 = body2;
818 }
819
820 QuicStreamFrame frame1(GetNthClientInitiatedBidirectionalId(0), false, 0,
821 QuicStringPiece(data));
822 stream_->OnStreamFrame(frame1);
823 QuicStreamFrame frame2(GetNthClientInitiatedBidirectionalId(1), false, 0,
824 QuicStringPiece(data));
825 stream2_->OnStreamFrame(frame2);
826
827 // Now receive a further single byte on one stream - again this does not
828 // trigger a stream WINDOW_UPDATE, but now the connection flow control window
829 // is over half full and thus a connection WINDOW_UPDATE is sent.
830 EXPECT_CALL(*connection_, SendControlFrame(_));
831 QuicStreamFrame frame3(GetNthClientInitiatedBidirectionalId(0), false,
832 body.length() + header_length, QuicStringPiece(data2));
833 stream_->OnStreamFrame(frame3);
834}
835
bnc677451a2019-06-07 10:13:30 -0700836// Tests that on if the peer sends too much data (i.e. violates the flow control
837// protocol), then we terminate the connection.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500838TEST_P(QuicSpdyStreamTest, StreamFlowControlViolation) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500839 // Stream should not process data, so that data gets buffered in the
840 // sequencer, triggering flow control limits.
841 Initialize(!kShouldProcessData);
842
843 // Set a small flow control limit.
844 const uint64_t kWindow = 50;
845 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(),
846 kWindow);
847
848 ProcessHeaders(false, headers_);
849
850 // Receive data to overflow the window, violating flow control.
vasilvvc48c8712019-03-11 13:38:16 -0700851 std::string body(kWindow + 1, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500852 std::unique_ptr<char[]> buf;
853 QuicByteCount header_length =
854 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700855 std::string header = std::string(buf.get(), header_length);
856 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500857 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
858 QuicStringPiece(data));
859 EXPECT_CALL(*connection_,
860 CloseConnection(QUIC_FLOW_CONTROL_RECEIVED_TOO_MUCH_DATA, _, _));
861 stream_->OnStreamFrame(frame);
862}
863
864TEST_P(QuicSpdyStreamTest, TestHandlingQuicRstStreamNoError) {
865 Initialize(kShouldProcessData);
866 ProcessHeaders(false, headers_);
867
868 stream_->OnStreamReset(QuicRstStreamFrame(
869 kInvalidControlFrameId, stream_->id(), QUIC_STREAM_NO_ERROR, 0));
870 EXPECT_TRUE(stream_->write_side_closed());
871 EXPECT_FALSE(stream_->reading_stopped());
872}
873
bnc677451a2019-06-07 10:13:30 -0700874// Tests that on if the peer sends too much data (i.e. violates the flow control
875// protocol), at the connection level (rather than the stream level) then we
876// terminate the connection.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500877TEST_P(QuicSpdyStreamTest, ConnectionFlowControlViolation) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500878 // Stream should not process data, so that data gets buffered in the
879 // sequencer, triggering flow control limits.
880 Initialize(!kShouldProcessData);
881
882 // Set a small flow control window on streams, and connection.
883 const uint64_t kStreamWindow = 50;
884 const uint64_t kConnectionWindow = 10;
885 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(),
886 kStreamWindow);
887 QuicFlowControllerPeer::SetReceiveWindowOffset(session_->flow_controller(),
888 kConnectionWindow);
889
890 ProcessHeaders(false, headers_);
891
892 // Send enough data to overflow the connection level flow control window.
vasilvvc48c8712019-03-11 13:38:16 -0700893 std::string body(kConnectionWindow + 1, 'a');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500894 std::unique_ptr<char[]> buf;
895 QuicByteCount header_length =
896 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700897 std::string header = std::string(buf.get(), header_length);
898 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500899
900 EXPECT_LT(data.size(), kStreamWindow);
901 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), false, 0,
902 QuicStringPiece(data));
903
904 EXPECT_CALL(*connection_,
905 CloseConnection(QUIC_FLOW_CONTROL_RECEIVED_TOO_MUCH_DATA, _, _));
906 stream_->OnStreamFrame(frame);
907}
908
bnc677451a2019-06-07 10:13:30 -0700909// An attempt to write a FIN with no data should not be flow control blocked,
910// even if the send window is 0.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500911TEST_P(QuicSpdyStreamTest, StreamFlowControlFinNotBlocked) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500912 Initialize(kShouldProcessData);
913
914 // Set a flow control limit of zero.
915 QuicFlowControllerPeer::SetReceiveWindowOffset(stream_->flow_controller(), 0);
916 EXPECT_EQ(0u, QuicFlowControllerPeer::ReceiveWindowOffset(
917 stream_->flow_controller()));
918
919 // Send a frame with a FIN but no data. This should not be blocked.
vasilvvc48c8712019-03-11 13:38:16 -0700920 std::string body = "";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500921 bool fin = true;
922
923 EXPECT_CALL(*connection_,
924 SendBlocked(GetNthClientInitiatedBidirectionalId(0)))
925 .Times(0);
926 EXPECT_CALL(*session_, WritevData(_, _, 0, _, FIN));
927
928 stream_->WriteOrBufferBody(body, fin);
929}
930
bnc677451a2019-06-07 10:13:30 -0700931// Test that receiving trailing headers from the peer via OnStreamHeaderList()
932// works, and can be read from the stream and consumed.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500933TEST_P(QuicSpdyStreamTest, ReceivingTrailersViaHeaderList) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500934 Initialize(kShouldProcessData);
935
936 // Receive initial headers.
937 size_t total_bytes = 0;
938 QuicHeaderList headers;
939 for (const auto& p : headers_) {
940 headers.OnHeader(p.first, p.second);
941 total_bytes += p.first.size() + p.second.size();
942 }
943
944 stream_->OnStreamHeadersPriority(kV3HighestPriority);
945 stream_->OnStreamHeaderList(/*fin=*/false, total_bytes, headers);
946 stream_->ConsumeHeaderList();
947
948 // Receive trailing headers.
949 SpdyHeaderBlock trailers_block;
950 trailers_block["key1"] = "value1";
951 trailers_block["key2"] = "value2";
952 trailers_block["key3"] = "value3";
953 SpdyHeaderBlock trailers_block_with_final_offset = trailers_block.Clone();
renjietang2abedac2019-05-20 14:04:50 -0700954 if (!VersionUsesQpack(GetParam().transport_version)) {
955 // :final-offset pseudo-header is only added if trailers are sent
956 // on the headers stream.
957 trailers_block_with_final_offset[kFinalOffsetHeaderKey] = "0";
958 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500959 total_bytes = 0;
960 QuicHeaderList trailers;
961 for (const auto& p : trailers_block_with_final_offset) {
962 trailers.OnHeader(p.first, p.second);
963 total_bytes += p.first.size() + p.second.size();
964 }
965 stream_->OnStreamHeaderList(/*fin=*/true, total_bytes, trailers);
966
967 // The trailers should be decompressed, and readable from the stream.
968 EXPECT_TRUE(stream_->trailers_decompressed());
969 EXPECT_EQ(trailers_block, stream_->received_trailers());
970
971 // IsDoneReading() returns false until trailers marked consumed.
972 EXPECT_FALSE(stream_->IsDoneReading());
973 stream_->MarkTrailersConsumed();
974 EXPECT_TRUE(stream_->IsDoneReading());
975}
976
bnc677451a2019-06-07 10:13:30 -0700977// Test that when receiving trailing headers with an offset before response
978// body, stream is closed at the right offset.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500979TEST_P(QuicSpdyStreamTest, ReceivingTrailersWithOffset) {
renjietang2abedac2019-05-20 14:04:50 -0700980 // kFinalOffsetHeaderKey is not used when HEADERS are sent on the
981 // request/response stream.
982 if (VersionUsesQpack(GetParam().transport_version)) {
983 return;
984 }
985
bnc677451a2019-06-07 10:13:30 -0700986 Initialize(kShouldProcessData);
987
QUICHE teama6ef0a62019-03-07 20:34:33 -0500988 // Receive initial headers.
989 QuicHeaderList headers = ProcessHeaders(false, headers_);
990 stream_->ConsumeHeaderList();
991
vasilvvc48c8712019-03-11 13:38:16 -0700992 const std::string body = "this is the body";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500993 std::unique_ptr<char[]> buf;
994 QuicByteCount header_length =
995 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -0700996 std::string header = std::string(buf.get(), header_length);
997 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500998
999 // Receive trailing headers.
1000 SpdyHeaderBlock trailers_block;
1001 trailers_block["key1"] = "value1";
1002 trailers_block["key2"] = "value2";
1003 trailers_block["key3"] = "value3";
1004 trailers_block[kFinalOffsetHeaderKey] =
1005 QuicTextUtils::Uint64ToString(data.size());
1006
1007 QuicHeaderList trailers = ProcessHeaders(true, trailers_block);
1008
1009 // The trailers should be decompressed, and readable from the stream.
1010 EXPECT_TRUE(stream_->trailers_decompressed());
1011
1012 // The final offset trailer will be consumed by QUIC.
1013 trailers_block.erase(kFinalOffsetHeaderKey);
1014 EXPECT_EQ(trailers_block, stream_->received_trailers());
1015
1016 // Consuming the trailers erases them from the stream.
1017 stream_->MarkTrailersConsumed();
1018 EXPECT_TRUE(stream_->FinishedReadingTrailers());
1019
1020 EXPECT_FALSE(stream_->IsDoneReading());
1021 // Receive and consume body.
1022 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), /*fin=*/false,
1023 0, data);
1024 stream_->OnStreamFrame(frame);
1025 EXPECT_EQ(body, stream_->data());
1026 EXPECT_TRUE(stream_->IsDoneReading());
1027}
1028
bnc677451a2019-06-07 10:13:30 -07001029// Test that receiving trailers without a final offset field is an error.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001030TEST_P(QuicSpdyStreamTest, ReceivingTrailersWithoutOffset) {
renjietang2abedac2019-05-20 14:04:50 -07001031 // kFinalOffsetHeaderKey is not used when HEADERS are sent on the
1032 // request/response stream.
1033 if (VersionUsesQpack(GetParam().transport_version)) {
1034 return;
1035 }
1036
bnc677451a2019-06-07 10:13:30 -07001037 Initialize(kShouldProcessData);
1038
QUICHE teama6ef0a62019-03-07 20:34:33 -05001039 // Receive initial headers.
1040 ProcessHeaders(false, headers_);
1041 stream_->ConsumeHeaderList();
1042
1043 // Receive trailing headers, without kFinalOffsetHeaderKey.
1044 SpdyHeaderBlock trailers_block;
1045 trailers_block["key1"] = "value1";
1046 trailers_block["key2"] = "value2";
1047 trailers_block["key3"] = "value3";
1048 auto trailers = AsHeaderList(trailers_block);
1049
1050 // Verify that the trailers block didn't contain a final offset.
1051 EXPECT_EQ("", trailers_block[kFinalOffsetHeaderKey].as_string());
1052
1053 // Receipt of the malformed trailers will close the connection.
1054 EXPECT_CALL(*connection_,
1055 CloseConnection(QUIC_INVALID_HEADERS_STREAM_DATA, _, _))
1056 .Times(1);
1057 stream_->OnStreamHeaderList(/*fin=*/true,
1058 trailers.uncompressed_header_bytes(), trailers);
1059}
1060
renjietang2abedac2019-05-20 14:04:50 -07001061TEST_P(QuicSpdyStreamTest, ReceivingTrailersOnRequestStream) {
renjietang2abedac2019-05-20 14:04:50 -07001062 if (!VersionUsesQpack(GetParam().transport_version)) {
1063 return;
1064 }
1065
bnc677451a2019-06-07 10:13:30 -07001066 Initialize(kShouldProcessData);
1067
renjietang2abedac2019-05-20 14:04:50 -07001068 // Receive initial headers.
1069 QuicHeaderList headers = ProcessHeaders(false, headers_);
1070 stream_->ConsumeHeaderList();
1071
1072 const std::string body = "this is the body";
1073 std::unique_ptr<char[]> buf;
1074 QuicByteCount header_length =
1075 encoder_.SerializeDataFrameHeader(body.length(), &buf);
1076 std::string header = std::string(buf.get(), header_length);
1077 std::string data = HasFrameHeader() ? header + body : body;
1078
1079 // Receive trailing headers.
1080 SpdyHeaderBlock trailers_block;
1081 trailers_block["key1"] = "value1";
1082 trailers_block["key2"] = "value2";
1083 trailers_block["key3"] = "value3";
1084
1085 QuicHeaderList trailers = ProcessHeaders(true, trailers_block);
1086
1087 // The trailers should be decompressed, and readable from the stream.
1088 EXPECT_TRUE(stream_->trailers_decompressed());
1089
1090 EXPECT_EQ(trailers_block, stream_->received_trailers());
1091
1092 // Consuming the trailers erases them from the stream.
1093 stream_->MarkTrailersConsumed();
1094 EXPECT_TRUE(stream_->FinishedReadingTrailers());
1095 EXPECT_TRUE(stream_->IsDoneReading());
1096
1097 // Receive and consume body.
1098 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), /*fin=*/false,
1099 0, data);
1100 stream_->OnStreamFrame(frame);
1101 EXPECT_EQ(body, stream_->data());
1102 EXPECT_TRUE(stream_->IsDoneReading());
1103}
1104
bnc677451a2019-06-07 10:13:30 -07001105// Test that received Trailers must always have the FIN set.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001106TEST_P(QuicSpdyStreamTest, ReceivingTrailersWithoutFin) {
renjietang2abedac2019-05-20 14:04:50 -07001107 // In IETF QUIC, there is no such thing as FIN flag on HTTP/3 frames like the
1108 // HEADERS frame.
1109 if (VersionUsesQpack(GetParam().transport_version)) {
1110 return;
1111 }
1112
bnc677451a2019-06-07 10:13:30 -07001113 Initialize(kShouldProcessData);
1114
QUICHE teama6ef0a62019-03-07 20:34:33 -05001115 // Receive initial headers.
1116 auto headers = AsHeaderList(headers_);
1117 stream_->OnStreamHeaderList(/*fin=*/false,
1118 headers.uncompressed_header_bytes(), headers);
1119 stream_->ConsumeHeaderList();
1120
1121 // Receive trailing headers with FIN deliberately set to false.
1122 SpdyHeaderBlock trailers_block;
1123 trailers_block["foo"] = "bar";
1124 auto trailers = AsHeaderList(trailers_block);
1125
1126 EXPECT_CALL(*connection_,
1127 CloseConnection(QUIC_INVALID_HEADERS_STREAM_DATA, _, _))
1128 .Times(1);
1129 stream_->OnStreamHeaderList(/*fin=*/false,
1130 trailers.uncompressed_header_bytes(), trailers);
1131}
1132
1133TEST_P(QuicSpdyStreamTest, ReceivingTrailersAfterHeadersWithFin) {
1134 // If headers are received with a FIN, no trailers should then arrive.
1135 Initialize(kShouldProcessData);
1136
1137 // Receive initial headers with FIN set.
1138 ProcessHeaders(true, headers_);
1139 stream_->ConsumeHeaderList();
1140
1141 // Receive trailing headers after FIN already received.
1142 SpdyHeaderBlock trailers_block;
1143 trailers_block["foo"] = "bar";
1144 EXPECT_CALL(*connection_,
1145 CloseConnection(QUIC_INVALID_HEADERS_STREAM_DATA, _, _))
1146 .Times(1);
1147 ProcessHeaders(true, trailers_block);
1148}
1149
bnc677451a2019-06-07 10:13:30 -07001150// If body data are received with a FIN, no trailers should then arrive.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001151TEST_P(QuicSpdyStreamTest, ReceivingTrailersAfterBodyWithFin) {
renjietang2abedac2019-05-20 14:04:50 -07001152 // If HEADERS frames are sent on the request/response stream,
1153 // then the sequencer will block them from reaching QuicSpdyStream
1154 // after the stream is closed.
1155 if (VersionUsesQpack(GetParam().transport_version)) {
1156 return;
1157 }
1158
bnc677451a2019-06-07 10:13:30 -07001159 Initialize(kShouldProcessData);
1160
QUICHE teama6ef0a62019-03-07 20:34:33 -05001161 // Receive initial headers without FIN set.
1162 ProcessHeaders(false, headers_);
1163 stream_->ConsumeHeaderList();
1164
1165 // Receive body data, with FIN.
1166 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), /*fin=*/true,
1167 0, "body");
1168 stream_->OnStreamFrame(frame);
1169
1170 // Receive trailing headers after FIN already received.
1171 SpdyHeaderBlock trailers_block;
1172 trailers_block["foo"] = "bar";
1173 EXPECT_CALL(*connection_,
1174 CloseConnection(QUIC_INVALID_HEADERS_STREAM_DATA, _, _))
1175 .Times(1);
1176 ProcessHeaders(true, trailers_block);
1177}
1178
1179TEST_P(QuicSpdyStreamTest, ClosingStreamWithNoTrailers) {
1180 // Verify that a stream receiving headers, body, and no trailers is correctly
1181 // marked as done reading on consumption of headers and body.
1182 Initialize(kShouldProcessData);
1183
1184 // Receive and consume initial headers with FIN not set.
1185 auto h = AsHeaderList(headers_);
1186 stream_->OnStreamHeaderList(/*fin=*/false, h.uncompressed_header_bytes(), h);
1187 stream_->ConsumeHeaderList();
1188
1189 // Receive and consume body with FIN set, and no trailers.
vasilvvc48c8712019-03-11 13:38:16 -07001190 std::string body(1024, 'x');
QUICHE teama6ef0a62019-03-07 20:34:33 -05001191 std::unique_ptr<char[]> buf;
1192 QuicByteCount header_length =
1193 encoder_.SerializeDataFrameHeader(body.length(), &buf);
vasilvvc48c8712019-03-11 13:38:16 -07001194 std::string header = std::string(buf.get(), header_length);
1195 std::string data = HasFrameHeader() ? header + body : body;
QUICHE teama6ef0a62019-03-07 20:34:33 -05001196
1197 QuicStreamFrame frame(GetNthClientInitiatedBidirectionalId(0), /*fin=*/true,
1198 0, data);
1199 stream_->OnStreamFrame(frame);
1200
1201 EXPECT_TRUE(stream_->IsDoneReading());
1202}
1203
bnc677451a2019-06-07 10:13:30 -07001204// Test that writing trailers will send a FIN, as Trailers are the last thing to
1205// be sent on a stream.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001206TEST_P(QuicSpdyStreamTest, WritingTrailersSendsAFin) {
nharperf5e68452019-05-29 17:24:18 -07001207 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1208 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1209 // enabled and fix it.
1210 return;
1211 }
bnc677451a2019-06-07 10:13:30 -07001212
QUICHE teama6ef0a62019-03-07 20:34:33 -05001213 Initialize(kShouldProcessData);
1214
renjietang2abedac2019-05-20 14:04:50 -07001215 if (VersionUsesQpack(GetParam().transport_version)) {
1216 // In this case, TestStream::WriteHeadersImpl() does not prevent writes.
1217 EXPECT_CALL(*session_, WritevData(stream_, stream_->id(), _, _, _))
1218 .Times(AtLeast(1));
1219 }
1220
QUICHE teama6ef0a62019-03-07 20:34:33 -05001221 // Write the initial headers, without a FIN.
1222 EXPECT_CALL(*stream_, WriteHeadersMock(false));
1223 stream_->WriteHeaders(SpdyHeaderBlock(), /*fin=*/false, nullptr);
1224
1225 // Writing trailers implicitly sends a FIN.
1226 SpdyHeaderBlock trailers;
1227 trailers["trailer key"] = "trailer value";
1228 EXPECT_CALL(*stream_, WriteHeadersMock(true));
1229 stream_->WriteTrailers(std::move(trailers), nullptr);
1230 EXPECT_TRUE(stream_->fin_sent());
1231}
1232
bnc677451a2019-06-07 10:13:30 -07001233// Test that when writing trailers, the trailers that are actually sent to the
1234// peer contain the final offset field indicating last byte of data.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001235TEST_P(QuicSpdyStreamTest, WritingTrailersFinalOffset) {
nharperf5e68452019-05-29 17:24:18 -07001236 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1237 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1238 // enabled and fix it.
1239 return;
1240 }
bnc677451a2019-06-07 10:13:30 -07001241
QUICHE teama6ef0a62019-03-07 20:34:33 -05001242 Initialize(kShouldProcessData);
1243
renjietang2abedac2019-05-20 14:04:50 -07001244 if (VersionUsesQpack(GetParam().transport_version)) {
1245 // In this case, TestStream::WriteHeadersImpl() does not prevent writes.
1246 EXPECT_CALL(*session_, WritevData(stream_, stream_->id(), _, _, _))
1247 .Times(AtLeast(1));
1248 }
1249
QUICHE teama6ef0a62019-03-07 20:34:33 -05001250 // Write the initial headers.
1251 EXPECT_CALL(*stream_, WriteHeadersMock(false));
1252 stream_->WriteHeaders(SpdyHeaderBlock(), /*fin=*/false, nullptr);
1253
1254 // Write non-zero body data to force a non-zero final offset.
1255 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
vasilvvc48c8712019-03-11 13:38:16 -07001256 std::string body(1024, 'x'); // 1 kB
QUICHE teama6ef0a62019-03-07 20:34:33 -05001257 QuicByteCount header_length = 0;
1258 if (HasFrameHeader()) {
1259 std::unique_ptr<char[]> buf;
1260 header_length = encoder_.SerializeDataFrameHeader(body.length(), &buf);
1261 }
1262
1263 stream_->WriteOrBufferBody(body, false);
1264
1265 // The final offset field in the trailing headers is populated with the
1266 // number of body bytes written (including queued bytes).
1267 SpdyHeaderBlock trailers;
1268 trailers["trailer key"] = "trailer value";
renjietang2abedac2019-05-20 14:04:50 -07001269
1270 SpdyHeaderBlock expected_trailers(trailers.Clone());
1271 // :final-offset pseudo-header is only added if trailers are sent
1272 // on the headers stream.
1273 if (!VersionUsesQpack(GetParam().transport_version)) {
1274 expected_trailers[kFinalOffsetHeaderKey] =
1275 QuicTextUtils::Uint64ToString(body.length() + header_length);
1276 }
1277
QUICHE teama6ef0a62019-03-07 20:34:33 -05001278 EXPECT_CALL(*stream_, WriteHeadersMock(true));
1279 stream_->WriteTrailers(std::move(trailers), nullptr);
renjietang2abedac2019-05-20 14:04:50 -07001280 EXPECT_EQ(expected_trailers, stream_->saved_headers());
QUICHE teama6ef0a62019-03-07 20:34:33 -05001281}
1282
bnc677451a2019-06-07 10:13:30 -07001283// Test that if trailers are written after all other data has been written
1284// (headers and body), that this closes the stream for writing.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001285TEST_P(QuicSpdyStreamTest, WritingTrailersClosesWriteSide) {
nharperf5e68452019-05-29 17:24:18 -07001286 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1287 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1288 // enabled and fix it.
1289 return;
1290 }
bnc677451a2019-06-07 10:13:30 -07001291
QUICHE teama6ef0a62019-03-07 20:34:33 -05001292 Initialize(kShouldProcessData);
1293
renjietang2abedac2019-05-20 14:04:50 -07001294 // Expect data being written on the stream. In addition to that, headers are
1295 // also written on the stream in case of IETF QUIC.
1296 EXPECT_CALL(*session_, WritevData(stream_, stream_->id(), _, _, _))
1297 .Times(AtLeast(1));
1298
QUICHE teama6ef0a62019-03-07 20:34:33 -05001299 // Write the initial headers.
1300 EXPECT_CALL(*stream_, WriteHeadersMock(false));
1301 stream_->WriteHeaders(SpdyHeaderBlock(), /*fin=*/false, nullptr);
1302
1303 // Write non-zero body data.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001304 const int kBodySize = 1 * 1024; // 1 kB
vasilvvc48c8712019-03-11 13:38:16 -07001305 stream_->WriteOrBufferBody(std::string(kBodySize, 'x'), false);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001306 EXPECT_EQ(0u, stream_->BufferedDataBytes());
1307
1308 // Headers and body have been fully written, there is no queued data. Writing
1309 // trailers marks the end of this stream, and thus the write side is closed.
1310 EXPECT_CALL(*stream_, WriteHeadersMock(true));
1311 stream_->WriteTrailers(SpdyHeaderBlock(), nullptr);
1312 EXPECT_TRUE(stream_->write_side_closed());
1313}
1314
bnc677451a2019-06-07 10:13:30 -07001315// Test that the stream is not closed for writing when trailers are sent while
1316// there are still body bytes queued.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001317TEST_P(QuicSpdyStreamTest, WritingTrailersWithQueuedBytes) {
renjietang2abedac2019-05-20 14:04:50 -07001318 // This test exercises sending trailers on the headers stream while data is
1319 // still queued on the response/request stream. In IETF QUIC, data and
1320 // trailers are sent on the same stream, so this test does not apply.
1321 if (VersionUsesQpack(GetParam().transport_version)) {
1322 return;
1323 }
1324
QUICHE teama6ef0a62019-03-07 20:34:33 -05001325 testing::InSequence seq;
1326 Initialize(kShouldProcessData);
1327
1328 // Write the initial headers.
1329 EXPECT_CALL(*stream_, WriteHeadersMock(false));
1330 stream_->WriteHeaders(SpdyHeaderBlock(), /*fin=*/false, nullptr);
1331
1332 // Write non-zero body data, but only consume partially, ensuring queueing.
1333 const int kBodySize = 1 * 1024; // 1 kB
1334 if (HasFrameHeader()) {
1335 EXPECT_CALL(*session_, WritevData(_, _, 3, _, NO_FIN));
1336 }
1337 EXPECT_CALL(*session_, WritevData(_, _, kBodySize, _, NO_FIN))
1338 .WillOnce(Return(QuicConsumedData(kBodySize - 1, false)));
vasilvvc48c8712019-03-11 13:38:16 -07001339 stream_->WriteOrBufferBody(std::string(kBodySize, 'x'), false);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001340 EXPECT_EQ(1u, stream_->BufferedDataBytes());
1341
1342 // Writing trailers will send a FIN, but not close the write side of the
1343 // stream as there are queued bytes.
1344 EXPECT_CALL(*stream_, WriteHeadersMock(true));
1345 stream_->WriteTrailers(SpdyHeaderBlock(), nullptr);
1346 EXPECT_TRUE(stream_->fin_sent());
1347 EXPECT_FALSE(stream_->write_side_closed());
1348
1349 // Writing the queued bytes will close the write side of the stream.
1350 EXPECT_CALL(*session_, WritevData(_, _, 1, _, NO_FIN));
1351 stream_->OnCanWrite();
1352 EXPECT_TRUE(stream_->write_side_closed());
1353}
1354
bnc677451a2019-06-07 10:13:30 -07001355// Test that it is not possible to write Trailers after a FIN has been sent.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001356TEST_P(QuicSpdyStreamTest, WritingTrailersAfterFIN) {
1357 // EXPECT_QUIC_BUG tests are expensive so only run one instance of them.
renjietang2abedac2019-05-20 14:04:50 -07001358 // In IETF QUIC, there is no such thing as FIN flag on HTTP/3 frames like the
1359 // HEADERS frame. That is version 99, which is element 0 of the array, so
1360 // pick another element.
1361 if (GetParam() != AllSupportedVersions()[1]) {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001362 return;
1363 }
1364
QUICHE teama6ef0a62019-03-07 20:34:33 -05001365 Initialize(kShouldProcessData);
1366
1367 // Write the initial headers, with a FIN.
1368 EXPECT_CALL(*stream_, WriteHeadersMock(true));
1369 stream_->WriteHeaders(SpdyHeaderBlock(), /*fin=*/true, nullptr);
1370 EXPECT_TRUE(stream_->fin_sent());
1371
1372 // Writing Trailers should fail, as the FIN has already been sent.
1373 // populated with the number of body bytes written.
1374 EXPECT_QUIC_BUG(stream_->WriteTrailers(SpdyHeaderBlock(), nullptr),
1375 "Trailers cannot be sent after a FIN");
1376}
1377
1378TEST_P(QuicSpdyStreamTest, HeaderStreamNotiferCorrespondingSpdyStream) {
nharperf5e68452019-05-29 17:24:18 -07001379 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1380 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1381 // enabled and fix it.
1382 return;
1383 }
bnc677451a2019-06-07 10:13:30 -07001384
QUICHE teama6ef0a62019-03-07 20:34:33 -05001385 Initialize(kShouldProcessData);
1386 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1387 testing::InSequence s;
1388 QuicReferenceCountedPointer<MockAckListener> ack_listener1(
1389 new MockAckListener());
1390 QuicReferenceCountedPointer<MockAckListener> ack_listener2(
1391 new MockAckListener());
1392 stream_->set_ack_listener(ack_listener1);
1393 stream2_->set_ack_listener(ack_listener2);
1394
1395 session_->headers_stream()->WriteOrBufferData("Header1", false,
1396 ack_listener1);
1397 stream_->WriteOrBufferBody("Test1", true);
1398
1399 session_->headers_stream()->WriteOrBufferData("Header2", false,
1400 ack_listener2);
1401 stream2_->WriteOrBufferBody("Test2", false);
1402
1403 QuicStreamFrame frame1(
1404 QuicUtils::GetHeadersStreamId(connection_->transport_version()), false, 0,
1405 "Header1");
vasilvvc48c8712019-03-11 13:38:16 -07001406 std::string header = "";
QUICHE teama6ef0a62019-03-07 20:34:33 -05001407 if (HasFrameHeader()) {
1408 std::unique_ptr<char[]> buffer;
1409 QuicByteCount header_length = encoder_.SerializeDataFrameHeader(5, &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001410 header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001411 }
1412 QuicStreamFrame frame2(stream_->id(), true, 0, header + "Test1");
1413 QuicStreamFrame frame3(
1414 QuicUtils::GetHeadersStreamId(connection_->transport_version()), false, 7,
1415 "Header2");
1416 QuicStreamFrame frame4(stream2_->id(), false, 0, header + "Test2");
1417
1418 EXPECT_CALL(*ack_listener1, OnPacketRetransmitted(7));
1419 session_->OnStreamFrameRetransmitted(frame1);
1420
1421 EXPECT_CALL(*ack_listener1, OnPacketAcked(7, _));
QUICHE team9467db02019-05-30 09:38:45 -07001422 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame1), QuicTime::Delta::Zero(),
1423 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001424 EXPECT_CALL(*ack_listener1, OnPacketAcked(5, _));
QUICHE team9467db02019-05-30 09:38:45 -07001425 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame2), QuicTime::Delta::Zero(),
1426 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001427 EXPECT_CALL(*ack_listener2, OnPacketAcked(7, _));
QUICHE team9467db02019-05-30 09:38:45 -07001428 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame3), QuicTime::Delta::Zero(),
1429 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001430 EXPECT_CALL(*ack_listener2, OnPacketAcked(5, _));
QUICHE team9467db02019-05-30 09:38:45 -07001431 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame4), QuicTime::Delta::Zero(),
1432 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001433}
1434
1435TEST_P(QuicSpdyStreamTest, StreamBecomesZombieWithWriteThatCloses) {
nharperf5e68452019-05-29 17:24:18 -07001436 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1437 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1438 // enabled and fix it.
1439 return;
1440 }
bnc677451a2019-06-07 10:13:30 -07001441
QUICHE teama6ef0a62019-03-07 20:34:33 -05001442 Initialize(kShouldProcessData);
1443 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1444 QuicStreamPeer::CloseReadSide(stream_);
1445 // This write causes stream to be closed.
1446 stream_->WriteOrBufferBody("Test1", true);
1447 // stream_ has unacked data and should become zombie.
1448 EXPECT_TRUE(QuicContainsKey(QuicSessionPeer::zombie_streams(session_.get()),
1449 stream_->id()));
1450 EXPECT_TRUE(QuicSessionPeer::closed_streams(session_.get()).empty());
1451}
1452
1453TEST_P(QuicSpdyStreamTest, OnPriorityFrame) {
1454 Initialize(kShouldProcessData);
1455 stream_->OnPriorityFrame(kV3HighestPriority);
1456 EXPECT_EQ(kV3HighestPriority, stream_->priority());
1457}
1458
1459TEST_P(QuicSpdyStreamTest, OnPriorityFrameAfterSendingData) {
nharperf5e68452019-05-29 17:24:18 -07001460 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1461 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1462 // enabled and fix it.
1463 return;
1464 }
bnc677451a2019-06-07 10:13:30 -07001465
QUICHE teama6ef0a62019-03-07 20:34:33 -05001466 testing::InSequence seq;
1467 Initialize(kShouldProcessData);
1468
1469 if (HasFrameHeader()) {
1470 EXPECT_CALL(*session_, WritevData(_, _, 2, _, NO_FIN));
1471 }
1472 EXPECT_CALL(*session_, WritevData(_, _, 4, _, FIN));
1473 stream_->WriteOrBufferBody("data", true);
1474 stream_->OnPriorityFrame(kV3HighestPriority);
1475 EXPECT_EQ(kV3HighestPriority, stream_->priority());
1476}
1477
1478TEST_P(QuicSpdyStreamTest, SetPriorityBeforeUpdateStreamPriority) {
1479 MockQuicConnection* connection = new StrictMock<MockQuicConnection>(
1480 &helper_, &alarm_factory_, Perspective::IS_SERVER,
1481 SupportedVersions(GetParam()));
1482 std::unique_ptr<TestMockUpdateStreamSession> session(
1483 new StrictMock<TestMockUpdateStreamSession>(connection));
1484 auto stream = new StrictMock<TestStream>(
1485 GetNthClientInitiatedBidirectionalStreamId(
1486 session->connection()->transport_version(), 0),
1487 session.get(),
1488 /*should_process_data=*/true);
1489 session->ActivateStream(QuicWrapUnique(stream));
1490
1491 // QuicSpdyStream::SetPriority() should eventually call UpdateStreamPriority()
1492 // on the session. Make sure stream->priority() returns the updated priority
1493 // if called within UpdateStreamPriority(). This expectation is enforced in
1494 // TestMockUpdateStreamSession::UpdateStreamPriority().
1495 session->SetExpectedStream(stream);
1496 session->SetExpectedPriority(kV3HighestPriority);
1497 stream->SetPriority(kV3HighestPriority);
1498
1499 session->SetExpectedPriority(kV3LowestPriority);
1500 stream->SetPriority(kV3LowestPriority);
1501}
1502
1503TEST_P(QuicSpdyStreamTest, StreamWaitsForAcks) {
nharperf5e68452019-05-29 17:24:18 -07001504 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1505 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1506 // enabled and fix it.
1507 return;
1508 }
bnc677451a2019-06-07 10:13:30 -07001509
QUICHE teama6ef0a62019-03-07 20:34:33 -05001510 Initialize(kShouldProcessData);
1511 QuicReferenceCountedPointer<MockAckListener> mock_ack_listener(
1512 new StrictMock<MockAckListener>);
1513 stream_->set_ack_listener(mock_ack_listener);
1514 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1515 // Stream is not waiting for acks initially.
1516 EXPECT_FALSE(stream_->IsWaitingForAcks());
1517 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1518
1519 // Send kData1.
1520 stream_->WriteOrBufferData("FooAndBar", false, nullptr);
1521 EXPECT_EQ(1u, QuicStreamPeer::SendBuffer(stream_).size());
1522 EXPECT_TRUE(stream_->IsWaitingForAcks());
1523 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(9, _));
1524 QuicByteCount newly_acked_length = 0;
1525 EXPECT_TRUE(stream_->OnStreamFrameAcked(0, 9, false, QuicTime::Delta::Zero(),
1526 &newly_acked_length));
1527 // Stream is not waiting for acks as all sent data is acked.
1528 EXPECT_FALSE(stream_->IsWaitingForAcks());
1529 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1530
1531 // Send kData2.
1532 stream_->WriteOrBufferData("FooAndBar", false, nullptr);
1533 EXPECT_TRUE(stream_->IsWaitingForAcks());
1534 EXPECT_EQ(1u, QuicStreamPeer::SendBuffer(stream_).size());
1535 // Send FIN.
1536 stream_->WriteOrBufferData("", true, nullptr);
1537 // Fin only frame is not stored in send buffer.
1538 EXPECT_EQ(1u, QuicStreamPeer::SendBuffer(stream_).size());
1539
1540 // kData2 is retransmitted.
1541 EXPECT_CALL(*mock_ack_listener, OnPacketRetransmitted(9));
1542 stream_->OnStreamFrameRetransmitted(9, 9, false);
1543
1544 // kData2 is acked.
1545 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(9, _));
1546 EXPECT_TRUE(stream_->OnStreamFrameAcked(9, 9, false, QuicTime::Delta::Zero(),
1547 &newly_acked_length));
1548 // Stream is waiting for acks as FIN is not acked.
1549 EXPECT_TRUE(stream_->IsWaitingForAcks());
1550 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1551
1552 // FIN is acked.
1553 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(0, _));
1554 EXPECT_TRUE(stream_->OnStreamFrameAcked(18, 0, true, QuicTime::Delta::Zero(),
1555 &newly_acked_length));
1556 EXPECT_FALSE(stream_->IsWaitingForAcks());
1557 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1558}
1559
1560TEST_P(QuicSpdyStreamTest, StreamDataGetAckedMultipleTimes) {
nharperf5e68452019-05-29 17:24:18 -07001561 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1562 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1563 // enabled and fix it.
1564 return;
1565 }
bnc677451a2019-06-07 10:13:30 -07001566
QUICHE teama6ef0a62019-03-07 20:34:33 -05001567 Initialize(kShouldProcessData);
1568 QuicReferenceCountedPointer<MockAckListener> mock_ack_listener(
1569 new StrictMock<MockAckListener>);
1570 stream_->set_ack_listener(mock_ack_listener);
1571 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1572 // Send [0, 27) and fin.
1573 stream_->WriteOrBufferData("FooAndBar", false, nullptr);
1574 stream_->WriteOrBufferData("FooAndBar", false, nullptr);
1575 stream_->WriteOrBufferData("FooAndBar", true, nullptr);
1576
1577 // Ack [0, 9), [5, 22) and [18, 26)
1578 // Verify [0, 9) 9 bytes are acked.
1579 QuicByteCount newly_acked_length = 0;
1580 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(9, _));
1581 EXPECT_TRUE(stream_->OnStreamFrameAcked(0, 9, false, QuicTime::Delta::Zero(),
1582 &newly_acked_length));
1583 EXPECT_EQ(2u, QuicStreamPeer::SendBuffer(stream_).size());
1584 // Verify [9, 22) 13 bytes are acked.
1585 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(13, _));
1586 EXPECT_TRUE(stream_->OnStreamFrameAcked(5, 17, false, QuicTime::Delta::Zero(),
1587 &newly_acked_length));
1588 EXPECT_EQ(1u, QuicStreamPeer::SendBuffer(stream_).size());
1589 // Verify [22, 26) 4 bytes are acked.
1590 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(4, _));
1591 EXPECT_TRUE(stream_->OnStreamFrameAcked(18, 8, false, QuicTime::Delta::Zero(),
1592 &newly_acked_length));
1593 EXPECT_EQ(1u, QuicStreamPeer::SendBuffer(stream_).size());
1594 EXPECT_TRUE(stream_->IsWaitingForAcks());
1595
1596 // Ack [0, 27).
1597 // Verify [26, 27) 1 byte is acked.
1598 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(1, _));
1599 EXPECT_TRUE(stream_->OnStreamFrameAcked(26, 1, false, QuicTime::Delta::Zero(),
1600 &newly_acked_length));
1601 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1602 EXPECT_TRUE(stream_->IsWaitingForAcks());
1603
1604 // Ack Fin. Verify OnPacketAcked is called.
1605 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(0, _));
1606 EXPECT_TRUE(stream_->OnStreamFrameAcked(27, 0, true, QuicTime::Delta::Zero(),
1607 &newly_acked_length));
1608 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1609 EXPECT_FALSE(stream_->IsWaitingForAcks());
1610
1611 // Ack [10, 27) and fin.
1612 // No new data is acked, verify OnPacketAcked is not called.
1613 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(_, _)).Times(0);
1614 EXPECT_FALSE(stream_->OnStreamFrameAcked(
1615 10, 17, true, QuicTime::Delta::Zero(), &newly_acked_length));
1616 EXPECT_EQ(0u, QuicStreamPeer::SendBuffer(stream_).size());
1617 EXPECT_FALSE(stream_->IsWaitingForAcks());
1618}
1619
1620// HTTP/3 only.
1621TEST_P(QuicSpdyStreamTest, HeadersAckNotReportedWriteOrBufferBody) {
nharperf5e68452019-05-29 17:24:18 -07001622 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1623 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1624 // enabled and fix it.
1625 return;
1626 }
bnc677451a2019-06-07 10:13:30 -07001627
QUICHE teama6ef0a62019-03-07 20:34:33 -05001628 if (!HasFrameHeader()) {
1629 return;
1630 }
bnc677451a2019-06-07 10:13:30 -07001631
1632 Initialize(kShouldProcessData);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001633 QuicReferenceCountedPointer<MockAckListener> mock_ack_listener(
1634 new StrictMock<MockAckListener>);
1635 stream_->set_ack_listener(mock_ack_listener);
vasilvvc48c8712019-03-11 13:38:16 -07001636 std::string body = "Test1";
1637 std::string body2(100, 'x');
QUICHE teama6ef0a62019-03-07 20:34:33 -05001638
1639 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1640 stream_->WriteOrBufferBody(body, false);
1641 stream_->WriteOrBufferBody(body2, true);
1642
1643 std::unique_ptr<char[]> buffer;
1644 QuicByteCount header_length =
1645 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001646 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001647
1648 header_length = encoder_.SerializeDataFrameHeader(body2.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001649 std::string header2 = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001650
1651 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(body.length(), _));
1652 QuicStreamFrame frame(stream_->id(), false, 0, header + body);
QUICHE team9467db02019-05-30 09:38:45 -07001653 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame), QuicTime::Delta::Zero(),
1654 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001655
1656 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(0, _));
1657 QuicStreamFrame frame2(stream_->id(), false, (header + body).length(),
1658 header2);
QUICHE team9467db02019-05-30 09:38:45 -07001659 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame2), QuicTime::Delta::Zero(),
1660 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001661
1662 EXPECT_CALL(*mock_ack_listener, OnPacketAcked(body2.length(), _));
1663 QuicStreamFrame frame3(stream_->id(), true,
1664 (header + body).length() + header2.length(), body2);
QUICHE team9467db02019-05-30 09:38:45 -07001665 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame3), QuicTime::Delta::Zero(),
1666 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001667
1668 EXPECT_TRUE(
1669 QuicSpdyStreamPeer::unacked_frame_headers_offsets(stream_).Empty());
1670}
1671
1672// HTTP/3 only.
1673TEST_P(QuicSpdyStreamTest, HeadersAckNotReportedWriteBodySlices) {
nharperf5e68452019-05-29 17:24:18 -07001674 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1675 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1676 // enabled and fix it.
1677 return;
1678 }
bnc677451a2019-06-07 10:13:30 -07001679
QUICHE teama6ef0a62019-03-07 20:34:33 -05001680 if (!HasFrameHeader()) {
1681 return;
1682 }
bnc677451a2019-06-07 10:13:30 -07001683
1684 Initialize(kShouldProcessData);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001685 QuicReferenceCountedPointer<MockAckListener> mock_ack_listener(
1686 new StrictMock<MockAckListener>);
1687 stream_->set_ack_listener(mock_ack_listener);
vasilvvc48c8712019-03-11 13:38:16 -07001688 std::string body = "Test1";
1689 std::string body2(100, 'x');
QUICHE teama6ef0a62019-03-07 20:34:33 -05001690 struct iovec body1_iov = {const_cast<char*>(body.data()), body.length()};
1691 struct iovec body2_iov = {const_cast<char*>(body2.data()), body2.length()};
1692 QuicMemSliceStorage storage(&body1_iov, 1,
1693 helper_.GetStreamSendBufferAllocator(), 1024);
1694 QuicMemSliceStorage storage2(&body2_iov, 1,
1695 helper_.GetStreamSendBufferAllocator(), 1024);
1696 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1697 stream_->WriteBodySlices(storage.ToSpan(), false);
1698 stream_->WriteBodySlices(storage2.ToSpan(), true);
1699
1700 std::unique_ptr<char[]> buffer;
1701 QuicByteCount header_length =
1702 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001703 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001704
1705 header_length = encoder_.SerializeDataFrameHeader(body2.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001706 std::string header2 = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001707
1708 EXPECT_CALL(*mock_ack_listener,
1709 OnPacketAcked(body.length() + body2.length(), _));
1710 QuicStreamFrame frame(stream_->id(), true, 0,
1711 header + body + header2 + body2);
QUICHE team9467db02019-05-30 09:38:45 -07001712 EXPECT_TRUE(session_->OnFrameAcked(QuicFrame(frame), QuicTime::Delta::Zero(),
1713 QuicTime::Zero()));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001714
1715 EXPECT_TRUE(
1716 QuicSpdyStreamPeer::unacked_frame_headers_offsets(stream_).Empty());
1717}
1718
1719// HTTP/3 only.
1720TEST_P(QuicSpdyStreamTest, HeaderBytesNotReportedOnRetransmission) {
nharperf5e68452019-05-29 17:24:18 -07001721 if (GetParam().handshake_protocol == PROTOCOL_TLS1_3) {
1722 // TODO(nharper, b/112643533): Figure out why this test fails when TLS is
1723 // enabled and fix it.
1724 return;
1725 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001726 if (!HasFrameHeader()) {
1727 return;
1728 }
bnc677451a2019-06-07 10:13:30 -07001729
1730 Initialize(kShouldProcessData);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001731 QuicReferenceCountedPointer<MockAckListener> mock_ack_listener(
1732 new StrictMock<MockAckListener>);
1733 stream_->set_ack_listener(mock_ack_listener);
vasilvvc48c8712019-03-11 13:38:16 -07001734 std::string body = "Test1";
1735 std::string body2(100, 'x');
QUICHE teama6ef0a62019-03-07 20:34:33 -05001736
1737 EXPECT_CALL(*session_, WritevData(_, _, _, _, _)).Times(AtLeast(1));
1738 stream_->WriteOrBufferBody(body, false);
1739 stream_->WriteOrBufferBody(body2, true);
1740
1741 std::unique_ptr<char[]> buffer;
1742 QuicByteCount header_length =
1743 encoder_.SerializeDataFrameHeader(body.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001744 std::string header = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001745
1746 header_length = encoder_.SerializeDataFrameHeader(body2.length(), &buffer);
vasilvvc48c8712019-03-11 13:38:16 -07001747 std::string header2 = std::string(buffer.get(), header_length);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001748
1749 EXPECT_CALL(*mock_ack_listener, OnPacketRetransmitted(body.length()));
1750 QuicStreamFrame frame(stream_->id(), false, 0, header + body);
1751 session_->OnStreamFrameRetransmitted(frame);
1752
1753 EXPECT_CALL(*mock_ack_listener, OnPacketRetransmitted(body2.length()));
1754 QuicStreamFrame frame2(stream_->id(), true, (header + body).length(),
1755 header2 + body2);
1756 session_->OnStreamFrameRetransmitted(frame2);
1757
1758 EXPECT_FALSE(
1759 QuicSpdyStreamPeer::unacked_frame_headers_offsets(stream_).Empty());
1760}
1761
renjietang2abedac2019-05-20 14:04:50 -07001762TEST_P(QuicSpdyStreamTest, HeadersFrameOnRequestStream) {
1763 if (!VersionUsesQpack(GetParam().transport_version)) {
1764 return;
1765 }
1766
1767 Initialize(kShouldProcessData);
1768
1769 // QPACK encoded header block with single header field "foo: bar".
1770 std::string headers_frame_payload =
1771 QuicTextUtils::HexDecode("00002a94e703626172");
1772 std::unique_ptr<char[]> headers_buffer;
1773 QuicByteCount headers_frame_header_length =
1774 encoder_.SerializeHeadersFrameHeader(headers_frame_payload.length(),
1775 &headers_buffer);
1776 QuicStringPiece headers_frame_header(headers_buffer.get(),
1777 headers_frame_header_length);
1778
1779 std::string data_frame_payload = "some data";
1780 std::unique_ptr<char[]> data_buffer;
1781 QuicByteCount data_frame_header_length = encoder_.SerializeDataFrameHeader(
1782 data_frame_payload.length(), &data_buffer);
1783 QuicStringPiece data_frame_header(data_buffer.get(),
1784 data_frame_header_length);
1785
1786 // QPACK encoded header block with single header field
1787 // "custom-key: custom-value".
1788 std::string trailers_frame_payload =
1789 QuicTextUtils::HexDecode("00002f0125a849e95ba97d7f8925a849e95bb8e8b4bf");
1790 std::unique_ptr<char[]> trailers_buffer;
1791 QuicByteCount trailers_frame_header_length =
1792 encoder_.SerializeHeadersFrameHeader(trailers_frame_payload.length(),
1793 &trailers_buffer);
1794 QuicStringPiece trailers_frame_header(trailers_buffer.get(),
1795 trailers_frame_header_length);
1796
1797 std::string stream_frame_payload = QuicStrCat(
1798 headers_frame_header, headers_frame_payload, data_frame_header,
1799 data_frame_payload, trailers_frame_header, trailers_frame_payload);
1800 QuicStreamFrame frame(stream_->id(), false, 0, stream_frame_payload);
1801 stream_->OnStreamFrame(frame);
1802
1803 auto it = stream_->header_list().begin();
1804 ASSERT_TRUE(it != stream_->header_list().end());
1805 EXPECT_EQ("foo", it->first);
1806 EXPECT_EQ("bar", it->second);
1807 ++it;
1808 EXPECT_TRUE(it == stream_->header_list().end());
1809
1810 // QuicSpdyStream only calls OnBodyAvailable()
1811 // after the header list has been consumed.
1812 EXPECT_EQ("", stream_->data());
1813 stream_->ConsumeHeaderList();
1814 EXPECT_EQ("some data", stream_->data());
1815
1816 const spdy::SpdyHeaderBlock& trailers = stream_->received_trailers();
1817 EXPECT_THAT(trailers, testing::ElementsAre(
1818 testing::Pair("custom-key", "custom-value")));
1819}
1820
renjietangbd1a0392019-05-31 11:36:24 -07001821TEST_P(QuicSpdyStreamTest, ProcessBodyAfterTrailers) {
1822 if (!VersionUsesQpack(GetParam().transport_version)) {
1823 return;
1824 }
bnc677451a2019-06-07 10:13:30 -07001825
1826 Initialize(!kShouldProcessData);
renjietangbd1a0392019-05-31 11:36:24 -07001827
1828 // QPACK encoded header block with single header field "foo: bar".
1829 std::string headers_frame_payload =
1830 QuicTextUtils::HexDecode("00002a94e703626172");
1831 std::unique_ptr<char[]> headers_buffer;
1832 QuicByteCount headers_frame_header_length =
1833 encoder_.SerializeHeadersFrameHeader(headers_frame_payload.length(),
1834 &headers_buffer);
1835 QuicStringPiece headers_frame_header(headers_buffer.get(),
1836 headers_frame_header_length);
1837
1838 std::string data_frame_payload = "some data";
1839 std::unique_ptr<char[]> data_buffer;
1840 QuicByteCount data_frame_header_length = encoder_.SerializeDataFrameHeader(
1841 data_frame_payload.length(), &data_buffer);
1842 QuicStringPiece data_frame_header(data_buffer.get(),
1843 data_frame_header_length);
1844
1845 // A header block that will take more than one block of sequencer buffer.
1846 // This ensures that when the trailers are consumed, some buffer buckets will
1847 // be freed.
1848 SpdyHeaderBlock trailers_block;
1849 trailers_block["key1"] = std::string(10000, 'x');
1850 std::string trailers_frame_payload =
1851 EncodeQpackHeaders(stream_->id(), &trailers_block);
1852
1853 std::unique_ptr<char[]> trailers_buffer;
1854 QuicByteCount trailers_frame_header_length =
1855 encoder_.SerializeHeadersFrameHeader(trailers_frame_payload.length(),
1856 &trailers_buffer);
1857 QuicStringPiece trailers_frame_header(trailers_buffer.get(),
1858 trailers_frame_header_length);
1859
1860 std::string stream_frame_payload = QuicStrCat(
1861 headers_frame_header, headers_frame_payload, data_frame_header,
1862 data_frame_payload, trailers_frame_header, trailers_frame_payload);
1863 QuicStreamFrame frame(stream_->id(), false, 0, stream_frame_payload);
1864 stream_->OnStreamFrame(frame);
1865
1866 stream_->ConsumeHeaderList();
1867 stream_->MarkTrailersConsumed();
1868 char buffer[2048];
1869 struct iovec vec;
1870 vec.iov_base = buffer;
1871 vec.iov_len = QUIC_ARRAYSIZE(buffer);
1872 size_t bytes_read = stream_->Readv(&vec, 1);
1873 std::string data(buffer, bytes_read);
1874 EXPECT_EQ("some data", data);
1875}
1876
bnc677451a2019-06-07 10:13:30 -07001877// The test stream will receive a stream frame containing malformed headers and
1878// normal body. Make sure the http decoder stops processing body after the
1879// connection shuts down.
renjietang546a6282019-06-03 10:21:21 -07001880TEST_P(QuicSpdyStreamTest, MalformedHeadersStopHttpDecoder) {
renjietang546a6282019-06-03 10:21:21 -07001881 if (!VersionUsesQpack(GetParam().transport_version)) {
1882 return;
1883 }
1884
bnc677451a2019-06-07 10:13:30 -07001885 testing::InSequence s;
1886
renjietang546a6282019-06-03 10:21:21 -07001887 Initialize(kShouldProcessData);
1888 connection_->AdvanceTime(QuicTime::Delta::FromSeconds(1));
1889
1890 // Random bad headers.
1891 std::string headers_frame_payload =
1892 QuicTextUtils::HexDecode("00002a94e7036261");
1893 std::unique_ptr<char[]> headers_buffer;
1894 QuicByteCount headers_frame_header_length =
1895 encoder_.SerializeHeadersFrameHeader(headers_frame_payload.length(),
1896 &headers_buffer);
1897 QuicStringPiece headers_frame_header(headers_buffer.get(),
1898 headers_frame_header_length);
1899
1900 std::string data_frame_payload = "some data";
1901 std::unique_ptr<char[]> data_buffer;
1902 QuicByteCount data_frame_header_length = encoder_.SerializeDataFrameHeader(
1903 data_frame_payload.length(), &data_buffer);
1904 QuicStringPiece data_frame_header(data_buffer.get(),
1905 data_frame_header_length);
1906
1907 std::string stream_frame_payload =
1908 QuicStrCat(headers_frame_header, headers_frame_payload, data_frame_header,
1909 data_frame_payload);
1910 QuicStreamFrame frame(stream_->id(), false, 0, stream_frame_payload);
1911
1912 EXPECT_CALL(*connection_,
1913 CloseConnection(QUIC_DECOMPRESSION_FAILURE,
1914 "Error decompressing header block on stream 4: "
1915 "Incomplete header block.",
1916 _))
1917 .WillOnce(
1918 (Invoke([this](QuicErrorCode error, const std::string& error_details,
1919 ConnectionCloseBehavior connection_close_behavior) {
1920 connection_->ReallyCloseConnection(error, error_details,
1921 connection_close_behavior);
1922 })));
1923 EXPECT_CALL(*connection_, SendConnectionClosePacket(_, _));
1924 EXPECT_CALL(*session_, OnConnectionClosed(_, _, _))
1925 .WillOnce(
1926 Invoke([this](QuicErrorCode error, const std::string& error_details,
1927 ConnectionCloseSource source) {
1928 session_->ReallyOnConnectionClosed(error, error_details, source);
1929 }));
1930 EXPECT_CALL(*session_, SendRstStream(_, _, _));
1931 EXPECT_CALL(*session_, SendRstStream(_, _, _));
1932 stream_->OnStreamFrame(frame);
1933}
1934
QUICHE teama6ef0a62019-03-07 20:34:33 -05001935} // namespace
1936} // namespace test
1937} // namespace quic