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. 4 5 #include "net/http/http_request_headers.h" 6 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" 14 15 namespace net { 16 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"; 39 40 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair() { 41 } 42 43 HttpRequestHeaders::HeaderKeyValuePair::HeaderKeyValuePair( 44 const base::StringPiece& key, const base::StringPiece& value) 45 : key(key.data(), key.size()), value(value.data(), value.size()) { 46 } 47 48 49 HttpRequestHeaders::Iterator::Iterator(const HttpRequestHeaders& headers) 50 : started_(false), 51 curr_(headers.headers_.begin()), 52 end_(headers.headers_.end()) {} 53 54 HttpRequestHeaders::Iterator::~Iterator() {} 55 56 bool HttpRequestHeaders::Iterator::GetNext() { 57 if (!started_) { 58 started_ = true; 59 return curr_ != end_; 60 } 61 62 if (curr_ == end_) 63 return false; 64 65 ++curr_; 66 return curr_ != end_; 67 } 68 69 HttpRequestHeaders::HttpRequestHeaders() {} 70 HttpRequestHeaders::~HttpRequestHeaders() {} 71 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()) 76 return false; 77 out->assign(it->value); 78 return true; 79 } 80 81 void HttpRequestHeaders::Clear() { 82 headers_.clear(); 83 } 84 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()); 92 else 93 headers_.push_back(HeaderKeyValuePair(key, value)); 94 } 95 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)); 103 } 104 105 void HttpRequestHeaders::RemoveHeader(const base::StringPiece& key) { 106 HeaderVector::iterator it = FindHeader(key); 107 if (it != headers_.end()) 108 headers_.erase(it); 109 } 110 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."; 115 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."; 119 return; 120 } 121 122 if (key_end_index == 0) { 123 LOG(DFATAL) << "\"" << header_line << "\" is missing header key."; 124 return; 125 } 126 127 const base::StringPiece header_key(header_line.data(), key_end_index); 128 129 const std::string::size_type value_index = key_end_index + 1; 130 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 = 137 header_value.end(); 138 HttpUtil::TrimLWS(&header_value_begin, &header_value_end); 139 140 if (header_value_begin == header_value_end) { 141 // Value was all LWS. 142 SetHeader(header_key, ""); 143 } else { 144 SetHeader(header_key, 145 base::StringPiece(&*header_value_begin, 146 header_value_end - header_value_begin)); 147 } 148 } else if (value_index == header_line.size()) { 149 SetHeader(header_key, ""); 150 } else { 151 NOTREACHED(); 152 } 153 } 154 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) { 164 if (!it->empty()) 165 AddHeaderFromString(*it); 166 } 167 } 168 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); 173 } 174 } 175 176 std::string HttpRequestHeaders::ToString() const { 177 std::string output; 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()); 183 } else { 184 base::StringAppendF(&output, "%s:\r\n", it->key.c_str()); 185 } 186 } 187 output.append("\r\n"); 188 return output; 189 } 190 191 base::Value* HttpRequestHeaders::NetLogCallback( 192 const std::string* request_line, 193 NetLog::LogLevel log_level) const { 194 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 = ElideHeaderValueForNetLog( 200 log_level, it->key, it->value); 201 headers->Append(new base::StringValue( 202 base::StringPrintf("%s: %s", 203 it->key.c_str(), log_value.c_str()))); 204 } 205 dict->Set("headers", headers); 206 return dict; 207 } 208 209 // static 210 bool HttpRequestHeaders::FromNetLogParam(const base::Value* event_param, 211 HttpRequestHeaders* headers, 212 std::string* request_line) { 213 headers->Clear(); 214 *request_line = ""; 215 216 const base::DictionaryValue* dict = NULL; 217 const base::ListValue* header_list = NULL; 218 219 if (!event_param || 220 !event_param->GetAsDictionary(&dict) || 221 !dict->GetList("headers", &header_list) || 222 !dict->GetString("line", request_line)) { 223 return false; 224 } 225 226 for (base::ListValue::const_iterator it = header_list->begin(); 227 it != header_list->end(); 228 ++it) { 229 std::string header_line; 230 if (!(*it)->GetAsString(&header_line)) { 231 headers->Clear(); 232 *request_line = ""; 233 return false; 234 } 235 headers->AddHeaderFromString(header_line); 236 } 237 return true; 238 } 239 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())) 246 return it; 247 } 248 249 return headers_.end(); 250 } 251 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())) 258 return it; 259 } 260 261 return headers_.end(); 262 } 263 264 } // namespace net 265