blob: 2435ea9c0434b6aa2df4223c4885a4903652101c [file] [log] [blame]
QUICHE team82dee2f2019-01-18 12:35:12 -05001// Copyright (c) 2012 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
QUICHE team5be974e2020-12-29 18:35:24 -05005#include "spdy/core/spdy_header_block.h"
QUICHE team82dee2f2019-01-18 12:35:12 -05006
7#include <memory>
8#include <utility>
9
QUICHE team5be974e2020-12-29 18:35:24 -050010#include "common/platform/api/quiche_test.h"
11#include "spdy/core/spdy_test_utils.h"
QUICHE team82dee2f2019-01-18 12:35:12 -050012
13using ::testing::ElementsAre;
14
15namespace spdy {
16namespace test {
17
18class ValueProxyPeer {
19 public:
QUICHE team557dfbc2020-10-20 10:52:30 -070020 static absl::string_view key(Http2HeaderBlock::ValueProxy* p) {
bnc7f82d042020-01-03 12:18:53 -080021 return p->key_;
22 }
QUICHE team82dee2f2019-01-18 12:35:12 -050023};
24
vasilvvc8ccdb22020-10-12 16:42:34 -070025std::pair<absl::string_view, absl::string_view> Pair(absl::string_view k,
26 absl::string_view v) {
QUICHE team82dee2f2019-01-18 12:35:12 -050027 return std::make_pair(k, v);
28}
29
QUICHE team557dfbc2020-10-20 10:52:30 -070030// This test verifies that Http2HeaderBlock behaves correctly when empty.
31TEST(Http2HeaderBlockTest, EmptyBlock) {
32 Http2HeaderBlock block;
QUICHE team82dee2f2019-01-18 12:35:12 -050033 EXPECT_TRUE(block.empty());
34 EXPECT_EQ(0u, block.size());
35 EXPECT_EQ(block.end(), block.find("foo"));
36 EXPECT_TRUE(block.end() == block.begin());
37
38 // Should have no effect.
39 block.erase("bar");
40}
41
QUICHE team557dfbc2020-10-20 10:52:30 -070042TEST(Http2HeaderBlockTest, KeyMemoryReclaimedOnLookup) {
43 Http2HeaderBlock block;
vasilvvc8ccdb22020-10-12 16:42:34 -070044 absl::string_view copied_key1;
QUICHE team82dee2f2019-01-18 12:35:12 -050045 {
46 auto proxy1 = block["some key name"];
47 copied_key1 = ValueProxyPeer::key(&proxy1);
48 }
vasilvvc8ccdb22020-10-12 16:42:34 -070049 absl::string_view copied_key2;
QUICHE team82dee2f2019-01-18 12:35:12 -050050 {
51 auto proxy2 = block["some other key name"];
52 copied_key2 = ValueProxyPeer::key(&proxy2);
53 }
54 // Because proxy1 was never used to modify the block, the memory used for the
55 // key could be reclaimed and used for the second call to operator[].
vasilvvc8ccdb22020-10-12 16:42:34 -070056 // Therefore, we expect the pointers of the two absl::string_views to be
bnc7f82d042020-01-03 12:18:53 -080057 // equal.
QUICHE team82dee2f2019-01-18 12:35:12 -050058 EXPECT_EQ(copied_key1.data(), copied_key2.data());
59
60 {
61 auto proxy1 = block["some key name"];
62 block["some other key name"] = "some value";
63 }
64 // Nothing should blow up when proxy1 is destructed, and we should be able to
QUICHE team557dfbc2020-10-20 10:52:30 -070065 // modify and access the Http2HeaderBlock.
QUICHE team82dee2f2019-01-18 12:35:12 -050066 block["key"] = "value";
67 EXPECT_EQ("value", block["key"]);
68 EXPECT_EQ("some value", block["some other key name"]);
69 EXPECT_TRUE(block.find("some key name") == block.end());
70}
71
72// This test verifies that headers can be set in a variety of ways.
QUICHE team557dfbc2020-10-20 10:52:30 -070073TEST(Http2HeaderBlockTest, AddHeaders) {
74 Http2HeaderBlock block;
bnc44712912019-08-15 18:58:14 -070075 block["foo"] = std::string(300, 'x');
QUICHE team82dee2f2019-01-18 12:35:12 -050076 block["bar"] = "baz";
77 block["qux"] = "qux1";
78 block["qux"] = "qux2";
79 block.insert(std::make_pair("key", "value"));
80
bnc44712912019-08-15 18:58:14 -070081 EXPECT_EQ(Pair("foo", std::string(300, 'x')), *block.find("foo"));
QUICHE team82dee2f2019-01-18 12:35:12 -050082 EXPECT_EQ("baz", block["bar"]);
bnc44712912019-08-15 18:58:14 -070083 std::string qux("qux");
QUICHE team82dee2f2019-01-18 12:35:12 -050084 EXPECT_EQ("qux2", block[qux]);
85 ASSERT_NE(block.end(), block.find("key"));
86 EXPECT_EQ(Pair("key", "value"), *block.find("key"));
87
88 block.erase("key");
89 EXPECT_EQ(block.end(), block.find("key"));
90}
91
QUICHE team557dfbc2020-10-20 10:52:30 -070092// This test verifies that Http2HeaderBlock can be copied using Clone().
93TEST(Http2HeaderBlockTest, CopyBlocks) {
94 Http2HeaderBlock block1;
bnc44712912019-08-15 18:58:14 -070095 block1["foo"] = std::string(300, 'x');
QUICHE team82dee2f2019-01-18 12:35:12 -050096 block1["bar"] = "baz";
97 block1.insert(std::make_pair("qux", "qux1"));
98
QUICHE team557dfbc2020-10-20 10:52:30 -070099 Http2HeaderBlock block2 = block1.Clone();
100 Http2HeaderBlock block3(block1.Clone());
QUICHE team82dee2f2019-01-18 12:35:12 -0500101
102 EXPECT_EQ(block1, block2);
103 EXPECT_EQ(block1, block3);
104}
105
QUICHE team557dfbc2020-10-20 10:52:30 -0700106TEST(Http2HeaderBlockTest, Equality) {
QUICHE team82dee2f2019-01-18 12:35:12 -0500107 // Test equality and inequality operators.
QUICHE team557dfbc2020-10-20 10:52:30 -0700108 Http2HeaderBlock block1;
QUICHE team82dee2f2019-01-18 12:35:12 -0500109 block1["foo"] = "bar";
110
QUICHE team557dfbc2020-10-20 10:52:30 -0700111 Http2HeaderBlock block2;
QUICHE team82dee2f2019-01-18 12:35:12 -0500112 block2["foo"] = "bar";
113
QUICHE team557dfbc2020-10-20 10:52:30 -0700114 Http2HeaderBlock block3;
QUICHE team82dee2f2019-01-18 12:35:12 -0500115 block3["baz"] = "qux";
116
117 EXPECT_EQ(block1, block2);
118 EXPECT_NE(block1, block3);
119
120 block2["baz"] = "qux";
121 EXPECT_NE(block1, block2);
122}
123
QUICHE team557dfbc2020-10-20 10:52:30 -0700124Http2HeaderBlock ReturnTestHeaderBlock() {
125 Http2HeaderBlock block;
QUICHE team0b745502019-12-10 11:02:08 -0800126 block["foo"] = "bar";
127 block.insert(std::make_pair("foo2", "baz"));
128 return block;
129}
130
QUICHE team82dee2f2019-01-18 12:35:12 -0500131// Test that certain methods do not crash on moved-from instances.
QUICHE team557dfbc2020-10-20 10:52:30 -0700132TEST(Http2HeaderBlockTest, MovedFromIsValid) {
133 Http2HeaderBlock block1;
QUICHE team82dee2f2019-01-18 12:35:12 -0500134 block1["foo"] = "bar";
135
QUICHE team557dfbc2020-10-20 10:52:30 -0700136 Http2HeaderBlock block2(std::move(block1));
QUICHE team82dee2f2019-01-18 12:35:12 -0500137 EXPECT_THAT(block2, ElementsAre(Pair("foo", "bar")));
138
139 block1["baz"] = "qux"; // NOLINT testing post-move behavior
140
QUICHE team557dfbc2020-10-20 10:52:30 -0700141 Http2HeaderBlock block3(std::move(block1));
QUICHE team82dee2f2019-01-18 12:35:12 -0500142
143 block1["foo"] = "bar"; // NOLINT testing post-move behavior
144
QUICHE team557dfbc2020-10-20 10:52:30 -0700145 Http2HeaderBlock block4(std::move(block1));
QUICHE team82dee2f2019-01-18 12:35:12 -0500146
147 block1.clear(); // NOLINT testing post-move behavior
148 EXPECT_TRUE(block1.empty());
149
150 block1["foo"] = "bar";
151 EXPECT_THAT(block1, ElementsAre(Pair("foo", "bar")));
QUICHE team0b745502019-12-10 11:02:08 -0800152
QUICHE team557dfbc2020-10-20 10:52:30 -0700153 Http2HeaderBlock block5 = ReturnTestHeaderBlock();
QUICHE team0b745502019-12-10 11:02:08 -0800154 block5.AppendValueOrAddHeader("foo", "bar2");
155 EXPECT_THAT(block5, ElementsAre(Pair("foo", std::string("bar\0bar2", 8)),
156 Pair("foo2", "baz")));
QUICHE team82dee2f2019-01-18 12:35:12 -0500157}
158
159// This test verifies that headers can be appended to no matter how they were
160// added originally.
QUICHE team557dfbc2020-10-20 10:52:30 -0700161TEST(Http2HeaderBlockTest, AppendHeaders) {
162 Http2HeaderBlock block;
QUICHE team82dee2f2019-01-18 12:35:12 -0500163 block["foo"] = "foo";
164 block.AppendValueOrAddHeader("foo", "bar");
bnc44712912019-08-15 18:58:14 -0700165 EXPECT_EQ(Pair("foo", std::string("foo\0bar", 7)), *block.find("foo"));
QUICHE team82dee2f2019-01-18 12:35:12 -0500166
167 block.insert(std::make_pair("foo", "baz"));
168 EXPECT_EQ("baz", block["foo"]);
169 EXPECT_EQ(Pair("foo", "baz"), *block.find("foo"));
170
171 // Try all four methods of adding an entry.
172 block["cookie"] = "key1=value1";
173 block.AppendValueOrAddHeader("h1", "h1v1");
174 block.insert(std::make_pair("h2", "h2v1"));
175
176 block.AppendValueOrAddHeader("h3", "h3v2");
177 block.AppendValueOrAddHeader("h2", "h2v2");
178 block.AppendValueOrAddHeader("h1", "h1v2");
179 block.AppendValueOrAddHeader("cookie", "key2=value2");
180
181 block.AppendValueOrAddHeader("cookie", "key3=value3");
182 block.AppendValueOrAddHeader("h1", "h1v3");
183 block.AppendValueOrAddHeader("h2", "h2v3");
184 block.AppendValueOrAddHeader("h3", "h3v3");
185 block.AppendValueOrAddHeader("h4", "singleton");
186
187 EXPECT_EQ("key1=value1; key2=value2; key3=value3", block["cookie"]);
188 EXPECT_EQ("baz", block["foo"]);
bnc44712912019-08-15 18:58:14 -0700189 EXPECT_EQ(std::string("h1v1\0h1v2\0h1v3", 14), block["h1"]);
190 EXPECT_EQ(std::string("h2v1\0h2v2\0h2v3", 14), block["h2"]);
191 EXPECT_EQ(std::string("h3v2\0h3v3", 9), block["h3"]);
QUICHE team82dee2f2019-01-18 12:35:12 -0500192 EXPECT_EQ("singleton", block["h4"]);
193}
194
QUICHE team557dfbc2020-10-20 10:52:30 -0700195TEST(Http2HeaderBlockTest, CompareValueToStringPiece) {
196 Http2HeaderBlock block;
QUICHE team3f57c752019-12-16 14:09:53 -0800197 block["foo"] = "foo";
198 block.AppendValueOrAddHeader("foo", "bar");
199 const auto& val = block["foo"];
200 const char expected[] = "foo\0bar";
vasilvvc8ccdb22020-10-12 16:42:34 -0700201 EXPECT_TRUE(absl::string_view(expected, 7) == val);
202 EXPECT_TRUE(val == absl::string_view(expected, 7));
203 EXPECT_FALSE(absl::string_view(expected, 3) == val);
204 EXPECT_FALSE(val == absl::string_view(expected, 3));
QUICHE team3f57c752019-12-16 14:09:53 -0800205 const char not_expected[] = "foo\0barextra";
vasilvvc8ccdb22020-10-12 16:42:34 -0700206 EXPECT_FALSE(absl::string_view(not_expected, 12) == val);
207 EXPECT_FALSE(val == absl::string_view(not_expected, 12));
QUICHE team3f57c752019-12-16 14:09:53 -0800208
209 const auto& val2 = block["foo2"];
vasilvvc8ccdb22020-10-12 16:42:34 -0700210 EXPECT_FALSE(absl::string_view(expected, 7) == val2);
211 EXPECT_FALSE(val2 == absl::string_view(expected, 7));
212 EXPECT_FALSE(absl::string_view("") == val2);
213 EXPECT_FALSE(val2 == absl::string_view(""));
QUICHE team3f57c752019-12-16 14:09:53 -0800214}
215
QUICHE team557dfbc2020-10-20 10:52:30 -0700216// This test demonstrates that the Http2HeaderBlock data structure does not
217// place any limitations on the characters present in the header names.
218TEST(Http2HeaderBlockTest, UpperCaseNames) {
219 Http2HeaderBlock block;
QUICHE teamef8416a2019-11-26 09:04:19 -0800220 block["Foo"] = "foo";
221 block.AppendValueOrAddHeader("Foo", "bar");
QUICHE teama0795f02020-01-03 13:01:12 -0800222 EXPECT_NE(block.end(), block.find("foo"));
QUICHE teamef8416a2019-11-26 09:04:19 -0800223 EXPECT_EQ(Pair("Foo", std::string("foo\0bar", 7)), *block.find("Foo"));
224
QUICHE teama0795f02020-01-03 13:01:12 -0800225 // The map is case insensitive, so updating "foo" modifies the entry
226 // previously added.
QUICHE teamef8416a2019-11-26 09:04:19 -0800227 block.AppendValueOrAddHeader("foo", "baz");
QUICHE teama0795f02020-01-03 13:01:12 -0800228 EXPECT_THAT(block,
229 ElementsAre(Pair("Foo", std::string("foo\0bar\0baz", 11))));
QUICHE teamef8416a2019-11-26 09:04:19 -0800230}
231
QUICHE team82dee2f2019-01-18 12:35:12 -0500232namespace {
QUICHE team557dfbc2020-10-20 10:52:30 -0700233size_t Http2HeaderBlockSize(const Http2HeaderBlock& block) {
QUICHE team82dee2f2019-01-18 12:35:12 -0500234 size_t size = 0;
235 for (const auto& pair : block) {
236 size += pair.first.size() + pair.second.size();
237 }
238 return size;
239}
240} // namespace
241
QUICHE team557dfbc2020-10-20 10:52:30 -0700242// Tests Http2HeaderBlock SizeEstimate().
243TEST(Http2HeaderBlockTest, TotalBytesUsed) {
244 Http2HeaderBlock block;
QUICHE team82dee2f2019-01-18 12:35:12 -0500245 const size_t value_size = 300;
bnc44712912019-08-15 18:58:14 -0700246 block["foo"] = std::string(value_size, 'x');
QUICHE team557dfbc2020-10-20 10:52:30 -0700247 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
bnc44712912019-08-15 18:58:14 -0700248 block.insert(std::make_pair("key", std::string(value_size, 'x')));
QUICHE team557dfbc2020-10-20 10:52:30 -0700249 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
bnc44712912019-08-15 18:58:14 -0700250 block.AppendValueOrAddHeader("abc", std::string(value_size, 'x'));
QUICHE team557dfbc2020-10-20 10:52:30 -0700251 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
QUICHE team82dee2f2019-01-18 12:35:12 -0500252
253 // Replace value for existing key.
bnc44712912019-08-15 18:58:14 -0700254 block["foo"] = std::string(value_size, 'x');
QUICHE team557dfbc2020-10-20 10:52:30 -0700255 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
bnc44712912019-08-15 18:58:14 -0700256 block.insert(std::make_pair("key", std::string(value_size, 'x')));
QUICHE team557dfbc2020-10-20 10:52:30 -0700257 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
QUICHE team82dee2f2019-01-18 12:35:12 -0500258 // Add value for existing key.
bnc44712912019-08-15 18:58:14 -0700259 block.AppendValueOrAddHeader("abc", std::string(value_size, 'x'));
QUICHE team557dfbc2020-10-20 10:52:30 -0700260 EXPECT_EQ(block.TotalBytesUsed(), Http2HeaderBlockSize(block));
QUICHE team82dee2f2019-01-18 12:35:12 -0500261
QUICHE team557dfbc2020-10-20 10:52:30 -0700262 // Copies/clones Http2HeaderBlock.
QUICHE team82dee2f2019-01-18 12:35:12 -0500263 size_t block_size = block.TotalBytesUsed();
QUICHE team557dfbc2020-10-20 10:52:30 -0700264 Http2HeaderBlock block_copy = std::move(block);
QUICHE team82dee2f2019-01-18 12:35:12 -0500265 EXPECT_EQ(block_size, block_copy.TotalBytesUsed());
266
267 // Erases key.
268 block_copy.erase("foo");
QUICHE team557dfbc2020-10-20 10:52:30 -0700269 EXPECT_EQ(block_copy.TotalBytesUsed(), Http2HeaderBlockSize(block_copy));
QUICHE team82dee2f2019-01-18 12:35:12 -0500270 block_copy.erase("key");
QUICHE team557dfbc2020-10-20 10:52:30 -0700271 EXPECT_EQ(block_copy.TotalBytesUsed(), Http2HeaderBlockSize(block_copy));
QUICHE team82dee2f2019-01-18 12:35:12 -0500272 block_copy.erase("abc");
QUICHE team557dfbc2020-10-20 10:52:30 -0700273 EXPECT_EQ(block_copy.TotalBytesUsed(), Http2HeaderBlockSize(block_copy));
QUICHE team82dee2f2019-01-18 12:35:12 -0500274}
275
276} // namespace test
277} // namespace spdy