mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 06:16:30 +03:00
510 lines
18 KiB
C++
510 lines
18 KiB
C++
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "net/base/escape.h"
|
|
|
|
#include "base/logging.h"
|
|
#include "base/strings/string_util.h"
|
|
#include "base/strings/utf_string_conversions.h"
|
|
|
|
namespace net {
|
|
|
|
namespace {
|
|
|
|
const char kHexString[] = "0123456789ABCDEF";
|
|
inline char IntToHex(int i) {
|
|
DCHECK_GE(i, 0) << i << " not a hex value";
|
|
DCHECK_LE(i, 15) << i << " not a hex value";
|
|
return kHexString[i];
|
|
}
|
|
|
|
// A fast bit-vector map for ascii characters.
|
|
//
|
|
// Internally stores 256 bits in an array of 8 ints.
|
|
// Does quick bit-flicking to lookup needed characters.
|
|
struct Charmap {
|
|
bool Contains(unsigned char c) const {
|
|
return ((map[c >> 5] & (1 << (c & 31))) != 0);
|
|
}
|
|
|
|
uint32_t map[8];
|
|
};
|
|
|
|
// Given text to escape and a Charmap defining which values to escape,
|
|
// return an escaped string. If use_plus is true, spaces are converted
|
|
// to +, otherwise, if spaces are in the charmap, they are converted to
|
|
// %20. And if keep_escaped is true, %XX will be kept as it is, otherwise, if
|
|
// '%' is in the charmap, it is converted to %25.
|
|
std::string Escape(base::StringPiece text,
|
|
const Charmap& charmap,
|
|
bool use_plus,
|
|
bool keep_escaped = false) {
|
|
std::string escaped;
|
|
escaped.reserve(text.length() * 3);
|
|
for (unsigned int i = 0; i < text.length(); ++i) {
|
|
unsigned char c = static_cast<unsigned char>(text[i]);
|
|
if (use_plus && ' ' == c) {
|
|
escaped.push_back('+');
|
|
} else if (keep_escaped && '%' == c && i + 2 < text.length() &&
|
|
base::IsHexDigit(text[i + 1]) && base::IsHexDigit(text[i + 2])) {
|
|
escaped.push_back('%');
|
|
} else if (charmap.Contains(c)) {
|
|
escaped.push_back('%');
|
|
escaped.push_back(IntToHex(c >> 4));
|
|
escaped.push_back(IntToHex(c & 0xf));
|
|
} else {
|
|
escaped.push_back(c);
|
|
}
|
|
}
|
|
return escaped;
|
|
}
|
|
|
|
// Contains nonzero when the corresponding character is unescapable for normal
|
|
// URLs. These characters are the ones that may change the parsing of a URL, so
|
|
// we don't want to unescape them sometimes. In many case we won't want to
|
|
// unescape spaces, but that is controlled by parameters to Unescape*.
|
|
//
|
|
// The basic rule is that we can't unescape anything that would changing parsing
|
|
// like # or ?. We also can't unescape &, =, or + since that could be part of a
|
|
// query and that could change the server's parsing of the query. Nor can we
|
|
// unescape \ since src/url/ will convert it to a /.
|
|
//
|
|
// Lastly, we can't unescape anything that doesn't have a canonical
|
|
// representation in a URL. This means that unescaping will change the URL, and
|
|
// you could get different behavior if you copy and paste the URL, or press
|
|
// enter in the URL bar. The list of characters that fall into this category
|
|
// are the ones labeled PASS (allow either escaped or unescaped) in the big
|
|
// lookup table at the top of url/url_canon_path.cc. Also, characters
|
|
// that have CHAR_QUERY set in url/url_canon_internal.cc but are not
|
|
// allowed in query strings according to http://www.ietf.org/rfc/rfc3261.txt are
|
|
// not unescaped, to avoid turning a valid url according to spec into an
|
|
// invalid one.
|
|
const char kUrlUnescape[128] = {
|
|
// NULL, control chars...
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
// ' ' ! " # $ % & ' ( ) * + , - . /
|
|
0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0,
|
|
// 0 1 2 3 4 5 6 7 8 9 : ; < = > ?
|
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 0, 1, 0,
|
|
// @ A B C D E F G H I J K L M N O
|
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
|
// P Q R S T U V W X Y Z [ \ ] ^ _
|
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 1,
|
|
// ` a b c d e f g h i j k l m n o
|
|
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
|
// p q r s t u v w x y z { | } ~ <NBSP>
|
|
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 0
|
|
};
|
|
|
|
// Attempts to unescape the sequence at |index| within |escaped_text|. If
|
|
// successful, sets |value| to the unescaped value. Returns whether
|
|
// unescaping succeeded.
|
|
template <typename STR>
|
|
bool UnescapeUnsignedCharAtIndex(STR escaped_text,
|
|
size_t index,
|
|
unsigned char* value) {
|
|
if ((index + 2) >= escaped_text.size())
|
|
return false;
|
|
if (escaped_text[index] != '%')
|
|
return false;
|
|
const typename STR::value_type most_sig_digit(
|
|
static_cast<typename STR::value_type>(escaped_text[index + 1]));
|
|
const typename STR::value_type least_sig_digit(
|
|
static_cast<typename STR::value_type>(escaped_text[index + 2]));
|
|
if (base::IsHexDigit(most_sig_digit) && base::IsHexDigit(least_sig_digit)) {
|
|
*value = base::HexDigitToInt(most_sig_digit) * 16 +
|
|
base::HexDigitToInt(least_sig_digit);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Returns true if there is an Arabic Language Mark at |index|. |first_byte|
|
|
// is the byte at |index|.
|
|
template <typename STR>
|
|
bool HasArabicLanguageMarkAtIndex(STR escaped_text,
|
|
unsigned char first_byte,
|
|
size_t index) {
|
|
if (first_byte != 0xD8)
|
|
return false;
|
|
unsigned char second_byte;
|
|
if (!UnescapeUnsignedCharAtIndex(escaped_text, index + 3, &second_byte))
|
|
return false;
|
|
return second_byte == 0x9c;
|
|
}
|
|
|
|
// Returns true if there is a BiDi control char at |index|. |first_byte| is the
|
|
// byte at |index|.
|
|
template <typename STR>
|
|
bool HasThreeByteBidiControlCharAtIndex(STR escaped_text,
|
|
unsigned char first_byte,
|
|
size_t index) {
|
|
if (first_byte != 0xE2)
|
|
return false;
|
|
unsigned char second_byte;
|
|
if (!UnescapeUnsignedCharAtIndex(escaped_text, index + 3, &second_byte))
|
|
return false;
|
|
if (second_byte != 0x80 && second_byte != 0x81)
|
|
return false;
|
|
unsigned char third_byte;
|
|
if (!UnescapeUnsignedCharAtIndex(escaped_text, index + 6, &third_byte))
|
|
return false;
|
|
if (second_byte == 0x80) {
|
|
return third_byte == 0x8E ||
|
|
third_byte == 0x8F ||
|
|
(third_byte >= 0xAA && third_byte <= 0xAE);
|
|
}
|
|
return third_byte >= 0xA6 && third_byte <= 0xA9;
|
|
}
|
|
|
|
// Returns true if there is a four-byte banned char at |index|. |first_byte| is
|
|
// the byte at |index|.
|
|
template <typename STR>
|
|
bool HasFourByteBannedCharAtIndex(STR escaped_text,
|
|
unsigned char first_byte,
|
|
size_t index) {
|
|
// The following characters are blacklisted for spoofability concerns.
|
|
// U+1F50F LOCK WITH INK PEN (%F0%9F%94%8F)
|
|
// U+1F510 CLOSED LOCK WITH KEY (%F0%9F%94%90)
|
|
// U+1F512 LOCK (%F0%9F%94%92)
|
|
// U+1F513 OPEN LOCK (%F0%9F%94%93)
|
|
if (first_byte != 0xF0)
|
|
return false;
|
|
|
|
unsigned char second_byte;
|
|
if (!UnescapeUnsignedCharAtIndex(escaped_text, index + 3, &second_byte) ||
|
|
second_byte != 0x9F) {
|
|
return false;
|
|
}
|
|
|
|
unsigned char third_byte;
|
|
if (!UnescapeUnsignedCharAtIndex(escaped_text, index + 6, &third_byte) ||
|
|
third_byte != 0x94) {
|
|
return false;
|
|
}
|
|
|
|
unsigned char fourth_byte;
|
|
return UnescapeUnsignedCharAtIndex(escaped_text, index + 9, &fourth_byte) &&
|
|
(fourth_byte == 0x8F || fourth_byte == 0x90 || fourth_byte == 0x92 ||
|
|
fourth_byte == 0x93);
|
|
}
|
|
|
|
// Unescapes |escaped_text| according to |rules|, returning the resulting
|
|
// string. Fills in an |adjustments| parameter, if non-NULL, so it reflects
|
|
// the alterations done to the string that are not one-character-to-one-
|
|
// character. The resulting |adjustments| will always be sorted by increasing
|
|
// offset.
|
|
template <typename STR>
|
|
STR UnescapeURLWithAdjustmentsImpl(
|
|
base::BasicStringPiece<STR> escaped_text,
|
|
UnescapeRule::Type rules,
|
|
base::OffsetAdjuster::Adjustments* adjustments) {
|
|
if (adjustments)
|
|
adjustments->clear();
|
|
// Do not unescape anything, return the |escaped_text| text.
|
|
if (rules == UnescapeRule::NONE)
|
|
return escaped_text.as_string();
|
|
|
|
// The output of the unescaping is always smaller than the input, so we can
|
|
// reserve the input size to make sure we have enough buffer and don't have
|
|
// to allocate in the loop below.
|
|
STR result;
|
|
result.reserve(escaped_text.length());
|
|
|
|
// Locations of adjusted text.
|
|
for (size_t i = 0, max = escaped_text.size(); i < max; ++i) {
|
|
if (static_cast<unsigned char>(escaped_text[i]) >= 128) {
|
|
// Non ASCII character, append as is.
|
|
result.push_back(escaped_text[i]);
|
|
continue;
|
|
}
|
|
|
|
unsigned char first_byte;
|
|
if (UnescapeUnsignedCharAtIndex(escaped_text, i, &first_byte)) {
|
|
// Per http://tools.ietf.org/html/rfc3987#section-4.1, the following BiDi
|
|
// control characters are not allowed to appear unescaped in URLs:
|
|
//
|
|
// U+200E LEFT-TO-RIGHT MARK (%E2%80%8E)
|
|
// U+200F RIGHT-TO-LEFT MARK (%E2%80%8F)
|
|
// U+202A LEFT-TO-RIGHT EMBEDDING (%E2%80%AA)
|
|
// U+202B RIGHT-TO-LEFT EMBEDDING (%E2%80%AB)
|
|
// U+202C POP DIRECTIONAL FORMATTING (%E2%80%AC)
|
|
// U+202D LEFT-TO-RIGHT OVERRIDE (%E2%80%AD)
|
|
// U+202E RIGHT-TO-LEFT OVERRIDE (%E2%80%AE)
|
|
//
|
|
// Additionally, the Unicode Technical Report (TR9) as referenced by RFC
|
|
// 3987 above has since added some new BiDi control characters.
|
|
// http://www.unicode.org/reports/tr9
|
|
//
|
|
// U+061C ARABIC LETTER MARK (%D8%9C)
|
|
// U+2066 LEFT-TO-RIGHT ISOLATE (%E2%81%A6)
|
|
// U+2067 RIGHT-TO-LEFT ISOLATE (%E2%81%A7)
|
|
// U+2068 FIRST STRONG ISOLATE (%E2%81%A8)
|
|
// U+2069 POP DIRECTIONAL ISOLATE (%E2%81%A9)
|
|
//
|
|
// The following spoofable characters are also banned, because they could
|
|
// be used to imitate parts of a web browser's UI.
|
|
//
|
|
// U+1F50F LOCK WITH INK PEN (%F0%9F%94%8F)
|
|
// U+1F510 CLOSED LOCK WITH KEY (%F0%9F%94%90)
|
|
// U+1F512 LOCK (%F0%9F%94%92)
|
|
// U+1F513 OPEN LOCK (%F0%9F%94%93)
|
|
//
|
|
// However, some schemes such as data: and file: need to parse the exact
|
|
// binary data when loading the URL. For that reason,
|
|
// SPOOFING_AND_CONTROL_CHARS allows unescaping BiDi control characters.
|
|
// DO NOT use SPOOFING_AND_CONTROL_CHARS if the parsed URL is going to be
|
|
// displayed in the UI.
|
|
if (!(rules & UnescapeRule::SPOOFING_AND_CONTROL_CHARS)) {
|
|
if (HasArabicLanguageMarkAtIndex(escaped_text, first_byte, i)) {
|
|
// Keep Arabic Language Mark escaped.
|
|
escaped_text.substr(i, 6).AppendToString(&result);
|
|
i += 5;
|
|
continue;
|
|
}
|
|
if (HasThreeByteBidiControlCharAtIndex(escaped_text, first_byte, i)) {
|
|
// Keep BiDi control char escaped.
|
|
escaped_text.substr(i, 9).AppendToString(&result);
|
|
i += 8;
|
|
continue;
|
|
}
|
|
if (HasFourByteBannedCharAtIndex(escaped_text, first_byte, i)) {
|
|
// Keep banned char escaped.
|
|
escaped_text.substr(i, 12).AppendToString(&result);
|
|
i += 11;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (first_byte >= 0x80 || // Unescape all high-bit characters.
|
|
// For 7-bit characters, the lookup table tells us all valid chars.
|
|
(kUrlUnescape[first_byte] ||
|
|
// ...and we allow some additional unescaping when flags are set.
|
|
(first_byte == ' ' && (rules & UnescapeRule::SPACES)) ||
|
|
// Allow any of the prohibited but non-control characters when
|
|
// we're doing "special" chars.
|
|
((first_byte == '/' || first_byte == '\\') &&
|
|
(rules & UnescapeRule::PATH_SEPARATORS)) ||
|
|
(first_byte > ' ' && first_byte != '/' && first_byte != '\\' &&
|
|
(rules & UnescapeRule::URL_SPECIAL_CHARS_EXCEPT_PATH_SEPARATORS)) ||
|
|
// Additionally allow non-display characters if requested.
|
|
(first_byte < ' ' &&
|
|
(rules & UnescapeRule::SPOOFING_AND_CONTROL_CHARS)))) {
|
|
// Use the unescaped version of the character.
|
|
if (adjustments)
|
|
adjustments->push_back(base::OffsetAdjuster::Adjustment(i, 3, 1));
|
|
result.push_back(first_byte);
|
|
i += 2;
|
|
} else {
|
|
// Keep escaped. Append a percent and we'll get the following two
|
|
// digits on the next loops through.
|
|
result.push_back('%');
|
|
}
|
|
} else if ((rules & UnescapeRule::REPLACE_PLUS_WITH_SPACE) &&
|
|
escaped_text[i] == '+') {
|
|
result.push_back(' ');
|
|
} else {
|
|
// Normal case for unescaped characters.
|
|
result.push_back(escaped_text[i]);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
template <class str>
|
|
void AppendEscapedCharForHTMLImpl(typename str::value_type c, str* output) {
|
|
static const struct {
|
|
char key;
|
|
const char* replacement;
|
|
} kCharsToEscape[] = {
|
|
{ '<', "<" },
|
|
{ '>', ">" },
|
|
{ '&', "&" },
|
|
{ '"', """ },
|
|
{ '\'', "'" },
|
|
};
|
|
size_t k;
|
|
for (k = 0; k < arraysize(kCharsToEscape); ++k) {
|
|
if (c == kCharsToEscape[k].key) {
|
|
const char* p = kCharsToEscape[k].replacement;
|
|
while (*p)
|
|
output->push_back(*p++);
|
|
break;
|
|
}
|
|
}
|
|
if (k == arraysize(kCharsToEscape))
|
|
output->push_back(c);
|
|
}
|
|
|
|
template <class str>
|
|
str EscapeForHTMLImpl(base::BasicStringPiece<str> input) {
|
|
str result;
|
|
result.reserve(input.size()); // Optimize for no escaping.
|
|
|
|
for (auto c : input) {
|
|
AppendEscapedCharForHTMLImpl(c, &result);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
// Everything except alphanumerics and !'()*-._~
|
|
// See RFC 2396 for the list of reserved characters.
|
|
static const Charmap kQueryCharmap = {{
|
|
0xffffffffL, 0xfc00987dL, 0x78000001L, 0xb8000001L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
|
|
// non-printable, non-7bit, and (including space) "#%:<>?[\]^`{|}
|
|
static const Charmap kPathCharmap = {{
|
|
0xffffffffL, 0xd400002dL, 0x78000000L, 0xb8000001L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
|
|
#if defined(OS_MACOSX)
|
|
// non-printable, non-7bit, and (including space) "#%<>[\]^`{|}
|
|
static const Charmap kNSURLCharmap = {{
|
|
0xffffffffL, 0x5000002dL, 0x78000000L, 0xb8000001L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
#endif // defined(OS_MACOSX)
|
|
|
|
// non-printable, non-7bit, and (including space) ?>=<;+'&%$#"![\]^`{|}
|
|
static const Charmap kUrlEscape = {{
|
|
0xffffffffL, 0xf80008fdL, 0x78000001L, 0xb8000001L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
|
|
// non-7bit
|
|
static const Charmap kNonASCIICharmap = {{
|
|
0x00000000L, 0x00000000L, 0x00000000L, 0x00000000L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
|
|
// Everything except alphanumerics, the reserved characters(;/?:@&=+$,) and
|
|
// !'()*-._~#[]
|
|
static const Charmap kExternalHandlerCharmap = {{
|
|
0xffffffffL, 0x50000025L, 0x50000000L, 0xb8000001L,
|
|
0xffffffffL, 0xffffffffL, 0xffffffffL, 0xffffffffL
|
|
}};
|
|
|
|
} // namespace
|
|
|
|
std::string EscapeQueryParamValue(base::StringPiece text, bool use_plus) {
|
|
return Escape(text, kQueryCharmap, use_plus);
|
|
}
|
|
|
|
std::string EscapePath(base::StringPiece path) {
|
|
return Escape(path, kPathCharmap, false);
|
|
}
|
|
|
|
#if defined(OS_MACOSX)
|
|
std::string EscapeNSURLPrecursor(base::StringPiece precursor) {
|
|
return Escape(precursor, kNSURLCharmap, false, true);
|
|
}
|
|
#endif // defined(OS_MACOSX)
|
|
|
|
std::string EscapeUrlEncodedData(base::StringPiece path, bool use_plus) {
|
|
return Escape(path, kUrlEscape, use_plus);
|
|
}
|
|
|
|
std::string EscapeNonASCII(base::StringPiece input) {
|
|
return Escape(input, kNonASCIICharmap, false);
|
|
}
|
|
|
|
std::string EscapeExternalHandlerValue(base::StringPiece text) {
|
|
return Escape(text, kExternalHandlerCharmap, false, true);
|
|
}
|
|
|
|
void AppendEscapedCharForHTML(char c, std::string* output) {
|
|
AppendEscapedCharForHTMLImpl(c, output);
|
|
}
|
|
|
|
std::string EscapeForHTML(base::StringPiece input) {
|
|
return EscapeForHTMLImpl(input);
|
|
}
|
|
|
|
base::string16 EscapeForHTML(base::StringPiece16 input) {
|
|
return EscapeForHTMLImpl(input);
|
|
}
|
|
|
|
std::string UnescapeURLComponent(base::StringPiece escaped_text,
|
|
UnescapeRule::Type rules) {
|
|
return UnescapeURLWithAdjustmentsImpl(escaped_text, rules, NULL);
|
|
}
|
|
|
|
base::string16 UnescapeURLComponent(base::StringPiece16 escaped_text,
|
|
UnescapeRule::Type rules) {
|
|
return UnescapeURLWithAdjustmentsImpl(escaped_text, rules, NULL);
|
|
}
|
|
|
|
base::string16 UnescapeAndDecodeUTF8URLComponent(base::StringPiece text,
|
|
UnescapeRule::Type rules) {
|
|
return UnescapeAndDecodeUTF8URLComponentWithAdjustments(text, rules, NULL);
|
|
}
|
|
|
|
base::string16 UnescapeAndDecodeUTF8URLComponentWithAdjustments(
|
|
base::StringPiece text,
|
|
UnescapeRule::Type rules,
|
|
base::OffsetAdjuster::Adjustments* adjustments) {
|
|
base::string16 result;
|
|
base::OffsetAdjuster::Adjustments unescape_adjustments;
|
|
std::string unescaped_url(UnescapeURLWithAdjustmentsImpl(
|
|
text, rules, &unescape_adjustments));
|
|
if (base::UTF8ToUTF16WithAdjustments(unescaped_url.data(),
|
|
unescaped_url.length(),
|
|
&result, adjustments)) {
|
|
// Character set looks like it's valid.
|
|
if (adjustments) {
|
|
base::OffsetAdjuster::MergeSequentialAdjustments(unescape_adjustments,
|
|
adjustments);
|
|
}
|
|
return result;
|
|
}
|
|
// Character set is not valid. Return the escaped version.
|
|
return base::UTF8ToUTF16WithAdjustments(text, adjustments);
|
|
}
|
|
|
|
base::string16 UnescapeForHTML(base::StringPiece16 input) {
|
|
static const struct {
|
|
const char* ampersand_code;
|
|
const char replacement;
|
|
} kEscapeToChars[] = {
|
|
{ "<", '<' },
|
|
{ ">", '>' },
|
|
{ "&", '&' },
|
|
{ """, '"' },
|
|
{ "'", '\''},
|
|
};
|
|
|
|
if (input.find(base::ASCIIToUTF16("&")) == std::string::npos)
|
|
return input.as_string();
|
|
|
|
base::string16 ampersand_chars[arraysize(kEscapeToChars)];
|
|
base::string16 text = input.as_string();
|
|
for (base::string16::iterator iter = text.begin();
|
|
iter != text.end(); ++iter) {
|
|
if (*iter == '&') {
|
|
// Potential ampersand encode char.
|
|
size_t index = iter - text.begin();
|
|
for (size_t i = 0; i < arraysize(kEscapeToChars); i++) {
|
|
if (ampersand_chars[i].empty()) {
|
|
ampersand_chars[i] =
|
|
base::ASCIIToUTF16(kEscapeToChars[i].ampersand_code);
|
|
}
|
|
if (text.find(ampersand_chars[i], index) == index) {
|
|
text.replace(iter, iter + ampersand_chars[i].length(),
|
|
1, kEscapeToChars[i].replacement);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return text;
|
|
}
|
|
|
|
} // namespace net
|