// Copyright (c) 2019-2022 The Bitcoin Core developers // Distributed under the MIT software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #ifndef BITCOIN_UTIL_STRING_H #define BITCOIN_UTIL_STRING_H #include #include #include #include #include #include #include // IWYU pragma: export #include // IWYU pragma: export #include namespace util { void ReplaceAll(std::string& in_out, const std::string& search, const std::string& substitute); /** Split a string on any char found in separators, returning a vector. * * If sep does not occur in sp, a singleton with the entirety of sp is returned. * * Note that this function does not care about braces, so splitting * "foo(bar(1),2),3) on ',' will return {"foo(bar(1)", "2)", "3)"}. */ template > std::vector Split(const Span& sp, std::string_view separators) { std::vector ret; auto it = sp.begin(); auto start = it; while (it != sp.end()) { if (separators.find(*it) != std::string::npos) { ret.emplace_back(start, it); start = it + 1; } ++it; } ret.emplace_back(start, it); return ret; } /** Split a string on every instance of sep, returning a vector. * * If sep does not occur in sp, a singleton with the entirety of sp is returned. * * Note that this function does not care about braces, so splitting * "foo(bar(1),2),3) on ',' will return {"foo(bar(1)", "2)", "3)"}. */ template > std::vector Split(const Span& sp, char sep) { return Split(sp, std::string_view{&sep, 1}); } [[nodiscard]] inline std::vector SplitString(std::string_view str, char sep) { return Split(str, sep); } [[nodiscard]] inline std::vector SplitString(std::string_view str, std::string_view separators) { return Split(str, separators); } [[nodiscard]] inline std::string_view TrimStringView(std::string_view str, std::string_view pattern = " \f\n\r\t\v") { std::string::size_type front = str.find_first_not_of(pattern); if (front == std::string::npos) { return {}; } std::string::size_type end = str.find_last_not_of(pattern); return str.substr(front, end - front + 1); } [[nodiscard]] inline std::string TrimString(std::string_view str, std::string_view pattern = " \f\n\r\t\v") { return std::string(TrimStringView(str, pattern)); } [[nodiscard]] inline std::string_view RemoveSuffixView(std::string_view str, std::string_view suffix) { if (str.ends_with(suffix)) { return str.substr(0, str.size() - suffix.size()); } return str; } [[nodiscard]] inline std::string_view RemovePrefixView(std::string_view str, std::string_view prefix) { if (str.substr(0, prefix.size()) == prefix) { return str.substr(prefix.size()); } return str; } [[nodiscard]] inline std::string RemovePrefix(std::string_view str, std::string_view prefix) { return std::string(RemovePrefixView(str, prefix)); } /** * Join all container items. Typically used to concatenate strings but accepts * containers with elements of any type. * * @param container The items to join * @param separator The separator * @param unary_op Apply this operator to each item */ template // NOLINTNEXTLINE(misc-no-recursion) auto Join(const C& container, const S& separator, UnaryOp unary_op) { decltype(unary_op(*container.begin())) ret; bool first{true}; for (const auto& item : container) { if (!first) ret += separator; ret += unary_op(item); first = false; } return ret; } template auto Join(const C& container, const S& separator) { return Join(container, separator, [](const auto& i) { return i; }); } /** * Create an unordered multi-line list of items. */ inline std::string MakeUnorderedList(const std::vector& items) { return Join(items, "\n", [](const std::string& item) { return "- " + item; }); } /** * Check if a string does not contain any embedded NUL (\0) characters */ [[nodiscard]] inline bool ContainsNoNUL(std::string_view str) noexcept { for (auto c : str) { if (c == 0) return false; } return true; } /** * Locale-independent version of std::to_string */ template std::string ToString(const T& t) { std::ostringstream oss; oss.imbue(std::locale::classic()); oss << t; return oss.str(); } /** * Check whether a container begins with the given prefix. */ template [[nodiscard]] inline bool HasPrefix(const T1& obj, const std::array& prefix) { return obj.size() >= PREFIX_LEN && std::equal(std::begin(prefix), std::end(prefix), std::begin(obj)); } } // namespace util #endif // BITCOIN_UTIL_STRING_H