Lightweight 0.1.0
Loading...
Searching...
No Matches
Utils.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "Api.hpp"
6
7#include <reflection-cpp/reflection.hpp>
8
9#include <ranges>
10#include <source_location>
11#include <string_view>
12#include <type_traits>
13#include <utility>
14
15#include <sql.h>
16
17namespace detail
18{
19
20template <typename T, typename... Comps>
21concept OneOf = (std::same_as<T, Comps> || ...);
22
23template <typename T>
24constexpr auto AlwaysFalse = std::false_type::value;
25
26constexpr auto Finally(auto&& cleanupRoutine) noexcept
27{
28 // NOLINTNEXTLINE(cppcoreguidelines-special-member-functions)
29 struct Finally
30 {
31 std::remove_cvref_t<decltype(cleanupRoutine)> cleanup;
32 ~Finally()
33 {
34 cleanup();
35 }
36 };
37 return Finally { std::forward<decltype(cleanupRoutine)>(cleanupRoutine) };
38}
39
40// is_specialization_of<> is inspired by:
41// https://www.open-std.org/jtc1/sc22/wg21/docs/papers/2020/p2098r1.pdf
42
43template <template <typename...> class T, typename U>
44struct is_specialization_of: std::false_type
45{
46};
47
48template <template <typename...> class T, typename... Us>
49struct is_specialization_of<T, T<Us...>>: std::true_type
50{
51};
52
53template <typename T>
54struct MemberClassTypeHelper;
55
56template <typename M, typename T>
57struct MemberClassTypeHelper<M T::*>
58{
59 using type = std::remove_cvref_t<T>;
60};
61
62template <typename Record>
63struct RecordTableNameImpl
64{
65 static constexpr std::string_view Value = []() {
66 if constexpr (requires { Record::TableName; })
67 return Record::TableName;
68 else
69 return []() {
70 // TODO: Build plural
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);
74 return typeName;
75 }();
76 }();
77};
78
79template <std::size_t I, typename Record>
80struct BelongsToNameImpl
81{
82 static constexpr auto baseName = Reflection::MemberNameOf<I, Record>;
83 static constexpr auto storage = []() -> std::array<char, baseName.size() + 4>
84 {
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';
89 return storage;
90 }
91 ();
92 static constexpr auto name = std::string_view(storage.data(), storage.size() - 1);
93};
94
95template <typename FieldType>
96constexpr auto ColumnNameOverride = []() consteval {
97 if constexpr (requires { FieldType::ColumnNameOverride; })
98 return FieldType::ColumnNameOverride;
99 else
100 return std::string_view {};
101}();
102
103template <typename ReferencedFieldType, auto F>
104struct FieldNameOfImpl;
105
106template <typename T, auto F, typename R>
107struct FieldNameOfImpl<R T::*, F>
108{
109 static constexpr std::string_view value = []() constexpr -> std::string_view {
110 if constexpr (requires { R::ColumnNameOverride; })
111 {
112 if constexpr (!R::ColumnNameOverride.empty())
113 return R::ColumnNameOverride;
114 }
115 return Reflection::NameOf<F>;
116 }();
117};
118
119template <std::size_t I, typename Record>
120consteval std::string_view FieldNameAt()
121{
122 using FieldType = Reflection::MemberTypeOf<I, Record>;
123
124 if constexpr (!std::string_view(ColumnNameOverride<FieldType>).empty())
125 {
126 return FieldType::ColumnNameOverride;
127 }
128 else if constexpr (requires { FieldType::ReferencedField; }) // check isBelongsTo
129 {
130 return detail::BelongsToNameImpl<I, Record>::name;
131 }
132 else
133 return Reflection::MemberNameOf<I, Record>;
134}
135
136} // namespace detail
137
138/// @brief Returns the SQL field name of the given field index in the record.
139///
140/// @ingroup DataMapper
141template <std::size_t I, typename Record>
142constexpr inline std::string_view FieldNameAt = detail::FieldNameAt<I, Record>();
143
144/// @brief Returns the name of the field referenced by the given pointer-to-member.
145///
146/// This also supports custom column name overrides.
147template <auto ReferencedField>
148constexpr inline std::string_view FieldNameOf =
149 detail::FieldNameOfImpl<decltype(ReferencedField), ReferencedField>::value;
150
151/// @brief Holds the SQL tabl ename for the given record type.
152///
153/// @ingroup DataMapper
154template <typename Record>
155constexpr std::string_view RecordTableName = detail::RecordTableNameImpl<Record>::Value;
156
157template <template <typename...> class S, class T>
158concept IsSpecializationOf = detail::is_specialization_of<S, T>::value;
159
160template <typename T>
161using MemberClassType = typename detail::MemberClassTypeHelper<T>::type;
162
163namespace detail
164{
165template <auto ReferencedField>
166struct FullFieldNameOfImpl
167{
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;
171
172 // Holds the full field name in the format "ClassName"."FieldName"
173 static constexpr auto Storage = []() constexpr -> std::array<char, StorageSize> {
174 // clang-format off
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';
182 // clang-format on
183 return storage;
184 }();
185 static constexpr auto value = std::string_view(Storage.data(), Storage.size() - 1);
186};
187} // namespace detail
188
189struct SqlRawColumnNameView
190{
191 std::string_view value;
192
193 std::weak_ordering operator<=>(SqlRawColumnNameView const& other) const = default;
194};
195
196constexpr bool operator==(SqlRawColumnNameView const& lhs, std::string_view rhs) noexcept
197{
198 return lhs.value == rhs;
199}
200
201constexpr bool operator!=(SqlRawColumnNameView const& lhs, std::string_view rhs) noexcept
202{
203 return lhs.value != rhs;
204}
205
206template <auto ReferencedField>
207constexpr inline auto FullFieldNameOf = SqlRawColumnNameView {
208 .value = detail::FullFieldNameOfImpl<ReferencedField>::value,
209};
210
211LIGHTWEIGHT_API void LogIfFailed(SQLHSTMT hStmt, SQLRETURN error, std::source_location sourceLocation);
212
213LIGHTWEIGHT_API void RequireSuccess(SQLHSTMT hStmt,
214 SQLRETURN error,
215 std::source_location sourceLocation = std::source_location::current());
constexpr std::string_view RecordTableName
Holds the SQL tabl ename for the given record type.
Definition Utils.hpp:155
constexpr std::string_view FieldNameAt
Returns the SQL field name of the given field index in the record.
Definition Utils.hpp:142