/* * 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_H /// JavaScript Object Notation: formal grammars & tools /// namespace ircd::json { IRCD_EXCEPTION(ircd::error, error); IRCD_EXCEPTION(error, parse_error); IRCD_EXCEPTION(error, print_error); IRCD_EXCEPTION(error, type_error); IRCD_EXCEPTION(error, not_found); struct value; struct member; struct array; struct object; struct vector; struct iov; enum type { STRING = 0, OBJECT = 1, ARRAY = 2, NUMBER = 3, LITERAL = 4, }; enum type type(const string_view &); enum type type(const string_view &, std::nothrow_t); string_view reflect(const enum type &); using name_hash_t = size_t; constexpr name_hash_t name_hash(const char *const name, const size_t len = 0); constexpr name_hash_t name_hash(const string_view &name); constexpr name_hash_t operator ""_(const char *const name, const size_t len); /// Higher order type beyond a string to cleanly delimit multiple keys. using path = std::initializer_list; std::ostream &operator<<(std::ostream &, const path &); /// These templates are generic frontends for building a JSON string. They /// eventually all lead to the stringify() friend function of the argument /// you pass to the template. template string_view stringify(const mutable_buffer &&mb, T&&... t); template size_t print(char *const &buf, const size_t &max, T&&... t); template size_t print(const mutable_buffer &buf, T&&... t); template struct buffer; struct strung; size_t serialized(const string_view &); string_view stringify(mutable_buffer &, const string_view &); struct string; using members = std::initializer_list; // Validate JSON - checks if canonical value. bool valid(const string_view &, std::nothrow_t) noexcept; void valid(const string_view &); // Convert to canonical JSON string_view canonize(const mutable_buffer &out, const string_view &in); std::string canonize(const string_view &in); } /// Strong type representing quoted strings in JSON (which may be unquoted /// automatically when this type is encountered in a tuple etc) struct ircd::json::string :string_view { using string_view::string_view; }; #include "array.h" #include "object.h" #include "vector.h" /// Convenience template to allocate std::string and print() arguments to it. /// struct ircd::json::strung :std::string { template strung(T&&... t); }; #include "value.h" #include "member.h" #include "property.h" #include "iov.h" #include "tuple.h" namespace ircd { using json::operator ""_; using json::operator<<; using json::defined; using json::for_each; using json::until; } template struct ircd::json::buffer :string_view { std::array b; template buffer(T&&... t) :string_view{stringify(b, std::forward(t)...)} {} }; /// /// Convenience template for const rvalue mutable_buffers or basically /// allowing a bracket initialization of a mutable_buffer in the argument /// to stringify(). The const rvalue reference is on purpose. The stringify() /// family of friends all use a non-const lvalue mutable_buffer as an append /// only "stream" buffer. We don't want to modify any non-const instances of /// the mutable_buffer you pass here by offering a `mutable_buffer &` overload /// template ircd::string_view ircd::json::stringify(const mutable_buffer &&mb, T&&... t) { mutable_buffer mbc{mb}; return stringify(mbc, std::forward(t)...); } /// /// Convenience template using the syntax print(mutable_buffer, ...) /// which stringifies with null termination into buffer. /// template size_t ircd::json::print(const mutable_buffer &buf, T&&... t) { return print(data(buf), size(buf), std::forward(t)...); } /// /// Convenience template using the syntax print(buf, sizeof(buf), ...) /// which stringifies with null termination into buffer. /// template size_t ircd::json::print(char *const &buf, const size_t &max, T&&... t) { if(unlikely(!max)) return 0; mutable_buffer mb { buf, max - 1 }; const auto sv { stringify(mb, std::forward(t)...) }; assert(sv.size() < max); assert(valid(sv, std::nothrow)); buf[sv.size()] = '\0'; return sv.size(); } template ircd::json::strung::strung(T&&... t) :std::string{[&]() -> std::string { const auto size { serialized(std::forward(t)...) }; std::string ret(size, char{}); const auto buf{const_cast(ret.data())}; const auto max{ret.size() + 1}; const auto printed { print(buf, max, std::forward(t)...) }; #ifdef RB_DEBUG if(unlikely(printed != ret.size())) std::cerr << printed << " != " << ret.size() << std::endl << ret << std::endl; #endif assert(printed == ret.size()); return ret; }()} { } inline std::ostream & ircd::json::operator<<(std::ostream &s, const path &p) { auto it(std::begin(p)); if(it != std::end(p)) { s << *it; ++it; } for(; it != std::end(p); ++it) s << '.' << *it; return s; } constexpr ircd::json::name_hash_t ircd::json::operator ""_(const char *const text, const size_t len) { return name_hash(text, len); } constexpr ircd::json::name_hash_t ircd::json::name_hash(const string_view &name) { return ircd::hash(name); } constexpr ircd::json::name_hash_t ircd::json::name_hash(const char *const name, const size_t len) { return ircd::hash(name); }