9#ifndef BITCOIN_UTIL_STRENCODINGS_H
10#define BITCOIN_UTIL_STRENCODINGS_H
25#include <system_error>
51 t = 1'000'000'000'000ULL,
64template <
typename Byte = std::
byte>
65std::optional<std::vector<Byte>>
TryParseHex(std::string_view str);
67template <
typename Byte = u
int8_t>
68std::vector<Byte>
ParseHex(std::string_view hex_str)
70 return TryParseHex<Byte>(hex_str).value_or(std::vector<Byte>{});
74bool IsHex(std::string_view str);
75std::optional<std::vector<unsigned char>>
DecodeBase64(std::string_view str);
79std::optional<std::vector<unsigned char>>
DecodeBase32(std::string_view str);
93std::string
EncodeBase32(std::string_view str,
bool pad =
true);
104bool SplitHostPort(std::string_view in, uint16_t& portOut, std::string& hostOut);
120 static_assert(std::is_integral<T>::value);
124 if (!
s.empty() &&
s[0] ==
'+') {
125 if (
s.length() >= 2 &&
s[1] ==
'-') {
130 auto [
_, error_condition] = std::from_chars(
s.data(),
s.data() +
s.size(), result);
131 if (error_condition == std::errc::result_out_of_range) {
132 if (
s.length() >= 1 &&
s[0] ==
'-') {
134 return std::numeric_limits<T>::min();
137 return std::numeric_limits<T>::max();
139 }
else if (error_condition != std::errc{}) {
152 return c >=
'0' && c <=
'9';
166constexpr inline bool IsSpace(
char c)
noexcept {
167 return c ==
' ' || c ==
'\f' || c ==
'\n' || c ==
'\r' || c ==
'\t' || c ==
'\v';
181 static_assert(std::is_integral<T>::value);
183 const auto [first_nonmatching, error_condition] = std::from_chars(str.data(), str.data() + str.size(), result);
184 if (first_nonmatching != str.data() + str.size() || error_condition != std::errc{}) {
195[[nodiscard]]
bool ParseInt32(std::string_view str, int32_t *
out);
202[[nodiscard]]
bool ParseInt64(std::string_view str, int64_t *
out);
209[[nodiscard]]
bool ParseUInt8(std::string_view str, uint8_t *
out);
216[[nodiscard]]
bool ParseUInt16(std::string_view str, uint16_t*
out);
223[[nodiscard]]
bool ParseUInt32(std::string_view str, uint32_t *
out);
230[[nodiscard]]
bool ParseUInt64(std::string_view str, uint64_t *
out);
236std::string
FormatParagraph(std::string_view in,
size_t width = 79,
size_t indent = 0);
246 if (b.size() == 0)
return a.size() == 0;
247 size_t accumulator = a.size() ^ b.size();
248 for (
size_t i = 0; i < a.size(); i++)
249 accumulator |=
size_t(a[i] ^ b[i%b.size()]);
250 return accumulator == 0;
257[[nodiscard]]
bool ParseFixedPoint(std::string_view,
int decimals, int64_t *amount_out);
263 [[maybe_unused]]
int operator()(
int x)
const {
return x; }
269template<
int frombits,
int tobits,
bool pad,
typename O,
typename It,
typename I = IntIdentity>
273 constexpr size_t maxv = (1 << tobits) - 1;
274 constexpr size_t max_acc = (1 << (frombits + tobits - 1)) - 1;
277 if (v < 0)
return false;
278 acc = ((acc << frombits) | v) & max_acc;
280 while (bits >= tobits) {
282 outfn((acc >> bits) & maxv);
287 if (bits) outfn((acc << (tobits - bits)) & maxv);
288 }
else if (bits >= frombits || ((acc << (tobits - bits)) & maxv)) {
306 return (c >=
'A' && c <=
'Z' ? (c -
'A') +
'a' : c);
318std::string
ToLower(std::string_view str);
332 return (c >=
'a' && c <=
'z' ? (c -
'a') +
'A' : c);
344std::string
ToUpper(std::string_view str);
374 if (c >=
'0' && c <=
'9')
return c -
'0';
375 if (c >=
'a' && c <=
'f')
return c -
'a' + 0xa;
377 throw "Only lowercase hex digits are allowed, for consistency";
383 std::array<std::byte, N / 2>
bytes{};
384 consteval Hex(
const char (&hex_str)[N])
388 if (hex_str[N - 1])
throw "null terminator required";
389 for (std::size_t i = 0; i <
bytes.size(); ++i) {
390 bytes[i] =
static_cast<std::byte
>(
427inline namespace hex_literals {
429template <util::detail::Hex str>
430constexpr auto operator""_hex() {
return str.bytes; }
432template <util::detail::Hex str>
433constexpr auto operator""_hex_u8() {
return std::bit_cast<std::array<uint8_t, str.bytes.size()>>(str.bytes); }
435template <util::detail::Hex str>
436constexpr auto operator""_hex_v() {
return std::vector<std::byte>{str.bytes.begin(), str.bytes.end()}; }
438template <util::detail::Hex str>
439inline auto operator""_hex_v_u8() {
return std::vector<uint8_t>{
UCharCast(str.bytes.data()),
UCharCast(str.bytes.data() + str.bytes.size())}; }
#define T(expected, seed, data)
consteval uint8_t ConstevalHexDigit(const char c)
consteval version of HexDigit() without the lookup table.
std::string_view TrimStringView(std::string_view str, std::string_view pattern=" \f\n\r\t\v")
constexpr auto MakeUCharSpan(V &&v) -> decltype(UCharSpanCast(Span{std::forward< V >(v)}))
Like the Span constructor, but for (const) unsigned char member types only.
unsigned char * UCharCast(char *c)
std::string EncodeBase32(Span< const unsigned char > input, bool pad=true)
Base32 encode.
std::string Capitalize(std::string str)
Capitalizes the first character of the given string.
bool ParseInt32(std::string_view str, int32_t *out)
Convert string to signed 32-bit integer with strict parse error feedback.
bool ParseUInt16(std::string_view str, uint16_t *out)
Convert decimal string to unsigned 16-bit integer with strict parse error feedback.
constexpr char ToLower(char c)
Converts the given character to its lowercase equivalent.
constexpr bool IsDigit(char c)
Tests if the given character is a decimal digit.
std::string EncodeBase64(Span< const unsigned char > input)
constexpr char ToUpper(char c)
Converts the given character to its uppercase equivalent.
std::optional< std::vector< unsigned char > > DecodeBase32(std::string_view str)
T LocaleIndependentAtoi(std::string_view str)
bool ParseInt64(std::string_view str, int64_t *out)
Convert string to signed 64-bit integer with strict parse error feedback.
ByteUnit
Used by ParseByteUnits() Lowercase base 1000 Uppercase base 1024.
bool TimingResistantEqual(const T &a, const T &b)
Timing-attack-resistant comparison.
bool ParseUInt8(std::string_view str, uint8_t *out)
Convert decimal string to unsigned 8-bit integer with strict parse error feedback.
std::vector< Byte > ParseHex(std::string_view hex_str)
Like TryParseHex, but returns an empty vector on invalid input.
bool ParseFixedPoint(std::string_view, int decimals, int64_t *amount_out)
Parse number as fixed point according to JSON number syntax.
bool ParseUInt64(std::string_view str, uint64_t *out)
Convert decimal string to unsigned 64-bit integer with strict parse error feedback.
constexpr bool IsSpace(char c) noexcept
Tests if the given character is a whitespace character.
bool IsHex(std::string_view str)
std::optional< std::vector< unsigned char > > DecodeBase64(std::string_view str)
std::optional< T > ToIntegral(std::string_view str)
Convert string to integral type T.
bool SplitHostPort(std::string_view in, uint16_t &portOut, std::string &hostOut)
Splits socket address string into host string and port value.
bool ParseUInt32(std::string_view str, uint32_t *out)
Convert decimal string to unsigned 32-bit integer with strict parse error feedback.
bool ConvertBits(O outfn, It it, It end, I infn={})
Convert from one power-of-2 number base to another.
std::string FormatParagraph(std::string_view in, size_t width=79, size_t indent=0)
Format a paragraph of text to a fixed width, adding spaces for indentation to any added line.
std::string SanitizeString(std::string_view str, int rule=SAFE_CHARS_DEFAULT)
Remove unsafe chars.
std::optional< std::vector< Byte > > TryParseHex(std::string_view str)
Parse the hex string into bytes (uint8_t or std::byte).
std::optional< uint64_t > ParseByteUnits(std::string_view str, ByteUnit default_multiplier)
Parse a string with suffix unit [k|K|m|M|g|G|t|T].
SafeChars
Utilities for converting data from/to strings.
@ SAFE_CHARS_DEFAULT
The full set of allowed chars.
@ SAFE_CHARS_UA_COMMENT
BIP-0014 subset.
@ SAFE_CHARS_URI
Chars allowed in URIs (RFC 3986)
@ SAFE_CHARS_FILENAME
Chars allowed in filenames.
consteval Hex(const char(&hex_str)[N])
std::array< std::byte, N/2 > bytes
consteval auto _(util::TranslatedLiteral str)