LCOV - code coverage report
Current view: top level - json/impl - conversion.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 100.0 % 18 18
Test Date: 2026-06-16 12:19:39 Functions: 96.5 % 86 83 3

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

Generated by: LCOV version 2.3