// Copyright 2015 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/base/apple/http_response_headers_util.h" #import #include "base/check.h" #include "base/format_macros.h" #include "base/strings/stringprintf.h" #include "base/strings/sys_string_conversions.h" #include "net/http/http_util.h" namespace { // String format used to create the http status line from the status code and // its localized description. char const kHttpStatusLineFormat[] = "HTTP %" PRIdNS " DummyStatusDescription"; } // anonymous namespace namespace net { NSString* FixNSStringIncorrectlyDecodedAsLatin1(NSString* string) { // Check if the string contains any character above '\u007f'. If not, // then the "latin1" and "utf-8" representation are identical, and there // is no need to allocate memory for the conversion. static NSCharacterSet* non_ascii_characters = [NSCharacterSet characterSetWithRange:NSMakeRange(0x0080, 0x10ffff - 0x0080)]; NSRange range = [string rangeOfCharacterFromSet:non_ascii_characters]; if (range.location == NSNotFound && range.length == 0) return string; // Try to save the string as "latin1". Will fail if it does contains // characters that falls out of the "latin1" range (i.e. after '\u00ff'). NSData* data = [string dataUsingEncoding:NSISOLatin1StringEncoding allowLossyConversion:NO]; if (!data) return string; // Try to load the saved data as "utf-8". Will fail if the string is // not encoded in "utf-8". In that case, it was probably not garbled // and the original string needs to be used. This will be the case for // strings that are genuinely encoded in "latin1". NSString* decoded = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]; if (!decoded) return string; return decoded; } scoped_refptr CreateHeadersFromNSHTTPURLResponse( NSHTTPURLResponse* response) { DCHECK(response); // Initialize the header with a generated status line. scoped_refptr http_headers(new HttpResponseHeaders( base::StringPrintf(kHttpStatusLineFormat, response.statusCode))); // Iterate through |response|'s headers and add them to |http_headers|. [response.allHeaderFields enumerateKeysAndObjectsUsingBlock:^( NSString* name, NSString* value, BOOL*) { const std::string header_name = base::SysNSStringToUTF8(name); if (!HttpUtil::IsValidHeaderName(header_name)) return; const std::string header_value = base::SysNSStringToUTF8(FixNSStringIncorrectlyDecodedAsLatin1(value)); if (!HttpUtil::IsValidHeaderValue(header_value)) return; http_headers->AddHeader(header_name, header_value); }]; return http_headers; } } // namespace net