LCOV - code coverage report
Current view: top level - json/impl - conversion.hpp (source / functions) Coverage Total Hit
Test: coverage_filtered.info Lines: 100.0 % 18 18
Test Date: 2025-12-23 17:38:56 Functions: 96.5 % 86 83

            Line data    Source code
       1              : //
       2              : // Copyright (c) 2020 Krystian Stasiowski (sdkrystian@gmail.com)
       3              : // Copyright (c) 2022 Dmitry Arkhipov (grisumbras@yandex.ru)
       4              : //
       5              : // Distributed under the Boost Software License, Version 1.0. (See accompanying
       6              : // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
       7              : //
       8              : // Official repository: https://github.com/boostorg/json
       9              : //
      10              : 
      11              : #ifndef BOOST_JSON_IMPL_CONVERSION_HPP
      12              : #define BOOST_JSON_IMPL_CONVERSION_HPP
      13              : 
      14              : #include <boost/json/fwd.hpp>
      15              : #include <boost/json/value.hpp>
      16              : #include <boost/json/string_view.hpp>
      17              : #include <boost/describe/enumerators.hpp>
      18              : #include <boost/describe/members.hpp>
      19              : #include <boost/describe/bases.hpp>
      20              : #include <boost/mp11/algorithm.hpp>
      21              : #include <boost/mp11/utility.hpp>
      22              : #include <boost/system/result.hpp>
      23              : 
      24              : #include <iterator>
      25              : #include <tuple>
      26              : #include <utility>
      27              : #ifndef BOOST_NO_CXX17_HDR_VARIANT
      28              : # include <variant>
      29              : #endif // BOOST_NO_CXX17_HDR_VARIANT
      30              : 
      31              : namespace boost {
      32              : namespace json {
      33              : namespace detail {
      34              : 
      35              : #ifdef __cpp_lib_nonmember_container_access
      36              : using std::size;
      37              : #endif
      38              : 
      39              : template<std::size_t I, class T>
      40              : using tuple_element_t = typename std::tuple_element<I, T>::type;
      41              : 
      42              : template<class T>
      43              : using iterator_type = decltype(std::begin(std::declval<T&>()));
      44              : template<class T>
      45              : using iterator_traits = std::iterator_traits< iterator_type<T> >;
      46              : 
      47              : template<class T>
      48              : using value_type = typename iterator_traits<T>::value_type;
      49              : template<class T>
      50              : using mapped_type = tuple_element_t< 1, value_type<T> >;
      51              : 
      52              : // had to make the metafunction always succeeding in order to make it work
      53              : // with msvc 14.0
      54              : template<class T>
      55              : using key_type_helper = tuple_element_t< 0, value_type<T> >;
      56              : template<class T>
      57              : using key_type = mp11::mp_eval_or<
      58              :     void,
      59              :     key_type_helper,
      60              :     T>;
      61              : 
      62              : template<class T>
      63              : using are_begin_and_end_same = std::is_same<
      64              :     iterator_type<T>,
      65              :     decltype(std::end(std::declval<T&>()))>;
      66              : 
      67              : // msvc 14.0 gets confused when std::is_same is used directly
      68              : template<class A, class B>
      69              : using is_same_msvc_140 = std::is_same<A, B>;
      70              : template<class T>
      71              : using is_its_own_value = is_same_msvc_140<value_type<T>, T>;
      72              : 
      73              : template<class T>
      74              : using not_its_own_value = mp11::mp_not< is_its_own_value<T> >;
      75              : 
      76              : template<class T>
      77              : using begin_iterator_category = typename std::iterator_traits<
      78              :     iterator_type<T>>::iterator_category;
      79              : 
      80              : template<class T>
      81              : using has_positive_tuple_size = mp11::mp_bool<
      82              :     (std::tuple_size<T>::value > 0) >;
      83              : 
      84              : template<class T>
      85              : using has_unique_keys = has_positive_tuple_size<decltype(
      86              :     std::declval<T&>().emplace(
      87              :         std::declval<value_type<T>>()))>;
      88              : 
      89              : template<class T>
      90              : using has_string_type = std::is_same<
      91              :     typename T::string_type, std::basic_string<typename T::value_type> >;
      92              : 
      93              : template<class T>
      94              : struct is_value_type_pair_helper : std::false_type
      95              : { };
      96              : template<class T1, class T2>
      97              : struct is_value_type_pair_helper<std::pair<T1, T2>> : std::true_type
      98              : { };
      99              : template<class T>
     100              : using is_value_type_pair = is_value_type_pair_helper<value_type<T>>;
     101              : 
     102              : template<class T>
     103              : using has_size_member_helper
     104              :     = std::is_convertible<decltype(std::declval<T&>().size()), std::size_t>;
     105              : template<class T>
     106              : using has_size_member = mp11::mp_valid_and_true<has_size_member_helper, T>;
     107              : template<class T>
     108              : using has_free_size_helper
     109              :     = std::is_convertible<
     110              :         decltype(size(std::declval<T const&>())),
     111              :         std::size_t>;
     112              : template<class T>
     113              : using has_free_size = mp11::mp_valid_and_true<has_free_size_helper, T>;
     114              : template<class T>
     115              : using size_implementation = mp11::mp_cond<
     116              :     has_size_member<T>, mp11::mp_int<3>,
     117              :     has_free_size<T>,   mp11::mp_int<2>,
     118              :     std::is_array<T>,   mp11::mp_int<1>,
     119              :     mp11::mp_true,      mp11::mp_int<0>>;
     120              : 
     121              : template<class T>
     122              : std::size_t
     123          138 : try_size(T&& cont, mp11::mp_int<3>)
     124              : {
     125          138 :     return cont.size();
     126              : }
     127              : 
     128              : template<class T>
     129              : std::size_t
     130            1 : try_size(T& cont, mp11::mp_int<2>)
     131              : {
     132            1 :     return size(cont);
     133              : }
     134              : 
     135              : template<class T, std::size_t N>
     136              : std::size_t
     137            1 : try_size(T(&)[N], mp11::mp_int<1>)
     138              : {
     139            1 :     return N;
     140              : }
     141              : 
     142              : template<class T>
     143              : std::size_t
     144            7 : try_size(T&, mp11::mp_int<0>)
     145              : {
     146            7 :     return 0;
     147              : }
     148              : 
     149              : template<class T>
     150              : using has_push_back_helper
     151              :     = decltype(std::declval<T&>().push_back(std::declval<value_type<T>>()));
     152              : template<class T>
     153              : using has_push_back = mp11::mp_valid<has_push_back_helper, T>;
     154              : template<class T>
     155              : using inserter_implementation = mp11::mp_cond<
     156              :     is_tuple_like<T>, mp11::mp_int<2>,
     157              :     has_push_back<T>, mp11::mp_int<1>,
     158              :     mp11::mp_true,    mp11::mp_int<0>>;
     159              : 
     160              : template<class T>
     161              : iterator_type<T>
     162           56 : inserter(
     163              :     T& target,
     164              :     mp11::mp_int<2>)
     165              : {
     166           56 :     return target.begin();
     167              : }
     168              : 
     169              : template<class T>
     170              : std::back_insert_iterator<T>
     171          569 : inserter(
     172              :     T& target,
     173              :     mp11::mp_int<1>)
     174              : {
     175          569 :     return std::back_inserter(target);
     176              : }
     177              : 
     178              : template<class T>
     179              : std::insert_iterator<T>
     180           62 : inserter(
     181              :     T& target,
     182              :     mp11::mp_int<0>)
     183              : {
     184           62 :     return std::inserter( target, target.end() );
     185              : }
     186              : 
     187              : using value_from_conversion = mp11::mp_true;
     188              : using value_to_conversion = mp11::mp_false;
     189              : 
     190              : struct user_conversion_tag { };
     191              : struct context_conversion_tag : user_conversion_tag { };
     192              : struct full_context_conversion_tag : context_conversion_tag { };
     193              : struct native_conversion_tag { };
     194              : struct value_conversion_tag : native_conversion_tag { };
     195              : struct object_conversion_tag : native_conversion_tag { };
     196              : struct array_conversion_tag : native_conversion_tag { };
     197              : struct string_conversion_tag : native_conversion_tag { };
     198              : struct bool_conversion_tag : native_conversion_tag { };
     199              : struct number_conversion_tag : native_conversion_tag { };
     200              : struct integral_conversion_tag : number_conversion_tag { };
     201              : struct floating_point_conversion_tag : number_conversion_tag { };
     202              : struct null_like_conversion_tag { };
     203              : struct string_like_conversion_tag { };
     204              : struct map_like_conversion_tag { };
     205              : struct path_conversion_tag { };
     206              : struct sequence_conversion_tag { };
     207              : struct tuple_conversion_tag { };
     208              : struct described_class_conversion_tag { };
     209              : struct described_enum_conversion_tag { };
     210              : struct variant_conversion_tag { };
     211              : struct optional_conversion_tag { };
     212              : struct no_conversion_tag { };
     213              : 
     214              : template<class... Args>
     215              : using supports_tag_invoke = decltype(tag_invoke( std::declval<Args>()... ));
     216              : 
     217              : template<class T>
     218              : using has_user_conversion_from_impl = supports_tag_invoke<
     219              :     value_from_tag, value&, T&& >;
     220              : template<class T>
     221              : using has_user_conversion_to_impl = supports_tag_invoke<
     222              :     value_to_tag<T>, value const& >;
     223              : template<class T>
     224              : using has_nonthrowing_user_conversion_to_impl = supports_tag_invoke<
     225              :     try_value_to_tag<T>, value const& >;
     226              : template< class T, class Dir >
     227              : using has_user_conversion1 = mp11::mp_if<
     228              :     std::is_same<Dir, value_from_conversion>,
     229              :     mp11::mp_valid<has_user_conversion_from_impl, T>,
     230              :     mp11::mp_or<
     231              :         mp11::mp_valid<has_user_conversion_to_impl, T>,
     232              :         mp11::mp_valid<has_nonthrowing_user_conversion_to_impl, T>>>;
     233              : 
     234              : template< class Ctx, class T >
     235              : using has_context_conversion_from_impl = supports_tag_invoke<
     236              :     value_from_tag, value&, T&&, Ctx const& >;
     237              : template< class Ctx, class T >
     238              : using has_context_conversion_to_impl = supports_tag_invoke<
     239              :     value_to_tag<T>, value const&, Ctx const& >;
     240              : template< class Ctx, class T >
     241              : using has_nonthrowing_context_conversion_to_impl = supports_tag_invoke<
     242              :     try_value_to_tag<T>, value const&, Ctx const& >;
     243              : template< class Ctx, class T, class Dir >
     244              : using has_user_conversion2 = mp11::mp_if<
     245              :     std::is_same<Dir, value_from_conversion>,
     246              :     mp11::mp_valid<has_context_conversion_from_impl, Ctx, T>,
     247              :     mp11::mp_or<
     248              :         mp11::mp_valid<has_context_conversion_to_impl, Ctx, T>,
     249              :         mp11::mp_valid<has_nonthrowing_context_conversion_to_impl, Ctx, T>>>;
     250              : 
     251              : template< class Ctx, class T >
     252              : using has_full_context_conversion_from_impl = supports_tag_invoke<
     253              :     value_from_tag, value&, T&&, Ctx const&, Ctx const& >;
     254              : template< class Ctx, class T >
     255              : using has_full_context_conversion_to_impl = supports_tag_invoke<
     256              :     value_to_tag<T>, value const&, Ctx const&,  Ctx const& >;
     257              : template< class Ctx, class T >
     258              : using has_nonthrowing_full_context_conversion_to_impl = supports_tag_invoke<
     259              :     try_value_to_tag<T>, value const&, Ctx const&, Ctx const& >;
     260              : template< class Ctx, class T, class Dir >
     261              : using has_user_conversion3 = mp11::mp_if<
     262              :     std::is_same<Dir, value_from_conversion>,
     263              :     mp11::mp_valid<has_full_context_conversion_from_impl, Ctx, T>,
     264              :     mp11::mp_or<
     265              :         mp11::mp_valid<has_full_context_conversion_to_impl, Ctx, T>,
     266              :         mp11::mp_valid<
     267              :             has_nonthrowing_full_context_conversion_to_impl, Ctx, T>>>;
     268              : 
     269              : template< class T >
     270              : using described_non_public_members = describe::describe_members<
     271              :     T,
     272              :     describe::mod_private
     273              :         | describe::mod_protected
     274              :         | boost::describe::mod_inherited>;
     275              : 
     276              : #if defined(BOOST_MSVC) && BOOST_MSVC < 1920
     277              : 
     278              : template< class T >
     279              : struct described_member_t_impl;
     280              : 
     281              : template< class T, class C >
     282              : struct described_member_t_impl<T C::*>
     283              : {
     284              :     using type = T;
     285              : };
     286              : 
     287              : template< class T, class D >
     288              : using described_member_t = remove_cvref<
     289              :     typename described_member_t_impl<
     290              :         remove_cvref<decltype(D::pointer)> >::type>;
     291              : 
     292              : #else
     293              : 
     294              : template< class T, class D >
     295              : using described_member_t = remove_cvref<decltype(
     296              :     std::declval<T&>().* D::pointer )>;
     297              : 
     298              : #endif
     299              : 
     300              : template< class T >
     301              : using described_members = describe::describe_members<
     302              :     T, describe::mod_any_access | describe::mod_inherited>;
     303              : 
     304              : #ifdef BOOST_DESCRIBE_CXX14
     305              : 
     306              : constexpr
     307              : bool
     308              : compare_strings(char const* l, char const* r)
     309              : {
     310              : #if defined(_MSC_VER) && (_MSC_VER <= 1900) && !defined(__clang__)
     311              :     return *l == *r && ( (*l == 0) | compare_strings(l + 1, r + 1) );
     312              : #else
     313              :     do
     314              :     {
     315              :         if( *l != *r )
     316              :             return false;
     317              :         if( *l == 0 )
     318              :             return true;
     319              :         ++l;
     320              :         ++r;
     321              :     } while(true);
     322              : #endif
     323              : }
     324              : 
     325              : template< class L, class R >
     326              : struct equal_member_names
     327              :     : mp11::mp_bool< compare_strings(L::name, R::name) >
     328              : {};
     329              : 
     330              : template< class T >
     331              : using uniquely_named_members = mp11::mp_same<
     332              :     mp11::mp_unique_if< described_members<T>, equal_member_names >,
     333              :     described_members<T> >;
     334              : 
     335              : #else
     336              : 
     337              : // we only check this in C++14, but the template should exist nevertheless
     338              : template< class T >
     339              : using uniquely_named_members = std::true_type;
     340              : 
     341              : #endif // BOOST_DESCRIBE_CXX14
     342              : 
     343              : // user conversion (via tag_invoke)
     344              : template< class Ctx, class T, class Dir >
     345              : using user_conversion_category = mp11::mp_cond<
     346              :     has_user_conversion3<Ctx, T, Dir>, full_context_conversion_tag,
     347              :     has_user_conversion2<Ctx, T, Dir>, context_conversion_tag,
     348              :     has_user_conversion1<T, Dir>,      user_conversion_tag>;
     349              : 
     350              : // native conversions (constructors and member functions of value)
     351              : template< class T >
     352              : using native_conversion_category = mp11::mp_cond<
     353              :     std::is_same<T, value>,  value_conversion_tag,
     354              :     std::is_same<T, array>,  array_conversion_tag,
     355              :     std::is_same<T, object>, object_conversion_tag,
     356              :     std::is_same<T, string>, string_conversion_tag>;
     357              : 
     358              : // generic conversions
     359              : template< class T >
     360              : using generic_conversion_category = mp11::mp_cond<
     361              :     std::is_same<T, bool>,     bool_conversion_tag,
     362              :     std::is_integral<T>,       integral_conversion_tag,
     363              :     std::is_floating_point<T>, floating_point_conversion_tag,
     364              :     is_null_like<T>,           null_like_conversion_tag,
     365              :     is_string_like<T>,         string_like_conversion_tag,
     366              :     is_variant_like<T>,        variant_conversion_tag,
     367              :     is_optional_like<T>,       optional_conversion_tag,
     368              :     is_map_like<T>,            map_like_conversion_tag,
     369              :     is_sequence_like<T>,       sequence_conversion_tag,
     370              :     is_tuple_like<T>,          tuple_conversion_tag,
     371              :     is_described_class<T>,     described_class_conversion_tag,
     372              :     is_described_enum<T>,      described_enum_conversion_tag,
     373              :     is_path_like<T>,           path_conversion_tag,
     374              :     // failed to find a suitable implementation
     375              :     mp11::mp_true,             no_conversion_tag>;
     376              : 
     377              : template< class T >
     378              : using nested_type = typename T::type;
     379              : template< class T1, class T2 >
     380              : using conversion_category_impl_helper = mp11::mp_eval_if_not<
     381              :     std::is_same<detail::no_conversion_tag, T1>,
     382              :     T1,
     383              :     mp11::mp_eval_or_q, T1, mp11::mp_quote<nested_type>, T2>;
     384              : template< class Ctx, class T, class Dir >
     385              : struct conversion_category_impl
     386              : {
     387              :     using type = mp11::mp_fold<
     388              :         mp11::mp_list<
     389              :             mp11::mp_defer<user_conversion_category, Ctx, T, Dir>,
     390              :             mp11::mp_defer<native_conversion_category, T>,
     391              :             mp11::mp_defer<generic_conversion_category, T>>,
     392              :         no_conversion_tag,
     393              :         conversion_category_impl_helper>;
     394              : };
     395              : template< class Ctx, class T, class Dir >
     396              : using conversion_category =
     397              :     typename conversion_category_impl< Ctx, T, Dir >::type;
     398              : 
     399              : template< class T >
     400              : using any_conversion_tag = mp11::mp_not<
     401              :     std::is_same< T, no_conversion_tag > >;
     402              : 
     403              : template< class T, class Dir, class... Ctxs >
     404              : struct conversion_category_impl< std::tuple<Ctxs...>, T, Dir >
     405              : {
     406              :     using ctxs = mp11::mp_list< remove_cvref<Ctxs>... >;
     407              :     using cats = mp11::mp_list<
     408              :         conversion_category<remove_cvref<Ctxs>, T, Dir>... >;
     409              : 
     410              :     template< class I >
     411              :     using exists = mp11::mp_less< I, mp11::mp_size<cats> >;
     412              : 
     413              :     using context2 = mp11::mp_find< cats, full_context_conversion_tag >;
     414              :     using context1 = mp11::mp_find< cats, context_conversion_tag >;
     415              :     using context0 = mp11::mp_find< cats, user_conversion_tag >;
     416              :     using index = mp11::mp_cond<
     417              :         exists<context2>, context2,
     418              :         exists<context1>, context1,
     419              :         exists<context0>, context0,
     420              :         mp11::mp_true, mp11::mp_find_if< cats, any_conversion_tag > >;
     421              :     using type = mp11::mp_eval_or<
     422              :         no_conversion_tag,
     423              :         mp11::mp_at, cats, index >;
     424              : };
     425              : 
     426              : struct no_context
     427              : {};
     428              : 
     429              : template <class T, class Dir>
     430              : using can_convert = mp11::mp_not<
     431              :     std::is_same<
     432              :         detail::conversion_category<no_context, T, Dir>,
     433              :         detail::no_conversion_tag>>;
     434              : 
     435              : template<class Impl1, class Impl2>
     436              : using conversion_round_trips_helper = mp11::mp_or<
     437              :     std::is_same<Impl1, Impl2>,
     438              :     std::is_base_of<user_conversion_tag, Impl1>,
     439              :     std::is_base_of<user_conversion_tag, Impl2>>;
     440              : template< class Ctx, class T, class Dir >
     441              : using conversion_round_trips  = conversion_round_trips_helper<
     442              :     conversion_category<Ctx, T, Dir>,
     443              :     conversion_category<Ctx, T, mp11::mp_not<Dir>>>;
     444              : 
     445              : template< class T1, class T2 >
     446              : struct copy_cref_helper
     447              : {
     448              :     using type = remove_cvref<T2>;
     449              : };
     450              : template< class T1, class T2 >
     451              : using copy_cref = typename copy_cref_helper< T1, T2 >::type;
     452              : 
     453              : template< class T1, class T2 >
     454              : struct copy_cref_helper<T1 const, T2>
     455              : {
     456              :     using type = remove_cvref<T2> const;
     457              : };
     458              : template< class T1, class T2 >
     459              : struct copy_cref_helper<T1&, T2>
     460              : {
     461              :     using type = copy_cref<T1, T2>&;
     462              : };
     463              : template< class T1, class T2 >
     464              : struct copy_cref_helper<T1&&, T2>
     465              : {
     466              :     using type = copy_cref<T1, T2>&&;
     467              : };
     468              : 
     469              : template< class Rng, class Traits >
     470              : using forwarded_value_helper = mp11::mp_if<
     471              :     std::is_convertible<
     472              :         typename Traits::reference,
     473              :         copy_cref<Rng, typename Traits::value_type> >,
     474              :     copy_cref<Rng, typename Traits::value_type>,
     475              :     typename Traits::value_type >;
     476              : 
     477              : template< class Rng >
     478              : using forwarded_value = forwarded_value_helper<
     479              :     Rng, iterator_traits< Rng > >;
     480              : 
     481              : template< class Ctx, class T, class Dir >
     482              : struct supported_context
     483              : {
     484              :     using type = Ctx;
     485              : 
     486              :     static
     487              :     type const&
     488           32 :     get( Ctx const& ctx ) noexcept
     489              :     {
     490           32 :         return ctx;
     491              :     }
     492              : };
     493              : 
     494              : template< class T, class Dir, class... Ctxs >
     495              : struct supported_context< std::tuple<Ctxs...>, T, Dir >
     496              : {
     497              :     using Ctx = std::tuple<Ctxs...>;
     498              :     using impl = conversion_category_impl<Ctx, T, Dir>;
     499              :     using index = typename impl::index;
     500              :     using next_supported = supported_context<
     501              :         mp11::mp_at< typename impl::ctxs, index >, T, Dir >;
     502              :     using type = typename next_supported::type;
     503              : 
     504              :     static
     505              :     type const&
     506           19 :     get( Ctx const& ctx ) noexcept
     507              :     {
     508           19 :         return next_supported::get( std::get<index::value>( ctx ) );
     509              :     }
     510              : };
     511              : 
     512              : template< class T >
     513              : using value_result_type = typename std::decay<
     514              :     decltype( std::declval<T&>().value() )>::type;
     515              : 
     516              : template< class T >
     517              : using can_reset = decltype( std::declval<T&>().reset() );
     518              : 
     519              : template< class T >
     520              : using has_valueless_by_exception =
     521              :     decltype( std::declval<T const&>().valueless_by_exception() );
     522              : 
     523              : } // namespace detail
     524              : 
     525              : template <class T>
     526              : struct result_for<T, value>
     527              : {
     528              :     using type = system::result< detail::remove_cvref<T> >;
     529              : };
     530              : 
     531              : template<class T>
     532              : struct is_string_like
     533              :     : std::is_convertible<T, string_view>
     534              : { };
     535              : 
     536              : template<class T>
     537              : struct is_path_like
     538              :     : mp11::mp_all<
     539              :         mp11::mp_valid_and_true<detail::is_its_own_value, T>,
     540              :         mp11::mp_valid_and_true<detail::has_string_type, T>>
     541              : { };
     542              : template<class T>
     543              : struct is_sequence_like
     544              :     : mp11::mp_all<
     545              :         mp11::mp_valid_and_true<detail::are_begin_and_end_same, T>,
     546              :         mp11::mp_valid_and_true<detail::not_its_own_value, T>,
     547              :         mp11::mp_valid<detail::begin_iterator_category, T>>
     548              : { };
     549              : 
     550              : template<class T>
     551              : struct is_map_like
     552              :     : mp11::mp_all<
     553              :         is_sequence_like<T>,
     554              :         mp11::mp_valid_and_true<detail::is_value_type_pair, T>,
     555              :         is_string_like<detail::key_type<T>>,
     556              :         mp11::mp_valid_and_true<detail::has_unique_keys, T>>
     557              : { };
     558              : 
     559              : template<class T>
     560              : struct is_tuple_like
     561              :     : mp11::mp_valid_and_true<detail::has_positive_tuple_size, T>
     562              : { };
     563              : 
     564              : template<>
     565              : struct is_null_like<std::nullptr_t>
     566              :     : std::true_type
     567              : { };
     568              : 
     569              : #ifndef BOOST_NO_CXX17_HDR_VARIANT
     570              : template<>
     571              : struct is_null_like<std::monostate>
     572              :     : std::true_type
     573              : { };
     574              : #endif // BOOST_NO_CXX17_HDR_VARIANT
     575              : 
     576              : template<class T>
     577              : struct is_described_class
     578              :     : mp11::mp_and<
     579              :         describe::has_describe_members<T>,
     580              :         mp11::mp_not< std::is_union<T> >,
     581              :         mp11::mp_empty<
     582              :             mp11::mp_eval_or<
     583              :                 mp11::mp_list<>, detail::described_non_public_members, T>>>
     584              : { };
     585              : 
     586              : template<class T>
     587              : struct is_described_enum
     588              :     : describe::has_describe_enumerators<T>
     589              : { };
     590              : 
     591              : template<class T>
     592              : struct is_variant_like : mp11::mp_valid<detail::has_valueless_by_exception, T>
     593              : { };
     594              : 
     595              : template<class T>
     596              : struct is_optional_like
     597              :     : mp11::mp_and<
     598              :         mp11::mp_not<std::is_void<
     599              :             mp11::mp_eval_or<void, detail::value_result_type, T>>>,
     600              :         mp11::mp_valid<detail::can_reset, T>>
     601              : { };
     602              : 
     603              : } // namespace json
     604              : } // namespace boost
     605              : 
     606              : #endif // BOOST_JSON_IMPL_CONVERSION_HPP
        

Generated by: LCOV version 2.1