19#include <userver/utils/meta_light.hpp>
21USERVER_NAMESPACE_BEGIN
27[[noreturn]]
void ThrowFromStringException(std::string_view message,
28 std::string_view input,
29 std::type_index resultType);
32std::enable_if_t<std::is_floating_point_v<T>, T> FromString(
const char* str) {
33 static_assert(!std::is_const_v<T> && !std::is_volatile_v<T>);
34 static_assert(!std::is_reference_v<T>);
37 impl::ThrowFromStringException(
"nullptr string",
"<null>",
typeid(T));
40 impl::ThrowFromStringException(
"empty string", str,
typeid(T));
42 if (std::isspace(str[0])) {
43 impl::ThrowFromStringException(
"leading spaces are not allowed", str,
50 const auto result = [&] {
51 if constexpr (std::is_same_v<T,
float>) {
52 return std::strtof(str, &end);
53 }
else if constexpr (std::is_same_v<T,
double>) {
54 return std::strtod(str, &end);
55 }
else if constexpr (std::is_same_v<T,
long double>) {
56 return std::strtold(str, &end);
60 if (errno == ERANGE) {
61 impl::ThrowFromStringException(
"overflow", str,
typeid(T));
64 impl::ThrowFromStringException(
"no number found", str,
typeid(T));
68 if (std::isspace(*end)) {
69 impl::ThrowFromStringException(
"trailing spaces are not allowed", str,
72 impl::ThrowFromStringException(
73 "extra junk at the end of the string is not allowed", str,
typeid(T));
81std::enable_if_t<std::is_floating_point_v<T>, T> FromString(
82 const std::string& str) {
83 return FromString<T>(str.data());
87std::enable_if_t<std::is_floating_point_v<T>, T> FromString(
88 std::string_view str) {
89 static constexpr std::size_t kSmallBufferSize = 32;
91 if (str.size() >= kSmallBufferSize) {
92 return FromString<T>(std::string{str});
95 char buffer[kSmallBufferSize];
96 std::copy(str.data(), str.data() + str.size(), buffer);
97 buffer[str.size()] =
'\0';
99 return FromString<T>(buffer);
103std::enable_if_t<meta::kIsInteger<T>, T> FromString(std::string_view str) {
104 static_assert(!std::is_const_v<T> && !std::is_volatile_v<T>);
105 static_assert(!std::is_reference_v<T>);
108 impl::ThrowFromStringException(
"empty string", str,
typeid(T));
110 if (std::isspace(str[0])) {
111 impl::ThrowFromStringException(
"leading spaces are not allowed", str,
115 std::size_t offset = 0;
118 if (str.size() > 1 && str[0] ==
'+' && str[1] ==
'-') {
119 impl::ThrowFromStringException(
"no number found", str,
typeid(T));
121 if (str[0] ==
'+') offset = 1;
124 if (std::is_unsigned_v<T> && str[0] ==
'-') offset = 1;
127 const auto [end, error_code] =
128 std::from_chars(str.data() + offset, str.data() + str.size(), result);
130 if (error_code == std::errc::result_out_of_range) {
131 impl::ThrowFromStringException(
"overflow", str,
typeid(T));
133 if (error_code == std::errc::invalid_argument) {
134 impl::ThrowFromStringException(
"no number found", str,
typeid(T));
137 if (std::is_unsigned_v<T> && str[0] ==
'-' && result != 0) {
138 impl::ThrowFromStringException(
"overflow", str,
typeid(T));
141 if (end != str.data() + str.size()) {
142 if (std::isspace(*end)) {
143 impl::ThrowFromStringException(
"trailing spaces are not allowed", str,
146 impl::ThrowFromStringException(
147 "extra junk at the end of the string is not allowed", str,
typeid(T));
170template <
typename T,
typename StringType,
174 return impl::FromString<T>(str);
177std::int64_t FromHexString(
const std::string& str);