#ifndef RFL_BSON_READER_HPP_ #define RFL_BSON_READER_HPP_ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../Box.hpp" #include "../Result.hpp" #include "../always_false.hpp" namespace rfl { namespace bson { /// Please refer to https://mongoc.org/libbson/current/api.html struct Reader { struct BSONValue { bson_value_t val_; }; struct BSONInputArray { BSONValue* val_; }; struct BSONInputObject { BSONValue* val_; }; struct BSONInputVar { BSONValue* val_; }; using InputArrayType = BSONInputArray; using InputObjectType = BSONInputObject; using InputVarType = BSONInputVar; template static constexpr bool has_custom_constructor = (requires(InputVarType var) { T::from_bson_obj(var); }); rfl::Result get_field( const std::string& _name, const InputObjectType& _obj) const noexcept { bson_t b; bson_iter_t iter; const auto doc = _obj.val_->val_.value.v_doc; if (bson_init_static(&b, doc.data, doc.data_len)) { if (bson_iter_init(&iter, &b)) { while (bson_iter_next(&iter)) { auto key = std::string(bson_iter_key(&iter)); if (key == _name) { return to_input_var(&iter); } } } } return Error("No field named '" + _name + "' was found."); } bool is_empty(const InputVarType& _var) const noexcept { return _var.val_->val_.value_type == BSON_TYPE_NULL; } template rfl::Result to_basic_type(const InputVarType& _var) const noexcept { const auto btype = _var.val_->val_.value_type; const auto value = _var.val_->val_.value; if constexpr (std::is_same, std::string>()) { switch (btype) { case BSON_TYPE_UTF8: return std::string(value.v_utf8.str, value.v_utf8.len); case BSON_TYPE_SYMBOL: return std::string(value.v_symbol.symbol, value.v_symbol.len); default: return rfl::Error( "Could not cast to string. The type must be UTF8 or symbol."); } } else if constexpr (std::is_same, bool>()) { if (btype != BSON_TYPE_BOOL) { return rfl::Error("Could not cast to boolean."); } return value.v_bool; } else if constexpr (std::is_floating_point>() || std::is_integral>()) { switch (btype) { case BSON_TYPE_DOUBLE: return static_cast(value.v_double); case BSON_TYPE_INT32: return static_cast(value.v_int32); case BSON_TYPE_INT64: return static_cast(value.v_int64); case BSON_TYPE_DATE_TIME: return static_cast(value.v_datetime); default: return rfl::Error( "Could not cast to numeric value. The type must be double, " "int32, int64 or date_time."); } } else if constexpr (std::is_same, bson_oid_t>()) { if (btype != BSON_TYPE_OID) { return rfl::Error("Could not cast to OID."); } return value.v_oid; } else { static_assert(rfl::always_false_v, "Unsupported type."); } } rfl::Result to_array( const InputVarType& _var) const noexcept { const auto btype = _var.val_->val_.value_type; if (btype != BSON_TYPE_ARRAY && btype != BSON_TYPE_DOCUMENT) { return Error("Could not cast to an array."); } return InputArrayType {_var.val_}; } template std::optional read_array( const ArrayReader& _array_reader, const InputArrayType& _arr) const noexcept { bson_t b; bson_iter_t iter; const auto doc = _arr.val_->val_.value.v_doc; if (bson_init_static(&b, doc.data, doc.data_len)) { if (bson_iter_init(&iter, &b)) { while (bson_iter_next(&iter)) { const auto err = _array_reader.read(to_input_var(&iter)); if (err) { return err; } } } } return std::nullopt; } template std::optional read_object( const ObjectReader& _object_reader, const InputObjectType& _obj) const noexcept { bson_t b; bson_iter_t iter; const auto doc = _obj.val_->val_.value.v_doc; if (bson_init_static(&b, doc.data, doc.data_len)) { if (bson_iter_init(&iter, &b)) { while (bson_iter_next(&iter)) { const char* k = bson_iter_key(&iter); _object_reader.read(std::string_view(k), to_input_var(&iter)); } } } return std::nullopt; } rfl::Result to_object( const InputVarType& _var) const noexcept { const auto btype = _var.val_->val_.value_type; if (btype != BSON_TYPE_DOCUMENT) { return Error("Could not cast to a document."); } return InputObjectType {_var.val_}; } template rfl::Result use_custom_constructor( const InputVarType& _var) const noexcept { try { return T::from_bson_obj(_var); } catch (std::exception& e) { return rfl::Error(e.what()); } } private: struct BSONValues { std::vector> vec_; ~BSONValues() { for (auto& v : vec_) { bson_value_destroy(&(v->val_)); } } }; private: InputVarType to_input_var(bson_iter_t* _iter) const noexcept { values_->vec_.emplace_back(rfl::Box::make()); auto* last_value = values_->vec_.back().get(); bson_value_copy(bson_iter_value(_iter), &last_value->val_); return InputVarType {last_value}; } private: /// Contains the values inside the object. rfl::Ref values_; }; } // namespace bson } // namespace rfl #endif // JSON_PARSER_HPP_