// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_JSON_TUPLE_AT_H namespace ircd { namespace json { template inline enable_if_tuple()> &> at(const tuple &t) { constexpr size_t idx { indexof() }; const auto &ret { val(t) }; if(unlikely(!defined(json::value(ret)))) throw not_found { "%s", key(t) }; return ret; } template inline enable_if_tuple()> &> at(tuple &t) { constexpr size_t idx { indexof() }; auto &ret { val(t) }; if(unlikely(!defined(json::value(ret)))) throw not_found { "%s", key(t) }; return ret; } template inline enable_if_tuple()> &> at(const tuple &t) { return at(t); } template inline enable_if_tuple()> &> at(tuple &t) { return at(t); } template inline enable_if_tuple at(tuple &t, const string_view &name, function&& f) { if constexpr(i < size()) { if(_constexpr_equal(name, key())) f(val(t)); else at(t, name, std::forward(f)); } else throw not_found { "%s", name }; } template inline enable_if_tuple at(const tuple &t, const string_view &name, function&& f) { if constexpr(i < size()) { if(_constexpr_equal(name, key())) f(val(t)); else at(t, name, std::forward(f)); } else throw not_found { "%s", name }; } template inline enable_if_tuple at(const tuple &t, const string_view &name) { const R *ret; at(t, name, [&ret](auto&& val) noexcept { //XXX is_pointer_interconvertible_base_of? (C++20) if constexpr(std::is_assignable()) ret = std::addressof(val); else assert(false); }); return *ret; } template inline enable_if_tuple at(tuple &t, const string_view &name) { R *ret; at(t, name, [&ret] (auto &val) noexcept { //XXX is_pointer_interconvertible_base_of? (C++20) if constexpr(std::is_assignable()) ret = std::addressof(val); else assert(false); }); return *ret; } } // namespace json } // namespace ircd