#pragma once #include #include "../loader/Log.hpp" #include #include #include namespace geode { struct JsonChecker; template struct is_iterable : std::false_type {}; template struct is_iterable< T, std::void_t())), decltype(std::end(std::declval()))>> : std::true_type {}; template constexpr bool is_iterable_v = is_iterable::value; namespace { using value_t = matjson::Type; constexpr char const* jsonValueTypeToString(value_t type) { switch (type) { default: case value_t::Null: return "null"; case value_t::Object: return "object"; case value_t::Array: return "array"; case value_t::String: return "string"; case value_t::Bool: return "boolean"; case value_t::Number: return "number"; } } template constexpr value_t getJsonType() { if constexpr (std::is_same_v) { return value_t::Bool; } else if constexpr (std::is_floating_point_v) { return value_t::Number; } else if constexpr (std::is_unsigned_v) { return value_t::Number; } else if constexpr (std::is_integral_v) { return value_t::Number; } else if constexpr (std::is_constructible_v) { return value_t::String; } else if constexpr (is_iterable_v) { return value_t::Array; } return value_t::Null; } bool jsonConvertibleTo(value_t value, value_t to) { // if we don't know the type we're passing into, // everything's valid if (to == value_t::Null) return true; if (value == value_t::Number) { return to == value_t::Number; } return value == to; } } template using JsonValueValidator = utils::MiniFunction; struct JsonMaybeObject; struct JsonMaybeValue; struct GEODE_DLL JsonMaybeSomething { protected: JsonChecker& m_checker; matjson::Value& m_json; std::string m_hierarchy; bool m_hasValue; friend struct JsonMaybeObject; friend struct JsonMaybeValue; void setError(std::string const& error); public: matjson::Value& json(); JsonMaybeSomething( JsonChecker& checker, matjson::Value& json, std::string const& hierarchy, bool hasValue ); bool isError() const; std::string getError() const; operator bool() const; }; struct GEODE_DLL JsonMaybeValue : public JsonMaybeSomething { bool m_inferType = true; JsonMaybeValue( JsonChecker& checker, matjson::Value& json, std::string const& hierarchy, bool hasValue ); JsonMaybeSomething& self(); template JsonMaybeValue& as() { if (this->isError()) return *this; if (!jsonConvertibleTo(self().m_json.type(), T)) { this->setError( self().m_hierarchy + ": Invalid type \"" + jsonValueTypeToString(self().m_json.type()) + "\", expected \"" + jsonValueTypeToString(T) + "\"" ); } m_inferType = false; return *this; } JsonMaybeValue& array(); template JsonMaybeValue& asOneOf() { if (this->isError()) return *this; bool isOneOf = (... || jsonConvertibleTo(self().m_json.type(), T)); if (!isOneOf) { this->setError( self().m_hierarchy + ": Invalid type \"" + jsonValueTypeToString(self().m_json.type()) + "\", expected one of \"" + (jsonValueTypeToString(T), ...) + "\"" ); } m_inferType = false; return *this; } template bool is() { if (this->isError()) return false; return self().m_json.template is(); } template JsonMaybeValue& validate(JsonValueValidator validator) { if (this->isError()) return *this; if (self().m_json.template is()) { if (!validator(self().m_json.template as())) { this->setError(self().m_hierarchy + ": Invalid value format"); } } else { this->setError( self().m_hierarchy + ": Invalid type \"" + std::string(jsonValueTypeToString(self().m_json.type())) + "\"" ); } return *this; } template JsonMaybeValue& inferType() { if (this->isError() || !m_inferType) return *this; return this->as()>(); } template JsonMaybeValue& intoRaw(T& target) { if (this->isError()) return *this; target = self().m_json; return *this; } template JsonMaybeValue& into(T& target) { return this->intoAs(target); } template JsonMaybeValue& into(std::optional& target) { return this->intoAs>(target); } template JsonMaybeValue& intoAs(T& target) { this->inferType(); if (this->isError()) return *this; if (self().m_json.template is()) { try { target = self().m_json.template as(); } catch(matjson::JsonException const& e) { this->setError( self().m_hierarchy + ": Error parsing JSON: " + std::string(e.what()) ); } } else { this->setError( self().m_hierarchy + ": Invalid type \"" + std::string(jsonValueTypeToString(self().m_json.type())) + "\"" ); } return *this; } template T get() { this->inferType(); if (this->isError()) return T(); if (self().m_json.template is()) { return self().m_json.template as(); } return T(); } JsonMaybeObject obj(); template struct Iterator { std::vector m_values; using iterator = typename std::vector::iterator; using const_iterator = typename std::vector::const_iterator; iterator begin() { return m_values.begin(); } iterator end() { return m_values.end(); } const_iterator begin() const { return m_values.begin(); } const_iterator end() const { return m_values.end(); } }; JsonMaybeValue at(size_t i); Iterator iterate(); Iterator> items(); }; struct GEODE_DLL JsonMaybeObject : JsonMaybeSomething { std::set m_knownKeys; JsonMaybeObject( JsonChecker& checker, matjson::Value& json, std::string const& hierarchy, bool hasValue ); JsonMaybeSomething& self(); void addKnownKey(std::string const& key); matjson::Value& json(); JsonMaybeValue emptyValue(); JsonMaybeValue has(std::string const& key); JsonMaybeValue needs(std::string const& key); void checkUnknownKeys(); }; struct GEODE_DLL JsonChecker { std::variant m_result; matjson::Value& m_json; JsonChecker(matjson::Value& json); bool isError() const; std::string getError() const; JsonMaybeValue root(std::string const& hierarchy); }; }