summaryrefslogtreecommitdiff
path: root/include/llmr/style/value.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'include/llmr/style/value.hpp')
-rw-r--r--include/llmr/style/value.hpp130
1 files changed, 13 insertions, 117 deletions
diff --git a/include/llmr/style/value.hpp b/include/llmr/style/value.hpp
index fac316d3d5..ed14c233e1 100644
--- a/include/llmr/style/value.hpp
+++ b/include/llmr/style/value.hpp
@@ -4,6 +4,9 @@
#include <llmr/util/variant.hpp>
#include <llmr/util/pbf.hpp>
+#include <cstdlib>
+#include <cerrno>
+
namespace llmr {
typedef util::variant<bool, int64_t, uint64_t, double, std::string> Value;
@@ -13,128 +16,21 @@ std::string toString(const Value &value);
Value parseValue(pbf data);
namespace util {
-
-namespace detail {
-
-inline int string_to_bool(std::string str) {
- std::transform(str.begin(), str.end(), str.begin(), ::tolower);
- if (str == "0" || str == "0.0" || str == "false" || str == "null" || !str.length()) return 0;
- else if (str == "true") return 1;
- else return -1;
+inline bool parseNumericString(const std::string &str, double &result) {
+ char *end = nullptr;
+ const char *begin = str.c_str();
+ result = std::strtod(begin, &end);
+ while (*end != '\0' && isspace(*end)) end++; // eat whitespace after the end
+ return errno == 0 && end - begin == long(str.size());
}
-
-template <typename T>
-struct string_to_number {};
-
-template <> struct string_to_number<bool> {
- bool operator()(const std::string &str) const {
- // Converts 0 => false, 1 => true and -1 => true.
- return string_to_bool(str);
- }
-};
-
-template <> struct string_to_number<double> {
- double operator()(std::string const &str) const {
- int val = string_to_bool(str);
- return val < 0 ? std::stod(str) : val;
- }
-};
-
-template <> struct string_to_number<int64_t> {
- int64_t operator()(std::string const &str) const {
- int val = string_to_bool(str);
- return val < 0 ? std::stoll(str) : val;
- }
-};
-
-template <> struct string_to_number<uint64_t> {
- uint64_t operator()(std::string const &str) const {
- int val = string_to_bool(str);
- return val < 0 ? std::stoull(str) : val;
- }
-};
-
-template <typename Operator>
-struct relaxed_operator_visitor {
- typedef bool result_type;
-
- bool operator()(bool lhs, bool rhs) const { return Operator()(lhs, rhs); }
-
- template <typename T, class = typename std::enable_if<std::is_arithmetic<T>::value>::type>
- bool operator()(bool lhs, T rhs) const { return Operator()(lhs, bool(rhs)); }
-
- template <typename T, class = typename std::enable_if<std::is_arithmetic<T>::value>::type>
- bool operator()(T lhs, bool rhs) const { return Operator()(bool(lhs), rhs); }
-
- bool operator()(int64_t lhs, uint64_t rhs) const { return lhs < 0 ? false : Operator()(uint64_t(lhs), rhs); }
- bool operator()(uint64_t lhs, int64_t rhs) const { return rhs < 0 ? false : Operator()(lhs, uint64_t(rhs)); }
-
- template <typename T, class = typename std::enable_if<std::is_arithmetic<T>::value>::type>
- bool operator()(const std::string &lhs, T rhs) const try { return Operator()(string_to_number<T>()(lhs), rhs); }
- catch(...) { return false; }
-
- template <typename T, class = typename std::enable_if<std::is_arithmetic<T>::value>::type>
- bool operator()(T lhs, const std::string &rhs) const try { return Operator()(lhs, string_to_number<T>()(rhs)); }
- catch(...) { return false; }
-
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return Operator()(lhs, rhs); }
-};
-
-struct relaxed_equal_operator {
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return lhs == rhs; }
-};
-
-struct relaxed_greater_operator {
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return lhs > rhs; }
-};
-
-struct relaxed_greater_equal_operator {
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return lhs >= rhs; }
-};
-
-struct relaxed_less_operator {
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return lhs < rhs; }
-};
-
-struct relaxed_less_equal_operator {
- template <typename T0, typename T1>
- bool operator()(T0 lhs, T1 rhs) const { return lhs <= rhs; }
-};
-
-
-} // end namespace detail
-
-inline bool relaxed_equal(Value const &lhs, Value const &rhs) {
- return apply_visitor(detail::relaxed_operator_visitor<detail::relaxed_equal_operator>(), lhs, rhs);
-}
-
-inline bool relaxed_greater(Value const &lhs, Value const &rhs) {
- return apply_visitor(detail::relaxed_operator_visitor<detail::relaxed_greater_operator>(), lhs, rhs);
}
-inline bool relaxed_greater_equal(Value const &lhs, Value const &rhs) {
- return apply_visitor(detail::relaxed_operator_visitor<detail::relaxed_greater_equal_operator>(), lhs, rhs);
-}
-
-inline bool relaxed_less(Value const &lhs, Value const &rhs) {
- return apply_visitor(detail::relaxed_operator_visitor<detail::relaxed_less_operator>(), lhs, rhs);
-}
-
-inline bool relaxed_less_equal(Value const &lhs, Value const &rhs) {
- return apply_visitor(detail::relaxed_operator_visitor<detail::relaxed_less_equal_operator>(), lhs, rhs);
-}
-
-}
-
-
template <typename T>
T toNumber(const Value &value) {
- if (value.is<std::string>()) return util::detail::string_to_number<T>()(value.get<std::string>());
+ if (value.is<std::string>()) {
+ double val;
+ return util::parseNumericString(value.get<std::string>(), val) ? val : 0;
+ }
else if (value.is<bool>()) return value.get<bool>();
else if (value.is<int64_t>()) return value.get<int64_t>();
else if (value.is<uint64_t>()) return value.get<uint64_t>();