oxen-core/contrib/epee/include/epee/storages/portable_storage_val_conver...

132 lines
4.9 KiB
C++

// Copyright (c) 2006-2013, Andrey N. Sabelnikov, www.sabelnikov.net
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// * Neither the name of the Andrey N. Sabelnikov nor the
// names of its contributors may be used to endorse or promote products
// derived from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER BE LIABLE FOR ANY
// DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
// ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
//
#pragma once
#include <iomanip>
#include <regex>
#include <type_traits>
#include <charconv>
#include "../misc_language.h"
#include "portable_storage_base.h"
#include "parserse_base_utils.h"
#include "../warnings.h"
namespace epee
{
namespace serialization
{
#define ASSERT_AND_THROW_WRONG_CONVERSION() ASSERT_MES_AND_THROW("WRONG DATA CONVERSION @ " << __FILE__ << ":" << __LINE__ << ": " << typeid(from).name() << " to " << typeid(to).name())
template<typename From, typename To, typename SFINAE = void>
struct converter
{
void operator()(const From& from, To& to)
{
ASSERT_AND_THROW_WRONG_CONVERSION();
}
};
template<typename From, typename To>
struct converter<From, To, std::enable_if_t<
!std::is_same_v<To, From> && std::is_integral_v<To> && std::is_integral_v<From> &&
!std::is_same_v<From, bool> && !std::is_same_v<To, bool>>>
{
void operator()(const From& from, To& to)
{
PUSH_WARNINGS
DISABLE_VS_WARNINGS(4018)
DISABLE_CLANG_WARNING(tautological-constant-out-of-range-compare)
DISABLE_GCC_AND_CLANG_WARNING(sign-compare)
bool in_range;
if constexpr (std::is_signed_v<From> == std::is_signed_v<To>) // signed -> signed or unsigned -> unsigned
in_range = from >= std::numeric_limits<To>::min() && from <= std::numeric_limits<To>::max();
else if constexpr (std::is_signed_v<To>) // unsigned -> signed
in_range = from <= std::numeric_limits<To>::max();
else // signed -> unsigned
in_range = from >= 0 && from <= std::numeric_limits<To>::max();
CHECK_AND_ASSERT_THROW_MES(in_range,
"int value overflow: cannot convert value " << +from << " to integer type with range ["
<< +std::numeric_limits<To>::min() << "," << +std::numeric_limits<To>::max() << "]");
to = static_cast<To>(from);
POP_WARNINGS
}
};
// For MyMonero/OpenMonero backend compatibility
// MyMonero backend sends amount, fees and timestamp values as strings.
// Until MM backend is updated, this is needed for compatibility between OpenMonero and MyMonero.
template<>
struct converter<std::string, uint64_t>
{
// MyMonero ISO 8061 timestamp (2017-05-06T16:27:06Z)
inline static std::regex mymonero_iso8061_timestamp{R"(\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d:[0-5]\dZ)"};
void operator()(const std::string& from, uint64_t& to)
{
MTRACE("Converting std::string to uint64_t. Source: " << from);
const auto* strend = from.data() + from.size();
if (auto [p, ec] = std::from_chars(from.data(), strend, to); ec == std::errc{} && p == strend)
return; // Good: successfully consumed the whole string.
if (std::regex_match(from, mymonero_iso8061_timestamp))
{
// Convert to unix timestamp
std::tm tm{};
std::istringstream ss{from};
if (ss >> std::get_time(&tm, "%Y-%m-%dT%H:%M:%S"))
{
to = std::mktime(&tm);
return;
}
}
ASSERT_AND_THROW_WRONG_CONVERSION();
}
};
template<typename From, typename To>
struct converter<From, To, std::enable_if_t<std::is_same<To, From>::value>>
{
void operator()(const From& from, To& to)
{
to = from;
}
};
template<class From, class To>
void convert_t(const From& from, To& to)
{
converter<From, To>{}(from, to);
}
}
}