| // Copyright 2013 The Chromium Authors | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/strings/string_util.h" | 
 |  | 
 | #include <errno.h> | 
 | #include <math.h> | 
 | #include <stdarg.h> | 
 | #include <stdint.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include <time.h> | 
 | #include <wchar.h> | 
 |  | 
 | #include <limits> | 
 | #include <type_traits> | 
 | #include <vector> | 
 |  | 
 | #include "polyfills/base/check_op.h" | 
 | #include "base/no_destructor.h" | 
 | #include "base/ranges/algorithm.h" | 
 | #include "base/strings/string_util_impl_helpers.h" | 
 | #include "base/strings/string_util_internal.h" | 
 | #include "base/strings/utf_string_conversion_utils.h" | 
 | #include "base/strings/utf_string_conversions.h" | 
 | #include "base/third_party/icu/icu_utf.h" | 
 | #include "build/build_config.h" | 
 | #include "absl/types/optional.h" | 
 |  | 
 | namespace gurl_base { | 
 |  | 
 | bool IsWprintfFormatPortable(const wchar_t* format) { | 
 |   for (const wchar_t* position = format; *position != '\0'; ++position) { | 
 |     if (*position == '%') { | 
 |       bool in_specification = true; | 
 |       bool modifier_l = false; | 
 |       while (in_specification) { | 
 |         // Eat up characters until reaching a known specifier. | 
 |         if (*++position == '\0') { | 
 |           // The format string ended in the middle of a specification.  Call | 
 |           // it portable because no unportable specifications were found.  The | 
 |           // string is equally broken on all platforms. | 
 |           return true; | 
 |         } | 
 |  | 
 |         if (*position == 'l') { | 
 |           // 'l' is the only thing that can save the 's' and 'c' specifiers. | 
 |           modifier_l = true; | 
 |         } else if (((*position == 's' || *position == 'c') && !modifier_l) || | 
 |                    *position == 'S' || *position == 'C' || *position == 'F' || | 
 |                    *position == 'D' || *position == 'O' || *position == 'U') { | 
 |           // Not portable. | 
 |           return false; | 
 |         } | 
 |  | 
 |         if (wcschr(L"diouxXeEfgGaAcspn%", *position)) { | 
 |           // Portable, keep scanning the rest of the format string. | 
 |           in_specification = false; | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | std::string ToLowerASCII(StringPiece str) { | 
 |   return internal::ToLowerASCIIImpl(str); | 
 | } | 
 |  | 
 | std::u16string ToLowerASCII(StringPiece16 str) { | 
 |   return internal::ToLowerASCIIImpl(str); | 
 | } | 
 |  | 
 | std::string ToUpperASCII(StringPiece str) { | 
 |   return internal::ToUpperASCIIImpl(str); | 
 | } | 
 |  | 
 | std::u16string ToUpperASCII(StringPiece16 str) { | 
 |   return internal::ToUpperASCIIImpl(str); | 
 | } | 
 |  | 
 | const std::string& EmptyString() { | 
 |   static const gurl_base::NoDestructor<std::string> s; | 
 |   return *s; | 
 | } | 
 |  | 
 | const std::u16string& EmptyString16() { | 
 |   static const gurl_base::NoDestructor<std::u16string> s16; | 
 |   return *s16; | 
 | } | 
 |  | 
 | bool ReplaceChars(StringPiece16 input, | 
 |                   StringPiece16 replace_chars, | 
 |                   StringPiece16 replace_with, | 
 |                   std::u16string* output) { | 
 |   return internal::ReplaceCharsT(input, replace_chars, replace_with, output); | 
 | } | 
 |  | 
 | bool ReplaceChars(StringPiece input, | 
 |                   StringPiece replace_chars, | 
 |                   StringPiece replace_with, | 
 |                   std::string* output) { | 
 |   return internal::ReplaceCharsT(input, replace_chars, replace_with, output); | 
 | } | 
 |  | 
 | bool RemoveChars(StringPiece16 input, | 
 |                  StringPiece16 remove_chars, | 
 |                  std::u16string* output) { | 
 |   return internal::ReplaceCharsT(input, remove_chars, StringPiece16(), output); | 
 | } | 
 |  | 
 | bool RemoveChars(StringPiece input, | 
 |                  StringPiece remove_chars, | 
 |                  std::string* output) { | 
 |   return internal::ReplaceCharsT(input, remove_chars, StringPiece(), output); | 
 | } | 
 |  | 
 | bool TrimString(StringPiece16 input, | 
 |                 StringPiece16 trim_chars, | 
 |                 std::u16string* output) { | 
 |   return internal::TrimStringT(input, trim_chars, TRIM_ALL, output) != | 
 |          TRIM_NONE; | 
 | } | 
 |  | 
 | bool TrimString(StringPiece input, | 
 |                 StringPiece trim_chars, | 
 |                 std::string* output) { | 
 |   return internal::TrimStringT(input, trim_chars, TRIM_ALL, output) != | 
 |          TRIM_NONE; | 
 | } | 
 |  | 
 | StringPiece16 TrimString(StringPiece16 input, | 
 |                          StringPiece16 trim_chars, | 
 |                          TrimPositions positions) { | 
 |   return internal::TrimStringPieceT(input, trim_chars, positions); | 
 | } | 
 |  | 
 | StringPiece TrimString(StringPiece input, | 
 |                        StringPiece trim_chars, | 
 |                        TrimPositions positions) { | 
 |   return internal::TrimStringPieceT(input, trim_chars, positions); | 
 | } | 
 |  | 
 | void TruncateUTF8ToByteSize(const std::string& input, | 
 |                             const size_t byte_size, | 
 |                             std::string* output) { | 
 |   GURL_DCHECK(output); | 
 |   if (byte_size > input.length()) { | 
 |     *output = input; | 
 |     return; | 
 |   } | 
 |   GURL_DCHECK_LE(byte_size, | 
 |             static_cast<uint32_t>(std::numeric_limits<int32_t>::max())); | 
 |   // Note: This cast is necessary because CBU8_NEXT uses int32_ts. | 
 |   int32_t truncation_length = static_cast<int32_t>(byte_size); | 
 |   int32_t char_index = truncation_length - 1; | 
 |   const char* data = input.data(); | 
 |  | 
 |   // Using CBU8, we will move backwards from the truncation point | 
 |   // to the beginning of the string looking for a valid UTF8 | 
 |   // character.  Once a full UTF8 character is found, we will | 
 |   // truncate the string to the end of that character. | 
 |   while (char_index >= 0) { | 
 |     int32_t prev = char_index; | 
 |     base_icu::UChar32 code_point = 0; | 
 |     CBU8_NEXT(reinterpret_cast<const uint8_t*>(data), char_index, | 
 |               truncation_length, code_point); | 
 |     if (!IsValidCharacter(code_point)) { | 
 |       char_index = prev - 1; | 
 |     } else { | 
 |       break; | 
 |     } | 
 |   } | 
 |  | 
 |   if (char_index >= 0 ) | 
 |     *output = input.substr(0, static_cast<size_t>(char_index)); | 
 |   else | 
 |     output->clear(); | 
 | } | 
 |  | 
 | TrimPositions TrimWhitespace(StringPiece16 input, | 
 |                              TrimPositions positions, | 
 |                              std::u16string* output) { | 
 |   return internal::TrimStringT(input, StringPiece16(kWhitespaceUTF16), | 
 |                                positions, output); | 
 | } | 
 |  | 
 | StringPiece16 TrimWhitespace(StringPiece16 input, | 
 |                              TrimPositions positions) { | 
 |   return internal::TrimStringPieceT(input, StringPiece16(kWhitespaceUTF16), | 
 |                                     positions); | 
 | } | 
 |  | 
 | TrimPositions TrimWhitespaceASCII(StringPiece input, | 
 |                                   TrimPositions positions, | 
 |                                   std::string* output) { | 
 |   return internal::TrimStringT(input, StringPiece(kWhitespaceASCII), positions, | 
 |                                output); | 
 | } | 
 |  | 
 | StringPiece TrimWhitespaceASCII(StringPiece input, TrimPositions positions) { | 
 |   return internal::TrimStringPieceT(input, StringPiece(kWhitespaceASCII), | 
 |                                     positions); | 
 | } | 
 |  | 
 | std::u16string CollapseWhitespace(StringPiece16 text, | 
 |                                   bool trim_sequences_with_line_breaks) { | 
 |   return internal::CollapseWhitespaceT(text, trim_sequences_with_line_breaks); | 
 | } | 
 |  | 
 | std::string CollapseWhitespaceASCII(StringPiece text, | 
 |                                     bool trim_sequences_with_line_breaks) { | 
 |   return internal::CollapseWhitespaceT(text, trim_sequences_with_line_breaks); | 
 | } | 
 |  | 
 | bool ContainsOnlyChars(StringPiece input, StringPiece characters) { | 
 |   return input.find_first_not_of(characters) == StringPiece::npos; | 
 | } | 
 |  | 
 | bool ContainsOnlyChars(StringPiece16 input, StringPiece16 characters) { | 
 |   return input.find_first_not_of(characters) == StringPiece16::npos; | 
 | } | 
 |  | 
 |  | 
 | bool IsStringASCII(StringPiece str) { | 
 |   return internal::DoIsStringASCII(str.data(), str.length()); | 
 | } | 
 |  | 
 | bool IsStringASCII(StringPiece16 str) { | 
 |   return internal::DoIsStringASCII(str.data(), str.length()); | 
 | } | 
 |  | 
 | #if defined(WCHAR_T_IS_UTF32) | 
 | bool IsStringASCII(WStringPiece str) { | 
 |   return internal::DoIsStringASCII(str.data(), str.length()); | 
 | } | 
 | #endif | 
 |  | 
 | bool IsStringUTF8(StringPiece str) { | 
 |   return internal::DoIsStringUTF8<IsValidCharacter>(str); | 
 | } | 
 |  | 
 | bool IsStringUTF8AllowingNoncharacters(StringPiece str) { | 
 |   return internal::DoIsStringUTF8<IsValidCodepoint>(str); | 
 | } | 
 |  | 
 | bool EqualsASCII(StringPiece16 str, StringPiece ascii) { | 
 |   return ranges::equal(ascii, str); | 
 | } | 
 |  | 
 | bool StartsWith(StringPiece str, | 
 |                 StringPiece search_for, | 
 |                 CompareCase case_sensitivity) { | 
 |   return internal::StartsWithT(str, search_for, case_sensitivity); | 
 | } | 
 |  | 
 | bool StartsWith(StringPiece16 str, | 
 |                 StringPiece16 search_for, | 
 |                 CompareCase case_sensitivity) { | 
 |   return internal::StartsWithT(str, search_for, case_sensitivity); | 
 | } | 
 |  | 
 | bool EndsWith(StringPiece str, | 
 |               StringPiece search_for, | 
 |               CompareCase case_sensitivity) { | 
 |   return internal::EndsWithT(str, search_for, case_sensitivity); | 
 | } | 
 |  | 
 | bool EndsWith(StringPiece16 str, | 
 |               StringPiece16 search_for, | 
 |               CompareCase case_sensitivity) { | 
 |   return internal::EndsWithT(str, search_for, case_sensitivity); | 
 | } | 
 |  | 
 | char HexDigitToInt(char c) { | 
 |   GURL_DCHECK(IsHexDigit(c)); | 
 |   if (c >= '0' && c <= '9') | 
 |     return static_cast<char>(c - '0'); | 
 |   return (c >= 'A' && c <= 'F') ? static_cast<char>(c - 'A' + 10) | 
 |                                 : static_cast<char>(c - 'a' + 10); | 
 | } | 
 |  | 
 | static const char* const kByteStringsUnlocalized[] = { | 
 |   " B", | 
 |   " kB", | 
 |   " MB", | 
 |   " GB", | 
 |   " TB", | 
 |   " PB" | 
 | }; | 
 |  | 
 | std::u16string FormatBytesUnlocalized(int64_t bytes) { | 
 |   double unit_amount = static_cast<double>(bytes); | 
 |   size_t dimension = 0; | 
 |   const int kKilo = 1024; | 
 |   while (unit_amount >= kKilo && | 
 |          dimension < std::size(kByteStringsUnlocalized) - 1) { | 
 |     unit_amount /= kKilo; | 
 |     dimension++; | 
 |   } | 
 |  | 
 |   char buf[64]; | 
 |   if (bytes != 0 && dimension > 0 && unit_amount < 100) { | 
 |     gurl_base::snprintf(buf, std::size(buf), "%.1lf%s", unit_amount, | 
 |                    kByteStringsUnlocalized[dimension]); | 
 |   } else { | 
 |     gurl_base::snprintf(buf, std::size(buf), "%.0lf%s", unit_amount, | 
 |                    kByteStringsUnlocalized[dimension]); | 
 |   } | 
 |  | 
 |   return ASCIIToUTF16(buf); | 
 | } | 
 |  | 
 | void ReplaceFirstSubstringAfterOffset(std::u16string* str, | 
 |                                       size_t start_offset, | 
 |                                       StringPiece16 find_this, | 
 |                                       StringPiece16 replace_with) { | 
 |   internal::DoReplaceMatchesAfterOffset( | 
 |       str, start_offset, internal::MakeSubstringMatcher(find_this), | 
 |       replace_with, internal::ReplaceType::REPLACE_FIRST); | 
 | } | 
 |  | 
 | void ReplaceFirstSubstringAfterOffset(std::string* str, | 
 |                                       size_t start_offset, | 
 |                                       StringPiece find_this, | 
 |                                       StringPiece replace_with) { | 
 |   internal::DoReplaceMatchesAfterOffset( | 
 |       str, start_offset, internal::MakeSubstringMatcher(find_this), | 
 |       replace_with, internal::ReplaceType::REPLACE_FIRST); | 
 | } | 
 |  | 
 | void ReplaceSubstringsAfterOffset(std::u16string* str, | 
 |                                   size_t start_offset, | 
 |                                   StringPiece16 find_this, | 
 |                                   StringPiece16 replace_with) { | 
 |   internal::DoReplaceMatchesAfterOffset( | 
 |       str, start_offset, internal::MakeSubstringMatcher(find_this), | 
 |       replace_with, internal::ReplaceType::REPLACE_ALL); | 
 | } | 
 |  | 
 | void ReplaceSubstringsAfterOffset(std::string* str, | 
 |                                   size_t start_offset, | 
 |                                   StringPiece find_this, | 
 |                                   StringPiece replace_with) { | 
 |   internal::DoReplaceMatchesAfterOffset( | 
 |       str, start_offset, internal::MakeSubstringMatcher(find_this), | 
 |       replace_with, internal::ReplaceType::REPLACE_ALL); | 
 | } | 
 |  | 
 | char* WriteInto(std::string* str, size_t length_with_null) { | 
 |   return internal::WriteIntoT(str, length_with_null); | 
 | } | 
 |  | 
 | char16_t* WriteInto(std::u16string* str, size_t length_with_null) { | 
 |   return internal::WriteIntoT(str, length_with_null); | 
 | } | 
 |  | 
 | std::string JoinString(span<const std::string> parts, StringPiece separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::u16string JoinString(span<const std::u16string> parts, | 
 |                           StringPiece16 separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::string JoinString(span<const StringPiece> parts, StringPiece separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::u16string JoinString(span<const StringPiece16> parts, | 
 |                           StringPiece16 separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::string JoinString(std::initializer_list<StringPiece> parts, | 
 |                        StringPiece separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::u16string JoinString(std::initializer_list<StringPiece16> parts, | 
 |                           StringPiece16 separator) { | 
 |   return internal::JoinStringT(parts, separator); | 
 | } | 
 |  | 
 | std::u16string ReplaceStringPlaceholders( | 
 |     StringPiece16 format_string, | 
 |     const std::vector<std::u16string>& subst, | 
 |     std::vector<size_t>* offsets) { | 
 |   absl::optional<std::u16string> replacement = | 
 |       internal::DoReplaceStringPlaceholders( | 
 |           format_string, subst, | 
 |           /*placeholder_prefix*/ u'$', | 
 |           /*should_escape_multiple_placeholder_prefixes*/ true, | 
 |           /*is_strict_mode*/ false, offsets); | 
 |  | 
 |   GURL_DCHECK(replacement); | 
 |   return replacement.value(); | 
 | } | 
 |  | 
 | std::string ReplaceStringPlaceholders(StringPiece format_string, | 
 |                                       const std::vector<std::string>& subst, | 
 |                                       std::vector<size_t>* offsets) { | 
 |   absl::optional<std::string> replacement = | 
 |       internal::DoReplaceStringPlaceholders( | 
 |           format_string, subst, | 
 |           /*placeholder_prefix*/ '$', | 
 |           /*should_escape_multiple_placeholder_prefixes*/ true, | 
 |           /*is_strict_mode*/ false, offsets); | 
 |  | 
 |   GURL_DCHECK(replacement); | 
 |   return replacement.value(); | 
 | } | 
 |  | 
 | std::u16string ReplaceStringPlaceholders(const std::u16string& format_string, | 
 |                                          const std::u16string& a, | 
 |                                          size_t* offset) { | 
 |   std::vector<size_t> offsets; | 
 |   std::u16string result = | 
 |       ReplaceStringPlaceholders(format_string, {a}, &offsets); | 
 |  | 
 |   GURL_DCHECK_EQ(1U, offsets.size()); | 
 |   if (offset) | 
 |     *offset = offsets[0]; | 
 |   return result; | 
 | } | 
 |  | 
 | size_t strlcpy(char* dst, const char* src, size_t dst_size) { | 
 |   return internal::lcpyT(dst, src, dst_size); | 
 | } | 
 |  | 
 | size_t u16cstrlcpy(char16_t* dst, const char16_t* src, size_t dst_size) { | 
 |   return internal::lcpyT(dst, src, dst_size); | 
 | } | 
 |  | 
 | size_t wcslcpy(wchar_t* dst, const wchar_t* src, size_t dst_size) { | 
 |   return internal::lcpyT(dst, src, dst_size); | 
 | } | 
 |  | 
 | }  // namespace base |