| // Copyright (c) 2018 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "http2/hpack/huffman/hpack_huffman_encoder.h" |
| |
| #include "absl/base/macros.h" |
| #include "absl/strings/escaping.h" |
| #include "common/platform/api/quiche_test.h" |
| |
| namespace http2 { |
| namespace { |
| |
| class HuffmanEncoderTest : public ::testing::TestWithParam<bool> { |
| protected: |
| HuffmanEncoderTest() : use_fast_encoder_(GetParam()) {} |
| virtual ~HuffmanEncoderTest() = default; |
| |
| void Encode(absl::string_view input, |
| size_t encoded_size, |
| std::string* output) { |
| use_fast_encoder_ ? HuffmanEncodeFast(input, encoded_size, output) |
| : HuffmanEncode(input, encoded_size, output); |
| } |
| |
| const bool use_fast_encoder_; |
| }; |
| |
| INSTANTIATE_TEST_SUITE_P(TwoEncoders, HuffmanEncoderTest, ::testing::Bool()); |
| |
| TEST_P(HuffmanEncoderTest, Empty) { |
| std::string empty(""); |
| size_t encoded_size = HuffmanSize(empty); |
| EXPECT_EQ(0u, encoded_size); |
| |
| std::string buffer; |
| Encode(empty, encoded_size, &buffer); |
| EXPECT_EQ("", buffer); |
| } |
| |
| TEST_P(HuffmanEncoderTest, SpecRequestExamples) { |
| std::string test_table[] = { |
| absl::HexStringToBytes("f1e3c2e5f23a6ba0ab90f4ff"), |
| "www.example.com", |
| absl::HexStringToBytes("a8eb10649cbf"), |
| "no-cache", |
| absl::HexStringToBytes("25a849e95ba97d7f"), |
| "custom-key", |
| absl::HexStringToBytes("25a849e95bb8e8b4bf"), |
| "custom-value", |
| }; |
| for (size_t i = 0; i != ABSL_ARRAYSIZE(test_table); i += 2) { |
| const std::string& huffman_encoded(test_table[i]); |
| const std::string& plain_string(test_table[i + 1]); |
| size_t encoded_size = HuffmanSize(plain_string); |
| EXPECT_EQ(huffman_encoded.size(), encoded_size); |
| std::string buffer; |
| buffer.reserve(); |
| Encode(plain_string, encoded_size, &buffer); |
| EXPECT_EQ(buffer, huffman_encoded) << "Error encoding " << plain_string; |
| } |
| } |
| |
| TEST_P(HuffmanEncoderTest, SpecResponseExamples) { |
| // clang-format off |
| std::string test_table[] = { |
| absl::HexStringToBytes("6402"), |
| "302", |
| absl::HexStringToBytes("aec3771a4b"), |
| "private", |
| absl::HexStringToBytes("d07abe941054d444a8200595040b8166" |
| "e082a62d1bff"), |
| "Mon, 21 Oct 2013 20:13:21 GMT", |
| absl::HexStringToBytes("9d29ad171863c78f0b97c8e9ae82ae43" |
| "d3"), |
| "https://www.example.com", |
| absl::HexStringToBytes("94e7821dd7f2e6c7b335dfdfcd5b3960" |
| "d5af27087f3672c1ab270fb5291f9587" |
| "316065c003ed4ee5b1063d5007"), |
| "foo=ASDJKHQKBZXOQWEOPIUAXQWEOIU; max-age=3600; version=1", |
| }; |
| // clang-format on |
| for (size_t i = 0; i != ABSL_ARRAYSIZE(test_table); i += 2) { |
| const std::string& huffman_encoded(test_table[i]); |
| const std::string& plain_string(test_table[i + 1]); |
| size_t encoded_size = HuffmanSize(plain_string); |
| EXPECT_EQ(huffman_encoded.size(), encoded_size); |
| std::string buffer; |
| Encode(plain_string, encoded_size, &buffer); |
| EXPECT_EQ(buffer, huffman_encoded) << "Error encoding " << plain_string; |
| } |
| } |
| |
| TEST_P(HuffmanEncoderTest, EncodedSizeAgreesWithEncodeString) { |
| std::string test_table[] = { |
| "", |
| "Mon, 21 Oct 2013 20:13:21 GMT", |
| "https://www.example.com", |
| "foo=ASDJKHQKBZXOQWEOPIUAXQWEOIU; max-age=3600; version=1", |
| std::string(1, '\0'), |
| std::string("foo\0bar", 7), |
| std::string(256, '\0'), |
| }; |
| // Modify last |test_table| entry to cover all codes. |
| for (size_t i = 0; i != 256; ++i) { |
| test_table[ABSL_ARRAYSIZE(test_table) - 1][i] = static_cast<char>(i); |
| } |
| |
| for (size_t i = 0; i != ABSL_ARRAYSIZE(test_table); ++i) { |
| const std::string& plain_string = test_table[i]; |
| size_t encoded_size = HuffmanSize(plain_string); |
| std::string huffman_encoded; |
| Encode(plain_string, encoded_size, &huffman_encoded); |
| EXPECT_EQ(encoded_size, huffman_encoded.size()); |
| } |
| } |
| |
| // Test that encoding appends to output without overwriting it. |
| TEST_P(HuffmanEncoderTest, AppendToOutput) { |
| size_t encoded_size = HuffmanSize("foo"); |
| std::string buffer; |
| Encode("foo", encoded_size, &buffer); |
| EXPECT_EQ(absl::HexStringToBytes("94e7"), buffer); |
| |
| encoded_size = HuffmanSize("bar"); |
| Encode("bar", encoded_size, &buffer); |
| EXPECT_EQ(absl::HexStringToBytes("94e78c767f"), buffer); |
| } |
| |
| } // namespace |
| } // namespace http2 |