1 // Copyright (c) 2010 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
5 #include "net/http/http_request_headers.h"
6
7 #include "base/logging.h"
8 #include "base/string_split.h"
9 #include "base/string_util.h"
10 #include "net/http/http_util.h"
11
12 namespace net {
13
14 const char HttpRequestHeaders::kGetMethod[] = "GET";
15 const char HttpRequestHeaders::kAcceptCharset[] = "Accept-Charset";
16 const char HttpRequestHeaders::kAcceptEncoding[] = "Accept-Encoding";
17 const char HttpRequestHeaders::kAcceptLanguage[] = "Accept-Language";
18 const char HttpRequestHeaders::kCacheControl[] = "Cache-Control";
19 const char HttpRequestHeaders::kConnection[] = "Connection";
20 const char HttpRequestHeaders::kContentLength[] = "Content-Length";
21 const char HttpRequestHeaders::kContentType[] = "Content-Type";
22 const char HttpRequestHeaders::kCookie[] = "Cookie";
23 const char HttpRequestHeaders::kHost[] = "Host";
24 const char HttpRequestHeaders::kIfModifiedSince[] = "If-Modified-Since";
25 const char HttpRequestHeaders::kIfNoneMatch[] = "If-None-Match";
26 const char HttpRequestHeaders::kIfRange[] = "If-Range";
27 const char HttpRequestHeaders::kOrigin[] = "Origin";
28 const char HttpRequestHeaders::kPragma[] = "Pragma";
29 const char HttpRequestHeaders::kProxyConnection[] = "Proxy-Connection";
30 const char HttpRequestHeaders::kRange[] = "Range";
31 const char HttpRequestHeaders::kReferer[] = "Referer";
32 const char HttpRequestHeaders::kUserAgent[] = "User-Agent";
33 const char HttpRequestHeaders::kTransferEncoding[] = "Transfer-Encoding";
34
HeaderKeyValuePair()35 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair() {
36 }
37
HeaderKeyValuePair(const base::StringPiece & key,const base::StringPiece & value)38 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair(
39 const base::StringPiece& key, const base::StringPiece& value)
40 : key(key.data(), key.size()), value(value.data(), value.size()) {
41 }
42
43
Iterator(const HttpRequestHeaders & headers)44 HttpRequestHeaders::Iterator::Iterator(const HttpRequestHeaders& headers)
45 : started_(false),
46 curr_(headers.headers_.begin()),
47 end_(headers.headers_.end()) {}
48
~Iterator()49 HttpRequestHeaders::Iterator::~Iterator() {}
50
GetNext()51 bool HttpRequestHeaders::Iterator::GetNext() {
52 if (!started_) {
53 started_ = true;
54 return curr_ != end_;
55 }
56
57 if (curr_ == end_)
58 return false;
59
60 ++curr_;
61 return curr_ != end_;
62 }
63
HttpRequestHeaders()64 HttpRequestHeaders::HttpRequestHeaders() {}
~HttpRequestHeaders()65 HttpRequestHeaders::~HttpRequestHeaders() {}
66
GetHeader(const base::StringPiece & key,std::string * out) const67 bool HttpRequestHeaders::GetHeader(const base::StringPiece& key,
68 std::string* out) const {
69 HeaderVector::const_iterator it = FindHeader(key);
70 if (it == headers_.end())
71 return false;
72 out->assign(it->value);
73 return true;
74 }
75
Clear()76 void HttpRequestHeaders::Clear() {
77 headers_.clear();
78 }
79
SetHeader(const base::StringPiece & key,const base::StringPiece & value)80 void HttpRequestHeaders::SetHeader(const base::StringPiece& key,
81 const base::StringPiece& value) {
82 HeaderVector::iterator it = FindHeader(key);
83 if (it != headers_.end())
84 it->value = value.as_string();
85 else
86 headers_.push_back(HeaderKeyValuePair(key.as_string(), value.as_string()));
87 }
88
SetHeaderIfMissing(const base::StringPiece & key,const base::StringPiece & value)89 void HttpRequestHeaders::SetHeaderIfMissing(const base::StringPiece& key,
90 const base::StringPiece& value) {
91 HeaderVector::iterator it = FindHeader(key);
92 if (it == headers_.end())
93 headers_.push_back(HeaderKeyValuePair(key.as_string(), value.as_string()));
94 }
95
RemoveHeader(const base::StringPiece & key)96 void HttpRequestHeaders::RemoveHeader(const base::StringPiece& key) {
97 HeaderVector::iterator it = FindHeader(key);
98 if (it != headers_.end())
99 headers_.erase(it);
100 }
101
AddHeaderFromString(const base::StringPiece & header_line)102 void HttpRequestHeaders::AddHeaderFromString(
103 const base::StringPiece& header_line) {
104 DCHECK_EQ(std::string::npos, header_line.find("\r\n"))
105 << "\"" << header_line << "\" contains CRLF.";
106
107 const std::string::size_type key_end_index = header_line.find(":");
108 if (key_end_index == std::string::npos) {
109 LOG(DFATAL) << "\"" << header_line << "\" is missing colon delimiter.";
110 return;
111 }
112
113 if (key_end_index == 0) {
114 LOG(DFATAL) << "\"" << header_line << "\" is missing header key.";
115 return;
116 }
117
118 const base::StringPiece header_key(header_line.data(), key_end_index);
119
120 const std::string::size_type value_index = key_end_index + 1;
121
122 if (value_index < header_line.size()) {
123 std::string header_value(header_line.data() + value_index,
124 header_line.size() - value_index);
125 std::string::const_iterator header_value_begin =
126 header_value.begin();
127 std::string::const_iterator header_value_end =
128 header_value.end();
129 HttpUtil::TrimLWS(&header_value_begin, &header_value_end);
130
131 if (header_value_begin == header_value_end) {
132 // Value was all LWS.
133 SetHeader(header_key, "");
134 } else {
135 SetHeader(header_key,
136 base::StringPiece(&*header_value_begin,
137 header_value_end - header_value_begin));
138 }
139 } else if (value_index == header_line.size()) {
140 SetHeader(header_key, "");
141 } else {
142 NOTREACHED();
143 }
144 }
145
AddHeadersFromString(const base::StringPiece & headers)146 void HttpRequestHeaders::AddHeadersFromString(
147 const base::StringPiece& headers) {
148 // TODO(willchan): Consider adding more StringPiece support in string_util.h
149 // to eliminate copies.
150 std::vector<std::string> header_line_vector;
151 base::SplitStringUsingSubstr(headers.as_string(), "\r\n",
152 &header_line_vector);
153 for (std::vector<std::string>::const_iterator it = header_line_vector.begin();
154 it != header_line_vector.end(); ++it) {
155 if (!it->empty())
156 AddHeaderFromString(*it);
157 }
158 }
159
MergeFrom(const HttpRequestHeaders & other)160 void HttpRequestHeaders::MergeFrom(const HttpRequestHeaders& other) {
161 for (HeaderVector::const_iterator it = other.headers_.begin();
162 it != other.headers_.end(); ++it ) {
163 SetHeader(it->key, it->value);
164 }
165 }
166
ToString() const167 std::string HttpRequestHeaders::ToString() const {
168 std::string output;
169 for (HeaderVector::const_iterator it = headers_.begin();
170 it != headers_.end(); ++it) {
171 if (!it->value.empty()) {
172 base::StringAppendF(&output, "%s: %s\r\n",
173 it->key.c_str(), it->value.c_str());
174 } else {
175 base::StringAppendF(&output, "%s:\r\n", it->key.c_str());
176 }
177 }
178 output.append("\r\n");
179 return output;
180 }
181
182 HttpRequestHeaders::HeaderVector::iterator
FindHeader(const base::StringPiece & key)183 HttpRequestHeaders::FindHeader(const base::StringPiece& key) {
184 for (HeaderVector::iterator it = headers_.begin();
185 it != headers_.end(); ++it) {
186 if (key.length() == it->key.length() &&
187 !base::strncasecmp(key.data(), it->key.data(), key.length()))
188 return it;
189 }
190
191 return headers_.end();
192 }
193
194 HttpRequestHeaders::HeaderVector::const_iterator
FindHeader(const base::StringPiece & key) const195 HttpRequestHeaders::FindHeader(const base::StringPiece& key) const {
196 for (HeaderVector::const_iterator it = headers_.begin();
197 it != headers_.end(); ++it) {
198 if (key.length() == it->key.length() &&
199 !base::strncasecmp(key.data(), it->key.data(), key.length()))
200 return it;
201 }
202
203 return headers_.end();
204 }
205
206 } // namespace net
207