0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-11-30 02:32:43 +01:00
construct/include/ircd/json/iov.h

146 lines
3.6 KiB
C
Raw Normal View History

2017-09-09 21:20:00 +02:00
/*
* Copyright (C) 2017 Charybdis Development Team
* Copyright (C) 2017 Jason Volk <jason@zemos.net>
*
* 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_IOV_H
namespace ircd::json
{
struct iov;
}
/// A forward list to compose JSON efficiently on the stack.
///
/// The IOV gathers members for a JSON object being assembled from various
/// sources and presents an iteration to a generator. This prevents the need
/// for multiple generations and copying to occur before the final JSON is
/// realized, if ever.
///
/// Add and remove items on the IOV by construction and destruction one of
/// the node objects. The IOV has a standard forward list interface, only use
/// that to observe and sort/rearrange the IOV. Do not add or remove things
/// that way.
///
/// Nodes support a single member each. To support initializer_list syntax
/// the iov allocates and internally manages the iov node that should have
/// been on your stack.
///
2017-09-09 21:20:00 +02:00
struct ircd::json::iov
:ircd::iov<ircd::json::member>
2017-09-09 21:20:00 +02:00
{
IRCD_EXCEPTION(json::error, error);
IRCD_EXCEPTION(error, exists);
struct push;
struct add;
struct add_if;
struct set;
struct set_if;
2017-09-09 21:20:00 +02:00
private:
std::forward_list<node> allocated;
2017-09-09 21:20:00 +02:00
public:
2017-09-09 21:20:00 +02:00
bool has(const string_view &key) const;
const value &at(const string_view &key) const;
2017-09-09 21:20:00 +02:00
iov() = default;
iov(member);
iov(members);
2017-09-09 21:20:00 +02:00
friend string_view stringify(mutable_buffer &, const iov &);
friend std::ostream &operator<<(std::ostream &, const iov &);
friend size_t serialized(const iov &);
2017-09-09 21:20:00 +02:00
};
struct ircd::json::iov::push
:protected ircd::json::iov::node
2017-09-09 21:20:00 +02:00
{
template<class... args>
push(iov &iov, args&&... a)
:node{iov, std::forward<args>(a)...}
{}
2017-09-09 21:20:00 +02:00
};
struct ircd::json::iov::add
:protected ircd::json::iov::node
2017-09-09 21:20:00 +02:00
{
add(iov &, member);
add() = default;
2017-09-09 21:20:00 +02:00
};
struct ircd::json::iov::add_if
:ircd::json::iov::add
2017-09-09 21:20:00 +02:00
{
add_if(iov &, const bool &, member);
add_if() = default;
2017-09-09 21:20:00 +02:00
};
struct ircd::json::iov::set
:protected ircd::json::iov::node
2017-09-09 21:20:00 +02:00
{
set(iov &, member);
set() = default;
};
2017-09-09 21:20:00 +02:00
struct ircd::json::iov::set_if
:ircd::json::iov::set
2017-09-09 21:20:00 +02:00
{
set_if(iov &, const bool &, member);
set_if() = default;
};
2017-09-09 21:20:00 +02:00
inline
ircd::json::iov::iov(json::member m)
2017-09-09 21:20:00 +02:00
{
allocated.emplace_front(*this, std::move(m));
2017-09-09 21:20:00 +02:00
}
inline
ircd::json::iov::iov(members m)
2017-09-09 21:20:00 +02:00
{
for(auto&& member : m)
allocated.emplace_front(*this, std::move(member));
2017-09-09 21:20:00 +02:00
}
inline size_t
ircd::json::serialized(const iov &iov)
2017-09-09 21:20:00 +02:00
{
const size_t ret
{
1U + !iov.empty()
};
2017-09-09 21:20:00 +02:00
return std::accumulate(std::begin(iov), std::end(iov), ret, []
(auto ret, const auto &member)
{
return ret += serialized(member);
});
2017-09-09 21:20:00 +02:00
}
inline std::ostream &
ircd::json::operator<<(std::ostream &s, const iov &iov)
2017-09-09 21:20:00 +02:00
{
s << string(iov);
return s;
2017-09-09 21:20:00 +02:00
}