1 // 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.
5 #include "net/http/http_request_headers.h"
7 #include "base/logging.h"
8 #include "base/strings/string_split.h"
9 #include "base/strings/string_util.h"
10 #include "base/strings/stringprintf.h"
11 #include "base/values.h"
12 #include "net/http/http_log_util.h"
13 #include "net/http/http_util.h"
17 const char HttpRequestHeaders::kGetMethod
[] = "GET";
18 const char HttpRequestHeaders::kAcceptCharset
[] = "Accept-Charset";
19 const char HttpRequestHeaders::kAcceptEncoding
[] = "Accept-Encoding";
20 const char HttpRequestHeaders::kAcceptLanguage
[] = "Accept-Language";
21 const char HttpRequestHeaders::kAuthorization
[] = "Authorization";
22 const char HttpRequestHeaders::kCacheControl
[] = "Cache-Control";
23 const char HttpRequestHeaders::kConnection
[] = "Connection";
24 const char HttpRequestHeaders::kContentLength
[] = "Content-Length";
25 const char HttpRequestHeaders::kContentType
[] = "Content-Type";
26 const char HttpRequestHeaders::kCookie
[] = "Cookie";
27 const char HttpRequestHeaders::kHost
[] = "Host";
28 const char HttpRequestHeaders::kIfModifiedSince
[] = "If-Modified-Since";
29 const char HttpRequestHeaders::kIfNoneMatch
[] = "If-None-Match";
30 const char HttpRequestHeaders::kIfRange
[] = "If-Range";
31 const char HttpRequestHeaders::kOrigin
[] = "Origin";
32 const char HttpRequestHeaders::kPragma
[] = "Pragma";
33 const char HttpRequestHeaders::kProxyAuthorization
[] = "Proxy-Authorization";
34 const char HttpRequestHeaders::kProxyConnection
[] = "Proxy-Connection";
35 const char HttpRequestHeaders::kRange
[] = "Range";
36 const char HttpRequestHeaders::kReferer
[] = "Referer";
37 const char HttpRequestHeaders::kUserAgent
[] = "User-Agent";
38 const char HttpRequestHeaders::kTransferEncoding
[] = "Transfer-Encoding";
40 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair() {
43 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair(
44 const base::StringPiece
& key
, const base::StringPiece
& value
)
45 : key(key
.data(), key
.size()), value(value
.data(), value
.size()) {
49 HttpRequestHeaders::Iterator::Iterator(const HttpRequestHeaders
& headers
)
51 curr_(headers
.headers_
.begin()),
52 end_(headers
.headers_
.end()) {}
54 HttpRequestHeaders::Iterator::~Iterator() {}
56 bool HttpRequestHeaders::Iterator::GetNext() {
69 HttpRequestHeaders::HttpRequestHeaders() {}
70 HttpRequestHeaders::~HttpRequestHeaders() {}
72 bool HttpRequestHeaders::GetHeader(const base::StringPiece
& key
,
73 std::string
* out
) const {
74 HeaderVector::const_iterator it
= FindHeader(key
);
75 if (it
== headers_
.end())
77 out
->assign(it
->value
);
81 void HttpRequestHeaders::Clear() {
85 void HttpRequestHeaders::SetHeader(const base::StringPiece
& key
,
86 const base::StringPiece
& value
) {
87 DCHECK(HttpUtil::IsValidHeaderName(key
.as_string()));
88 DCHECK(HttpUtil::IsValidHeaderValue(value
.as_string()));
89 HeaderVector::iterator it
= FindHeader(key
);
90 if (it
!= headers_
.end())
91 it
->value
.assign(value
.data(), value
.size());
93 headers_
.push_back(HeaderKeyValuePair(key
, value
));
96 void HttpRequestHeaders::SetHeaderIfMissing(const base::StringPiece
& key
,
97 const base::StringPiece
& value
) {
98 DCHECK(HttpUtil::IsValidHeaderName(key
.as_string()));
99 DCHECK(HttpUtil::IsValidHeaderValue(value
.as_string()));
100 HeaderVector::iterator it
= FindHeader(key
);
101 if (it
== headers_
.end())
102 headers_
.push_back(HeaderKeyValuePair(key
, value
));
105 void HttpRequestHeaders::RemoveHeader(const base::StringPiece
& key
) {
106 HeaderVector::iterator it
= FindHeader(key
);
107 if (it
!= headers_
.end())
111 void HttpRequestHeaders::AddHeaderFromString(
112 const base::StringPiece
& header_line
) {
113 DCHECK_EQ(std::string::npos
, header_line
.find("\r\n"))
114 << "\"" << header_line
<< "\" contains CRLF.";
116 const std::string::size_type key_end_index
= header_line
.find(":");
117 if (key_end_index
== std::string::npos
) {
118 LOG(DFATAL
) << "\"" << header_line
<< "\" is missing colon delimiter.";
122 if (key_end_index
== 0) {
123 LOG(DFATAL
) << "\"" << header_line
<< "\" is missing header key.";
127 const base::StringPiece
header_key(header_line
.data(), key_end_index
);
129 const std::string::size_type value_index
= key_end_index
+ 1;
131 if (value_index
< header_line
.size()) {
132 std::string
header_value(header_line
.data() + value_index
,
133 header_line
.size() - value_index
);
134 std::string::const_iterator header_value_begin
=
135 header_value
.begin();
136 std::string::const_iterator header_value_end
=
138 HttpUtil::TrimLWS(&header_value_begin
, &header_value_end
);
140 if (header_value_begin
== header_value_end
) {
141 // Value was all LWS.
142 SetHeader(header_key
, "");
144 SetHeader(header_key
,
145 base::StringPiece(&*header_value_begin
,
146 header_value_end
- header_value_begin
));
148 } else if (value_index
== header_line
.size()) {
149 SetHeader(header_key
, "");
155 void HttpRequestHeaders::AddHeadersFromString(
156 const base::StringPiece
& headers
) {
157 // TODO(willchan): Consider adding more StringPiece support in string_util.h
158 // to eliminate copies.
159 std::vector
<std::string
> header_line_vector
;
160 base::SplitStringUsingSubstr(headers
.as_string(), "\r\n",
161 &header_line_vector
);
162 for (std::vector
<std::string
>::const_iterator it
= header_line_vector
.begin();
163 it
!= header_line_vector
.end(); ++it
) {
165 AddHeaderFromString(*it
);
169 void HttpRequestHeaders::MergeFrom(const HttpRequestHeaders
& other
) {
170 for (HeaderVector::const_iterator it
= other
.headers_
.begin();
171 it
!= other
.headers_
.end(); ++it
) {
172 SetHeader(it
->key
, it
->value
);
176 std::string
HttpRequestHeaders::ToString() const {
178 for (HeaderVector::const_iterator it
= headers_
.begin();
179 it
!= headers_
.end(); ++it
) {
180 if (!it
->value
.empty()) {
181 base::StringAppendF(&output
, "%s: %s\r\n",
182 it
->key
.c_str(), it
->value
.c_str());
184 base::StringAppendF(&output
, "%s:\r\n", it
->key
.c_str());
187 output
.append("\r\n");
191 scoped_ptr
<base::Value
> HttpRequestHeaders::NetLogCallback(
192 const std::string
* request_line
,
193 NetLogCaptureMode capture_mode
) const {
194 scoped_ptr
<base::DictionaryValue
> dict(new base::DictionaryValue());
195 dict
->SetString("line", *request_line
);
196 base::ListValue
* headers
= new base::ListValue();
197 for (HeaderVector::const_iterator it
= headers_
.begin();
198 it
!= headers_
.end(); ++it
) {
199 std::string log_value
=
200 ElideHeaderValueForNetLog(capture_mode
, it
->key
, it
->value
);
201 headers
->Append(new base::StringValue(
202 base::StringPrintf("%s: %s",
203 it
->key
.c_str(), log_value
.c_str())));
205 dict
->Set("headers", headers
);
210 bool HttpRequestHeaders::FromNetLogParam(const base::Value
* event_param
,
211 HttpRequestHeaders
* headers
,
212 std::string
* request_line
) {
216 const base::DictionaryValue
* dict
= NULL
;
217 const base::ListValue
* header_list
= NULL
;
220 !event_param
->GetAsDictionary(&dict
) ||
221 !dict
->GetList("headers", &header_list
) ||
222 !dict
->GetString("line", request_line
)) {
226 for (base::ListValue::const_iterator it
= header_list
->begin();
227 it
!= header_list
->end();
229 std::string header_line
;
230 if (!(*it
)->GetAsString(&header_line
)) {
235 headers
->AddHeaderFromString(header_line
);
240 HttpRequestHeaders::HeaderVector::iterator
241 HttpRequestHeaders::FindHeader(const base::StringPiece
& key
) {
242 for (HeaderVector::iterator it
= headers_
.begin();
243 it
!= headers_
.end(); ++it
) {
244 if (key
.length() == it
->key
.length() &&
245 !base::strncasecmp(key
.data(), it
->key
.data(), key
.length()))
249 return headers_
.end();
252 HttpRequestHeaders::HeaderVector::const_iterator
253 HttpRequestHeaders::FindHeader(const base::StringPiece
& key
) const {
254 for (HeaderVector::const_iterator it
= headers_
.begin();
255 it
!= headers_
.end(); ++it
) {
256 if (key
.length() == it
->key
.length() &&
257 !base::strncasecmp(key
.data(), it
->key
.data(), key
.length()))
261 return headers_
.end();