2016-08-18 22:34:36 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <memory>
|
2016-08-25 22:29:45 +08:00
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
#include "data_structures/concurrent/concurrent_map.hpp"
|
2016-08-25 22:29:45 +08:00
|
|
|
#include "storage/indexes/index_holder.hpp"
|
2016-08-18 22:34:36 +08:00
|
|
|
#include "storage/model/properties/flags.hpp"
|
2016-08-29 02:36:52 +08:00
|
|
|
#include "storage/type_group_edge.hpp"
|
|
|
|
#include "storage/type_group_vertex.hpp"
|
2016-08-18 22:34:36 +08:00
|
|
|
#include "utils/option.hpp"
|
|
|
|
#include "utils/total_ordering.hpp"
|
|
|
|
#include "utils/underlying_cast.hpp"
|
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
// #include "storage/indexes/index_base.hpp"
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
// Family of properties with the same name but different types.
|
|
|
|
// Ordered on name.
|
2016-08-25 22:29:45 +08:00
|
|
|
// TG - group of types Edge/Vertex
|
|
|
|
template <class TG>
|
|
|
|
class PropertyFamily : public TotalOrdering<PropertyFamily<TG>>
|
2016-08-18 22:34:36 +08:00
|
|
|
{
|
|
|
|
friend class PropertyType;
|
|
|
|
friend class PropertyFamilyKey;
|
|
|
|
friend class PropertyTypeKey;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Type of property defined with his family and his type.
|
|
|
|
// Ordered on PropertyFamily and Type.
|
|
|
|
class PropertyType : public TotalOrdering<PropertyType>
|
|
|
|
{
|
|
|
|
friend class PropertyFamilyKey;
|
|
|
|
friend class PropertyTypeKey;
|
|
|
|
friend class PropertyFamily;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Ordered on POINTERS to PropertyFamily
|
2016-08-20 01:40:04 +08:00
|
|
|
class PropertyFamilyKey
|
|
|
|
: public TotalOrdering<PropertyFamilyKey>,
|
|
|
|
public TotalOrdering<PropertyFamilyKey, PropertyFamily>,
|
|
|
|
public TotalOrdering<PropertyFamily, PropertyFamilyKey>
|
2016-08-18 22:34:36 +08:00
|
|
|
{
|
|
|
|
friend class PropertyType;
|
|
|
|
friend class PropertyTypeKey;
|
|
|
|
|
|
|
|
PropertyFamilyKey(const PropertyType &type) : type(&type) {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
friend bool operator==(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type->family) == &(rhs.type->family);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type->family) < &(rhs.type->family);
|
|
|
|
}
|
|
|
|
|
2016-08-20 01:40:04 +08:00
|
|
|
friend bool operator==(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyFamily &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type->family) == &(rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyFamily &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type->family) < &(rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const PropertyFamily &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs) == &(rhs.type->family);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyFamily &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs) < &(rhs.type->family);
|
|
|
|
}
|
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
Type get_type() const { return type->type; }
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
Type prop_type() const { return type->type; }
|
|
|
|
|
|
|
|
std::string const &family_name() const
|
|
|
|
{
|
|
|
|
return type->family.name();
|
|
|
|
}
|
|
|
|
|
2016-08-23 02:03:45 +08:00
|
|
|
const PropertyFamily &get_family() const { return type->family; }
|
|
|
|
|
2016-09-05 22:56:08 +08:00
|
|
|
const std::string to_str() const
|
|
|
|
{
|
|
|
|
return std::string(family_name());
|
|
|
|
}
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
private:
|
|
|
|
const PropertyType *type;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Ordered on POINTERS to PropertyType.
|
|
|
|
// When compared with PropertyFamilyKey behaves as PropertyFamilyKey.
|
|
|
|
template <class T>
|
2016-09-19 06:22:36 +08:00
|
|
|
class PropertyTypeKey
|
|
|
|
: public TotalOrdering<PropertyTypeKey<T>>,
|
|
|
|
public TotalOrdering<PropertyFamilyKey, PropertyTypeKey<T>>,
|
|
|
|
public TotalOrdering<PropertyTypeKey<T>, PropertyFamilyKey>
|
2016-08-18 22:34:36 +08:00
|
|
|
{
|
|
|
|
friend class PropertyType;
|
|
|
|
|
|
|
|
PropertyTypeKey(const PropertyType &type) : type(type) {}
|
|
|
|
public:
|
2016-09-19 06:22:36 +08:00
|
|
|
PropertyFamilyKey family_key() const
|
|
|
|
{
|
|
|
|
return PropertyFamilyKey(type);
|
|
|
|
}
|
2016-08-18 22:34:36 +08:00
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
Type const &prop_type() const { return type.type; }
|
2016-08-18 22:34:36 +08:00
|
|
|
|
|
|
|
friend bool operator==(const PropertyTypeKey &lhs,
|
|
|
|
const PropertyTypeKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type) == &(rhs.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyTypeKey &lhs,
|
|
|
|
const PropertyTypeKey &rhs)
|
|
|
|
{
|
|
|
|
return &(lhs.type) < &(rhs.type);
|
|
|
|
}
|
|
|
|
|
2016-09-19 06:22:36 +08:00
|
|
|
friend bool operator==(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyTypeKey &rhs)
|
|
|
|
{
|
|
|
|
return lhs == rhs.family_key();
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyFamilyKey &lhs,
|
|
|
|
const PropertyTypeKey &rhs)
|
|
|
|
{
|
|
|
|
return lhs < rhs.family_key();
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const PropertyTypeKey &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return lhs.family_key() == rhs;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyTypeKey &lhs,
|
|
|
|
const PropertyFamilyKey &rhs)
|
|
|
|
{
|
|
|
|
return lhs.family_key() < rhs;
|
|
|
|
}
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
private:
|
|
|
|
const PropertyType &type;
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
|
|
|
PropertyType(PropertyFamily &family, Type type);
|
|
|
|
PropertyType(PropertyFamily &other) = delete;
|
|
|
|
PropertyType(PropertyFamily &&other) = delete;
|
|
|
|
|
|
|
|
public:
|
|
|
|
template <class T>
|
|
|
|
bool is() const
|
|
|
|
{
|
2016-08-20 01:40:04 +08:00
|
|
|
return type == T::type;
|
2016-08-18 22:34:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool is(Type &t) const;
|
|
|
|
|
|
|
|
// Returns key ordered on POINTERS to PropertyType.
|
|
|
|
// When compared with PropertyFamilyKey behaves as PropertyFamilyKey.
|
|
|
|
template <class T>
|
|
|
|
PropertyTypeKey<T> type_key()
|
|
|
|
{
|
|
|
|
assert(this->is<T>());
|
|
|
|
return PropertyTypeKey<T>(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns key ordered on POINTERS to PropertyFamily
|
|
|
|
PropertyFamilyKey family_key();
|
|
|
|
|
|
|
|
friend bool operator<(const PropertyType &lhs, const PropertyType &rhs)
|
|
|
|
{
|
|
|
|
return lhs.family < rhs.family ||
|
|
|
|
(lhs.family == rhs.family && lhs.type < rhs.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const PropertyType &lhs, const PropertyType &rhs)
|
|
|
|
{
|
|
|
|
return lhs.family == rhs.family && lhs.type == rhs.type;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const PropertyFamily &family;
|
|
|
|
const Type type;
|
|
|
|
};
|
|
|
|
|
|
|
|
PropertyFamily(std::string const &name_v);
|
|
|
|
PropertyFamily(std::string &&name_v);
|
|
|
|
PropertyFamily(PropertyFamily &other) = delete;
|
|
|
|
PropertyFamily(PropertyFamily &&other) = delete;
|
|
|
|
|
|
|
|
std::string const &name() const;
|
|
|
|
|
|
|
|
// Returns type if it exists otherwise creates it.
|
|
|
|
PropertyType &get(Type type);
|
|
|
|
|
2016-08-20 01:40:04 +08:00
|
|
|
// Return pointer for NULL type. Extremly fast.
|
2016-09-05 17:02:48 +08:00
|
|
|
PropertyType &getNull() const { return *null_type; }
|
2016-08-20 01:40:04 +08:00
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
friend bool operator<(const PropertyFamily &lhs, const PropertyFamily &rhs);
|
|
|
|
|
|
|
|
friend bool operator==(const PropertyFamily &lhs,
|
|
|
|
const PropertyFamily &rhs);
|
|
|
|
|
2016-09-19 06:22:36 +08:00
|
|
|
// Place for index of TG::elements which have property from this family.
|
2016-08-25 22:29:45 +08:00
|
|
|
IndexHolder<TG, std::nullptr_t> index;
|
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
private:
|
|
|
|
const std::string name_v;
|
2016-09-19 06:22:36 +08:00
|
|
|
|
2016-08-20 01:40:04 +08:00
|
|
|
// This is exclusivly for getNull method.
|
|
|
|
PropertyType *null_type{nullptr};
|
2016-09-19 06:22:36 +08:00
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
// TODO: Because types wont be removed this could be done with more efficent
|
|
|
|
// data structure.
|
|
|
|
ConcurrentMap<Type, std::unique_ptr<PropertyType>> types;
|
|
|
|
};
|
2016-08-23 02:03:45 +08:00
|
|
|
|
2016-08-29 02:36:52 +08:00
|
|
|
using VertexPropertyKey =
|
|
|
|
PropertyFamily<TypeGroupVertex>::PropertyType::PropertyFamilyKey;
|
|
|
|
using EdgePropertyKey =
|
|
|
|
PropertyFamily<TypeGroupEdge>::PropertyType::PropertyFamilyKey;
|
|
|
|
template <class T>
|
|
|
|
using VertexPropertyType =
|
|
|
|
PropertyFamily<TypeGroupVertex>::PropertyType::PropertyTypeKey<T>;
|
|
|
|
template <class T>
|
|
|
|
using EdgePropertyType =
|
|
|
|
PropertyFamily<TypeGroupEdge>::PropertyType::PropertyTypeKey<T>;
|