-
-
Notifications
You must be signed in to change notification settings - Fork 321
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Uniform syntax for column pointer expressions through table references
- Loading branch information
Showing
21 changed files
with
772 additions
and
292 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,109 +1,144 @@ | ||
#pragma once | ||
|
||
#include <type_traits> // std::remove_const, std::is_base_of, std::is_same | ||
#ifdef SQLITE_ORM_WITH_CPP20_ALIASES | ||
#include <concepts> | ||
#endif | ||
|
||
#include "functional/cxx_universal.h" | ||
#include "functional/cxx_type_traits_polyfill.h" | ||
#include "type_traits.h" | ||
|
||
namespace sqlite_orm { | ||
|
||
/** @short Base class for a custom table alias, column alias or expression alias. | ||
*/ | ||
struct alias_tag {}; | ||
|
||
namespace internal { | ||
|
||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_alias_v = std::is_base_of<alias_tag, A>::value; | ||
|
||
template<class A> | ||
using is_alias = polyfill::bool_constant<is_alias_v<A>>; | ||
|
||
/** @short Alias of a column in a record set, see `orm_column_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_column_alias_v = | ||
polyfill::conjunction_v<is_alias<A>, polyfill::negation<polyfill::is_detected<type_t, A>>>; | ||
|
||
template<class A> | ||
using is_column_alias = is_alias<A>; | ||
|
||
/** @short Alias of any type of record set, see `orm_recordset_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_recordset_alias_v = | ||
polyfill::conjunction_v<is_alias<A>, polyfill::is_detected<type_t, A>>; | ||
|
||
template<class A> | ||
using is_recordset_alias = polyfill::bool_constant<is_recordset_alias_v<A>>; | ||
|
||
/** @short Alias of a concrete table, see `orm_table_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_table_alias_v = polyfill::conjunction_v< | ||
is_recordset_alias<A>, | ||
polyfill::negation<std::is_same<polyfill::detected_t<type_t, A>, std::remove_const_t<A>>>>; | ||
|
||
template<class A> | ||
using is_table_alias = polyfill::bool_constant<is_table_alias_v<A>>; | ||
|
||
/** @short Alias of a CTE, see `orm_cte_moniker`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_cte_moniker_v = | ||
#ifdef SQLITE_ORM_WITH_CTE | ||
polyfill::conjunction_v<is_recordset_alias<A>, | ||
std::is_same<polyfill::detected_t<type_t, A>, std::remove_const_t<A>>>; | ||
#else | ||
false; | ||
#endif | ||
|
||
template<class A> | ||
using is_cte_moniker = polyfill::bool_constant<is_cte_moniker_v<A>>; | ||
} | ||
|
||
#ifdef SQLITE_ORM_WITH_CPP20_ALIASES | ||
template<class A> | ||
concept orm_alias = std::derived_from<A, alias_tag>; | ||
|
||
/** @short Alias of a column in a record set. | ||
* | ||
* A column alias has the following traits: | ||
* - is derived from `alias_tag` | ||
* - must not have a nested `type` typename | ||
*/ | ||
template<class A> | ||
concept orm_column_alias = (orm_alias<A> && !orm_names_type<A>); | ||
|
||
/** @short Alias of any type of record set. | ||
* | ||
* A record set alias has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a nested `type` typename, which refers to a mapped object. | ||
*/ | ||
template<class A> | ||
concept orm_recordset_alias = (orm_alias<A> && orm_names_type<A>); | ||
|
||
/** @short Alias of a concrete table. | ||
* | ||
* A concrete table alias has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a `type` typename, which refers to another mapped object (i.e. doesn't refer to itself). | ||
*/ | ||
template<class A> | ||
concept orm_table_alias = (orm_recordset_alias<A> && !std::same_as<typename A::type, std::remove_const_t<A>>); | ||
|
||
/** @short Moniker of a CTE. | ||
* | ||
* A CTE moniker has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a `type` typename, which refers to itself. | ||
*/ | ||
template<class A> | ||
concept orm_cte_moniker = (orm_recordset_alias<A> && std::same_as<typename A::type, std::remove_const_t<A>>); | ||
#endif | ||
} | ||
#pragma once | ||
|
||
#include <type_traits> // std::remove_const, std::is_base_of, std::is_same, std::type_identity | ||
#ifdef SQLITE_ORM_WITH_CPP20_ALIASES | ||
#include <concepts> | ||
#endif | ||
|
||
#include "functional/cxx_universal.h" | ||
#include "functional/cxx_type_traits_polyfill.h" | ||
#include "type_traits.h" | ||
|
||
namespace sqlite_orm { | ||
|
||
/** @short Base class for a custom table alias, column alias or expression alias. | ||
*/ | ||
struct alias_tag {}; | ||
|
||
namespace internal { | ||
|
||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_alias_v = std::is_base_of<alias_tag, A>::value; | ||
|
||
template<class A> | ||
using is_alias = polyfill::bool_constant<is_alias_v<A>>; | ||
|
||
/** @short Alias of a column in a record set, see `orm_column_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_column_alias_v = | ||
polyfill::conjunction_v<is_alias<A>, polyfill::negation<polyfill::is_detected<type_t, A>>>; | ||
|
||
template<class A> | ||
using is_column_alias = is_alias<A>; | ||
|
||
/** @short Alias of any type of record set, see `orm_recordset_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_recordset_alias_v = | ||
polyfill::conjunction_v<is_alias<A>, polyfill::is_detected<type_t, A>>; | ||
|
||
template<class A> | ||
using is_recordset_alias = polyfill::bool_constant<is_recordset_alias_v<A>>; | ||
|
||
/** @short Alias of a concrete table, see `orm_table_alias`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_table_alias_v = polyfill::conjunction_v< | ||
is_recordset_alias<A>, | ||
polyfill::negation<std::is_same<polyfill::detected_t<type_t, A>, std::remove_const_t<A>>>>; | ||
|
||
template<class A> | ||
using is_table_alias = polyfill::bool_constant<is_table_alias_v<A>>; | ||
|
||
#ifdef SQLITE_ORM_WITH_CPP20_ALIASES | ||
/* | ||
* Identity wrapper around a mapped object, facilitating uniform column pointer expressions. | ||
*/ | ||
template<class O> | ||
struct table_reference : std::type_identity<O> {}; | ||
|
||
template<class RecordSet> | ||
struct decay_table_reference : std::remove_const<RecordSet> {}; | ||
template<class O> | ||
struct decay_table_reference<table_reference<O>> : std::type_identity<O> {}; | ||
template<class O> | ||
struct decay_table_reference<const table_reference<O>> : std::type_identity<O> {}; | ||
|
||
template<auto recordset> | ||
using decay_table_reference_t = typename decay_table_reference<decltype(recordset)>::type; | ||
#endif | ||
|
||
/** @short Moniker of a CTE, see `orm_cte_moniker`. | ||
*/ | ||
template<class A> | ||
SQLITE_ORM_INLINE_VAR constexpr bool is_cte_moniker_v = | ||
#ifdef SQLITE_ORM_WITH_CTE | ||
polyfill::conjunction_v<is_recordset_alias<A>, | ||
std::is_same<polyfill::detected_t<type_t, A>, std::remove_const_t<A>>>; | ||
#else | ||
false; | ||
#endif | ||
|
||
template<class A> | ||
using is_cte_moniker = polyfill::bool_constant<is_cte_moniker_v<A>>; | ||
} | ||
|
||
#ifdef SQLITE_ORM_WITH_CPP20_ALIASES | ||
template<class A> | ||
concept orm_alias = std::derived_from<A, alias_tag>; | ||
|
||
/** @short Alias of a column in a record set. | ||
* | ||
* A column alias has the following traits: | ||
* - is derived from `alias_tag` | ||
* - must not have a nested `type` typename | ||
*/ | ||
template<class A> | ||
concept orm_column_alias = (orm_alias<A> && !orm_names_type<A>); | ||
|
||
/** @short Alias of any type of record set. | ||
* | ||
* A record set alias has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a nested `type` typename, which refers to a mapped object. | ||
*/ | ||
template<class A> | ||
concept orm_recordset_alias = (orm_alias<A> && orm_names_type<A>); | ||
|
||
/** @short Alias of a concrete table. | ||
* | ||
* A concrete table alias has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a `type` typename, which refers to another mapped object (i.e. doesn't refer to itself). | ||
*/ | ||
template<class A> | ||
concept orm_table_alias = (orm_recordset_alias<A> && !std::same_as<typename A::type, std::remove_const_t<A>>); | ||
|
||
/** @short Reference of a concrete table, especially of a derived class. | ||
* | ||
* A concrete table reference has the following traits: | ||
* - specialization of `table_reference`, whose `type` typename references a mapped object. | ||
*/ | ||
template<class R> | ||
concept orm_table_reference = polyfill::is_specialization_of_v<std::remove_const_t<R>, internal::table_reference>; | ||
|
||
/** @short Moniker of a CTE. | ||
* | ||
* A CTE moniker has the following traits: | ||
* - is derived from `alias_tag`. | ||
* - has a `type` typename, which refers to itself. | ||
*/ | ||
template<class A> | ||
concept orm_cte_moniker = (orm_recordset_alias<A> && std::same_as<typename A::type, std::remove_const_t<A>>); | ||
|
||
template<class T> | ||
concept orm_refers_to_table = (orm_table_reference<T> || orm_table_alias<T>); | ||
|
||
template<class T> | ||
concept orm_refers_to_recordset = (orm_table_reference<T> || orm_recordset_alias<T>); | ||
|
||
template<class T> | ||
concept orm_mapped_recordset = (orm_table_reference<T> || orm_cte_moniker<T>); | ||
#endif | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.