/* * Copyright (C) 2017 Charybdis Development Team * Copyright (C) 2017 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. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #pragma once #define HAVE_IRCD_JSON_MEMBER_H // json::member is a pair of values. The key value (member.first) should always // be a STRING type. We don't use string_view directly in member.first because // json::value can take ownership of a string or use a literal depending on // the circumstance and it's more consistent this way. // // json::member, like json::value, is a runtime construct though still very // lightweight and useful for non-deterministic composition to and extraction // from JSON strings. // namespace ircd::json { struct member; using members = std::initializer_list; size_t serialized(const member *const &begin, const member *const &end); size_t serialized(const members &); string_view stringify(mutable_buffer &, const member *const *const &begin, const member *const *const &end); string_view stringify(mutable_buffer &, const member *const &begin, const member *const &end); string_view stringify(mutable_buffer &, const members &); string_view stringify(mutable_buffer &, const std::vector &); } struct ircd::json::member :std::pair { using std::pair::pair; template member(const K &k, std::initializer_list v); template member(const K &k, V&& v); explicit member(const string_view &k); explicit member(const object::member &m); member() = default; friend bool operator==(const member &a, const member &b); friend bool operator!=(const member &a, const member &b); friend bool operator<(const member &a, const member &b); friend bool operator==(const member &a, const string_view &b); friend bool operator!=(const member &a, const string_view &b); friend bool operator<(const member &a, const string_view &b); friend size_t serialized(const member &); friend string_view stringify(mutable_buffer &, const member &); friend std::ostream &operator<<(std::ostream &, const member &); }; template ircd::json::member::member(const K &k, V&& v) :std::pair { value { k }, value { std::forward(v) } } {} template ircd::json::member::member(const K &k, std::initializer_list v) :std::pair { value { k }, value { std::make_unique(std::move(v)) } } {} inline ircd::json::member::member(const object::member &m) :std::pair { m.first, value { m.second, type(m.second) } } {} inline ircd::json::member::member(const string_view &k) :std::pair { k, string_view{} } {} inline bool ircd::json::operator<(const member &a, const member &b) { return a.first < b.first; } inline bool ircd::json::operator!=(const member &a, const member &b) { return a.first != b.first; } inline bool ircd::json::operator==(const member &a, const member &b) { return a.first == b.first; } inline bool ircd::json::operator<(const member &a, const string_view &b) { return string_view{a.first.string, a.first.len} < b; } inline bool ircd::json::operator!=(const member &a, const string_view &b) { return string_view{a.first.string, a.first.len} != b; } inline bool ircd::json::operator==(const member &a, const string_view &b) { return string_view{a.first.string, a.first.len} == b; }