7#include <reflection-cpp/reflection.hpp>
10#include <source_location>
20template <
typename T,
typename... Comps>
21concept OneOf = (std::same_as<T, Comps> || ...);
24constexpr auto AlwaysFalse = std::false_type::value;
26constexpr auto Finally(
auto&& cleanupRoutine)
noexcept
31 std::remove_cvref_t<
decltype(cleanupRoutine)> cleanup;
37 return Finally { std::forward<decltype(cleanupRoutine)>(cleanupRoutine) };
43template <
template <
typename...>
class T,
typename U>
44struct is_specialization_of: std::false_type
48template <
template <
typename...>
class T,
typename... Us>
49struct is_specialization_of<T, T<Us...>>: std::true_type
54struct MemberClassTypeHelper;
56template <
typename M,
typename T>
57struct MemberClassTypeHelper<M T::*>
59 using type = std::remove_cvref_t<T>;
62template <
typename Record>
63struct RecordTableNameImpl
65 static constexpr std::string_view Value = []() {
66 if constexpr (
requires { Record::TableName; })
67 return Record::TableName;
71 auto const typeName = Reflection::TypeNameOf<Record>;
72 if (
auto const i = typeName.rfind(
':'); i != std::string_view::npos)
73 return typeName.substr(i + 1);
79template <std::
size_t I,
typename Record>
80struct BelongsToNameImpl
82 static constexpr auto baseName = Reflection::MemberNameOf<I, Record>;
83 static constexpr auto storage = []() -> std::array<
char, baseName.size() + 4>
85 std::array<char, baseName.size() + 4> storage;
86 std::copy_n(baseName.begin(), baseName.size(), storage.begin());
87 std::copy_n(
"_id", 3, storage.begin() + baseName.size());
88 storage.back() =
'\0';
92 static constexpr auto name = std::string_view(storage.data(), storage.size() - 1);
95template <
typename FieldType>
96constexpr auto ColumnNameOverride = []()
consteval {
97 if constexpr (
requires { FieldType::ColumnNameOverride; })
98 return FieldType::ColumnNameOverride;
100 return std::string_view {};
103template <
typename ReferencedFieldType, auto F>
104struct FieldNameOfImpl;
106template <
typename T, auto F,
typename R>
107struct FieldNameOfImpl<R T::*, F>
109 static constexpr std::string_view value = []()
constexpr -> std::string_view {
110 if constexpr (
requires { R::ColumnNameOverride; })
112 if constexpr (!R::ColumnNameOverride.empty())
113 return R::ColumnNameOverride;
115 return Reflection::NameOf<F>;
119template <std::
size_t I,
typename Record>
122 using FieldType = Reflection::MemberTypeOf<I, Record>;
124 if constexpr (!std::string_view(ColumnNameOverride<FieldType>).empty())
126 return FieldType::ColumnNameOverride;
128 else if constexpr (
requires { FieldType::ReferencedField; })
130 return detail::BelongsToNameImpl<I, Record>::name;
133 return Reflection::MemberNameOf<I, Record>;
141template <std::
size_t I,
typename Record>
142constexpr inline std::string_view
FieldNameAt = detail::FieldNameAt<I, Record>();
147template <auto ReferencedField>
148constexpr inline std::string_view FieldNameOf =
149 detail::FieldNameOfImpl<
decltype(ReferencedField), ReferencedField>::value;
154template <
typename Record>
155constexpr std::string_view
RecordTableName = detail::RecordTableNameImpl<Record>::Value;
157template <
template <
typename...>
class S,
class T>
158concept IsSpecializationOf = detail::is_specialization_of<S, T>::value;
161using MemberClassType =
typename detail::MemberClassTypeHelper<T>::type;
165template <auto ReferencedField>
166struct FullFieldNameOfImpl
168 static constexpr auto ClassName =
RecordTableName<MemberClassType<
decltype(ReferencedField)>>;
169 static constexpr auto FieldName = FieldNameOf<ReferencedField>;
170 static constexpr auto StorageSize = ClassName.size() + FieldName.size() + 6;
173 static constexpr auto Storage = []()
constexpr -> std::array<char, StorageSize> {
175 auto storage = std::array<char, StorageSize> {};
176 std::ranges::copy(
"\"", storage.begin());
177 std::ranges::copy(ClassName, storage.begin() + 1);
178 std::ranges::copy(
"\".\"", storage.begin() + 1 + ClassName.size());
179 std::ranges::copy(FieldName, storage.begin() + 1 + ClassName.size() + 3);
180 std::ranges::copy(
"\"", storage.begin() + 1 + ClassName.size() + 3 + FieldName.size());
181 storage.back() =
'\0';
185 static constexpr auto value = std::string_view(Storage.data(), Storage.size() - 1);
189struct SqlRawColumnNameView
191 std::string_view value;
193 std::weak_ordering operator<=>(SqlRawColumnNameView
const& other)
const =
default;
196constexpr bool operator==(SqlRawColumnNameView
const& lhs, std::string_view rhs)
noexcept
198 return lhs.value == rhs;
201constexpr bool operator!=(SqlRawColumnNameView
const& lhs, std::string_view rhs)
noexcept
203 return lhs.value != rhs;
206template <auto ReferencedField>
207constexpr inline auto FullFieldNameOf = SqlRawColumnNameView {
208 .value = detail::FullFieldNameOfImpl<ReferencedField>::value,
211LIGHTWEIGHT_API
void LogIfFailed(SQLHSTMT hStmt, SQLRETURN error, std::source_location sourceLocation);
213LIGHTWEIGHT_API
void RequireSuccess(SQLHSTMT hStmt,
215 std::source_location sourceLocation = std::source_location::current());
constexpr std::string_view RecordTableName
Holds the SQL tabl ename for the given record type.
constexpr std::string_view FieldNameAt
Returns the SQL field name of the given field index in the record.