1 // Copyright 2014 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.
5 #include "net/spdy/hpack/hpack_encoder.h"
9 #include "base/logging.h"
10 #include "net/spdy/hpack/hpack_header_table.h"
11 #include "net/spdy/hpack/hpack_huffman_table.h"
12 #include "net/spdy/hpack/hpack_output_stream.h"
16 using base::StringPiece
;
19 HpackEncoder::HpackEncoder(const HpackHuffmanTable
& table
)
21 allow_huffman_compression_(true),
22 huffman_table_(table
),
24 total_char_counts_(NULL
) {}
26 HpackEncoder::~HpackEncoder() {}
28 bool HpackEncoder::EncodeHeaderSet(const SpdyHeaderBlock
& header_set
,
30 // Separate header set into pseudo-headers and regular headers.
31 Representations pseudo_headers
;
32 Representations regular_headers
;
33 for (const auto& header
: header_set
) {
34 if (header
.first
== "cookie") {
35 // Note that there can only be one "cookie" header, because header_set is
37 CookieToCrumbs(header
, ®ular_headers
);
38 } else if (header
.first
[0] == kPseudoHeaderPrefix
) {
39 DecomposeRepresentation(header
, &pseudo_headers
);
41 DecomposeRepresentation(header
, ®ular_headers
);
45 // Encode pseudo-headers.
46 for (const auto& header
: pseudo_headers
) {
47 const HpackEntry
* entry
=
48 header_table_
.GetByNameAndValue(header
.first
, header
.second
);
52 if (header
.first
== ":authority") {
53 // :authority is always present and rarely changes, and has moderate
54 // length, therefore it makes a lot of sense to index (insert in the
56 EmitIndexedLiteral(header
);
58 // Most common pseudo-header fields are represented in the static table,
59 // while uncommon ones are small, so do not index them.
60 EmitNonIndexedLiteral(header
);
65 // Encode regular headers.
66 for (const auto& header
: regular_headers
) {
67 const HpackEntry
* entry
=
68 header_table_
.GetByNameAndValue(header
.first
, header
.second
);
72 EmitIndexedLiteral(header
);
76 output_stream_
.TakeString(output
);
80 bool HpackEncoder::EncodeHeaderSetWithoutCompression(
81 const SpdyHeaderBlock
& header_set
,
83 allow_huffman_compression_
= false;
84 for (const auto& header
: header_set
) {
85 // Note that cookies are not crumbled in this case.
86 EmitNonIndexedLiteral(header
);
88 allow_huffman_compression_
= true;
89 output_stream_
.TakeString(output
);
93 void HpackEncoder::EmitIndex(const HpackEntry
* entry
) {
94 output_stream_
.AppendPrefix(kIndexedOpcode
);
95 output_stream_
.AppendUint32(header_table_
.IndexOf(entry
));
98 void HpackEncoder::EmitIndexedLiteral(const Representation
& representation
) {
99 output_stream_
.AppendPrefix(kLiteralIncrementalIndexOpcode
);
100 EmitLiteral(representation
);
101 header_table_
.TryAddEntry(representation
.first
, representation
.second
);
104 void HpackEncoder::EmitNonIndexedLiteral(const Representation
& representation
) {
105 output_stream_
.AppendPrefix(kLiteralNoIndexOpcode
);
106 output_stream_
.AppendUint32(0);
107 EmitString(representation
.first
);
108 EmitString(representation
.second
);
111 void HpackEncoder::EmitLiteral(const Representation
& representation
) {
112 const HpackEntry
* name_entry
= header_table_
.GetByName(representation
.first
);
113 if (name_entry
!= NULL
) {
114 output_stream_
.AppendUint32(header_table_
.IndexOf(name_entry
));
116 output_stream_
.AppendUint32(0);
117 EmitString(representation
.first
);
119 EmitString(representation
.second
);
122 void HpackEncoder::EmitString(StringPiece str
) {
123 size_t encoded_size
=
124 (!allow_huffman_compression_
? str
.size()
125 : huffman_table_
.EncodedSize(str
));
126 if (encoded_size
< str
.size()) {
127 output_stream_
.AppendPrefix(kStringLiteralHuffmanEncoded
);
128 output_stream_
.AppendUint32(encoded_size
);
129 huffman_table_
.EncodeString(str
, &output_stream_
);
131 output_stream_
.AppendPrefix(kStringLiteralIdentityEncoded
);
132 output_stream_
.AppendUint32(str
.size());
133 output_stream_
.AppendBytes(str
);
135 UpdateCharacterCounts(str
);
138 void HpackEncoder::SetCharCountsStorage(std::vector
<size_t>* char_counts
,
139 size_t* total_char_counts
) {
140 CHECK_LE(256u, char_counts
->size());
141 char_counts_
= char_counts
;
142 total_char_counts_
= total_char_counts
;
145 void HpackEncoder::UpdateCharacterCounts(base::StringPiece str
) {
146 if (char_counts_
== NULL
|| total_char_counts_
== NULL
) {
149 for (StringPiece::const_iterator it
= str
.begin(); it
!= str
.end(); ++it
) {
150 ++(*char_counts_
)[static_cast<uint8
>(*it
)];
152 (*total_char_counts_
) += str
.size();
156 void HpackEncoder::CookieToCrumbs(const Representation
& cookie
,
157 Representations
* out
) {
158 // See Section 8.1.2.5. "Compressing the Cookie Header Field" in the HTTP/2
159 // specification at https://tools.ietf.org/html/draft-ietf-httpbis-http2-14.
160 // Cookie values are split into individually-encoded HPACK representations.
161 StringPiece cookie_value
= cookie
.second
;
162 // Consume leading and trailing whitespace if present.
163 StringPiece::size_type first
= cookie_value
.find_first_not_of(" \t");
164 StringPiece::size_type last
= cookie_value
.find_last_not_of(" \t");
165 if (first
== StringPiece::npos
) {
166 cookie_value
.clear();
168 cookie_value
= cookie_value
.substr(first
, (last
- first
) + 1);
170 for (size_t pos
= 0;;) {
171 size_t end
= cookie_value
.find(";", pos
);
173 if (end
== StringPiece::npos
) {
174 out
->push_back(std::make_pair(cookie
.first
, cookie_value
.substr(pos
)));
178 std::make_pair(cookie
.first
, cookie_value
.substr(pos
, end
- pos
)));
180 // Consume next space if present.
182 if (pos
!= cookie_value
.size() && cookie_value
[pos
] == ' ') {
189 void HpackEncoder::DecomposeRepresentation(const Representation
& header_field
,
190 Representations
* out
) {
193 while (end
!= StringPiece::npos
) {
194 end
= header_field
.second
.find('\0', pos
);
195 out
->push_back(std::make_pair(header_field
.first
,
196 header_field
.second
.substr(pos
, end
- pos
)));