relation_completer progress
This commit is contained in:
parent
5a7ab5d4d0
commit
38cbffc18b
|
|
@ -12,6 +12,7 @@
|
||||||
namespace matador::object {
|
namespace matador::object {
|
||||||
|
|
||||||
class schema_node;
|
class schema_node;
|
||||||
|
class relation_endpoint;
|
||||||
|
|
||||||
class basic_object_info {
|
class basic_object_info {
|
||||||
public:
|
public:
|
||||||
|
|
@ -26,7 +27,7 @@ public:
|
||||||
[[nodiscard]] const object_definition& definition() const;
|
[[nodiscard]] const object_definition& definition() const;
|
||||||
[[nodiscard]] std::shared_ptr<attribute_definition> reference_column() const;
|
[[nodiscard]] std::shared_ptr<attribute_definition> reference_column() const;
|
||||||
|
|
||||||
bool has_primary_key() const;
|
[[nodiscard]] bool has_primary_key() const;
|
||||||
[[nodiscard]] const utils::identifier& primary_key() const;
|
[[nodiscard]] const utils::identifier& primary_key() const;
|
||||||
|
|
||||||
void register_relation_endpoint(const std::type_index &type, const relation_endpoint &endpoint);
|
void register_relation_endpoint(const std::type_index &type, const relation_endpoint &endpoint);
|
||||||
|
|
|
||||||
|
|
@ -33,5 +33,30 @@ private:
|
||||||
object_ptr<ForeignType> remote_;
|
object_ptr<ForeignType> remote_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template < class LocalType, class Type >
|
||||||
|
class many_to_relation {
|
||||||
|
public:
|
||||||
|
many_to_relation() = default;
|
||||||
|
many_to_relation(std::string local_name, std::string remote_name)
|
||||||
|
: local_name_(std::move(local_name))
|
||||||
|
, type_name_(std::move(remote_name)) {}
|
||||||
|
|
||||||
|
template<class Operator>
|
||||||
|
void process(Operator &op) {
|
||||||
|
namespace field = matador::access;
|
||||||
|
field::belongs_to(op, local_name_.c_str(), local_, utils::default_foreign_attributes);
|
||||||
|
field::attribute(op, type_name_.c_str(), value_);
|
||||||
|
}
|
||||||
|
|
||||||
|
object_ptr<LocalType> local() const { return local_; }
|
||||||
|
Type value() const { return value_; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string local_name_;
|
||||||
|
std::string type_name_;
|
||||||
|
object_ptr<LocalType> local_;
|
||||||
|
Type value_;
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
#endif //QUERY_HAS_MANY_TO_MANY_RELATION_HPP
|
#endif //QUERY_HAS_MANY_TO_MANY_RELATION_HPP
|
||||||
|
|
|
||||||
|
|
@ -10,21 +10,33 @@ class schema_node;
|
||||||
template<typename Type>
|
template<typename Type>
|
||||||
class object_info final : public basic_object_info {
|
class object_info final : public basic_object_info {
|
||||||
public:
|
public:
|
||||||
explicit object_info(const std::shared_ptr<schema_node>& node,
|
using create_func = std::function<std::unique_ptr<Type>()>;
|
||||||
std::shared_ptr<attribute_definition> &&ref_column)
|
|
||||||
|
object_info(const std::shared_ptr<schema_node>& node,
|
||||||
|
std::shared_ptr<attribute_definition> &&ref_column)
|
||||||
: basic_object_info(node, typeid(Type), {}, std::move(ref_column), {}) {
|
: basic_object_info(node, typeid(Type), {}, std::move(ref_column), {}) {
|
||||||
}
|
}
|
||||||
explicit object_info(const std::shared_ptr<schema_node>& node,
|
object_info(const std::shared_ptr<schema_node>& node,
|
||||||
utils::identifier &&pk,
|
utils::identifier &&pk,
|
||||||
std::shared_ptr<attribute_definition> &&ref_column,
|
std::shared_ptr<attribute_definition> &&ref_column,
|
||||||
object_definition &&definition)
|
object_definition &&definition)
|
||||||
: basic_object_info(node, typeid(Type), std::move(pk), std::move(ref_column), std::move(definition)) {
|
: basic_object_info(node, typeid(Type), std::move(pk), std::move(ref_column), std::move(definition)) {
|
||||||
}
|
}
|
||||||
|
object_info(const std::shared_ptr<schema_node>& node,
|
||||||
|
utils::identifier &&pk,
|
||||||
|
std::shared_ptr<attribute_definition> &&ref_column,
|
||||||
|
object_definition &&definition,
|
||||||
|
create_func&& creator)
|
||||||
|
: basic_object_info(node, typeid(Type), std::move(pk), std::move(ref_column), std::move(definition))
|
||||||
|
, creator_(std::move(creator)){
|
||||||
|
}
|
||||||
|
|
||||||
const Type &prototype() const { return prototype_; }
|
const Type &prototype() const { return prototype_; }
|
||||||
|
std::unique_ptr<Type> create() const { return creator_(); }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Type prototype_;
|
Type prototype_;
|
||||||
|
create_func creator_{[]{ return std::make_unique<Type>(); }};
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename Type>
|
template<typename Type>
|
||||||
|
|
|
||||||
|
|
@ -45,6 +45,14 @@ private:
|
||||||
utils::identifier pk_;
|
utils::identifier pk_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename>
|
||||||
|
struct is_object_ptr : std::false_type
|
||||||
|
{};
|
||||||
|
|
||||||
|
template<typename Type>
|
||||||
|
struct is_object_ptr<object_ptr<Type>> : std::true_type
|
||||||
|
{};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif //OBJECT_PTR_HPP
|
#endif //OBJECT_PTR_HPP
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,14 @@
|
||||||
#ifndef MATADOR_RELATION_ENDPOINT_HPP
|
#ifndef MATADOR_RELATION_ENDPOINT_HPP
|
||||||
#define MATADOR_RELATION_ENDPOINT_HPP
|
#define MATADOR_RELATION_ENDPOINT_HPP
|
||||||
|
|
||||||
#include "matador/object/schema_node.hpp"
|
|
||||||
|
|
||||||
#include "matador/utils/enum_mapper.hpp"
|
#include "matador/utils/enum_mapper.hpp"
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace matador::object {
|
namespace matador::object {
|
||||||
|
|
||||||
|
class schema_node;
|
||||||
|
|
||||||
enum class relation_type : uint8_t {
|
enum class relation_type : uint8_t {
|
||||||
BELONGS_TO,
|
BELONGS_TO,
|
||||||
HAS_ONE,
|
HAS_ONE,
|
||||||
|
|
|
||||||
|
|
@ -58,11 +58,11 @@ class relation_completer final {
|
||||||
public:
|
public:
|
||||||
using value_type = Type;
|
using value_type = Type;
|
||||||
|
|
||||||
static void prepare_expected_nodes(schema &scm) {
|
static void prepare_expected_nodes(schema_node &node) {
|
||||||
relation_completer analyzer(scm);
|
relation_completer completer(node);
|
||||||
|
|
||||||
Type obj;
|
Type obj;
|
||||||
access::process(analyzer, obj);
|
access::process(completer, obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
template < class PrimaryKeyType >
|
template < class PrimaryKeyType >
|
||||||
|
|
@ -86,11 +86,12 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class CollectionType>
|
template<class CollectionType>
|
||||||
void on_has_many(const char * /*id*/, CollectionType &, const char *, const utils::foreign_attributes &/*attr*/);
|
void on_has_many(const char *id, CollectionType &, const char *join_column, const utils::foreign_attributes &attr, std::enable_if_t<is_object_ptr<typename CollectionType::value_type>::value>* = nullptr );
|
||||||
|
template<class CollectionType>
|
||||||
|
void on_has_many(const char *id, CollectionType &, const char *join_column, const utils::foreign_attributes &attr, std::enable_if_t<!is_object_ptr<typename CollectionType::value_type>::value>* = nullptr );
|
||||||
|
|
||||||
template<class CollectionType>
|
template<class CollectionType>
|
||||||
void on_has_many_to_many(const char *id, CollectionType &collection, const char *join_column, const char *inverse_join_column, const utils::foreign_attributes &attr);
|
void on_has_many_to_many(const char *id, CollectionType &collection, const char *join_column, const char *inverse_join_column, const utils::foreign_attributes &attr);
|
||||||
|
|
||||||
template<class ContainerType>
|
template<class ContainerType>
|
||||||
void on_has_many_to_many(const char *id, ContainerType &collection, const utils::foreign_attributes &attr);
|
void on_has_many_to_many(const char *id, ContainerType &collection, const utils::foreign_attributes &attr);
|
||||||
|
|
||||||
|
|
@ -99,14 +100,14 @@ private:
|
||||||
void on_foreign_key();
|
void on_foreign_key();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
explicit relation_completer(schema& schema, const std::shared_ptr<schema_node> &node)
|
explicit relation_completer(schema_node& node)
|
||||||
: schema_(schema)
|
: node_(node)
|
||||||
, node_(node){}
|
, schema_(node.schema_){}
|
||||||
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
schema &schema_;
|
schema_node &node_;
|
||||||
std::shared_ptr<schema_node> node_;
|
schema& schema_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -134,9 +135,8 @@ public:
|
||||||
type_index_node_map_.insert({node->type_index(), node});
|
type_index_node_map_.insert({node->type_index(), node});
|
||||||
} else {
|
} else {
|
||||||
// analyze node (collect unknown types by type index)
|
// analyze node (collect unknown types by type index)
|
||||||
relation_completer<Type>::prepare_expected_nodes(*this);
|
|
||||||
|
|
||||||
const auto node = schema_node::make_node<Type>(*this, name);
|
const auto node = schema_node::make_node<Type>(*this, name);
|
||||||
|
relation_completer<Type>::prepare_expected_nodes(*node);
|
||||||
if (auto result = attach_node(node, parent); !result) {
|
if (auto result = attach_node(node, parent); !result) {
|
||||||
return utils::failure(result.err());
|
return utils::failure(result.err());
|
||||||
}
|
}
|
||||||
|
|
@ -245,8 +245,41 @@ private:
|
||||||
|
|
||||||
template<typename Type>
|
template<typename Type>
|
||||||
template<class CollectionType>
|
template<class CollectionType>
|
||||||
void relation_completer<Type>::on_has_many( const char*, CollectionType&, const char*, const utils::foreign_attributes& ) {
|
void relation_completer<Type>::on_has_many( const char *id, CollectionType&, const char *join_column, const utils::foreign_attributes&, std::enable_if_t<is_object_ptr<typename CollectionType::value_type>::value>* /*unused*/ ) {
|
||||||
|
using value_type = typename CollectionType::value_type;
|
||||||
|
|
||||||
|
// Process foreign key has many relation
|
||||||
|
// Check if foreign key type was already registered
|
||||||
|
auto result = schema_.find_node(typeid(value_type));
|
||||||
|
if (result) {
|
||||||
|
} else {
|
||||||
|
//
|
||||||
|
using relation_type = many_to_many_relation<value_type, Type>;
|
||||||
|
auto creator = [] {
|
||||||
|
return new many_to_many_relation<value_type, Type>();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename Type>
|
||||||
|
template<class CollectionType>
|
||||||
|
void relation_completer<Type>::on_has_many( const char *id, CollectionType&, const char *join_column, const utils::foreign_attributes&, std::enable_if_t<!is_object_ptr<typename CollectionType::value_type>::value>* /*unused*/ ) {
|
||||||
|
using value_type = typename CollectionType::value_type;
|
||||||
|
|
||||||
|
// Process values has many relation
|
||||||
|
// Register relation table
|
||||||
|
|
||||||
|
// many_to_relation<Type, value_type> *relation = new many_to_relation<Type, value_type>(join_column, "value");
|
||||||
|
auto result = schema_.find_node(typeid(value_type));
|
||||||
|
if (result) {
|
||||||
|
} else {
|
||||||
|
//
|
||||||
|
using relation_type = many_to_many_relation<value_type, Type>;
|
||||||
|
auto creator = [] {
|
||||||
|
return new many_to_many_relation<value_type, Type>();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,7 @@ public:
|
||||||
using node_ptr = std::shared_ptr<schema_node>;
|
using node_ptr = std::shared_ptr<schema_node>;
|
||||||
|
|
||||||
template < typename Type >
|
template < typename Type >
|
||||||
static std::shared_ptr<schema_node> make_node(schema& tree, const std::string& name) {
|
static std::shared_ptr<schema_node> make_node(object::schema& tree, const std::string& name) {
|
||||||
auto node = std::shared_ptr<schema_node>(new schema_node(tree, name));
|
auto node = std::shared_ptr<schema_node>(new schema_node(tree, name));
|
||||||
|
|
||||||
primary_key_resolver resolver;
|
primary_key_resolver resolver;
|
||||||
|
|
@ -34,7 +34,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
template < typename Type >
|
template < typename Type >
|
||||||
static std::shared_ptr<schema_node> make_relation_node(schema& tree, const std::string& name) {
|
static std::shared_ptr<schema_node> make_relation_node(object::schema& tree, const std::string& name) {
|
||||||
auto node = std::shared_ptr<schema_node>(new schema_node(tree, name));
|
auto node = std::shared_ptr<schema_node>(new schema_node(tree, name));
|
||||||
|
|
||||||
auto info = std::make_unique<object_info<Type>>(
|
auto info = std::make_unique<object_info<Type>>(
|
||||||
|
|
@ -69,15 +69,19 @@ public:
|
||||||
return std::ref(static_cast<const object_info<Type>&>(*info_));
|
return std::ref(static_cast<const object_info<Type>&>(*info_));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const object::schema& schema() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
explicit schema_node(schema& tree);
|
explicit schema_node(object::schema& tree);
|
||||||
schema_node(schema& tree, std::string name);
|
schema_node(object::schema& tree, std::string name);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class schema;
|
friend class schema;
|
||||||
|
template <typename Type>
|
||||||
|
friend class relation_completer;
|
||||||
friend class const_schema_node_iterator;
|
friend class const_schema_node_iterator;
|
||||||
|
|
||||||
schema &schema_;
|
object::schema &schema_;
|
||||||
std::unique_ptr<basic_object_info> info_;
|
std::unique_ptr<basic_object_info> info_;
|
||||||
|
|
||||||
std::shared_ptr<schema_node> parent_;
|
std::shared_ptr<schema_node> parent_;
|
||||||
|
|
|
||||||
|
|
@ -15,6 +15,7 @@ add_library(matador-core STATIC
|
||||||
../../include/matador/object/object_proxy.hpp
|
../../include/matador/object/object_proxy.hpp
|
||||||
../../include/matador/object/object_ptr.hpp
|
../../include/matador/object/object_ptr.hpp
|
||||||
../../include/matador/object/primary_key_resolver.hpp
|
../../include/matador/object/primary_key_resolver.hpp
|
||||||
|
../../include/matador/object/relation_endpoint.hpp
|
||||||
../../include/matador/object/schema.hpp
|
../../include/matador/object/schema.hpp
|
||||||
../../include/matador/object/schema_node.hpp
|
../../include/matador/object/schema_node.hpp
|
||||||
../../include/matador/object/schema_node_iterator.hpp
|
../../include/matador/object/schema_node_iterator.hpp
|
||||||
|
|
@ -57,6 +58,7 @@ add_library(matador-core STATIC
|
||||||
object/error_code.cpp
|
object/error_code.cpp
|
||||||
object/object_definition.cpp
|
object/object_definition.cpp
|
||||||
object/primary_key_resolver.cpp
|
object/primary_key_resolver.cpp
|
||||||
|
object/relation_endpoint.cpp
|
||||||
object/schema.cpp
|
object/schema.cpp
|
||||||
object/schema_node.cpp
|
object/schema_node.cpp
|
||||||
object/schema_node_iterator.cpp
|
object/schema_node_iterator.cpp
|
||||||
|
|
@ -75,8 +77,6 @@ add_library(matador-core STATIC
|
||||||
utils/uuid.cpp
|
utils/uuid.cpp
|
||||||
utils/value.cpp
|
utils/value.cpp
|
||||||
utils/version.cpp
|
utils/version.cpp
|
||||||
../../include/matador/object/relation_endpoint.hpp
|
|
||||||
object/relation_endpoint.cpp
|
|
||||||
)
|
)
|
||||||
|
|
||||||
target_link_libraries(matador-core ${CMAKE_DL_LIBS})
|
target_link_libraries(matador-core ${CMAKE_DL_LIBS})
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,7 @@
|
||||||
#include "matador/object/relation_endpoint.hpp"
|
#include "matador/object/relation_endpoint.hpp"
|
||||||
|
|
||||||
|
#include "matador/object/schema_node.hpp"
|
||||||
|
|
||||||
namespace matador::object {
|
namespace matador::object {
|
||||||
relation_endpoint::relation_endpoint(std::string field_name, relation_type type, const std::shared_ptr<schema_node> &node)
|
relation_endpoint::relation_endpoint(std::string field_name, relation_type type, const std::shared_ptr<schema_node> &node)
|
||||||
: field_name_(std::move(field_name))
|
: field_name_(std::move(field_name))
|
||||||
|
|
|
||||||
|
|
@ -3,11 +3,11 @@
|
||||||
#include "matador/object/schema_node.hpp"
|
#include "matador/object/schema_node.hpp"
|
||||||
|
|
||||||
namespace matador::object {
|
namespace matador::object {
|
||||||
schema_node::schema_node(schema &tree)
|
schema_node::schema_node(object::schema &tree)
|
||||||
: schema_(tree) {
|
: schema_(tree) {
|
||||||
}
|
}
|
||||||
|
|
||||||
schema_node::schema_node(schema &tree, std::string name)
|
schema_node::schema_node(object::schema &tree, std::string name)
|
||||||
: schema_(tree)
|
: schema_(tree)
|
||||||
, first_child_(std::shared_ptr<schema_node>(new schema_node(tree)))
|
, first_child_(std::shared_ptr<schema_node>(new schema_node(tree)))
|
||||||
, last_child_(std::shared_ptr<schema_node>(new schema_node(tree)))
|
, last_child_(std::shared_ptr<schema_node>(new schema_node(tree)))
|
||||||
|
|
@ -16,7 +16,7 @@ schema_node::schema_node(schema &tree, std::string name)
|
||||||
last_child_->previous_sibling_ = first_child_;
|
last_child_->previous_sibling_ = first_child_;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<schema_node> schema_node::make_null_node(schema &tree) {
|
std::shared_ptr<schema_node> schema_node::make_null_node(object::schema &tree) {
|
||||||
auto node = std::shared_ptr<schema_node>(new schema_node(tree));
|
auto node = std::shared_ptr<schema_node>(new schema_node(tree));
|
||||||
node->info_ = std::make_unique<null_info>(node, std::shared_ptr<attribute_definition>{});
|
node->info_ = std::make_unique<null_info>(node, std::shared_ptr<attribute_definition>{});
|
||||||
|
|
||||||
|
|
@ -40,6 +40,10 @@ void schema_node::update_name(const std::string& name) {
|
||||||
info_->reference_column()->name(name);
|
info_->reference_column()->name(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const object::schema& schema_node::schema() const {
|
||||||
|
return schema_;
|
||||||
|
}
|
||||||
|
|
||||||
schema_node::node_ptr schema_node::next() const {
|
schema_node::node_ptr schema_node::next() const {
|
||||||
// if we have a child, child is the next iterator to return
|
// if we have a child, child is the next iterator to return
|
||||||
// (if we don't do iterate over the siblings)
|
// (if we don't do iterate over the siblings)
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
#include "matador/object/schema.hpp"
|
#include "matador/object/schema.hpp"
|
||||||
|
|
||||||
|
#include "../../models/department.hpp"
|
||||||
|
|
||||||
struct node {};
|
struct node {};
|
||||||
|
|
||||||
using namespace matador;
|
using namespace matador;
|
||||||
|
|
@ -15,6 +17,16 @@ struct person {
|
||||||
struct student final : person {};
|
struct student final : person {};
|
||||||
struct teacher final : person {};
|
struct teacher final : person {};
|
||||||
|
|
||||||
|
struct names {
|
||||||
|
std::vector<std::string> names_list;
|
||||||
|
|
||||||
|
template<typename Operator>
|
||||||
|
void process(Operator &op) {
|
||||||
|
namespace field = matador::access;
|
||||||
|
field::has_many(op, "name_list", names_list, "names_id", utils::fetch_type::EAGER);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
TEST_CASE("Test empty prototype tree", "[schema_node][empty]") {
|
TEST_CASE("Test empty prototype tree", "[schema_node][empty]") {
|
||||||
const object::schema tree;
|
const object::schema tree;
|
||||||
|
|
||||||
|
|
@ -51,4 +63,22 @@ TEST_CASE("Test next and previous of schema node", "[schema_node][next][previous
|
||||||
REQUIRE( it->name() == "person" );
|
REQUIRE( it->name() == "person" );
|
||||||
REQUIRE( (--it)->name() == "person" );
|
REQUIRE( (--it)->name() == "person" );
|
||||||
REQUIRE( ++it == tree.end() );
|
REQUIRE( ++it == tree.end() );
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test automatic creating of a relation table with foreign key", "[schema][relation_table][foreign_key]") {
|
||||||
|
object::schema tree;
|
||||||
|
|
||||||
|
REQUIRE( tree.empty() );
|
||||||
|
|
||||||
|
auto res = tree.attach<test::department>("department");
|
||||||
|
REQUIRE( res.is_ok() );
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test automatic creating of a relation table with values", "[schema][relation_table][values]") {
|
||||||
|
object::schema tree;
|
||||||
|
|
||||||
|
REQUIRE( tree.empty() );
|
||||||
|
|
||||||
|
auto res = tree.attach<names>("names");
|
||||||
|
REQUIRE( res.is_ok() );
|
||||||
}
|
}
|
||||||
Loading…
Reference in New Issue