227 lines
6.3 KiB
C++
227 lines
6.3 KiB
C++
#ifndef MATADOR_IDENTIFIER_HPP
|
|
#define MATADOR_IDENTIFIER_HPP
|
|
|
|
#include "matador/utils/default_type_traits.hpp"
|
|
#include "matador/utils/field_attributes.hpp"
|
|
#include "matador/utils/basic_types.hpp"
|
|
#include "matador/utils/types.hpp"
|
|
|
|
#include <memory>
|
|
#include <string>
|
|
#include <typeindex>
|
|
|
|
namespace matador::utils {
|
|
|
|
class identifier_serializer
|
|
{
|
|
public:
|
|
virtual ~identifier_serializer() = default;
|
|
|
|
virtual void serialize(int8_t &, const field_attributes &) = 0;
|
|
virtual void serialize(int16_t &, const field_attributes &) = 0;
|
|
virtual void serialize(int32_t &, const field_attributes &) = 0;
|
|
virtual void serialize(int64_t &, const field_attributes &) = 0;
|
|
virtual void serialize(uint8_t &, const field_attributes &) = 0;
|
|
virtual void serialize(uint16_t &, const field_attributes &) = 0;
|
|
virtual void serialize(uint32_t &, const field_attributes &) = 0;
|
|
virtual void serialize(uint64_t &, const field_attributes &) = 0;
|
|
virtual void serialize(const char*, const field_attributes &) = 0;
|
|
virtual void serialize(std::string &, const field_attributes &) = 0;
|
|
virtual void serialize(null_type_t &, const field_attributes &) = 0;
|
|
};
|
|
|
|
template<typename Type, class Enabled = void>
|
|
struct identifier_type_traits;
|
|
|
|
template<typename Type>
|
|
struct identifier_type_traits<Type, std::enable_if_t<std::is_integral_v<Type>>> {
|
|
static bool is_valid(Type value) { return value > 0; }
|
|
static std::string to_string(const Type value) { return std::to_string(value); }
|
|
};
|
|
|
|
template<typename Type>
|
|
struct identifier_type_traits<Type, std::enable_if_t<std::is_same_v<Type, std::string>>> {
|
|
static bool is_valid(const Type &value) { return !value.empty(); }
|
|
static std::string to_string(const Type &value) { return value; }
|
|
};
|
|
|
|
template<>
|
|
struct identifier_type_traits<null_type_t, void> {
|
|
static bool is_valid() { return false; }
|
|
static std::string to_string() { return "null"; }
|
|
};
|
|
|
|
template<>
|
|
struct identifier_type_traits<const char*, void> {
|
|
static bool is_valid(const char *value);
|
|
static std::string to_string(const char *value);
|
|
};
|
|
|
|
namespace detail {
|
|
|
|
template<typename Type>
|
|
size_t hash(const Type &value) {
|
|
return std::hash<Type>()(value);
|
|
}
|
|
|
|
size_t hash(const char *value);
|
|
|
|
}
|
|
|
|
class identifier
|
|
{
|
|
private:
|
|
struct base
|
|
{
|
|
base(const std::type_index &ti, basic_type type);
|
|
base(const base &x) = delete;
|
|
base &operator=(const base &x) = delete;
|
|
base(base &&x) = delete;
|
|
base &operator=(base &&x) = delete;
|
|
virtual ~base() = default;
|
|
|
|
[[nodiscard]] virtual base *copy() const = 0;
|
|
[[nodiscard]] virtual bool equal_to(const base &x) const = 0;
|
|
[[nodiscard]] virtual bool less(const base &x) const = 0;
|
|
[[nodiscard]] virtual bool is_valid() const = 0;
|
|
virtual void serialize(identifier_serializer &s) = 0;
|
|
[[nodiscard]] virtual std::string str() const = 0;
|
|
[[nodiscard]] virtual size_t hash() const = 0;
|
|
|
|
std::type_index type_index_;
|
|
basic_type type_{basic_type::type_null};
|
|
};
|
|
|
|
template<class IdType>
|
|
struct pk final : base
|
|
{
|
|
using self = pk;
|
|
|
|
explicit pk(const IdType &id)
|
|
: base(std::type_index(typeid(IdType)), data_type_traits<IdType>::type(1))
|
|
, id_(id)
|
|
, size_(sizeof(IdType)) {}
|
|
|
|
[[nodiscard]] base *copy() const override {
|
|
return new self(id_);
|
|
}
|
|
|
|
[[nodiscard]] bool equal_to(const base &x) const override {
|
|
return static_cast<const pk&>(x).id_ == id_;
|
|
}
|
|
|
|
[[nodiscard]] bool less(const base &x) const override {
|
|
return id_ < static_cast<const pk&>(x).id_;
|
|
}
|
|
|
|
[[nodiscard]] bool is_valid() const override {
|
|
return identifier_type_traits<IdType>::is_valid(id_);
|
|
}
|
|
|
|
[[nodiscard]] std::string str() const override {
|
|
return identifier_type_traits<IdType>::to_string(id_);
|
|
}
|
|
|
|
void serialize(identifier_serializer &s) override {
|
|
s.serialize(id_, size_);
|
|
}
|
|
|
|
[[nodiscard]] size_t hash() const override {
|
|
return detail::hash(id_);
|
|
}
|
|
|
|
IdType id_;
|
|
size_t size_{};
|
|
};
|
|
|
|
struct null_pk final : base
|
|
{
|
|
null_pk();
|
|
[[nodiscard]] base *copy() const override;
|
|
[[nodiscard]] bool equal_to(const base &x) const override;
|
|
[[nodiscard]] bool less(const base &x) const override;
|
|
[[nodiscard]] bool is_valid() const override;
|
|
void serialize(identifier_serializer &s) override;
|
|
[[nodiscard]] std::string str() const override;
|
|
[[nodiscard]] size_t hash() const override;
|
|
null_type_t null_;
|
|
};
|
|
|
|
public:
|
|
identifier();
|
|
template<typename Type>
|
|
explicit identifier(const Type &id)
|
|
: id_(std::make_shared<pk<Type>>(id)) {}
|
|
explicit identifier(const char *id)
|
|
: id_(std::make_shared<pk<const char*>>(id)) {}
|
|
identifier(const identifier &x);
|
|
identifier &operator=(const identifier &x);
|
|
identifier(identifier &&x) noexcept ;
|
|
identifier &operator=(identifier &&x) noexcept;
|
|
|
|
template<typename Type>
|
|
identifier& operator=(const Type &value)
|
|
{
|
|
id_ = std::make_shared<pk<Type>>(value);
|
|
return *this;
|
|
}
|
|
|
|
identifier& operator=(const char *value)
|
|
{
|
|
id_ = std::make_shared<pk<const char*>>(value);
|
|
return *this;
|
|
}
|
|
|
|
~identifier() = default;
|
|
|
|
bool operator==(const identifier &x) const;
|
|
bool operator!=(const identifier &x) const;
|
|
bool operator<(const identifier &x) const;
|
|
bool operator<=(const identifier &x) const;
|
|
bool operator>(const identifier &x) const;
|
|
bool operator>=(const identifier &x) const;
|
|
|
|
[[nodiscard]] std::string str() const;
|
|
[[nodiscard]] const std::type_index &type_index() const;
|
|
[[nodiscard]] basic_type type() const;
|
|
|
|
[[nodiscard]] identifier share() const;
|
|
[[nodiscard]] size_t use_count() const;
|
|
|
|
[[nodiscard]] bool is_integer() const;
|
|
[[nodiscard]] bool is_floating_point() const;
|
|
[[nodiscard]] bool is_bool() const;
|
|
[[nodiscard]] bool is_varchar() const;
|
|
[[nodiscard]] bool is_date() const;
|
|
[[nodiscard]] bool is_time() const;
|
|
[[nodiscard]] bool is_blob() const;
|
|
[[nodiscard]] bool is_null() const;
|
|
|
|
[[nodiscard]] bool is_valid() const;
|
|
void clear();
|
|
|
|
void serialize(identifier_serializer &s) const;
|
|
|
|
[[nodiscard]] size_t hash() const;
|
|
|
|
friend std::ostream &operator<<(std::ostream &out, const identifier &id);
|
|
|
|
private:
|
|
explicit identifier(const std::shared_ptr<base>& id);
|
|
|
|
private:
|
|
std::shared_ptr<base> id_;
|
|
};
|
|
|
|
static identifier null_identifier{};
|
|
|
|
/// @cond MATADOR_DEV
|
|
struct id_pk_hash
|
|
{
|
|
size_t operator()(const identifier &id) const;
|
|
};
|
|
/// @endcond
|
|
}
|
|
|
|
#endif //MATADOR_IDENTIFIER_HPP
|