Lightweight 0.1.0
Loading...
Searching...
No Matches
Field.hpp
1// SPDX-License-Identifier: Apache-2.0
2#pragma once
3
4#include "../DataBinder/Core.hpp"
5
6#include <reflection-cpp/reflection.hpp>
7
8/// @brief Tells the data mapper that this field is a primary key with given semantics, or not a primary key.
9enum class PrimaryKey : uint8_t
10{
11 /// The field is not a primary key.
12 No,
13
14 /// @brief The field is a primary key.
15 ///
16 /// If the field is an auto-incrementable key and not manually set, it is automatically set to the
17 /// next available value on the client side, using a SELECT MAX() query.
18 /// This is happening transparently to the user.
19 ///
20 /// If the field is a GUID, it is automatically set to a new GUID value, if not manually set.
21 ///
22 /// @note If the field is neither auto-incrementable nor a GUID, it must be manually set.
23 AutoAssign,
24
25 /// The field is an integer primary key, and it is auto-incremented by the database.
26 ServerSideAutoIncrement,
27};
28
29namespace detail
30{
31
32// clang-format off
33
34template <typename T>
35struct IsStdOptionalType: std::false_type {};
36
37template <typename T>
38struct IsStdOptionalType<std::optional<T>>: std::true_type {};
39
40template <typename T>
41constexpr bool IsStdOptional = IsStdOptionalType<T>::value;
42
43template <typename T>
44concept FieldElementType = SqlInputParameterBinder<T> && SqlOutputColumnBinder<T>;
45
46// clang-format on
47
48template <typename TargetType, typename P1, typename P2>
49consteval auto Choose(TargetType defaultValue, P1 p1, P2 p2) noexcept
50{
51 if constexpr (!std::same_as<P1, std::nullopt_t> && requires { TargetType { p1 }; })
52 return p1;
53 else if constexpr (!std::same_as<P2, std::nullopt_t> && requires { TargetType { p2 }; })
54 return p2;
55 else
56 return defaultValue;
57}
58} // namespace detail
59
60/// @brief Represents a single column in a table.
61///
62/// This class is used to represent a single column in a table.
63/// It also keeps track of modified-state of the field.
64///
65/// The column name, index, nullability, and type are known at compile time.
66///
67/// @see DataMapper
68/// @ingroup DataMapper, DataTypes
69template <detail::FieldElementType T, auto P1 = std::nullopt, auto P2 = std::nullopt>
70struct Field
71{
72 using ValueType = T;
73
74 static constexpr auto IsPrimaryKeyValue = detail::Choose<PrimaryKey>(PrimaryKey::No, P1, P2);
75 static constexpr auto ColumnNameOverride = detail::Choose<std::string_view>({}, P1, P2);
76
77 // clang-format off
78 constexpr Field() noexcept = default;
79 constexpr Field(Field const&) noexcept = default;
80 constexpr Field& operator=(Field const&) noexcept = default;
81 constexpr Field(Field&&) noexcept = default;
82 constexpr Field& operator=(Field&&) noexcept = default;
83 constexpr ~Field() noexcept = default;
84 // clang-format on
85
86 /// Constructs a new field with the given value.
87 template <typename... S>
88 requires std::constructible_from<T, S...>
89 constexpr Field(S&&... value) noexcept;
90
91 /// Assigns a new value to the field.
92 template <typename S>
93 requires std::constructible_from<T, S>
94 constexpr Field& operator=(S&& value) noexcept;
95
96 /// Indicates if the field is optional, i.e., it can be NULL.
97 static constexpr auto IsOptional = detail::IsStdOptional<T>;
98
99 /// Indicates if the field is mandatory, i.e., it cannot be NULL.
100 static constexpr auto IsMandatory = !IsOptional;
101
102 /// Indicates if the field is a primary key.
103 static constexpr auto IsPrimaryKey = IsPrimaryKeyValue != PrimaryKey::No;
104
105 /// Indicates if this is a primary key, it also is auto-assigned by the client.
106 static constexpr auto IsAutoAssignPrimaryKey = IsPrimaryKeyValue == PrimaryKey::AutoAssign;
107
108 /// Indicates if this is a primary key, it also is auto-incremented by the database.
109 static constexpr auto IsAutoIncrementPrimaryKey = IsPrimaryKeyValue == PrimaryKey::ServerSideAutoIncrement;
110
111 /// Compares two fields for equality.
112 constexpr std::weak_ordering operator<=>(Field const& other) const noexcept;
113
114 /// Compares the field value with the given value for equality.
115 bool operator==(Field const& value) const noexcept = default;
116
117 /// Compares the field value with the given value for inequality.
118 bool operator!=(Field const& value) const noexcept = default;
119
120 /// Compares the field value with the given value for equality.
121 template <typename S>
122 requires std::convertible_to<S, T>
123 bool operator==(S const& value) const noexcept;
124
125 /// Compares the field value with the given value for inequality.
126 template <typename S>
127 requires std::convertible_to<S, T>
128 bool operator!=(S const& value) const noexcept;
129
130 /// Returns a string representation of the value, suitable for use in debugging and logging.
131 [[nodiscard]] std::string InspectValue() const;
132
133 /// Sets the modified state of the field.
134 constexpr void SetModified(bool value) noexcept;
135
136 /// Checks if the field has been modified.
137 [[nodiscard]] constexpr bool IsModified() const noexcept;
138
139 /// Returns the value of the field.
140 [[nodiscard]] constexpr T const& Value() const noexcept;
141
142 /// Returns a mutable reference to the value of the field.
143 ///
144 /// @note If the field value is changed through this method, it will not be automatically marked as modified.
145 [[nodiscard]] constexpr T& MutableValue() noexcept;
146
147 private:
148 ValueType _value {};
149 bool _modified { false };
150};
151
152// clang-format off
153namespace detail
154{
155
156template <typename T>
157struct IsAutoAssignPrimaryKeyField: std::false_type {};
158
159template <typename T, auto P>
160struct IsAutoAssignPrimaryKeyField<Field<T, PrimaryKey::AutoAssign, P>>: std::true_type {};
161
162template <typename T, auto P>
163struct IsAutoAssignPrimaryKeyField<Field<T, P, PrimaryKey::AutoAssign>>: std::true_type {};
164
165template <typename T>
166struct IsAutoIncrementPrimaryKeyField: std::false_type {};
167
168template <typename T, auto P>
169struct IsAutoIncrementPrimaryKeyField<Field<T, PrimaryKey::ServerSideAutoIncrement, P>>: std::true_type {};
170
171template <typename T, auto P>
172struct IsAutoIncrementPrimaryKeyField<Field<T, P, PrimaryKey::ServerSideAutoIncrement>>: std::true_type {};
173
174template <typename T>
175struct IsFieldType: std::false_type {};
176
177template <typename T, auto P1, auto P2>
178struct IsFieldType<Field<T, P1, P2>>: std::true_type {};
179
180} // namespace detail
181// clang-format on
182
183/// Tests if T is a Field<> that is a primary key.
184template <typename T>
185constexpr bool IsPrimaryKey =
186 detail::IsAutoAssignPrimaryKeyField<T>::value || detail::IsAutoIncrementPrimaryKeyField<T>::value;
187
188/// Requires that T satisfies to be a field with storage and is considered a primary key.
189template <typename T>
190constexpr bool IsAutoIncrementPrimaryKey = detail::IsAutoIncrementPrimaryKeyField<T>::value;
191
192template <typename T>
193constexpr bool IsField = detail::IsFieldType<std::remove_cvref_t<T>>::value;
194
195template <detail::FieldElementType T, auto P1, auto P2>
196template <typename... S>
197 requires std::constructible_from<T, S...>
198constexpr LIGHTWEIGHT_FORCE_INLINE Field<T, P1, P2>::Field(S&&... value) noexcept:
199 _value(std::forward<S>(value)...)
200{
201}
202
203template <detail::FieldElementType T, auto P1, auto P2>
204template <typename S>
205 requires std::constructible_from<T, S>
206constexpr LIGHTWEIGHT_FORCE_INLINE Field<T, P1, P2>& Field<T, P1, P2>::operator=(S&& value) noexcept
207{
208 _value = std::forward<S>(value);
209 SetModified(true);
210 return *this;
211}
212
213template <detail::FieldElementType T, auto P1, auto P2>
214constexpr std::weak_ordering LIGHTWEIGHT_FORCE_INLINE Field<T, P1, P2>::operator<=>(Field const& other) const noexcept
215{
216 return _value <=> other._value;
217}
218
219template <detail::FieldElementType T, auto P1, auto P2>
220template <typename S>
221 requires std::convertible_to<S, T>
222inline LIGHTWEIGHT_FORCE_INLINE bool Field<T, P1, P2>::operator==(S const& value) const noexcept
223{
224 return _value == value;
225}
226
227template <detail::FieldElementType T, auto P1, auto P2>
228template <typename S>
229 requires std::convertible_to<S, T>
230inline LIGHTWEIGHT_FORCE_INLINE bool Field<T, P1, P2>::operator!=(S const& value) const noexcept
231{
232 return _value != value;
233}
234
235template <detail::FieldElementType T, auto P1, auto P2>
236inline LIGHTWEIGHT_FORCE_INLINE std::string Field<T, P1, P2>::InspectValue() const
237{
238 if constexpr (std::is_same_v<T, std::string>)
239 {
240 std::stringstream result;
241 result << std::quoted(_value, '\'');
242 return result.str();
243 }
244 else if constexpr (std::is_same_v<T, SqlText>)
245 {
246 std::stringstream result;
247 result << std::quoted(_value.value, '\'');
248 return result.str();
249 }
250 else if constexpr (std::is_same_v<T, SqlDate>)
251 return std::format("\'{}\'", _value.value);
252 else if constexpr (std::is_same_v<T, SqlTime>)
253 return std::format("\'{}\'", _value.value);
254 else if constexpr (std::is_same_v<T, SqlDateTime>)
255 return std::format("\'{}\'", _value.value());
256 else if constexpr (requires { _value.has_value(); })
257 {
258 if (_value.has_value())
259 return std::format("{}", _value.value());
260 else
261 return "NULL";
262 }
263 else
264 return std::format("{}", _value);
265}
266
267// ------------------------------------------------------------------------------------------------
268
269template <detail::FieldElementType T, auto P1, auto P2>
270constexpr LIGHTWEIGHT_FORCE_INLINE void Field<T, P1, P2>::SetModified(bool value) noexcept
271{
272 _modified = value;
273}
274
275template <detail::FieldElementType T, auto P1, auto P2>
276constexpr LIGHTWEIGHT_FORCE_INLINE bool Field<T, P1, P2>::IsModified() const noexcept
277{
278 return _modified;
279}
280
281template <detail::FieldElementType T, auto P1, auto P2>
282constexpr LIGHTWEIGHT_FORCE_INLINE T const& Field<T, P1, P2>::Value() const noexcept
283{
284 return _value;
285}
286
287template <detail::FieldElementType T, auto P1, auto P2>
288constexpr LIGHTWEIGHT_FORCE_INLINE T& Field<T, P1, P2>::MutableValue() noexcept
289{
290 return _value;
291}
292
293template <detail::FieldElementType T, auto P1, auto P2>
294struct SqlDataBinder<Field<T, P1, P2>>
295{
296 using ValueType = Field<T, P1, P2>;
297
298 static constexpr auto ColumnType = SqlDataBinder<T>::ColumnType;
299
300 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
301 SQLUSMALLINT column,
302 ValueType const& value,
304 {
305 return SqlDataBinder<T>::InputParameter(stmt, column, value.Value(), cb);
306 }
307
308 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN
309 OutputColumn(SQLHSTMT stmt, SQLUSMALLINT column, ValueType* result, SQLLEN* indicator, SqlDataBinderCallback& cb)
310 {
311 return SqlDataBinder<T>::OutputColumn(stmt, column, &result->MutableValue(), indicator, cb);
312 }
313
314 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
315 SQLUSMALLINT column,
316 ValueType* result,
317 SQLLEN* indicator,
318 SqlDataBinderCallback const& cb) noexcept
319 {
320 return SqlDataBinder<T>::GetColumn(stmt, column, &result->emplace(), indicator, cb);
321 }
322
323 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(ValueType const& value) noexcept
324 {
325 return value.InspectValue();
326 }
327};
328
329template <detail::FieldElementType T, auto P1, auto P2>
330struct std::formatter<Field<T, P1, P2>>: std::formatter<T>
331{
332 template <typename FormatContext>
333 auto format(Field<T, P1, P2> const& field, FormatContext& ctx)
334 {
335 return formatter<T>::format(field.InspectValue(), ctx);
336 }
337};
Represents a single column in a table.
Definition Field.hpp:71
constexpr T & MutableValue() noexcept
Definition Field.hpp:288
std::string InspectValue() const
Returns a string representation of the value, suitable for use in debugging and logging.
Definition Field.hpp:236
constexpr T const & Value() const noexcept
Returns the value of the field.
Definition Field.hpp:282
bool operator!=(S const &value) const noexcept
Compares the field value with the given value for inequality.
bool operator==(S const &value) const noexcept
Compares the field value with the given value for equality.
constexpr void SetModified(bool value) noexcept
Sets the modified state of the field.
Definition Field.hpp:270
constexpr Field(S &&... value) noexcept
Constructs a new field with the given value.
static constexpr auto IsPrimaryKey
Indicates if the field is a primary key.
Definition Field.hpp:103
static constexpr auto IsAutoIncrementPrimaryKey
Indicates if this is a primary key, it also is auto-incremented by the database.
Definition Field.hpp:109
constexpr Field & operator=(S &&value) noexcept
Assigns a new value to the field.
constexpr std::weak_ordering operator<=>(Field const &other) const noexcept
Compares two fields for equality.
Definition Field.hpp:214
constexpr bool IsModified() const noexcept
Checks if the field has been modified.
Definition Field.hpp:276
bool operator==(Field const &value) const noexcept=default
Compares the field value with the given value for equality.
static constexpr auto IsMandatory
Indicates if the field is mandatory, i.e., it cannot be NULL.
Definition Field.hpp:100
bool operator!=(Field const &value) const noexcept=default
Compares the field value with the given value for inequality.
static constexpr auto IsAutoAssignPrimaryKey
Indicates if this is a primary key, it also is auto-assigned by the client.
Definition Field.hpp:106
static constexpr auto IsOptional
Indicates if the field is optional, i.e., it can be NULL.
Definition Field.hpp:97