#pragma once #include <functional> #include <glm/glm.hpp> #include <iosfwd> #include <map> #include <memory> #include <special_members.hpp> #include <stack> #include <stdexcept> #include <string> #include <string_view> #include <type_traits> #include <typeinfo> #include <utility> #include <vector> namespace glm { template<glm::length_t L, typename T, glm::qualifier Q> struct vec; } namespace Persistence { struct Selection; using SelectionPtr = std::unique_ptr<Selection>; using Stack = std::stack<SelectionPtr>; struct Writer { Writer() = default; virtual ~Writer() = default; DEFAULT_MOVE_COPY(Writer); virtual void beginObject() const = 0; virtual void beginArray() const = 0; virtual void pushValue(bool value) const = 0; virtual void pushValue(float value) const = 0; virtual void pushValue(std::nullptr_t) const = 0; virtual void pushValue(std::string_view value) const = 0; virtual void nextValue() const = 0; virtual void pushKey(std::string_view k) const = 0; virtual void endArray() const = 0; virtual void endObject() const = 0; }; struct Selection { Selection() = default; virtual ~Selection() = default; DEFAULT_MOVE_COPY(Selection); virtual void setValue(float); virtual void setValue(bool); virtual void setValue(std::nullptr_t); virtual void setValue(std::string &&); virtual void beginArray(Stack &); virtual void beginObject(Stack &); virtual void endObject(Stack &); virtual void beforeValue(Stack &); [[nodiscard]] virtual SelectionPtr select(const std::string &); [[nodiscard]] virtual bool needsWrite() const; virtual void write(const Writer &) const; }; template<typename T> struct SelectionT; template<typename T> struct SelectionV : public Selection { explicit SelectionV(T & value) : v {value} { } void beforeValue(Stack &) override { } [[nodiscard]] static SelectionPtr make(T & value) { return make_s<SelectionT<T>>(value); } template<typename S> [[nodiscard]] static SelectionPtr make_s(T & value) { return std::make_unique<S>(value); } T & v; }; template<typename T> struct SelectionT : public SelectionV<T> { using SelectionV<T>::SelectionV; using Selection::setValue; using P = std::conditional_t<std::is_scalar_v<T>, T, T &&>; void setValue(P evalue) override { std::swap(this->v, evalue); } void write(const Writer & out) const override { out.pushValue(this->v); } }; struct Persistable; struct PersistenceStore { PersistenceStore() = default; virtual ~PersistenceStore() = default; DEFAULT_MOVE_NO_COPY(PersistenceStore); template<typename T> [[nodiscard]] inline bool persistType(const T * const, const std::type_info & ti); enum class NameAction { Push, HandleAndContinue, Ignore }; template<typename T> [[nodiscard]] inline bool persistValue(const std::string_view key, T & value) { SelectionT<T> s {value}; const auto act {setName(key, s)}; if (act != NameAction::Ignore) { sel = std::make_unique<decltype(s)>(std::move(s)); if (act == NameAction::HandleAndContinue) { selHandler(); } } return (act != NameAction::Push); } virtual NameAction setName(const std::string_view key, const Selection &) = 0; virtual void selHandler() {}; virtual void setType(const std::string_view, const Persistable *) = 0; SelectionPtr sel {}; }; struct PersistenceSelect : public PersistenceStore { explicit PersistenceSelect(const std::string & n); NameAction setName(const std::string_view key, const Selection &) override; void setType(const std::string_view, const Persistable *) override; const std::string & name; }; struct PersistenceWrite : public PersistenceStore { explicit PersistenceWrite(const Writer & o, bool sh); NameAction setName(const std::string_view key, const Selection &) override; void selHandler() override; void setType(const std::string_view tn, const Persistable * p) override; bool first {true}; const Writer & out; bool shared; }; template<glm::length_t L, typename T, glm::qualifier Q> struct SelectionT<glm::vec<L, T, Q>> : public SelectionV<glm::vec<L, T, Q>> { using V = glm::vec<L, T, Q>; struct Members : public SelectionV<V> { using SelectionV<V>::SelectionV; void beforeValue(Stack & stk) override { stk.push(SelectionV<T>::make(this->v[idx++])); } glm::length_t idx {0}; void write(const Writer & out) const override { for (glm::length_t n = 0; n < L; n += 1) { if (n) { out.nextValue(); } SelectionT<T> {this->v[n]}.write(out); } } }; using SelectionV<V>::SelectionV; void beginArray(Stack & stk) override { stk.push(this->template make_s<Members>(this->v)); } void write(const Writer & out) const override { out.beginArray(); Members {this->v}.write(out); out.endArray(); } }; template<typename T> struct SelectionT<std::vector<T>> : public SelectionV<std::vector<T>> { using V = std::vector<T>; struct Members : public SelectionV<V> { using SelectionV<V>::SelectionV; void beforeValue(Stack & stk) override { stk.push(SelectionV<T>::make(this->v.emplace_back())); } void write(const Writer & out) const override { for (auto & member : this->v) { if (&member != &this->v.front()) { out.nextValue(); } SelectionT<T> {member}.write(out); } } }; using SelectionV<V>::SelectionV; void beginArray(Stack & stk) override { stk.push(this->template make_s<Members>(this->v)); } void write(const Writer & out) const override { out.beginArray(); Members {this->v}.write(out); out.endArray(); } }; struct Persistable { Persistable() = default; virtual ~Persistable() = default; DEFAULT_MOVE_COPY(Persistable); virtual bool persist(PersistenceStore & store) = 0; [[nodiscard]] virtual std::string getId() const; template<typename T> [[nodiscard]] constexpr static auto typeName() { constexpr std::string_view name {__PRETTY_FUNCTION__}; constexpr auto s {name.find("T = ") + 4}, e {name.rfind(']')}; return name.substr(s, e - s); } template<typename T> static void addFactory() __attribute__((constructor)); static void addFactory(const std::string_view, std::function<std::unique_ptr<Persistable>()>, std::function<std::shared_ptr<Persistable>()>); [[nodiscard]] static std::unique_ptr<Persistable> callFactory(const std::string_view); [[nodiscard]] static std::shared_ptr<Persistable> callSharedFactory(const std::string_view); }; template<typename T> void Persistable::addFactory() { addFactory(typeName<T>(), std::make_unique<T>, std::make_shared<T>); } template<typename T> inline bool PersistenceStore::persistType(const T * const v, const std::type_info & ti) { if constexpr (!std::is_abstract_v<T>) { [[maybe_unused]] constexpr auto f = &Persistable::addFactory<T>; } if (typeid(std::decay_t<T>) == ti) { setType(Persistable::typeName<T>(), v); } return true; } // TODO Move these using SharedObjects = std::map<std::string, std::shared_ptr<Persistable>>; inline SharedObjects sharedObjects; using SeenSharedObjects = std::map<void *, std::string>; inline SeenSharedObjects seenSharedObjects; template<typename Ptr, bool shared> struct SelectionPtrBase : public SelectionV<Ptr> { using T = typename Ptr::element_type; struct SelectionObj : public SelectionV<Ptr> { struct MakeObjectByTypeName : public SelectionV<Ptr> { using SelectionV<Ptr>::SelectionV; using Selection::setValue; void setValue(std::string && type) override { if constexpr (shared) { auto no = Persistable::callSharedFactory(type); if (auto tno = std::dynamic_pointer_cast<T>(no)) { this->v = std::move(tno); return; } } else { auto no = Persistable::callFactory(type); if (dynamic_cast<T *>(no.get())) { this->v.reset(static_cast<T *>(no.release())); return; } } throw std::runtime_error("Named type doesn't cast to target type"); } }; struct RememberObjectById : public SelectionV<Ptr> { using SelectionV<Ptr>::SelectionV; using Selection::setValue; void setValue(std::string && id) override { sharedObjects.emplace(id, this->v); } }; using SelectionV<Ptr>::SelectionV; [[nodiscard]] SelectionPtr select(const std::string & mbr) override { using namespace std::literals; if (mbr == "@typeid"sv) { if (this->v) { throw std::runtime_error("cannot set object type after creation"); } return this->template make_s<MakeObjectByTypeName>(this->v); } if constexpr (shared) { if (mbr == "@id"sv) { return this->template make_s<RememberObjectById>(this->v); } } make_default_as_needed(this->v); PersistenceSelect ps {mbr}; if (this->v->persist(ps)) { throw std::runtime_error("cannot find member: " + mbr); } return std::move(ps.sel); } void endObject(Stack & stk) override { make_default_as_needed(this->v); stk.pop(); } void write(const Writer & out) const override { out.beginObject(); PersistenceWrite pw {out, shared}; this->v->persist(pw); out.endObject(); } }; static inline void make_default_as_needed(Ptr & v) { if (!v) { if constexpr (std::is_abstract_v<T>) { throw std::runtime_error("cannot select member of null abstract object"); } else if constexpr (shared) { v = std::make_shared<T>(); } else { v = std::make_unique<T>(); } } } using SelectionV<Ptr>::SelectionV; using Selection::setValue; void setValue(std::nullptr_t) override { this->v.reset(); } void beginObject(Stack & stk) override { stk.push(this->template make_s<SelectionObj>(this->v)); } void endObject(Stack & stk) override { stk.pop(); } [[nodiscard]] bool needsWrite() const override { return this->v != nullptr; } void write(const Writer & out) const override { if (this->v) { if constexpr (shared) { if (const auto existing = seenSharedObjects.find(std::to_address(this->v)); existing != seenSharedObjects.end()) { out.pushValue(existing->second); return; } seenSharedObjects.emplace(std::to_address(this->v), this->v->getId()); } SelectionObj {this->v}.write(out); } else { out.pushValue(nullptr); } } }; template<typename T> struct SelectionT<std::unique_ptr<T>> : public SelectionPtrBase<std::unique_ptr<T>, false> { using SelectionPtrBase<std::unique_ptr<T>, false>::SelectionPtrBase; }; template<typename T> struct SelectionT<std::shared_ptr<T>> : public SelectionPtrBase<std::shared_ptr<T>, true> { using SelectionPtrBase<std::shared_ptr<T>, true>::SelectionPtrBase; using SelectionPtrBase<std::shared_ptr<T>, true>::setValue; void setValue(std::string && id) override { if (auto teo = std::dynamic_pointer_cast<T>(sharedObjects.at(id))) { this->v = std::move(teo); } else { throw std::runtime_error("Named type doesn't cast to target type"); } } }; } #define STORE_TYPE store.persistType(this, typeid(*this)) #define STORE_MEMBER(mbr) store.persistValue(#mbr, mbr)