1155 lines
		
	
	
		
			41 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			1155 lines
		
	
	
		
			41 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
								 | 
							
								//     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
							 |