|  | // Copyright 2014 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 <algorithm> | 
|  | #include <cmath> | 
|  | #include <ctime> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "quiche/http2/core/recording_headers_handler.h" | 
|  | #include "quiche/http2/hpack/hpack_decoder_adapter.h" | 
|  | #include "quiche/http2/hpack/hpack_encoder.h" | 
|  | #include "quiche/http2/test_tools/http2_random.h" | 
|  | #include "quiche/common/http/http_header_block.h" | 
|  | #include "quiche/common/platform/api/quiche_test.h" | 
|  |  | 
|  | namespace spdy { | 
|  | namespace test { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Supports testing with the input split at every byte boundary. | 
|  | enum InputSizeParam { ALL_INPUT, ONE_BYTE, ZERO_THEN_ONE_BYTE }; | 
|  |  | 
|  | class HpackRoundTripTest | 
|  | : public quiche::test::QuicheTestWithParam<InputSizeParam> { | 
|  | protected: | 
|  | void SetUp() override { | 
|  | // Use a small table size to tickle eviction handling. | 
|  | encoder_.ApplyHeaderTableSizeSetting(256); | 
|  | decoder_.ApplyHeaderTableSizeSetting(256); | 
|  | } | 
|  |  | 
|  | bool RoundTrip(const quiche::HttpHeaderBlock& header_set) { | 
|  | std::string encoded = encoder_.EncodeHeaderBlock(header_set); | 
|  |  | 
|  | bool success = true; | 
|  | decoder_.HandleControlFrameHeadersStart(&handler_); | 
|  | if (GetParam() == ALL_INPUT) { | 
|  | // Pass all the input to the decoder at once. | 
|  | success = decoder_.HandleControlFrameHeadersData(encoded.data(), | 
|  | encoded.size()); | 
|  | } else if (GetParam() == ONE_BYTE) { | 
|  | // Pass the input to the decoder one byte at a time. | 
|  | const char* data = encoded.data(); | 
|  | for (size_t ndx = 0; ndx < encoded.size() && success; ++ndx) { | 
|  | success = decoder_.HandleControlFrameHeadersData(data + ndx, 1); | 
|  | } | 
|  | } else if (GetParam() == ZERO_THEN_ONE_BYTE) { | 
|  | // Pass the input to the decoder one byte at a time, but before each | 
|  | // byte pass an empty buffer. | 
|  | const char* data = encoded.data(); | 
|  | for (size_t ndx = 0; ndx < encoded.size() && success; ++ndx) { | 
|  | success = (decoder_.HandleControlFrameHeadersData(data + ndx, 0) && | 
|  | decoder_.HandleControlFrameHeadersData(data + ndx, 1)); | 
|  | } | 
|  | } else { | 
|  | ADD_FAILURE() << "Unknown param: " << GetParam(); | 
|  | } | 
|  |  | 
|  | if (success) { | 
|  | success = decoder_.HandleControlFrameHeadersComplete(); | 
|  | } | 
|  |  | 
|  | EXPECT_EQ(header_set, handler_.decoded_block()); | 
|  | return success; | 
|  | } | 
|  |  | 
|  | size_t SampleExponential(size_t mean, size_t sanity_bound) { | 
|  | return std::min<size_t>(-std::log(random_.RandDouble()) * mean, | 
|  | sanity_bound); | 
|  | } | 
|  |  | 
|  | http2::test::Http2Random random_; | 
|  | HpackEncoder encoder_; | 
|  | HpackDecoderAdapter decoder_; | 
|  | RecordingHeadersHandler handler_; | 
|  | }; | 
|  |  | 
|  | INSTANTIATE_TEST_SUITE_P(Tests, HpackRoundTripTest, | 
|  | ::testing::Values(ALL_INPUT, ONE_BYTE, | 
|  | ZERO_THEN_ONE_BYTE)); | 
|  |  | 
|  | TEST_P(HpackRoundTripTest, ResponseFixtures) { | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":status"] = "302"; | 
|  | headers["cache-control"] = "private"; | 
|  | headers["date"] = "Mon, 21 Oct 2013 20:13:21 GMT"; | 
|  | headers["location"] = "https://www.example.com"; | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":status"] = "200"; | 
|  | headers["cache-control"] = "private"; | 
|  | headers["date"] = "Mon, 21 Oct 2013 20:13:21 GMT"; | 
|  | headers["location"] = "https://www.example.com"; | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":status"] = "200"; | 
|  | headers["cache-control"] = "private"; | 
|  | headers["content-encoding"] = "gzip"; | 
|  | headers["date"] = "Mon, 21 Oct 2013 20:13:22 GMT"; | 
|  | headers["location"] = "https://www.example.com"; | 
|  | headers["set-cookie"] = | 
|  | "foo=ASDJKHQKBZXOQWEOPIUAXQWEOIU;" | 
|  | " max-age=3600; version=1"; | 
|  | headers["multivalue"] = std::string("foo\0bar", 7); | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_P(HpackRoundTripTest, RequestFixtures) { | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":authority"] = "www.example.com"; | 
|  | headers[":method"] = "GET"; | 
|  | headers[":path"] = "/"; | 
|  | headers[":scheme"] = "http"; | 
|  | headers["cookie"] = "baz=bing; foo=bar"; | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":authority"] = "www.example.com"; | 
|  | headers[":method"] = "GET"; | 
|  | headers[":path"] = "/"; | 
|  | headers[":scheme"] = "http"; | 
|  | headers["cache-control"] = "no-cache"; | 
|  | headers["cookie"] = "foo=bar; spam=eggs"; | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | { | 
|  | quiche::HttpHeaderBlock headers; | 
|  | headers[":authority"] = "www.example.com"; | 
|  | headers[":method"] = "GET"; | 
|  | headers[":path"] = "/index.html"; | 
|  | headers[":scheme"] = "https"; | 
|  | headers["custom-key"] = "custom-value"; | 
|  | headers["cookie"] = "baz=bing; fizzle=fazzle; garbage"; | 
|  | headers["multivalue"] = std::string("foo\0bar", 7); | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_P(HpackRoundTripTest, RandomizedExamples) { | 
|  | // Grow vectors of names & values, which are seeded with fixtures and then | 
|  | // expanded with dynamically generated data. Samples are taken using the | 
|  | // exponential distribution. | 
|  | std::vector<std::string> pseudo_header_names, random_header_names; | 
|  | pseudo_header_names.push_back(":authority"); | 
|  | pseudo_header_names.push_back(":path"); | 
|  | pseudo_header_names.push_back(":status"); | 
|  |  | 
|  | // TODO(jgraettinger): Enable "cookie" as a name fixture. Crumbs may be | 
|  | // reconstructed in any order, which breaks the simple validation used here. | 
|  |  | 
|  | std::vector<std::string> values; | 
|  | values.push_back("/"); | 
|  | values.push_back("/index.html"); | 
|  | values.push_back("200"); | 
|  | values.push_back("404"); | 
|  | values.push_back(""); | 
|  | values.push_back("baz=bing; foo=bar; garbage"); | 
|  | values.push_back("baz=bing; fizzle=fazzle; garbage"); | 
|  |  | 
|  | for (size_t i = 0; i != 2000; ++i) { | 
|  | quiche::HttpHeaderBlock headers; | 
|  |  | 
|  | // Choose a random number of headers to add, and of these a random subset | 
|  | // will be HTTP/2 pseudo headers. | 
|  | size_t header_count = 1 + SampleExponential(7, 50); | 
|  | size_t pseudo_header_count = | 
|  | std::min(header_count, 1 + SampleExponential(7, 50)); | 
|  | EXPECT_LE(pseudo_header_count, header_count); | 
|  | for (size_t j = 0; j != header_count; ++j) { | 
|  | std::string name, value; | 
|  | // Pseudo headers must be added before regular headers. | 
|  | if (j < pseudo_header_count) { | 
|  | // Choose one of the defined pseudo headers at random. | 
|  | size_t name_index = random_.Uniform(pseudo_header_names.size()); | 
|  | name = pseudo_header_names[name_index]; | 
|  | } else { | 
|  | // Randomly reuse an existing header name, or generate a new one. | 
|  | size_t name_index = SampleExponential(20, 200); | 
|  | if (name_index >= random_header_names.size()) { | 
|  | name = random_.RandString(1 + SampleExponential(5, 30)); | 
|  | // A regular header cannot begin with the pseudo header prefix ":". | 
|  | if (name[0] == ':') { | 
|  | name[0] = 'x'; | 
|  | } | 
|  | random_header_names.push_back(name); | 
|  | } else { | 
|  | name = random_header_names[name_index]; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Randomly reuse an existing value, or generate a new one. | 
|  | size_t value_index = SampleExponential(20, 200); | 
|  | if (value_index >= values.size()) { | 
|  | std::string newvalue = | 
|  | random_.RandString(1 + SampleExponential(15, 75)); | 
|  | // Currently order is not preserved in the encoder.  In particular, | 
|  | // when a value is decomposed at \0 delimiters, its parts might get | 
|  | // encoded out of order if some but not all of them already exist in | 
|  | // the header table.  For now, avoid \0 bytes in values. | 
|  | std::replace(newvalue.begin(), newvalue.end(), '\x00', '\x01'); | 
|  | values.push_back(newvalue); | 
|  | value = values.back(); | 
|  | } else { | 
|  | value = values[value_index]; | 
|  | } | 
|  | headers[name] = value; | 
|  | } | 
|  | EXPECT_TRUE(RoundTrip(headers)); | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | }  // namespace test | 
|  | }  // namespace spdy |