blob: 8b546b0ed9f84d8cc924dd1a5939aefda8096e85 [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// Copyright (c) 2018 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/qpack/qpack_decoder.h"
6
7#include <algorithm>
8
vasilvv0fb44432019-03-13 22:47:36 -07009#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050010#include "net/third_party/quiche/src/quic/platform/api/quic_test.h"
bnce42f7ad2019-10-25 17:46:31 -070011#include "net/third_party/quiche/src/quic/test_tools/qpack/qpack_decoder_test_utils.h"
12#include "net/third_party/quiche/src/quic/test_tools/qpack/qpack_test_utils.h"
QUICHE team11f55d42019-12-11 10:36:09 -080013#include "net/third_party/quiche/src/common/platform/api/quiche_string_piece.h"
14#include "net/third_party/quiche/src/common/platform/api/quiche_text_utils.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050015#include "net/third_party/quiche/src/spdy/core/spdy_header_block.h"
16
bncfb4f4fc2019-11-18 17:14:56 -080017using ::testing::_;
QUICHE teama6ef0a62019-03-07 20:34:33 -050018using ::testing::Eq;
bncfb4f4fc2019-11-18 17:14:56 -080019using ::testing::Invoke;
bnc098ff612019-07-09 03:24:37 -070020using ::testing::Mock;
QUICHE teama6ef0a62019-03-07 20:34:33 -050021using ::testing::Sequence;
22using ::testing::StrictMock;
23using ::testing::Values;
24
25namespace quic {
26namespace test {
27namespace {
28
29// Header Acknowledgement decoder stream instruction with stream_id = 1.
30const char* const kHeaderAcknowledgement = "\x81";
31
bnc57b5f622019-08-21 14:07:44 -070032const uint64_t kMaximumDynamicTableCapacity = 1024;
33const uint64_t kMaximumBlockedStreams = 1;
bnc4c664c52019-08-04 18:14:12 -070034
QUICHE teama6ef0a62019-03-07 20:34:33 -050035class QpackDecoderTest : public QuicTestWithParam<FragmentMode> {
36 protected:
37 QpackDecoderTest()
bnc57b5f622019-08-21 14:07:44 -070038 : qpack_decoder_(kMaximumDynamicTableCapacity,
39 kMaximumBlockedStreams,
40 &encoder_stream_error_delegate_),
renjietang8a2df8f2019-08-07 10:43:52 -070041 fragment_mode_(GetParam()) {
42 qpack_decoder_.set_qpack_stream_sender_delegate(
43 &decoder_stream_sender_delegate_);
44 }
QUICHE teama6ef0a62019-03-07 20:34:33 -050045
46 ~QpackDecoderTest() override = default;
47
bncfb4f4fc2019-11-18 17:14:56 -080048 void SetUp() override {
49 // Destroy QpackProgressiveDecoder on error to test that it does not crash.
50 // See https://crbug.com/1025209.
51 ON_CALL(handler_, OnDecodingErrorDetected(_))
QUICHE team11f55d42019-12-11 10:36:09 -080052 .WillByDefault(
53 Invoke([this](quiche::QuicheStringPiece /* error_message */) {
54 progressive_decoder_.reset();
55 }));
bncfb4f4fc2019-11-18 17:14:56 -080056 }
57
QUICHE team11f55d42019-12-11 10:36:09 -080058 void DecodeEncoderStreamData(quiche::QuicheStringPiece data) {
bnc7d9e2a72019-08-09 08:20:10 -070059 qpack_decoder_.encoder_stream_receiver()->Decode(data);
QUICHE teama6ef0a62019-03-07 20:34:33 -050060 }
61
bnc57b5f622019-08-21 14:07:44 -070062 std::unique_ptr<QpackProgressiveDecoder> CreateProgressiveDecoder(
63 QuicStreamId stream_id) {
64 return qpack_decoder_.CreateProgressiveDecoder(stream_id, &handler_);
65 }
66
bnc098ff612019-07-09 03:24:37 -070067 // Set up |progressive_decoder_|.
68 void StartDecoding() {
bnc57b5f622019-08-21 14:07:44 -070069 progressive_decoder_ = CreateProgressiveDecoder(/* stream_id = */ 1);
bnc098ff612019-07-09 03:24:37 -070070 }
71
72 // Pass header block data to QpackProgressiveDecoder::Decode()
73 // in fragments dictated by |fragment_mode_|.
QUICHE team11f55d42019-12-11 10:36:09 -080074 void DecodeData(quiche::QuicheStringPiece data) {
QUICHE teama6ef0a62019-03-07 20:34:33 -050075 auto fragment_size_generator =
76 FragmentModeToFragmentSizeGenerator(fragment_mode_);
bncfb4f4fc2019-11-18 17:14:56 -080077 while (progressive_decoder_ && !data.empty()) {
QUICHE teama6ef0a62019-03-07 20:34:33 -050078 size_t fragment_size = std::min(fragment_size_generator(), data.size());
bnc098ff612019-07-09 03:24:37 -070079 progressive_decoder_->Decode(data.substr(0, fragment_size));
QUICHE teama6ef0a62019-03-07 20:34:33 -050080 data = data.substr(fragment_size);
81 }
bnc098ff612019-07-09 03:24:37 -070082 }
83
84 // Signal end of header block to QpackProgressiveDecoder.
85 void EndDecoding() {
bncfb4f4fc2019-11-18 17:14:56 -080086 if (progressive_decoder_) {
87 progressive_decoder_->EndHeaderBlock();
88 }
89 // If no error was detected, |*progressive_decoder_| is kept alive so that
90 // it can handle callbacks later in case of blocked decoding.
bnc098ff612019-07-09 03:24:37 -070091 }
92
93 // Decode an entire header block.
QUICHE team11f55d42019-12-11 10:36:09 -080094 void DecodeHeaderBlock(quiche::QuicheStringPiece data) {
bnc098ff612019-07-09 03:24:37 -070095 StartDecoding();
96 DecodeData(data);
97 EndDecoding();
QUICHE teama6ef0a62019-03-07 20:34:33 -050098 }
99
100 StrictMock<MockEncoderStreamErrorDelegate> encoder_stream_error_delegate_;
renjietangc2aa5cb2019-06-20 12:22:53 -0700101 StrictMock<MockQpackStreamSenderDelegate> decoder_stream_sender_delegate_;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500102 StrictMock<MockHeadersHandler> handler_;
103
104 private:
105 QpackDecoder qpack_decoder_;
106 const FragmentMode fragment_mode_;
bnc098ff612019-07-09 03:24:37 -0700107 std::unique_ptr<QpackProgressiveDecoder> progressive_decoder_;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500108};
109
QUICHE team1bfe0d72019-09-23 04:50:47 -0700110INSTANTIATE_TEST_SUITE_P(All,
vasilvvc48c8712019-03-11 13:38:16 -0700111 QpackDecoderTest,
112 Values(FragmentMode::kSingleChunk,
113 FragmentMode::kOctetByOctet));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500114
115TEST_P(QpackDecoderTest, NoPrefix) {
116 EXPECT_CALL(handler_,
117 OnDecodingErrorDetected(Eq("Incomplete header data prefix.")));
118
119 // Header Data Prefix is at least two bytes long.
QUICHE team11f55d42019-12-11 10:36:09 -0800120 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("00"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500121}
122
bncfb4f4fc2019-11-18 17:14:56 -0800123// Regression test for https://1025209: QpackProgressiveDecoder must not crash
124// in Decode() if it is destroyed by handler_.OnDecodingErrorDetected().
125TEST_P(QpackDecoderTest, InvalidPrefix) {
126 StartDecoding();
127
128 EXPECT_CALL(handler_,
129 OnDecodingErrorDetected(Eq("Encoded integer too large.")));
130
131 // Encoded Required Insert Count in Header Data Prefix is too large.
QUICHE team11f55d42019-12-11 10:36:09 -0800132 DecodeData(
133 quiche::QuicheTextUtils::HexDecode("ffffffffffffffffffffffffffff"));
bncfb4f4fc2019-11-18 17:14:56 -0800134}
135
QUICHE teama6ef0a62019-03-07 20:34:33 -0500136TEST_P(QpackDecoderTest, EmptyHeaderBlock) {
137 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500138
QUICHE team11f55d42019-12-11 10:36:09 -0800139 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("0000"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500140}
141
142TEST_P(QpackDecoderTest, LiteralEntryEmptyName) {
143 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(""), Eq("foo")));
144 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500145
QUICHE team11f55d42019-12-11 10:36:09 -0800146 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("00002003666f6f"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500147}
148
149TEST_P(QpackDecoderTest, LiteralEntryEmptyValue) {
150 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("")));
151 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500152
QUICHE team11f55d42019-12-11 10:36:09 -0800153 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("000023666f6f00"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500154}
155
156TEST_P(QpackDecoderTest, LiteralEntryEmptyNameAndValue) {
157 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(""), Eq("")));
158 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500159
QUICHE team11f55d42019-12-11 10:36:09 -0800160 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("00002000"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500161}
162
163TEST_P(QpackDecoderTest, SimpleLiteralEntry) {
164 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
165 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500166
QUICHE team11f55d42019-12-11 10:36:09 -0800167 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("000023666f6f03626172"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500168}
169
170TEST_P(QpackDecoderTest, MultipleLiteralEntries) {
171 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
vasilvvc48c8712019-03-11 13:38:16 -0700172 std::string str(127, 'a');
QUICHE team11f55d42019-12-11 10:36:09 -0800173 EXPECT_CALL(handler_,
174 OnHeaderDecoded(Eq("foobaar"), quiche::QuicheStringPiece(str)));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500175 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500176
QUICHE team11f55d42019-12-11 10:36:09 -0800177 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500178 "0000" // prefix
179 "23666f6f03626172" // foo: bar
180 "2700666f6f62616172" // 7 octet long header name, the smallest number
181 // that does not fit on a 3-bit prefix.
182 "7f0061616161616161" // 127 octet long header value, the smallest number
183 "616161616161616161" // that does not fit on a 7-bit prefix.
184 "6161616161616161616161616161616161616161616161616161616161616161616161"
185 "6161616161616161616161616161616161616161616161616161616161616161616161"
186 "6161616161616161616161616161616161616161616161616161616161616161616161"
187 "616161616161"));
188}
189
190// Name Length value is too large for varint decoder to decode.
191TEST_P(QpackDecoderTest, NameLenTooLargeForVarintDecoder) {
192 EXPECT_CALL(handler_,
193 OnDecodingErrorDetected(Eq("Encoded integer too large.")));
194
QUICHE team11f55d42019-12-11 10:36:09 -0800195 DecodeHeaderBlock(
196 quiche::QuicheTextUtils::HexDecode("000027ffffffffffffffffffff"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500197}
198
199// Name Length value can be decoded by varint decoder but exceeds 1 MB limit.
200TEST_P(QpackDecoderTest, NameLenExceedsLimit) {
201 EXPECT_CALL(handler_,
202 OnDecodingErrorDetected(Eq("String literal too long.")));
203
QUICHE team11f55d42019-12-11 10:36:09 -0800204 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("000027ffff7f"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500205}
206
207// Value Length value is too large for varint decoder to decode.
208TEST_P(QpackDecoderTest, ValueLenTooLargeForVarintDecoder) {
209 EXPECT_CALL(handler_,
210 OnDecodingErrorDetected(Eq("Encoded integer too large.")));
211
212 DecodeHeaderBlock(
QUICHE team11f55d42019-12-11 10:36:09 -0800213 quiche::QuicheTextUtils::HexDecode("000023666f6f7fffffffffffffffffffff"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500214}
215
216// Value Length value can be decoded by varint decoder but exceeds 1 MB limit.
217TEST_P(QpackDecoderTest, ValueLenExceedsLimit) {
218 EXPECT_CALL(handler_,
219 OnDecodingErrorDetected(Eq("String literal too long.")));
220
QUICHE team11f55d42019-12-11 10:36:09 -0800221 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("000023666f6f7fffff7f"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500222}
223
224TEST_P(QpackDecoderTest, IncompleteHeaderBlock) {
225 EXPECT_CALL(handler_,
226 OnDecodingErrorDetected(Eq("Incomplete header block.")));
227
QUICHE team11f55d42019-12-11 10:36:09 -0800228 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("00002366"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500229}
230
231TEST_P(QpackDecoderTest, HuffmanSimple) {
232 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("custom-key"), Eq("custom-value")));
233 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500234
QUICHE team11f55d42019-12-11 10:36:09 -0800235 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
236 "00002f0125a849e95ba97d7f8925a849e95bb8e8b4bf"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500237}
238
239TEST_P(QpackDecoderTest, AlternatingHuffmanNonHuffman) {
240 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("custom-key"), Eq("custom-value")))
241 .Times(4);
242 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500243
QUICHE team11f55d42019-12-11 10:36:09 -0800244 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500245 "0000" // Prefix.
246 "2f0125a849e95ba97d7f" // Huffman-encoded name.
247 "8925a849e95bb8e8b4bf" // Huffman-encoded value.
248 "2703637573746f6d2d6b6579" // Non-Huffman encoded name.
249 "0c637573746f6d2d76616c7565" // Non-Huffman encoded value.
250 "2f0125a849e95ba97d7f" // Huffman-encoded name.
251 "0c637573746f6d2d76616c7565" // Non-Huffman encoded value.
252 "2703637573746f6d2d6b6579" // Non-Huffman encoded name.
253 "8925a849e95bb8e8b4bf")); // Huffman-encoded value.
254}
255
256TEST_P(QpackDecoderTest, HuffmanNameDoesNotHaveEOSPrefix) {
QUICHE team11f55d42019-12-11 10:36:09 -0800257 EXPECT_CALL(handler_, OnDecodingErrorDetected(quiche::QuicheStringPiece(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500258 "Error in Huffman-encoded string.")));
259
260 // 'y' ends in 0b0 on the most significant bit of the last byte.
261 // The remaining 7 bits must be a prefix of EOS, which is all 1s.
QUICHE team11f55d42019-12-11 10:36:09 -0800262 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
263 "00002f0125a849e95ba97d7e8925a849e95bb8e8b4bf"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500264}
265
266TEST_P(QpackDecoderTest, HuffmanValueDoesNotHaveEOSPrefix) {
QUICHE team11f55d42019-12-11 10:36:09 -0800267 EXPECT_CALL(handler_, OnDecodingErrorDetected(quiche::QuicheStringPiece(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500268 "Error in Huffman-encoded string.")));
269
270 // 'e' ends in 0b101, taking up the 3 most significant bits of the last byte.
271 // The remaining 5 bits must be a prefix of EOS, which is all 1s.
QUICHE team11f55d42019-12-11 10:36:09 -0800272 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
273 "00002f0125a849e95ba97d7f8925a849e95bb8e8b4be"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500274}
275
276TEST_P(QpackDecoderTest, HuffmanNameEOSPrefixTooLong) {
QUICHE team11f55d42019-12-11 10:36:09 -0800277 EXPECT_CALL(handler_, OnDecodingErrorDetected(quiche::QuicheStringPiece(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500278 "Error in Huffman-encoded string.")));
279
280 // The trailing EOS prefix must be at most 7 bits long. Appending one octet
281 // with value 0xff is invalid, even though 0b111111111111111 (15 bits) is a
282 // prefix of EOS.
QUICHE team11f55d42019-12-11 10:36:09 -0800283 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500284 "00002f0225a849e95ba97d7fff8925a849e95bb8e8b4bf"));
285}
286
287TEST_P(QpackDecoderTest, HuffmanValueEOSPrefixTooLong) {
QUICHE team11f55d42019-12-11 10:36:09 -0800288 EXPECT_CALL(handler_, OnDecodingErrorDetected(quiche::QuicheStringPiece(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500289 "Error in Huffman-encoded string.")));
290
291 // The trailing EOS prefix must be at most 7 bits long. Appending one octet
292 // with value 0xff is invalid, even though 0b1111111111111 (13 bits) is a
293 // prefix of EOS.
QUICHE team11f55d42019-12-11 10:36:09 -0800294 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500295 "00002f0125a849e95ba97d7f8a25a849e95bb8e8b4bfff"));
296}
297
298TEST_P(QpackDecoderTest, StaticTable) {
299 // A header name that has multiple entries with different values.
300 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("GET")));
301 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("POST")));
302 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("TRACE")));
303
304 // A header name that has a single entry with non-empty value.
305 EXPECT_CALL(handler_,
306 OnHeaderDecoded(Eq("accept-encoding"), Eq("gzip, deflate, br")));
307 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("accept-encoding"), Eq("compress")));
308 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("accept-encoding"), Eq("")));
309
310 // A header name that has a single entry with empty value.
311 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("location"), Eq("")));
312 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("location"), Eq("foo")));
313
314 EXPECT_CALL(handler_, OnDecodingCompleted());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500315
QUICHE team11f55d42019-12-11 10:36:09 -0800316 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500317 "0000d1dfccd45f108621e9aec2a11f5c8294e75f000554524143455f1000"));
318}
319
320TEST_P(QpackDecoderTest, TooHighStaticTableIndex) {
321 // This is the last entry in the static table with index 98.
322 EXPECT_CALL(handler_,
323 OnHeaderDecoded(Eq("x-frame-options"), Eq("sameorigin")));
324
325 // Addressing entry 99 should trigger an error.
326 EXPECT_CALL(handler_,
327 OnDecodingErrorDetected(Eq("Static table entry not found.")));
328
QUICHE team11f55d42019-12-11 10:36:09 -0800329 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("0000ff23ff24"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500330}
331
332TEST_P(QpackDecoderTest, DynamicTable) {
QUICHE team11f55d42019-12-11 10:36:09 -0800333 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode(
bnc40e9a7b2019-08-30 05:27:08 -0700334 "3fe107" // Set dynamic table capacity to 1024.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500335 "6294e703626172" // Add literal entry with name "foo" and value "bar".
336 "80035a5a5a" // Add entry with name of dynamic table entry index 0
337 // (relative index) and value "ZZZ".
338 "cf8294e7" // Add entry with name of static table entry index 15
339 // and value "foo".
340 "01")); // Duplicate entry with relative index 1.
341
342 // Now there are four entries in the dynamic table.
343 // Entry 0: "foo", "bar"
344 // Entry 1: "foo", "ZZZ"
345 // Entry 2: ":method", "foo"
346 // Entry 3: "foo", "ZZZ"
347
348 // Use a Sequence to test that mock methods are called in order.
349 Sequence s;
350
351 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar"))).InSequence(s);
352 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
353 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("foo")))
354 .InSequence(s);
355 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
356 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("ZZ"))).InSequence(s);
357 EXPECT_CALL(decoder_stream_sender_delegate_,
renjietangc2aa5cb2019-06-20 12:22:53 -0700358 WriteStreamData(Eq(kHeaderAcknowledgement)))
QUICHE teama6ef0a62019-03-07 20:34:33 -0500359 .InSequence(s);
360 EXPECT_CALL(handler_, OnDecodingCompleted()).InSequence(s);
361
QUICHE team11f55d42019-12-11 10:36:09 -0800362 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500363 "0500" // Required Insert Count 4 and Delta Base 0.
364 // Base is 4 + 0 = 4.
365 "83" // Dynamic table entry with relative index 3, absolute index 0.
366 "82" // Dynamic table entry with relative index 2, absolute index 1.
367 "81" // Dynamic table entry with relative index 1, absolute index 2.
368 "80" // Dynamic table entry with relative index 0, absolute index 3.
369 "41025a5a")); // Name of entry 1 (relative index) from dynamic table,
370 // with value "ZZ".
371
372 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar"))).InSequence(s);
373 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
374 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("foo")))
375 .InSequence(s);
376 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
377 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("ZZ"))).InSequence(s);
378 EXPECT_CALL(decoder_stream_sender_delegate_,
renjietangc2aa5cb2019-06-20 12:22:53 -0700379 WriteStreamData(Eq(kHeaderAcknowledgement)))
QUICHE teama6ef0a62019-03-07 20:34:33 -0500380 .InSequence(s);
381 EXPECT_CALL(handler_, OnDecodingCompleted()).InSequence(s);
382
QUICHE team11f55d42019-12-11 10:36:09 -0800383 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500384 "0502" // Required Insert Count 4 and Delta Base 2.
385 // Base is 4 + 2 = 6.
386 "85" // Dynamic table entry with relative index 5, absolute index 0.
387 "84" // Dynamic table entry with relative index 4, absolute index 1.
388 "83" // Dynamic table entry with relative index 3, absolute index 2.
389 "82" // Dynamic table entry with relative index 2, absolute index 3.
390 "43025a5a")); // Name of entry 3 (relative index) from dynamic table,
391 // with value "ZZ".
392
393 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar"))).InSequence(s);
394 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
395 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("foo")))
396 .InSequence(s);
397 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("ZZZ"))).InSequence(s);
398 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("ZZ"))).InSequence(s);
399 EXPECT_CALL(decoder_stream_sender_delegate_,
renjietangc2aa5cb2019-06-20 12:22:53 -0700400 WriteStreamData(Eq(kHeaderAcknowledgement)))
QUICHE teama6ef0a62019-03-07 20:34:33 -0500401 .InSequence(s);
402 EXPECT_CALL(handler_, OnDecodingCompleted()).InSequence(s);
403
QUICHE team11f55d42019-12-11 10:36:09 -0800404 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500405 "0582" // Required Insert Count 4 and Delta Base 2 with sign bit set.
406 // Base is 4 - 2 - 1 = 1.
407 "80" // Dynamic table entry with relative index 0, absolute index 0.
408 "10" // Dynamic table entry with post-base index 0, absolute index 1.
409 "11" // Dynamic table entry with post-base index 1, absolute index 2.
410 "12" // Dynamic table entry with post-base index 2, absolute index 3.
411 "01025a5a")); // Name of entry 1 (post-base index) from dynamic table,
412 // with value "ZZ".
413}
414
415TEST_P(QpackDecoderTest, DecreasingDynamicTableCapacityEvictsEntries) {
bnc40e9a7b2019-08-30 05:27:08 -0700416 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800417 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500418 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800419 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500420
421 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
422 EXPECT_CALL(handler_, OnDecodingCompleted());
423 EXPECT_CALL(decoder_stream_sender_delegate_,
renjietangc2aa5cb2019-06-20 12:22:53 -0700424 WriteStreamData(Eq(kHeaderAcknowledgement)));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500425
QUICHE team11f55d42019-12-11 10:36:09 -0800426 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500427 "0200" // Required Insert Count 1 and Delta Base 0.
428 // Base is 1 + 0 = 1.
429 "80")); // Dynamic table entry with relative index 0, absolute index 0.
430
431 // Change dynamic table capacity to 32 bytes, smaller than the entry.
432 // This must cause the entry to be evicted.
QUICHE team11f55d42019-12-11 10:36:09 -0800433 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3f01"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500434
bnc098ff612019-07-09 03:24:37 -0700435 EXPECT_CALL(handler_, OnDecodingErrorDetected(
436 Eq("Dynamic table entry already evicted.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500437
QUICHE team11f55d42019-12-11 10:36:09 -0800438 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500439 "0200" // Required Insert Count 1 and Delta Base 0.
440 // Base is 1 + 0 = 1.
441 "80")); // Dynamic table entry with relative index 0, absolute index 0.
442}
443
444TEST_P(QpackDecoderTest, EncoderStreamErrorEntryTooLarge) {
445 EXPECT_CALL(encoder_stream_error_delegate_,
446 OnEncoderStreamError(Eq("Error inserting literal entry.")));
447
448 // Set dynamic table capacity to 34.
QUICHE team11f55d42019-12-11 10:36:09 -0800449 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3f03"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500450 // Add literal entry with name "foo" and value "bar", size is 32 + 3 + 3 = 38.
QUICHE team11f55d42019-12-11 10:36:09 -0800451 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500452}
453
454TEST_P(QpackDecoderTest, EncoderStreamErrorInvalidStaticTableEntry) {
455 EXPECT_CALL(encoder_stream_error_delegate_,
456 OnEncoderStreamError(Eq("Invalid static table entry.")));
457
458 // Address invalid static table entry index 99.
QUICHE team11f55d42019-12-11 10:36:09 -0800459 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("ff2400"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500460}
461
462TEST_P(QpackDecoderTest, EncoderStreamErrorInvalidDynamicTableEntry) {
463 EXPECT_CALL(encoder_stream_error_delegate_,
bncae04c652019-08-14 13:15:14 -0700464 OnEncoderStreamError(Eq("Invalid relative index.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500465
QUICHE team11f55d42019-12-11 10:36:09 -0800466 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode(
bnc40e9a7b2019-08-30 05:27:08 -0700467 "3fe107" // Set dynamic table capacity to 1024.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500468 "6294e703626172" // Add literal entry with name "foo" and value "bar".
469 "8100")); // Address dynamic table entry with relative index 1. Such
470 // entry does not exist. The most recently added and only
471 // dynamic table entry has relative index 0.
472}
473
474TEST_P(QpackDecoderTest, EncoderStreamErrorDuplicateInvalidEntry) {
475 EXPECT_CALL(encoder_stream_error_delegate_,
bncae04c652019-08-14 13:15:14 -0700476 OnEncoderStreamError(Eq("Invalid relative index.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500477
QUICHE team11f55d42019-12-11 10:36:09 -0800478 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode(
bnc40e9a7b2019-08-30 05:27:08 -0700479 "3fe107" // Set dynamic table capacity to 1024.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500480 "6294e703626172" // Add literal entry with name "foo" and value "bar".
481 "01")); // Duplicate dynamic table entry with relative index 1. Such
482 // entry does not exist. The most recently added and only
483 // dynamic table entry has relative index 0.
484}
485
486TEST_P(QpackDecoderTest, EncoderStreamErrorTooLargeInteger) {
487 EXPECT_CALL(encoder_stream_error_delegate_,
488 OnEncoderStreamError(Eq("Encoded integer too large.")));
489
QUICHE team11f55d42019-12-11 10:36:09 -0800490 DecodeEncoderStreamData(
491 quiche::QuicheTextUtils::HexDecode("3fffffffffffffffffffff"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500492}
493
494TEST_P(QpackDecoderTest, InvalidDynamicEntryWhenBaseIsZero) {
495 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq("Invalid relative index.")));
496
bnc40e9a7b2019-08-30 05:27:08 -0700497 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800498 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc098ff612019-07-09 03:24:37 -0700499 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800500 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700501
QUICHE team11f55d42019-12-11 10:36:09 -0800502 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500503 "0280" // Required Insert Count is 1. Base 1 - 1 - 0 = 0 is explicitly
504 // permitted by the spec.
505 "80")); // However, addressing entry with relative index 0 would point to
506 // absolute index -1, which is invalid.
507}
508
509TEST_P(QpackDecoderTest, InvalidNegativeBase) {
510 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq("Error calculating Base.")));
511
512 // Required Insert Count 1, Delta Base 1 with sign bit set, Base would
513 // be 1 - 1 - 1 = -1, but it is not allowed to be negative.
QUICHE team11f55d42019-12-11 10:36:09 -0800514 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("0281"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500515}
516
517TEST_P(QpackDecoderTest, InvalidDynamicEntryByRelativeIndex) {
bnc40e9a7b2019-08-30 05:27:08 -0700518 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800519 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500520 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800521 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500522
bnc098ff612019-07-09 03:24:37 -0700523 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq("Invalid relative index.")));
524
QUICHE team11f55d42019-12-11 10:36:09 -0800525 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700526 "0200" // Required Insert Count 1 and Delta Base 0.
527 // Base is 1 + 0 = 1.
528 "81")); // Indexed Header Field instruction addressing relative index 1.
529 // This is absolute index -1, which is invalid.
530
531 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq("Invalid relative index.")));
532
QUICHE team11f55d42019-12-11 10:36:09 -0800533 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700534 "0200" // Required Insert Count 1 and Delta Base 0.
535 // Base is 1 + 0 = 1.
536 "4100")); // Literal Header Field with Name Reference instruction
537 // addressing relative index 1. This is absolute index -1,
538 // which is invalid.
539}
540
541TEST_P(QpackDecoderTest, EvictedDynamicTableEntry) {
542 // Update dynamic table capacity to 128.
QUICHE team11f55d42019-12-11 10:36:09 -0800543 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3f61"));
bnc098ff612019-07-09 03:24:37 -0700544
545 // Add literal entry with name "foo" and value "bar", size 32 + 3 + 3 = 38.
546 // This fits in the table three times.
QUICHE team11f55d42019-12-11 10:36:09 -0800547 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700548 // Duplicate entry four times. This evicts the first two instances.
QUICHE team11f55d42019-12-11 10:36:09 -0800549 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("00000000"));
bnc098ff612019-07-09 03:24:37 -0700550
551 EXPECT_CALL(handler_, OnDecodingErrorDetected(
552 Eq("Dynamic table entry already evicted.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500553
QUICHE team11f55d42019-12-11 10:36:09 -0800554 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500555 "0500" // Required Insert Count 4 and Delta Base 0.
556 // Base is 4 + 0 = 4.
557 "82")); // Indexed Header Field instruction addressing relative index 2.
558 // This is absolute index 1. Such entry does not exist.
559
bnc098ff612019-07-09 03:24:37 -0700560 EXPECT_CALL(handler_, OnDecodingErrorDetected(
561 Eq("Dynamic table entry already evicted.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500562
QUICHE team11f55d42019-12-11 10:36:09 -0800563 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500564 "0500" // Required Insert Count 4 and Delta Base 0.
565 // Base is 4 + 0 = 4.
566 "4200")); // Literal Header Field with Name Reference instruction
567 // addressing relative index 2. This is absolute index 1. Such
568 // entry does not exist.
569
bnc098ff612019-07-09 03:24:37 -0700570 EXPECT_CALL(handler_, OnDecodingErrorDetected(
571 Eq("Dynamic table entry already evicted.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500572
QUICHE team11f55d42019-12-11 10:36:09 -0800573 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500574 "0380" // Required Insert Count 2 and Delta Base 0 with sign bit set.
575 // Base is 2 - 0 - 1 = 1
576 "10")); // Indexed Header Field instruction addressing dynamic table
577 // entry with post-base index 0, absolute index 1. Such entry
578 // does not exist.
579
bnc098ff612019-07-09 03:24:37 -0700580 EXPECT_CALL(handler_, OnDecodingErrorDetected(
581 Eq("Dynamic table entry already evicted.")));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500582
QUICHE team11f55d42019-12-11 10:36:09 -0800583 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500584 "0380" // Required Insert Count 2 and Delta Base 0 with sign bit set.
585 // Base is 2 - 0 - 1 = 1
586 "0000")); // Literal Header Field With Name Reference instruction
587 // addressing dynamic table entry with post-base index 0,
588 // absolute index 1. Such entry does not exist.
589}
590
591TEST_P(QpackDecoderTest, TableCapacityMustNotExceedMaximum) {
592 EXPECT_CALL(
593 encoder_stream_error_delegate_,
594 OnEncoderStreamError(Eq("Error updating dynamic table capacity.")));
595
596 // Try to update dynamic table capacity to 2048, which exceeds the maximum.
QUICHE team11f55d42019-12-11 10:36:09 -0800597 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe10f"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500598}
599
bnc4c664c52019-08-04 18:14:12 -0700600TEST_P(QpackDecoderTest, SetDynamicTableCapacity) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500601 // Update dynamic table capacity to 128, which does not exceed the maximum.
QUICHE team11f55d42019-12-11 10:36:09 -0800602 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3f61"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500603}
604
605TEST_P(QpackDecoderTest, InvalidEncodedRequiredInsertCount) {
606 // Maximum dynamic table capacity is 1024.
607 // MaxEntries is 1024 / 32 = 32.
608 // Required Insert Count is decoded modulo 2 * MaxEntries, that is, modulo 64.
609 // A value of 1 cannot be encoded as 65 even though it has the same remainder.
610 EXPECT_CALL(handler_, OnDecodingErrorDetected(
611 Eq("Error decoding Required Insert Count.")));
QUICHE team11f55d42019-12-11 10:36:09 -0800612 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("4100"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500613}
614
bnc32e69f82019-06-14 06:38:36 -0700615// Regression test for https://crbug.com/970218: Decoder must stop processing
616// after a Header Block Prefix with an invalid Encoded Required Insert Count.
617TEST_P(QpackDecoderTest, DataAfterInvalidEncodedRequiredInsertCount) {
618 EXPECT_CALL(handler_, OnDecodingErrorDetected(
619 Eq("Error decoding Required Insert Count.")));
620 // Header Block Prefix followed by some extra data.
QUICHE team11f55d42019-12-11 10:36:09 -0800621 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode("410000"));
bnc32e69f82019-06-14 06:38:36 -0700622}
623
QUICHE teama6ef0a62019-03-07 20:34:33 -0500624TEST_P(QpackDecoderTest, WrappedRequiredInsertCount) {
625 // Maximum dynamic table capacity is 1024.
626 // MaxEntries is 1024 / 32 = 32.
627
bnc40e9a7b2019-08-30 05:27:08 -0700628 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800629 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500630 // Add literal entry with name "foo" and a 600 byte long value. This will fit
631 // in the dynamic table once but not twice.
632 DecodeEncoderStreamData(
QUICHE team11f55d42019-12-11 10:36:09 -0800633 quiche::QuicheTextUtils::HexDecode("6294e7" // Name "foo".
634 "7fd903")); // Value length 600.
vasilvvc48c8712019-03-11 13:38:16 -0700635 std::string header_value(600, 'Z');
QUICHE teama6ef0a62019-03-07 20:34:33 -0500636 DecodeEncoderStreamData(header_value);
637
638 // Duplicate most recent entry 200 times.
vasilvvc48c8712019-03-11 13:38:16 -0700639 DecodeEncoderStreamData(std::string(200, '\x00'));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500640
641 // Now there is only one entry in the dynamic table, with absolute index 200.
642
643 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq(header_value)));
644 EXPECT_CALL(handler_, OnDecodingCompleted());
645 EXPECT_CALL(decoder_stream_sender_delegate_,
renjietangc2aa5cb2019-06-20 12:22:53 -0700646 WriteStreamData(Eq(kHeaderAcknowledgement)));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500647
648 // Send header block with Required Insert Count = 201.
QUICHE team11f55d42019-12-11 10:36:09 -0800649 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500650 "0a00" // Encoded Required Insert Count 10, Required Insert Count 201,
651 // Delta Base 0, Base 201.
652 "80")); // Emit dynamic table entry with relative index 0.
653}
654
655TEST_P(QpackDecoderTest, NonZeroRequiredInsertCountButNoDynamicEntries) {
bnc40e9a7b2019-08-30 05:27:08 -0700656 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800657 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc098ff612019-07-09 03:24:37 -0700658 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800659 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700660
QUICHE teama6ef0a62019-03-07 20:34:33 -0500661 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("GET")));
662 EXPECT_CALL(handler_,
663 OnDecodingErrorDetected(Eq("Required Insert Count too large.")));
664
QUICHE team11f55d42019-12-11 10:36:09 -0800665 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500666 "0200" // Required Insert Count is 1.
667 "d1")); // But the only instruction references the static table.
668}
669
670TEST_P(QpackDecoderTest, AddressEntryNotAllowedByRequiredInsertCount) {
bnc40e9a7b2019-08-30 05:27:08 -0700671 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800672 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc098ff612019-07-09 03:24:37 -0700673 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800674 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700675
QUICHE teama6ef0a62019-03-07 20:34:33 -0500676 EXPECT_CALL(
677 handler_,
678 OnDecodingErrorDetected(
679 Eq("Absolute Index must be smaller than Required Insert Count.")));
680
QUICHE team11f55d42019-12-11 10:36:09 -0800681 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500682 "0201" // Required Insert Count 1 and Delta Base 1.
683 // Base is 1 + 1 = 2.
684 "80")); // Indexed Header Field instruction addressing dynamic table
685 // entry with relative index 0, absolute index 1. This is not
686 // allowed by Required Insert Count.
687
688 EXPECT_CALL(
689 handler_,
690 OnDecodingErrorDetected(
691 Eq("Absolute Index must be smaller than Required Insert Count.")));
692
QUICHE team11f55d42019-12-11 10:36:09 -0800693 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500694 "0201" // Required Insert Count 1 and Delta Base 1.
695 // Base is 1 + 1 = 2.
696 "4000")); // Literal Header Field with Name Reference instruction
697 // addressing dynamic table entry with relative index 0,
698 // absolute index 1. This is not allowed by Required Index
699 // Count.
700
701 EXPECT_CALL(
702 handler_,
703 OnDecodingErrorDetected(
704 Eq("Absolute Index must be smaller than Required Insert Count.")));
705
QUICHE team11f55d42019-12-11 10:36:09 -0800706 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500707 "0200" // Required Insert Count 1 and Delta Base 0.
708 // Base is 1 + 0 = 1.
709 "10")); // Indexed Header Field with Post-Base Index instruction
710 // addressing dynamic table entry with post-base index 0,
711 // absolute index 1. This is not allowed by Required Insert
712 // Count.
713
714 EXPECT_CALL(
715 handler_,
716 OnDecodingErrorDetected(
717 Eq("Absolute Index must be smaller than Required Insert Count.")));
718
QUICHE team11f55d42019-12-11 10:36:09 -0800719 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500720 "0200" // Required Insert Count 1 and Delta Base 0.
721 // Base is 1 + 0 = 1.
722 "0000")); // Literal Header Field with Post-Base Name Reference
723 // instruction addressing dynamic table entry with post-base
724 // index 0, absolute index 1. This is not allowed by Required
725 // Index Count.
726}
727
728TEST_P(QpackDecoderTest, PromisedRequiredInsertCountLargerThanActual) {
bnc40e9a7b2019-08-30 05:27:08 -0700729 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800730 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500731 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800732 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700733 // Duplicate entry twice so that decoding of header blocks with Required
734 // Insert Count not exceeding 3 is not blocked.
QUICHE team11f55d42019-12-11 10:36:09 -0800735 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("00"));
736 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("00"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500737
738 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
739 EXPECT_CALL(handler_,
740 OnDecodingErrorDetected(Eq("Required Insert Count too large.")));
741
QUICHE team11f55d42019-12-11 10:36:09 -0800742 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500743 "0300" // Required Insert Count 2 and Delta Base 0.
744 // Base is 2 + 0 = 2.
745 "81")); // Indexed Header Field instruction addressing dynamic table
746 // entry with relative index 1, absolute index 0. Header block
747 // requires insert count of 1, even though Required Insert Count
748 // is 2.
749
750 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("")));
751 EXPECT_CALL(handler_,
752 OnDecodingErrorDetected(Eq("Required Insert Count too large.")));
753
QUICHE team11f55d42019-12-11 10:36:09 -0800754 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500755 "0300" // Required Insert Count 2 and Delta Base 0.
756 // Base is 2 + 0 = 2.
757 "4100")); // Literal Header Field with Name Reference instruction
758 // addressing dynamic table entry with relative index 1,
759 // absolute index 0. Header block requires insert count of 1,
760 // even though Required Insert Count is 2.
761
762 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
763 EXPECT_CALL(handler_,
764 OnDecodingErrorDetected(Eq("Required Insert Count too large.")));
765
QUICHE team11f55d42019-12-11 10:36:09 -0800766 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500767 "0481" // Required Insert Count 3 and Delta Base 1 with sign bit set.
768 // Base is 3 - 1 - 1 = 1.
769 "10")); // Indexed Header Field with Post-Base Index instruction
770 // addressing dynamic table entry with post-base index 0,
771 // absolute index 1. Header block requires insert count of 2,
772 // even though Required Insert Count is 3.
773
774 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("")));
775 EXPECT_CALL(handler_,
776 OnDecodingErrorDetected(Eq("Required Insert Count too large.")));
777
QUICHE team11f55d42019-12-11 10:36:09 -0800778 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
QUICHE teama6ef0a62019-03-07 20:34:33 -0500779 "0481" // Required Insert Count 3 and Delta Base 1 with sign bit set.
780 // Base is 3 - 1 - 1 = 1.
781 "0000")); // Literal Header Field with Post-Base Name Reference
782 // instruction addressing dynamic table entry with post-base
783 // index 0, absolute index 1. Header block requires insert
784 // count of 2, even though Required Insert Count is 3.
785}
786
bnc098ff612019-07-09 03:24:37 -0700787TEST_P(QpackDecoderTest, BlockedDecoding) {
QUICHE team11f55d42019-12-11 10:36:09 -0800788 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700789 "0200" // Required Insert Count 1 and Delta Base 0.
790 // Base is 1 + 0 = 1.
791 "80")); // Indexed Header Field instruction addressing dynamic table
792 // entry with relative index 0, absolute index 0.
793
794 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
795 EXPECT_CALL(handler_, OnDecodingCompleted());
796 EXPECT_CALL(decoder_stream_sender_delegate_,
797 WriteStreamData(Eq(kHeaderAcknowledgement)));
798
bnc40e9a7b2019-08-30 05:27:08 -0700799 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800800 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc098ff612019-07-09 03:24:37 -0700801 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800802 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700803}
804
805TEST_P(QpackDecoderTest, BlockedDecodingUnblockedBeforeEndOfHeaderBlock) {
806 StartDecoding();
QUICHE team11f55d42019-12-11 10:36:09 -0800807 DecodeData(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700808 "0200" // Required Insert Count 1 and Delta Base 0.
809 // Base is 1 + 0 = 1.
810 "80" // Indexed Header Field instruction addressing dynamic table
811 // entry with relative index 0, absolute index 0.
812 "d1")); // Static table entry with index 17.
813
bnc232ff9b2019-11-13 18:38:22 -0800814 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800815 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc232ff9b2019-11-13 18:38:22 -0800816
bnc098ff612019-07-09 03:24:37 -0700817 // Add literal entry with name "foo" and value "bar". Decoding is now
818 // unblocked because dynamic table Insert Count reached the Required Insert
819 // Count of the header block. |handler_| methods are called immediately for
820 // the already consumed part of the header block.
821 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
822 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":method"), Eq("GET")));
QUICHE team11f55d42019-12-11 10:36:09 -0800823 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700824 Mock::VerifyAndClearExpectations(&handler_);
825
826 // Rest of header block is processed by QpackProgressiveDecoder
827 // in the unblocked state.
828 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
829 EXPECT_CALL(handler_, OnHeaderDecoded(Eq(":scheme"), Eq("https")));
QUICHE team11f55d42019-12-11 10:36:09 -0800830 DecodeData(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700831 "80" // Indexed Header Field instruction addressing dynamic table
832 // entry with relative index 0, absolute index 0.
833 "d7")); // Static table entry with index 23.
834 Mock::VerifyAndClearExpectations(&handler_);
835
836 EXPECT_CALL(handler_, OnDecodingCompleted());
837 EXPECT_CALL(decoder_stream_sender_delegate_,
838 WriteStreamData(Eq(kHeaderAcknowledgement)));
839 EndDecoding();
840}
841
bnc232ff9b2019-11-13 18:38:22 -0800842// Regression test for https://crbug.com/1024263.
843TEST_P(QpackDecoderTest,
844 BlockedDecodingUnblockedAndErrorBeforeEndOfHeaderBlock) {
845 StartDecoding();
QUICHE team11f55d42019-12-11 10:36:09 -0800846 DecodeData(quiche::QuicheTextUtils::HexDecode(
bnc232ff9b2019-11-13 18:38:22 -0800847 "0200" // Required Insert Count 1 and Delta Base 0.
848 // Base is 1 + 0 = 1.
849 "80" // Indexed Header Field instruction addressing dynamic table
850 // entry with relative index 0, absolute index 0.
851 "81")); // Relative index 1 is equal to Base, therefore invalid.
852
853 // Set dynamic table capacity to 1024.
QUICHE team11f55d42019-12-11 10:36:09 -0800854 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3fe107"));
bnc232ff9b2019-11-13 18:38:22 -0800855
856 // Add literal entry with name "foo" and value "bar". Decoding is now
857 // unblocked because dynamic table Insert Count reached the Required Insert
858 // Count of the header block. |handler_| methods are called immediately for
859 // the already consumed part of the header block.
860 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
861 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq("Invalid relative index.")));
QUICHE team11f55d42019-12-11 10:36:09 -0800862 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc232ff9b2019-11-13 18:38:22 -0800863}
864
bnc098ff612019-07-09 03:24:37 -0700865// Make sure that Required Insert Count is compared to Insert Count,
866// not size of dynamic table.
867TEST_P(QpackDecoderTest, BlockedDecodingAndEvictedEntries) {
868 // Update dynamic table capacity to 128.
869 // At most three non-empty entries fit in the dynamic table.
QUICHE team11f55d42019-12-11 10:36:09 -0800870 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("3f61"));
bnc098ff612019-07-09 03:24:37 -0700871
QUICHE team11f55d42019-12-11 10:36:09 -0800872 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
bnc098ff612019-07-09 03:24:37 -0700873 "0700" // Required Insert Count 6 and Delta Base 0.
874 // Base is 6 + 0 = 6.
875 "80")); // Indexed Header Field instruction addressing dynamic table
876 // entry with relative index 0, absolute index 5.
877
878 // Add literal entry with name "foo" and value "bar".
QUICHE team11f55d42019-12-11 10:36:09 -0800879 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e703626172"));
bnc098ff612019-07-09 03:24:37 -0700880
881 // Duplicate entry four times. This evicts the first two instances.
QUICHE team11f55d42019-12-11 10:36:09 -0800882 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("00000000"));
bnc098ff612019-07-09 03:24:37 -0700883
884 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("baz")));
885 EXPECT_CALL(handler_, OnDecodingCompleted());
886 EXPECT_CALL(decoder_stream_sender_delegate_,
887 WriteStreamData(Eq(kHeaderAcknowledgement)));
888
889 // Add literal entry with name "foo" and value "bar".
890 // Insert Count is now 6, reaching Required Insert Count of the header block.
QUICHE team11f55d42019-12-11 10:36:09 -0800891 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode("6294e70362617a"));
bnc098ff612019-07-09 03:24:37 -0700892}
893
bnc57b5f622019-08-21 14:07:44 -0700894TEST_P(QpackDecoderTest, TooManyBlockedStreams) {
895 // Required Insert Count 1 and Delta Base 0.
896 // Without any dynamic table entries received, decoding is blocked.
QUICHE team11f55d42019-12-11 10:36:09 -0800897 std::string data = quiche::QuicheTextUtils::HexDecode("0200");
bnc57b5f622019-08-21 14:07:44 -0700898
899 auto progressive_decoder1 = CreateProgressiveDecoder(/* stream_id = */ 1);
900 progressive_decoder1->Decode(data);
901
902 EXPECT_CALL(handler_, OnDecodingErrorDetected(Eq(
903 "Limit on number of blocked streams exceeded.")));
904
905 auto progressive_decoder2 = CreateProgressiveDecoder(/* stream_id = */ 2);
906 progressive_decoder2->Decode(data);
907}
908
bnc45af7512019-10-08 06:59:58 -0700909TEST_P(QpackDecoderTest, InsertCountIncrement) {
QUICHE team11f55d42019-12-11 10:36:09 -0800910 DecodeEncoderStreamData(quiche::QuicheTextUtils::HexDecode(
bnc45af7512019-10-08 06:59:58 -0700911 "3fe107" // Set dynamic table capacity to 1024.
912 "6294e703626172" // Add literal entry with name "foo" and value "bar".
913 "00")); // Duplicate entry.
914
915 EXPECT_CALL(handler_, OnHeaderDecoded(Eq("foo"), Eq("bar")));
916 EXPECT_CALL(handler_, OnDecodingCompleted());
bnc45af7512019-10-08 06:59:58 -0700917
918 // Decoder received two insertions, but Header Acknowledgement only increases
919 // Known Insert Count to one. Decoder should send an Insert Count Increment
920 // instruction with increment of one to update Known Insert Count to two.
921 EXPECT_CALL(decoder_stream_sender_delegate_,
QUICHE team11f55d42019-12-11 10:36:09 -0800922 WriteStreamData(Eq(quiche::QuicheTextUtils::HexDecode(
bnc6b1fc8a2019-10-11 17:17:14 -0700923 "81" // Header Acknowledgement on stream 1
924 "01")))); // Insert Count Increment with increment of one
bnc45af7512019-10-08 06:59:58 -0700925
QUICHE team11f55d42019-12-11 10:36:09 -0800926 DecodeHeaderBlock(quiche::QuicheTextUtils::HexDecode(
bnc45af7512019-10-08 06:59:58 -0700927 "0200" // Required Insert Count 1 and Delta Base 0.
928 // Base is 1 + 0 = 1.
929 "80")); // Dynamic table entry with relative index 0, absolute index 0.
930}
931
QUICHE teama6ef0a62019-03-07 20:34:33 -0500932} // namespace
933} // namespace test
934} // namespace quic