draconisplusplus/include/rfl/parsing/TupleParser.hpp

110 lines
3.3 KiB
C++
Raw Normal View History

2024-05-31 22:59:00 -04:00
#ifndef RFL_PARSING_TUPLEPARSER_HPP_
#define RFL_PARSING_TUPLEPARSER_HPP_
#include <map>
#include <tuple>
#include <type_traits>
#include "../Result.hpp"
#include "../always_false.hpp"
#include "Parent.hpp"
#include "TupleReader.hpp"
#include "schema/Type.hpp"
namespace rfl::parsing {
2024-06-08 15:53:06 -04:00
template <
class R,
class W,
bool _ignore_empty_containers,
bool _all_required,
class ProcessorsType,
class... Ts>
2024-06-08 14:10:59 -04:00
requires AreReaderAndWriter<R, W, std::tuple<Ts...>>
struct TupleParser {
public:
using InputArrayType = typename R::InputArrayType;
using InputVarType = typename R::InputVarType;
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
using OutputArrayType = typename W::OutputArrayType;
using OutputVarType = typename W::OutputVarType;
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
using ParentType = Parent<W>;
2024-05-31 22:59:00 -04:00
2024-06-08 15:53:06 -04:00
static Result<std::tuple<Ts...>>
read(const R& _r, const InputVarType& _var) noexcept {
const auto parse = [&](const InputArrayType& _arr
) -> Result<std::tuple<Ts...>> {
2024-06-08 14:10:59 -04:00
alignas(std::tuple<Ts...>) unsigned char buf[sizeof(std::tuple<Ts...>)];
2024-06-08 15:53:06 -04:00
auto ptr = reinterpret_cast<std::tuple<Ts...>*>(buf);
const auto tuple_reader = TupleReader<
R, W, std::tuple<Ts...>, _ignore_empty_containers, _all_required,
ProcessorsType>(&_r, ptr);
2024-06-08 14:10:59 -04:00
auto err = _r.read_array(tuple_reader, _arr);
if (err) {
tuple_reader.call_destructors_where_necessary();
return *err;
}
err = tuple_reader.handle_missing_fields();
if (err) {
tuple_reader.call_destructors_where_necessary();
return *err;
}
return std::move(*ptr);
};
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
return _r.to_array(_var).and_then(parse);
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <class P>
2024-06-08 15:53:06 -04:00
static void write(
const W& _w,
const std::tuple<Ts...>& _tup,
const P& _parent
) noexcept {
auto arr = ParentType::add_array(_w, sizeof...(Ts), _parent);
2024-06-08 14:10:59 -04:00
const auto new_parent = typename ParentType::Array {&arr};
to_array<0>(_w, _tup, new_parent);
_w.end_array(&arr);
}
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
template <size_t _i = 0>
static schema::Type to_schema(
std::map<std::string, schema::Type>* _definitions,
2024-06-08 15:53:06 -04:00
std::vector<schema::Type> _types = {}
) {
2024-06-08 14:10:59 -04:00
using Type = schema::Type;
constexpr size_t size = sizeof...(Ts);
if constexpr (_i == size) {
return Type {Type::Tuple {.types_ = _types}};
} else {
using U =
std::remove_cvref_t<std::tuple_element_t<_i, std::tuple<Ts...>>>;
2024-06-08 15:53:06 -04:00
_types.push_back(Parser<R, W, U, ProcessorsType>::to_schema(_definitions
));
2024-06-08 14:10:59 -04:00
return to_schema<_i + 1>(_definitions, std::move(_types));
}
2024-05-31 22:59:00 -04:00
}
2024-06-08 14:10:59 -04:00
private:
template <int _i, class P>
2024-06-08 15:53:06 -04:00
static void to_array(
const W& _w,
const std::tuple<Ts...>& _tup,
const P& _parent
) noexcept {
2024-06-08 14:10:59 -04:00
if constexpr (_i < sizeof...(Ts)) {
using NewFieldType = std::remove_cvref_t<
typename std::tuple_element_t<_i, std::tuple<Ts...>>>;
Parser<R, W, NewFieldType, ProcessorsType>::write(
2024-06-08 15:53:06 -04:00
_w, std::get<_i>(_tup), _parent
);
2024-06-08 14:10:59 -04:00
to_array<_i + 1>(_w, _tup, _parent);
}
2024-05-31 22:59:00 -04:00
}
2024-06-08 14:10:59 -04:00
};
2024-05-31 22:59:00 -04:00
2024-06-08 14:10:59 -04:00
} // namespace rfl::parsing
2024-05-31 22:59:00 -04:00
#endif