mirror of
https://github.com/nlohmann/json.git
synced 2025-07-28 12:02:00 +03:00
Add support for deserialization of STL containers of non-default constructable types (fixes #2574).
This commit is contained in:
@ -1,14 +1,16 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <type_traits>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include <nlohmann/detail/conversions/from_json.hpp>
|
#include <nlohmann/detail/conversions/from_json.hpp>
|
||||||
#include <nlohmann/detail/conversions/to_json.hpp>
|
#include <nlohmann/detail/conversions/to_json.hpp>
|
||||||
|
#include <nlohmann/detail/meta/type_traits.hpp>
|
||||||
|
|
||||||
namespace nlohmann
|
namespace nlohmann
|
||||||
{
|
{
|
||||||
|
|
||||||
template<typename, typename>
|
template<typename ValueType, typename>
|
||||||
struct adl_serializer
|
struct adl_serializer
|
||||||
{
|
{
|
||||||
/*!
|
/*!
|
||||||
@ -20,14 +22,22 @@ struct adl_serializer
|
|||||||
@param[in] j JSON value to read from
|
@param[in] j JSON value to read from
|
||||||
@param[in,out] val value to write to
|
@param[in,out] val value to write to
|
||||||
*/
|
*/
|
||||||
template<typename BasicJsonType, typename ValueType>
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
static auto from_json(BasicJsonType&& j, ValueType& val) noexcept(
|
static auto from_json(BasicJsonType && j, U& val) noexcept(
|
||||||
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), val)))
|
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), val)))
|
||||||
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), val), void())
|
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), val), void())
|
||||||
{
|
{
|
||||||
::nlohmann::from_json(std::forward<BasicJsonType>(j), val);
|
::nlohmann::from_json(std::forward<BasicJsonType>(j), val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
|
static auto from_json(BasicJsonType && j) noexcept(
|
||||||
|
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {})))
|
||||||
|
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {}))
|
||||||
|
{
|
||||||
|
return ::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {});
|
||||||
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief convert any value type to a JSON value
|
@brief convert any value type to a JSON value
|
||||||
|
|
||||||
@ -37,13 +47,12 @@ struct adl_serializer
|
|||||||
@param[in,out] j JSON value to write to
|
@param[in,out] j JSON value to write to
|
||||||
@param[in] val value to read from
|
@param[in] val value to read from
|
||||||
*/
|
*/
|
||||||
template<typename BasicJsonType, typename ValueType>
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
static auto to_json(BasicJsonType& j, ValueType&& val) noexcept(
|
static auto to_json(BasicJsonType& j, U && val) noexcept(
|
||||||
noexcept(::nlohmann::to_json(j, std::forward<ValueType>(val))))
|
noexcept(::nlohmann::to_json(j, std::forward<U>(val))))
|
||||||
-> decltype(::nlohmann::to_json(j, std::forward<ValueType>(val)), void())
|
-> decltype(::nlohmann::to_json(j, std::forward<U>(val)), void())
|
||||||
{
|
{
|
||||||
::nlohmann::to_json(j, std::forward<ValueType>(val));
|
::nlohmann::to_json(j, std::forward<U>(val));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace nlohmann
|
} // namespace nlohmann
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#include <nlohmann/detail/exceptions.hpp>
|
#include <nlohmann/detail/exceptions.hpp>
|
||||||
#include <nlohmann/detail/macro_scope.hpp>
|
#include <nlohmann/detail/macro_scope.hpp>
|
||||||
#include <nlohmann/detail/meta/cpp_future.hpp>
|
#include <nlohmann/detail/meta/cpp_future.hpp>
|
||||||
|
#include <nlohmann/detail/meta/tag.hpp>
|
||||||
#include <nlohmann/detail/meta/type_traits.hpp>
|
#include <nlohmann/detail/meta/type_traits.hpp>
|
||||||
#include <nlohmann/detail/value_t.hpp>
|
#include <nlohmann/detail/value_t.hpp>
|
||||||
|
|
||||||
@ -248,6 +249,27 @@ void())
|
|||||||
from_json_array_impl(j, arr, priority_tag<3> {});
|
from_json_array_impl(j, arr, priority_tag<3> {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename Array, std::size_t... Is >
|
||||||
|
Array from_json_array_impl(BasicJsonType&& j, tag<Array> /*unused*/, index_sequence<Is...> /*unused*/)
|
||||||
|
{
|
||||||
|
return { std::forward<BasicJsonType>(j).at(Is).template get<typename Array::value_type>()... };
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename T, std::size_t N,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::array<T, N>>::value, int > = 0 >
|
||||||
|
auto from_json(BasicJsonType && j, tag<std::array<T, N>> t)
|
||||||
|
-> decltype(j.template get<T>(),
|
||||||
|
from_json_array_impl(std::forward<BasicJsonType>(j), t, make_index_sequence<N> {}))
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return from_json_array_impl(std::forward<BasicJsonType>(j), t, make_index_sequence<N> {});
|
||||||
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType>
|
template<typename BasicJsonType>
|
||||||
void from_json(const BasicJsonType& j, typename BasicJsonType::binary_t& bin)
|
void from_json(const BasicJsonType& j, typename BasicJsonType::binary_t& bin)
|
||||||
{
|
{
|
||||||
@ -323,22 +345,71 @@ void from_json(const BasicJsonType& j, ArithmeticType& val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename A1, typename A2>
|
template<typename BasicJsonType, typename A1, typename A2,
|
||||||
void from_json(const BasicJsonType& j, std::pair<A1, A2>& p)
|
enable_if_t<std::is_default_constructible<std::pair<A1, A2>>::value, int> = 0>
|
||||||
|
void from_json(BasicJsonType && j, std::pair<A1, A2>& p)
|
||||||
{
|
{
|
||||||
p = {j.at(0).template get<A1>(), j.at(1).template get<A2>()};
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
p = {std::forward<BasicJsonType>(j).at(0).template get<A1>(),
|
||||||
|
std::forward<BasicJsonType>(j).at(1).template get<A2>()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, class A1, class A2,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::pair<A1, A2>>::value, int > = 0 >
|
||||||
|
std::pair<A1, A2> from_json(BasicJsonType && j, tag<std::pair<A1, A2>> /*unused*/)
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return {std::forward<BasicJsonType>(j).at(0).template get<A1>(),
|
||||||
|
std::forward<BasicJsonType>(j).at(1).template get<A2>()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
||||||
void from_json_tuple_impl(const BasicJsonType& j, Tuple& t, index_sequence<Idx...> /*unused*/)
|
void from_json_tuple_impl(BasicJsonType&& j, Tuple& t, index_sequence<Idx...> /*unused*/)
|
||||||
{
|
{
|
||||||
t = std::make_tuple(j.at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
t = std::make_tuple(std::forward<BasicJsonType>(j).at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename... Args>
|
template<typename BasicJsonType, typename... Args,
|
||||||
void from_json(const BasicJsonType& j, std::tuple<Args...>& t)
|
enable_if_t < std::is_default_constructible<std::tuple<Args...>>::value, int > = 0 >
|
||||||
|
void from_json(BasicJsonType && j, std::tuple<Args...>& t)
|
||||||
{
|
{
|
||||||
from_json_tuple_impl(j, t, index_sequence_for<Args...> {});
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
from_json_tuple_impl(std::forward<BasicJsonType>(j), t, index_sequence_for<Args...> {});
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
||||||
|
Tuple from_json_tuple_impl(BasicJsonType&& j, tag<Tuple> /*unused*/, index_sequence<Idx...> /*unused*/)
|
||||||
|
{
|
||||||
|
return std::make_tuple(std::forward<BasicJsonType>(j).at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename... Args,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::tuple<Args...>>::value, int > = 0 >
|
||||||
|
std::tuple<Args...> from_json(BasicJsonType && j, tag<std::tuple<Args...>> t)
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return from_json_tuple_impl(std::forward<BasicJsonType>(j), t, index_sequence_for<Args...> {});
|
||||||
}
|
}
|
||||||
|
|
||||||
template < typename BasicJsonType, typename Key, typename Value, typename Compare, typename Allocator,
|
template < typename BasicJsonType, typename Key, typename Value, typename Compare, typename Allocator,
|
||||||
@ -390,6 +461,14 @@ struct from_json_fn
|
|||||||
{
|
{
|
||||||
return from_json(j, val);
|
return from_json(j, val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename T>
|
||||||
|
auto operator()(const BasicJsonType& j, detail::tag<T> t) const
|
||||||
|
noexcept(noexcept(from_json(j, t)))
|
||||||
|
-> decltype(from_json(j, t))
|
||||||
|
{
|
||||||
|
return from_json(j, t);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
|
10
include/nlohmann/detail/meta/tag.hpp
Normal file
10
include/nlohmann/detail/meta/tag.hpp
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace nlohmann
|
||||||
|
{
|
||||||
|
namespace detail
|
||||||
|
{
|
||||||
|
// dispatching helper struct
|
||||||
|
template <class T> struct tag {};
|
||||||
|
} // namespace detail
|
||||||
|
} // namespace nlohmann
|
@ -106,8 +106,7 @@ struct is_getable
|
|||||||
};
|
};
|
||||||
|
|
||||||
template<typename BasicJsonType, typename T>
|
template<typename BasicJsonType, typename T>
|
||||||
struct has_from_json < BasicJsonType, T,
|
struct has_from_json < BasicJsonType, T, enable_if_t < !is_basic_json<T>::value >>
|
||||||
enable_if_t < !is_basic_json<T>::value >>
|
|
||||||
{
|
{
|
||||||
using serializer = typename BasicJsonType::template json_serializer<T, void>;
|
using serializer = typename BasicJsonType::template json_serializer<T, void>;
|
||||||
|
|
||||||
|
@ -49,6 +49,7 @@ SOFTWARE.
|
|||||||
// #include <nlohmann/adl_serializer.hpp>
|
// #include <nlohmann/adl_serializer.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
#include <type_traits>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
// #include <nlohmann/detail/conversions/from_json.hpp>
|
// #include <nlohmann/detail/conversions/from_json.hpp>
|
||||||
@ -2812,6 +2813,18 @@ constexpr T static_const<T>::value;
|
|||||||
} // namespace detail
|
} // namespace detail
|
||||||
} // namespace nlohmann
|
} // namespace nlohmann
|
||||||
|
|
||||||
|
// #include <nlohmann/detail/meta/tag.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
namespace nlohmann
|
||||||
|
{
|
||||||
|
namespace detail
|
||||||
|
{
|
||||||
|
// dispatching helper struct
|
||||||
|
template <class T> struct tag {};
|
||||||
|
} // namespace detail
|
||||||
|
} // namespace nlohmann
|
||||||
|
|
||||||
// #include <nlohmann/detail/meta/type_traits.hpp>
|
// #include <nlohmann/detail/meta/type_traits.hpp>
|
||||||
|
|
||||||
|
|
||||||
@ -3129,8 +3142,7 @@ struct is_getable
|
|||||||
};
|
};
|
||||||
|
|
||||||
template<typename BasicJsonType, typename T>
|
template<typename BasicJsonType, typename T>
|
||||||
struct has_from_json < BasicJsonType, T,
|
struct has_from_json < BasicJsonType, T, enable_if_t < !is_basic_json<T>::value >>
|
||||||
enable_if_t < !is_basic_json<T>::value >>
|
|
||||||
{
|
{
|
||||||
using serializer = typename BasicJsonType::template json_serializer<T, void>;
|
using serializer = typename BasicJsonType::template json_serializer<T, void>;
|
||||||
|
|
||||||
@ -3734,6 +3746,27 @@ void())
|
|||||||
from_json_array_impl(j, arr, priority_tag<3> {});
|
from_json_array_impl(j, arr, priority_tag<3> {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename Array, std::size_t... Is >
|
||||||
|
Array from_json_array_impl(BasicJsonType&& j, tag<Array> /*unused*/, index_sequence<Is...> /*unused*/)
|
||||||
|
{
|
||||||
|
return { std::forward<BasicJsonType>(j).at(Is).template get<typename Array::value_type>()... };
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename T, std::size_t N,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::array<T, N>>::value, int > = 0 >
|
||||||
|
auto from_json(BasicJsonType && j, tag<std::array<T, N>> t)
|
||||||
|
-> decltype(j.template get<T>(),
|
||||||
|
from_json_array_impl(std::forward<BasicJsonType>(j), t, make_index_sequence<N> {}))
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return from_json_array_impl(std::forward<BasicJsonType>(j), t, make_index_sequence<N> {});
|
||||||
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType>
|
template<typename BasicJsonType>
|
||||||
void from_json(const BasicJsonType& j, typename BasicJsonType::binary_t& bin)
|
void from_json(const BasicJsonType& j, typename BasicJsonType::binary_t& bin)
|
||||||
{
|
{
|
||||||
@ -3809,22 +3842,71 @@ void from_json(const BasicJsonType& j, ArithmeticType& val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename A1, typename A2>
|
template<typename BasicJsonType, typename A1, typename A2,
|
||||||
void from_json(const BasicJsonType& j, std::pair<A1, A2>& p)
|
enable_if_t<std::is_default_constructible<std::pair<A1, A2>>::value, int> = 0>
|
||||||
|
void from_json(BasicJsonType && j, std::pair<A1, A2>& p)
|
||||||
{
|
{
|
||||||
p = {j.at(0).template get<A1>(), j.at(1).template get<A2>()};
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
p = {std::forward<BasicJsonType>(j).at(0).template get<A1>(),
|
||||||
|
std::forward<BasicJsonType>(j).at(1).template get<A2>()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, class A1, class A2,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::pair<A1, A2>>::value, int > = 0 >
|
||||||
|
std::pair<A1, A2> from_json(BasicJsonType && j, tag<std::pair<A1, A2>> /*unused*/)
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return {std::forward<BasicJsonType>(j).at(0).template get<A1>(),
|
||||||
|
std::forward<BasicJsonType>(j).at(1).template get<A2>()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
||||||
void from_json_tuple_impl(const BasicJsonType& j, Tuple& t, index_sequence<Idx...> /*unused*/)
|
void from_json_tuple_impl(BasicJsonType&& j, Tuple& t, index_sequence<Idx...> /*unused*/)
|
||||||
{
|
{
|
||||||
t = std::make_tuple(j.at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
t = std::make_tuple(std::forward<BasicJsonType>(j).at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename BasicJsonType, typename... Args>
|
template<typename BasicJsonType, typename... Args,
|
||||||
void from_json(const BasicJsonType& j, std::tuple<Args...>& t)
|
enable_if_t < std::is_default_constructible<std::tuple<Args...>>::value, int > = 0 >
|
||||||
|
void from_json(BasicJsonType && j, std::tuple<Args...>& t)
|
||||||
{
|
{
|
||||||
from_json_tuple_impl(j, t, index_sequence_for<Args...> {});
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
from_json_tuple_impl(std::forward<BasicJsonType>(j), t, index_sequence_for<Args...> {});
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename Tuple, std::size_t... Idx>
|
||||||
|
Tuple from_json_tuple_impl(BasicJsonType&& j, tag<Tuple> /*unused*/, index_sequence<Idx...> /*unused*/)
|
||||||
|
{
|
||||||
|
return std::make_tuple(std::forward<BasicJsonType>(j).at(Idx).template get<typename std::tuple_element<Idx, Tuple>::type>()...);
|
||||||
|
}
|
||||||
|
|
||||||
|
template < typename BasicJsonType, typename... Args,
|
||||||
|
enable_if_t < !std::is_default_constructible<std::tuple<Args...>>::value, int > = 0 >
|
||||||
|
std::tuple<Args...> from_json(BasicJsonType && j, tag<std::tuple<Args...>> t)
|
||||||
|
{
|
||||||
|
if (JSON_HEDLEY_UNLIKELY(!j.is_array()))
|
||||||
|
{
|
||||||
|
JSON_THROW(type_error::create(302, "type must be array, but is " +
|
||||||
|
std::string(j.type_name())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return from_json_tuple_impl(std::forward<BasicJsonType>(j), t, index_sequence_for<Args...> {});
|
||||||
}
|
}
|
||||||
|
|
||||||
template < typename BasicJsonType, typename Key, typename Value, typename Compare, typename Allocator,
|
template < typename BasicJsonType, typename Key, typename Value, typename Compare, typename Allocator,
|
||||||
@ -3876,6 +3958,14 @@ struct from_json_fn
|
|||||||
{
|
{
|
||||||
return from_json(j, val);
|
return from_json(j, val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename T>
|
||||||
|
auto operator()(const BasicJsonType& j, detail::tag<T> t) const
|
||||||
|
noexcept(noexcept(from_json(j, t)))
|
||||||
|
-> decltype(from_json(j, t))
|
||||||
|
{
|
||||||
|
return from_json(j, t);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
@ -4448,11 +4538,13 @@ constexpr const auto& to_json = detail::static_const<detail::to_json_fn>::value;
|
|||||||
} // namespace
|
} // namespace
|
||||||
} // namespace nlohmann
|
} // namespace nlohmann
|
||||||
|
|
||||||
|
// #include <nlohmann/detail/meta/type_traits.hpp>
|
||||||
|
|
||||||
|
|
||||||
namespace nlohmann
|
namespace nlohmann
|
||||||
{
|
{
|
||||||
|
|
||||||
template<typename, typename>
|
template<typename ValueType, typename>
|
||||||
struct adl_serializer
|
struct adl_serializer
|
||||||
{
|
{
|
||||||
/*!
|
/*!
|
||||||
@ -4464,14 +4556,22 @@ struct adl_serializer
|
|||||||
@param[in] j JSON value to read from
|
@param[in] j JSON value to read from
|
||||||
@param[in,out] val value to write to
|
@param[in,out] val value to write to
|
||||||
*/
|
*/
|
||||||
template<typename BasicJsonType, typename ValueType>
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
static auto from_json(BasicJsonType&& j, ValueType& val) noexcept(
|
static auto from_json(BasicJsonType && j, U& val) noexcept(
|
||||||
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), val)))
|
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), val)))
|
||||||
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), val), void())
|
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), val), void())
|
||||||
{
|
{
|
||||||
::nlohmann::from_json(std::forward<BasicJsonType>(j), val);
|
::nlohmann::from_json(std::forward<BasicJsonType>(j), val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
|
static auto from_json(BasicJsonType && j) noexcept(
|
||||||
|
noexcept(::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {})))
|
||||||
|
-> decltype(::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {}))
|
||||||
|
{
|
||||||
|
return ::nlohmann::from_json(std::forward<BasicJsonType>(j), detail::tag<U> {});
|
||||||
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief convert any value type to a JSON value
|
@brief convert any value type to a JSON value
|
||||||
|
|
||||||
@ -4481,15 +4581,14 @@ struct adl_serializer
|
|||||||
@param[in,out] j JSON value to write to
|
@param[in,out] j JSON value to write to
|
||||||
@param[in] val value to read from
|
@param[in] val value to read from
|
||||||
*/
|
*/
|
||||||
template<typename BasicJsonType, typename ValueType>
|
template<typename BasicJsonType, typename U = ValueType>
|
||||||
static auto to_json(BasicJsonType& j, ValueType&& val) noexcept(
|
static auto to_json(BasicJsonType& j, U && val) noexcept(
|
||||||
noexcept(::nlohmann::to_json(j, std::forward<ValueType>(val))))
|
noexcept(::nlohmann::to_json(j, std::forward<U>(val))))
|
||||||
-> decltype(::nlohmann::to_json(j, std::forward<ValueType>(val)), void())
|
-> decltype(::nlohmann::to_json(j, std::forward<U>(val)), void())
|
||||||
{
|
{
|
||||||
::nlohmann::to_json(j, std::forward<ValueType>(val));
|
::nlohmann::to_json(j, std::forward<U>(val));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace nlohmann
|
} // namespace nlohmann
|
||||||
|
|
||||||
// #include <nlohmann/byte_container_with_subtype.hpp>
|
// #include <nlohmann/byte_container_with_subtype.hpp>
|
||||||
|
@ -115,7 +115,7 @@ namespace nlohmann
|
|||||||
template <>
|
template <>
|
||||||
struct adl_serializer<NonDefaultFromJsonStruct>
|
struct adl_serializer<NonDefaultFromJsonStruct>
|
||||||
{
|
{
|
||||||
static NonDefaultFromJsonStruct from_json (json const&) noexcept
|
static NonDefaultFromJsonStruct from_json (const json&) noexcept
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
@ -133,6 +133,28 @@ struct NotSerializableData
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////
|
||||||
|
// for #2574
|
||||||
|
/////////////////////////////////////////////////////////////////////
|
||||||
|
struct NonDefaultConstructible
|
||||||
|
{
|
||||||
|
explicit NonDefaultConstructible (int x) : x(x) { }
|
||||||
|
int x;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace nlohmann
|
||||||
|
{
|
||||||
|
template <>
|
||||||
|
struct adl_serializer<NonDefaultConstructible>
|
||||||
|
{
|
||||||
|
static NonDefaultConstructible from_json (const json& j) noexcept
|
||||||
|
{
|
||||||
|
return NonDefaultConstructible(j.get<int>());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
TEST_CASE("regression tests 2")
|
TEST_CASE("regression tests 2")
|
||||||
{
|
{
|
||||||
SECTION("issue #1001 - Fix memory leak during parser callback")
|
SECTION("issue #1001 - Fix memory leak during parser callback")
|
||||||
@ -498,4 +520,59 @@ TEST_CASE("regression tests 2")
|
|||||||
CHECK(j.dump() == "\"Hello, world!\"");
|
CHECK(j.dump() == "\"Hello, world!\"");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
SECTION("issue #2574 - Deserialization to std::array, std::pair, and std::tuple with non-default constructable types fails")
|
||||||
|
{
|
||||||
|
SECTION("std::array")
|
||||||
|
{
|
||||||
|
json j = { 7, 4 };
|
||||||
|
auto arr = j.get<std::array<NonDefaultConstructible, 2>>();
|
||||||
|
CHECK(arr[0].x == 7);
|
||||||
|
CHECK(arr[1].x == 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
SECTION("std::pair")
|
||||||
|
{
|
||||||
|
{
|
||||||
|
json j = { 3, 8 };
|
||||||
|
auto x = j.at(0).get<NonDefaultConstructible>();
|
||||||
|
CHECK(x.x == 3);
|
||||||
|
|
||||||
|
auto p = j.get<std::pair<NonDefaultConstructible, NonDefaultConstructible>>();
|
||||||
|
CHECK(p.first.x == 3);
|
||||||
|
CHECK(p.second.x == 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
json j = { 4, 1 };
|
||||||
|
auto p = j.get<std::pair<int, NonDefaultConstructible>>();
|
||||||
|
CHECK(p.first == 4);
|
||||||
|
CHECK(p.second.x == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
json j = { 6, 7 };
|
||||||
|
auto p = j.get<std::pair<NonDefaultConstructible, int>>();
|
||||||
|
CHECK(p.first.x == 6);
|
||||||
|
CHECK(p.second == 7);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SECTION("std::tuple")
|
||||||
|
{
|
||||||
|
{
|
||||||
|
json j = { 9 };
|
||||||
|
auto t = j.get<std::tuple<NonDefaultConstructible>>();
|
||||||
|
CHECK(std::get<0>(t).x == 9);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
json j = { 9, 8, 7 };
|
||||||
|
auto t = j.get<std::tuple<NonDefaultConstructible, int, NonDefaultConstructible>>();
|
||||||
|
CHECK(std::get<0>(t).x == 9);
|
||||||
|
CHECK(std::get<1>(t) == 8);
|
||||||
|
CHECK(std::get<2>(t).x == 7);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user