Mercurial > minori
view dep/toml11/toml/get.hpp @ 337:a7d4e5107531
dep/animone: REFACTOR ALL THE THINGS
1: animone now has its own syntax divergent from anisthesia,
making different platforms actually have their own sections
2: process names in animone are now called `comm' (this will
probably break things). this is what its called in bsd/linux
so I'm just going to use it everywhere
3: the X11 code now checks for the existence of a UTF-8 window title
and passes it if available
4: ANYTHING THATS NOT LINUX IS 100% UNTESTED AND CAN AND WILL BREAK!
I still actually need to test the bsd code. to be honest I'm probably
going to move all of the bsds into separate files because they're
all essentially different operating systems at this point
author | Paper <paper@paper.us.eu.org> |
---|---|
date | Wed, 19 Jun 2024 12:51:15 -0400 |
parents | 3b355fa948c7 |
children |
line wrap: on
line source
// Copyright Toru Niina 2017. // Distributed under the MIT License. #ifndef TOML11_GET_HPP #define TOML11_GET_HPP #include <algorithm> #include "from.hpp" #include "result.hpp" #include "value.hpp" namespace toml { // ============================================================================ // exact toml::* type template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T> & get(basic_value<C, M, V>& v) { return v.template cast<detail::type_to_enum<T, basic_value<C, M, V>>::value>(); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T> const& get(const basic_value<C, M, V>& v) { return v.template cast<detail::type_to_enum<T, basic_value<C, M, V>>::value>(); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T> get(basic_value<C, M, V>&& v) { return T(std::move(v).template cast<detail::type_to_enum<T, basic_value<C, M, V>>::value>()); } // ============================================================================ // T == toml::value; identity transformation. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, basic_value<C, M, V>>::value, T>& get(basic_value<C, M, V>& v) { return v; } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, basic_value<C, M, V>>::value, T> const& get(const basic_value<C, M, V>& v) { return v; } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, basic_value<C, M, V>>::value, T> get(basic_value<C, M, V>&& v) { return basic_value<C, M, V>(std::move(v)); } // ============================================================================ // T == toml::basic_value<C2, M2, V2>; basic_value -> basic_value template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<detail::conjunction<detail::is_basic_value<T>, detail::negation<std::is_same<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { return T(v); } // ============================================================================ // integer convertible from toml::Integer template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<detail::conjunction< std::is_integral<T>, // T is integral detail::negation<std::is_same<T, bool>>, // but not bool detail::negation< // but not toml::integer detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { return static_cast<T>(v.as_integer()); } // ============================================================================ // floating point convertible from toml::Float template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<detail::conjunction< std::is_floating_point<T>, // T is floating_point detail::negation< // but not toml::floating detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { return static_cast<T>(v.as_floating()); } // ============================================================================ // std::string; toml uses its own toml::string, but it should be convertible to // std::string seamlessly template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, std::string>::value, std::string>& get(basic_value<C, M, V>& v) { return v.as_string().str; } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, std::string>::value, std::string> const& get(const basic_value<C, M, V>& v) { return v.as_string().str; } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, std::string>::value, std::string> get(basic_value<C, M, V>&& v) { return std::string(std::move(v.as_string().str)); } // ============================================================================ // std::string_view #if defined(TOML11_USING_STRING_VIEW) && TOML11_USING_STRING_VIEW>0 template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<std::is_same<T, std::string_view>::value, std::string_view> get(const basic_value<C, M, V>& v) { return std::string_view(v.as_string().str); } #endif // ============================================================================ // std::chrono::duration from toml::local_time. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t<detail::is_chrono_duration<T>::value, T> get(const basic_value<C, M, V>& v) { return std::chrono::duration_cast<T>( std::chrono::nanoseconds(v.as_local_time())); } // ============================================================================ // std::chrono::system_clock::time_point from toml::datetime variants template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> inline detail::enable_if_t< std::is_same<std::chrono::system_clock::time_point, T>::value, T> get(const basic_value<C, M, V>& v) { switch(v.type()) { case value_t::local_date: { return std::chrono::system_clock::time_point(v.as_local_date()); } case value_t::local_datetime: { return std::chrono::system_clock::time_point(v.as_local_datetime()); } case value_t::offset_datetime: { return std::chrono::system_clock::time_point(v.as_offset_datetime()); } default: { throw type_error(detail::format_underline("toml::value: " "bad_cast to std::chrono::system_clock::time_point", { {v.location(), concat_to_string("the actual type is ", v.type())} }), v.location()); } } } // ============================================================================ // forward declaration to use this recursively. ignore this and go ahead. // array-like type with push_back(value) method template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_container<T>, // T is a container detail::has_push_back_method<T>, // T::push_back(value) works detail::negation< // but not toml::array detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>&); // array-like type without push_back(value) method template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_container<T>, // T is a container detail::negation<detail::has_push_back_method<T>>, // w/o push_back(...) detail::negation<detail::has_specialized_from<T>>, // T does not have special conversion detail::negation< // not toml::array detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>&); // std::pair<T1, T2> template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_std_pair<T>::value, T> get(const basic_value<C, M, V>&); // std::tuple<T1, T2, ...> template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_std_tuple<T>::value, T> get(const basic_value<C, M, V>&); // map-like classes template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_map<T>, // T is map detail::negation< // but not toml::table detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>&); // T.from_toml(v) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation< // not a toml::* type detail::is_exact_toml_type<T, basic_value<C, M, V>>>, detail::has_from_toml_method<T, C, M, V>, // but has from_toml(toml::value) std::is_default_constructible<T> // and default constructible >::value, T> get(const basic_value<C, M, V>&); // toml::from<T>::from_toml(v) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::has_specialized_from<T>::value, T> get(const basic_value<C, M, V>&); template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::has_specialized_from<T>::value, T> get(basic_value<C, M, V>&); // T(const toml::value&) and T is not toml::basic_value, // and it does not have `from<T>` nor `from_toml`. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation<detail::is_basic_value<T>>, std::is_constructible<T, const basic_value<C, M, V>&>, detail::negation<detail::has_from_toml_method<T, C, M, V>>, detail::negation<detail::has_specialized_from<T>> >::value, T> get(const basic_value<C, M, V>&); template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation<detail::is_basic_value<T>>, std::is_constructible<T, basic_value<C, M, V>&>, detail::negation<detail::has_from_toml_method<T, C, M, V>>, detail::negation<detail::has_specialized_from<T>> >::value, T> get(basic_value<C, M, V>&); // ============================================================================ // array-like types; most likely STL container, like std::vector, etc. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_container<T>, // T is a container detail::has_push_back_method<T>, // container.push_back(elem) works detail::negation< // but not toml::array detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { using value_type = typename T::value_type; const auto& ary = v.as_array(); T container; try_reserve(container, ary.size()); for(const auto& elem : ary) { container.push_back(get<value_type>(elem)); } return container; } // ============================================================================ // std::forward_list does not have push_back, insert, or emplace. // It has insert_after, emplace_after, push_front. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_std_forward_list<T>::value, T> get(const basic_value<C, M, V>& v) { using value_type = typename T::value_type; T container; for(const auto& elem : v.as_array()) { container.push_front(get<value_type>(elem)); } container.reverse(); return container; } // ============================================================================ // array-like types, without push_back(). most likely [std|boost]::array. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_container<T>, // T is a container detail::negation<detail::has_push_back_method<T>>, // w/o push_back detail::negation<detail::has_specialized_from<T>>, // T does not have special conversion detail::negation< // T is not toml::array detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { using value_type = typename T::value_type; const auto& ar = v.as_array(); T container; if(ar.size() != container.size()) { throw std::out_of_range(detail::format_underline(concat_to_string( "toml::get: specified container size is ", container.size(), " but there are ", ar.size(), " elements in toml array."), { {v.location(), "here"} })); } for(std::size_t i=0; i<ar.size(); ++i) { container[i] = ::toml::get<value_type>(ar[i]); } return container; } // ============================================================================ // std::pair. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_std_pair<T>::value, T> get(const basic_value<C, M, V>& v) { using first_type = typename T::first_type; using second_type = typename T::second_type; const auto& ar = v.as_array(); if(ar.size() != 2) { throw std::out_of_range(detail::format_underline(concat_to_string( "toml::get: specified std::pair but there are ", ar.size(), " elements in toml array."), {{v.location(), "here"}})); } return std::make_pair(::toml::get<first_type >(ar.at(0)), ::toml::get<second_type>(ar.at(1))); } // ============================================================================ // std::tuple. namespace detail { template<typename T, typename Array, std::size_t ... I> T get_tuple_impl(const Array& a, index_sequence<I...>) { return std::make_tuple( ::toml::get<typename std::tuple_element<I, T>::type>(a.at(I))...); } } // detail template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_std_tuple<T>::value, T> get(const basic_value<C, M, V>& v) { const auto& ar = v.as_array(); if(ar.size() != std::tuple_size<T>::value) { throw std::out_of_range(detail::format_underline(concat_to_string( "toml::get: specified std::tuple with ", std::tuple_size<T>::value, " elements, but there are ", ar.size(), " elements in toml array."), {{v.location(), "here"}})); } return detail::get_tuple_impl<T>(ar, detail::make_index_sequence<std::tuple_size<T>::value>{}); } // ============================================================================ // map-like types; most likely STL map, like std::map or std::unordered_map. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::is_map<T>, // T is map detail::negation< // but not toml::array detail::is_exact_toml_type<T, basic_value<C, M, V>>> >::value, T> get(const basic_value<C, M, V>& v) { using key_type = typename T::key_type; using mapped_type = typename T::mapped_type; static_assert(std::is_convertible<std::string, key_type>::value, "toml::get only supports map type of which key_type is " "convertible from std::string."); T map; for(const auto& kv : v.as_table()) { map.emplace(key_type(kv.first), get<mapped_type>(kv.second)); } return map; } // ============================================================================ // user-defined, but compatible types. template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation< // not a toml::* type detail::is_exact_toml_type<T, basic_value<C, M, V>>>, detail::has_from_toml_method<T, C, M, V>, // but has from_toml(toml::value) memfn std::is_default_constructible<T> // and default constructible >::value, T> get(const basic_value<C, M, V>& v) { T ud; ud.from_toml(v); return ud; } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::has_specialized_from<T>::value, T> get(const basic_value<C, M, V>& v) { return ::toml::from<T>::from_toml(v); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::has_specialized_from<T>::value, T> get(basic_value<C, M, V>& v) { return ::toml::from<T>::from_toml(v); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation<detail::is_basic_value<T>>, // T is not a toml::value std::is_constructible<T, const basic_value<C, M, V>&>, // T is constructible from toml::value detail::negation<detail::has_from_toml_method<T, C, M, V>>, // and T does not have T.from_toml(v); detail::negation<detail::has_specialized_from<T>> // and T does not have toml::from<T>{}; >::value, T> get(const basic_value<C, M, V>& v) { return T(v); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation<detail::is_basic_value<T>>, // T is not a toml::value std::is_constructible<T, basic_value<C, M, V>&>, // T is constructible from toml::value detail::negation<detail::has_from_toml_method<T, C, M, V>>, // and T does not have T.from_toml(v); detail::negation<detail::has_specialized_from<T>> // and T does not have toml::from<T>{}; >::value, T> get(basic_value<C, M, V>& v) { return T(v); } // ============================================================================ // find // ---------------------------------------------------------------------------- // these overloads do not require to set T. and returns value itself. template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> const& find(const basic_value<C, M, V>& v, const key& ky) { const auto& tab = v.as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return tab.at(ky); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V>& find(basic_value<C, M, V>& v, const key& ky) { auto& tab = v.as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return tab.at(ky); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> find(basic_value<C, M, V>&& v, const key& ky) { typename basic_value<C, M, V>::table_type tab = std::move(v).as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return basic_value<C, M, V>(std::move(tab.at(ky))); } // ---------------------------------------------------------------------------- // find(value, idx) template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> const& find(const basic_value<C, M, V>& v, const std::size_t idx) { const auto& ary = v.as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return ary.at(idx); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V>& find(basic_value<C, M, V>& v, const std::size_t idx) { auto& ary = v.as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return ary.at(idx); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> find(basic_value<C, M, V>&& v, const std::size_t idx) { auto& ary = v.as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return basic_value<C, M, V>(std::move(ary.at(idx))); } // ---------------------------------------------------------------------------- // find<T>(value, key); template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V> const&>())) find(const basic_value<C, M, V>& v, const key& ky) { const auto& tab = v.as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return ::toml::get<T>(tab.at(ky)); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&>())) find(basic_value<C, M, V>& v, const key& ky) { auto& tab = v.as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return ::toml::get<T>(tab.at(ky)); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&&>())) find(basic_value<C, M, V>&& v, const key& ky) { typename basic_value<C, M, V>::table_type tab = std::move(v).as_table(); if(tab.count(ky) == 0) { detail::throw_key_not_found_error(v, ky); } return ::toml::get<T>(std::move(tab.at(ky))); } // ---------------------------------------------------------------------------- // find<T>(value, idx) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V> const&>())) find(const basic_value<C, M, V>& v, const std::size_t idx) { const auto& ary = v.as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return ::toml::get<T>(ary.at(idx)); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&>())) find(basic_value<C, M, V>& v, const std::size_t idx) { auto& ary = v.as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return ::toml::get<T>(ary.at(idx)); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&&>())) find(basic_value<C, M, V>&& v, const std::size_t idx) { typename basic_value<C, M, V>::array_type ary = std::move(v).as_array(); if(ary.size() <= idx) { throw std::out_of_range(detail::format_underline(concat_to_string( "index ", idx, " is out of range"), {{v.location(), "in this array"}})); } return ::toml::get<T>(std::move(ary.at(idx))); } // -------------------------------------------------------------------------- // toml::find(toml::value, toml::key, Ts&& ... keys) namespace detail { // It suppresses warnings by -Wsign-conversion. Let's say we have the following // code. // ```cpp // const auto x = toml::find<std::string>(data, "array", 0); // ``` // Here, the type of literal number `0` is `int`. `int` is a signed integer. // `toml::find` takes `std::size_t` as an index. So it causes implicit sign // conversion and `-Wsign-conversion` warns about it. Using `0u` instead of `0` // suppresses the warning, but it makes user code messy. // To suppress this warning, we need to be aware of type conversion caused // by `toml::find(v, key1, key2, ... keys)`. But the thing is that the types of // keys can be any combination of {string-like, size_t-like}. Of course we can't // write down all the combinations. Thus we need to use some function that // recognize the type of argument and cast it into `std::string` or // `std::size_t` depending on the context. // `key_cast` does the job. It has 2 overloads. One is invoked when the // argument type is an integer and cast the argument into `std::size_t`. The // other is invoked when the argument type is not an integer, possibly one of // std::string, const char[N] or const char*, and construct std::string from // the argument. // `toml::find(v, k1, k2, ... ks)` uses `key_cast` before passing `ks` to // `toml::find(v, k)` to suppress -Wsign-conversion. template<typename T> enable_if_t<conjunction<std::is_integral<remove_cvref_t<T>>, negation<std::is_same<remove_cvref_t<T>, bool>>>::value, std::size_t> key_cast(T&& v) noexcept { return std::size_t(v); } template<typename T> enable_if_t<negation<conjunction<std::is_integral<remove_cvref_t<T>>, negation<std::is_same<remove_cvref_t<T>, bool>>>>::value, std::string> key_cast(T&& v) noexcept { return std::string(std::forward<T>(v)); } } // detail template<typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> const basic_value<C, M, V>& find(const basic_value<C, M, V>& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find(::toml::find(v, detail::key_cast(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } template<typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> basic_value<C, M, V>& find(basic_value<C, M, V>& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find(::toml::find(v, detail::key_cast(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } template<typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> basic_value<C, M, V> find(basic_value<C, M, V>&& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find(::toml::find(std::move(v), std::forward<Key1>(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> decltype(::toml::get<T>(std::declval<const basic_value<C, M, V>&>())) find(const basic_value<C, M, V>& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find<T>(::toml::find(v, detail::key_cast(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&>())) find(basic_value<C, M, V>& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find<T>(::toml::find(v, detail::key_cast(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V, typename Key1, typename Key2, typename ... Keys> decltype(::toml::get<T>(std::declval<basic_value<C, M, V>&&>())) find(basic_value<C, M, V>&& v, Key1&& k1, Key2&& k2, Keys&& ... keys) { return ::toml::find<T>(::toml::find(std::move(v), detail::key_cast(k1)), detail::key_cast(k2), std::forward<Keys>(keys)...); } // ============================================================================ // get_or(value, fallback) template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> const& get_or(const basic_value<C, M, V>& v, const basic_value<C, M, V>&) { return v; } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V>& get_or(basic_value<C, M, V>& v, basic_value<C, M, V>&) { return v; } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> get_or(basic_value<C, M, V>&& v, basic_value<C, M, V>&&) { return v; } // ---------------------------------------------------------------------------- // specialization for the exact toml types (return type becomes lvalue ref) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T> const& get_or(const basic_value<C, M, V>& v, const T& opt) { try { return get<detail::remove_cvref_t<T>>(v); } catch(...) { return opt; } } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T>& get_or(basic_value<C, M, V>& v, T& opt) { try { return get<detail::remove_cvref_t<T>>(v); } catch(...) { return opt; } } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_exact_toml_type<detail::remove_cvref_t<T>, basic_value<C, M, V>>::value, detail::remove_cvref_t<T>> get_or(basic_value<C, M, V>&& v, T&& opt) { try { return get<detail::remove_cvref_t<T>>(std::move(v)); } catch(...) { return detail::remove_cvref_t<T>(std::forward<T>(opt)); } } // ---------------------------------------------------------------------------- // specialization for std::string (return type becomes lvalue ref) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<std::is_same<detail::remove_cvref_t<T>, std::string>::value, std::string> const& get_or(const basic_value<C, M, V>& v, const T& opt) { try { return v.as_string().str; } catch(...) { return opt; } } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<std::is_same<T, std::string>::value, std::string>& get_or(basic_value<C, M, V>& v, T& opt) { try { return v.as_string().str; } catch(...) { return opt; } } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< std::is_same<detail::remove_cvref_t<T>, std::string>::value, std::string> get_or(basic_value<C, M, V>&& v, T&& opt) { try { return std::move(v.as_string().str); } catch(...) { return std::string(std::forward<T>(opt)); } } // ---------------------------------------------------------------------------- // specialization for string literal template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::is_string_literal< typename std::remove_reference<T>::type>::value, std::string> get_or(const basic_value<C, M, V>& v, T&& opt) { try { return std::move(v.as_string().str); } catch(...) { return std::string(std::forward<T>(opt)); } } // ---------------------------------------------------------------------------- // others (require type conversion and return type cannot be lvalue reference) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< detail::negation<detail::is_exact_toml_type<detail::remove_cvref_t<T>, basic_value<C, M, V>>>, detail::negation<std::is_same<std::string, detail::remove_cvref_t<T>>>, detail::negation<detail::is_string_literal< typename std::remove_reference<T>::type>> >::value, detail::remove_cvref_t<T>> get_or(const basic_value<C, M, V>& v, T&& opt) { try { return get<detail::remove_cvref_t<T>>(v); } catch(...) { return detail::remove_cvref_t<T>(std::forward<T>(opt)); } } // =========================================================================== // find_or(value, key, fallback) template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> const& find_or(const basic_value<C, M, V>& v, const key& ky, const basic_value<C, M, V>& opt) { if(!v.is_table()) {return opt;} const auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return tab.at(ky); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V>& find_or(basic_value<C, M, V>& v, const toml::key& ky, basic_value<C, M, V>& opt) { if(!v.is_table()) {return opt;} auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return tab.at(ky); } template<typename C, template<typename ...> class M, template<typename ...> class V> basic_value<C, M, V> find_or(basic_value<C, M, V>&& v, const toml::key& ky, basic_value<C, M, V>&& opt) { if(!v.is_table()) {return opt;} auto tab = std::move(v).as_table(); if(tab.count(ky) == 0) {return opt;} return basic_value<C, M, V>(std::move(tab.at(ky))); } // --------------------------------------------------------------------------- // exact types (return type can be a reference) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T> const& find_or(const basic_value<C, M, V>& v, const key& ky, const T& opt) { if(!v.is_table()) {return opt;} const auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return get_or(tab.at(ky), opt); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, T>& find_or(basic_value<C, M, V>& v, const toml::key& ky, T& opt) { if(!v.is_table()) {return opt;} auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return get_or(tab.at(ky), opt); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_exact_toml_type<T, basic_value<C, M, V>>::value, detail::remove_cvref_t<T>> find_or(basic_value<C, M, V>&& v, const toml::key& ky, T&& opt) { if(!v.is_table()) {return std::forward<T>(opt);} auto tab = std::move(v).as_table(); if(tab.count(ky) == 0) {return std::forward<T>(opt);} return get_or(std::move(tab.at(ky)), std::forward<T>(opt)); } // --------------------------------------------------------------------------- // std::string (return type can be a reference) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<std::is_same<T, std::string>::value, std::string> const& find_or(const basic_value<C, M, V>& v, const key& ky, const T& opt) { if(!v.is_table()) {return opt;} const auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return get_or(tab.at(ky), opt); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<std::is_same<T, std::string>::value, std::string>& find_or(basic_value<C, M, V>& v, const toml::key& ky, T& opt) { if(!v.is_table()) {return opt;} auto& tab = v.as_table(); if(tab.count(ky) == 0) {return opt;} return get_or(tab.at(ky), opt); } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<std::is_same<T, std::string>::value, std::string> find_or(basic_value<C, M, V>&& v, const toml::key& ky, T&& opt) { if(!v.is_table()) {return std::forward<T>(opt);} auto tab = std::move(v).as_table(); if(tab.count(ky) == 0) {return std::forward<T>(opt);} return get_or(std::move(tab.at(ky)), std::forward<T>(opt)); } // --------------------------------------------------------------------------- // string literal (deduced as std::string) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t< detail::is_string_literal<typename std::remove_reference<T>::type>::value, std::string> find_or(const basic_value<C, M, V>& v, const toml::key& ky, T&& opt) { if(!v.is_table()) {return std::string(opt);} const auto& tab = v.as_table(); if(tab.count(ky) == 0) {return std::string(opt);} return get_or(tab.at(ky), std::forward<T>(opt)); } // --------------------------------------------------------------------------- // others (require type conversion and return type cannot be lvalue reference) template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> detail::enable_if_t<detail::conjunction< // T is not an exact toml type detail::negation<detail::is_exact_toml_type< detail::remove_cvref_t<T>, basic_value<C, M, V>>>, // T is not std::string detail::negation<std::is_same<std::string, detail::remove_cvref_t<T>>>, // T is not a string literal detail::negation<detail::is_string_literal< typename std::remove_reference<T>::type>> >::value, detail::remove_cvref_t<T>> find_or(const basic_value<C, M, V>& v, const toml::key& ky, T&& opt) { if(!v.is_table()) {return std::forward<T>(opt);} const auto& tab = v.as_table(); if(tab.count(ky) == 0) {return std::forward<T>(opt);} return get_or(tab.at(ky), std::forward<T>(opt)); } // --------------------------------------------------------------------------- // recursive find-or with type deduction (find_or(value, keys, opt)) template<typename Value, typename ... Ks, typename detail::enable_if_t<(sizeof...(Ks) > 1), std::nullptr_t> = nullptr> // here we need to add SFINAE in the template parameter to avoid // infinite recursion in type deduction on gcc auto find_or(Value&& v, const toml::key& ky, Ks&& ... keys) -> decltype(find_or(std::forward<Value>(v), ky, detail::last_one(std::forward<Ks>(keys)...))) { if(!v.is_table()) { return detail::last_one(std::forward<Ks>(keys)...); } auto&& tab = std::forward<Value>(v).as_table(); if(tab.count(ky) == 0) { return detail::last_one(std::forward<Ks>(keys)...); } return find_or(std::forward<decltype(tab)>(tab).at(ky), std::forward<Ks>(keys)...); } // --------------------------------------------------------------------------- // recursive find_or with explicit type specialization, find_or<int>(value, keys...) template<typename T, typename Value, typename ... Ks, typename detail::enable_if_t<(sizeof...(Ks) > 1), std::nullptr_t> = nullptr> // here we need to add SFINAE in the template parameter to avoid // infinite recursion in type deduction on gcc auto find_or(Value&& v, const toml::key& ky, Ks&& ... keys) -> decltype(find_or<T>(std::forward<Value>(v), ky, detail::last_one(std::forward<Ks>(keys)...))) { if(!v.is_table()) { return detail::last_one(std::forward<Ks>(keys)...); } auto&& tab = std::forward<Value>(v).as_table(); if(tab.count(ky) == 0) { return detail::last_one(std::forward<Ks>(keys)...); } return find_or(std::forward<decltype(tab)>(tab).at(ky), std::forward<Ks>(keys)...); } // ============================================================================ // expect template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> result<T, std::string> expect(const basic_value<C, M, V>& v) noexcept { try { return ok(get<T>(v)); } catch(const std::exception& e) { return err(e.what()); } } template<typename T, typename C, template<typename ...> class M, template<typename ...> class V> result<T, std::string> expect(const basic_value<C, M, V>& v, const toml::key& k) noexcept { try { return ok(find<T>(v, k)); } catch(const std::exception& e) { return err(e.what()); } } } // toml #endif// TOML11_GET