4#include "../DataBinder/Core.hpp"
6#include <reflection-cpp/reflection.hpp>
9enum class PrimaryKey : uint8_t
26 ServerSideAutoIncrement,
35struct IsStdOptionalType: std::false_type {};
38struct IsStdOptionalType<std::optional<T>>: std::true_type {};
41constexpr bool IsStdOptional = IsStdOptionalType<T>::value;
44concept FieldElementType = SqlInputParameterBinder<T> && SqlOutputColumnBinder<T>;
48template <
typename TargetType,
typename P1,
typename P2>
49consteval auto Choose(TargetType defaultValue, P1 p1, P2 p2)
noexcept
51 if constexpr (!std::same_as<P1, std::nullopt_t> &&
requires { TargetType { p1 }; })
53 else if constexpr (!std::same_as<P2, std::nullopt_t> &&
requires { TargetType { p2 }; })
69template <detail::FieldElementType T, auto P1 = std::nullopt, auto P2 = std::nullopt>
74 static constexpr auto IsPrimaryKeyValue = detail::Choose<PrimaryKey>(PrimaryKey::No, P1, P2);
75 static constexpr auto ColumnNameOverride = detail::Choose<std::string_view>({}, P1, P2);
78 constexpr Field()
noexcept =
default;
79 constexpr Field(
Field const&)
noexcept =
default;
80 constexpr Field& operator=(
Field const&)
noexcept =
default;
82 constexpr Field& operator=(
Field&&)
noexcept =
default;
83 constexpr ~Field()
noexcept =
default;
87 template <
typename... S>
88 requires std::constructible_from<T, S...>
89 constexpr Field(S&&... value)
noexcept;
93 requires std::constructible_from<T, S>
97 static constexpr auto IsOptional = detail::IsStdOptional<T>;
103 static constexpr auto IsPrimaryKey = IsPrimaryKeyValue != PrimaryKey::No;
112 constexpr std::weak_ordering
operator<=>(
Field const& other)
const noexcept;
121 template <
typename S>
122 requires std::convertible_to<S, T>
126 template <
typename S>
127 requires std::convertible_to<S, T>
137 [[nodiscard]]
constexpr bool IsModified() const noexcept;
140 [[nodiscard]] constexpr T const&
Value() const noexcept;
149 bool _modified {
false };
157struct IsAutoAssignPrimaryKeyField: std::false_type {};
159template <
typename T, auto P>
160struct IsAutoAssignPrimaryKeyField<
Field<T, PrimaryKey::AutoAssign, P>>: std::true_type {};
162template <
typename T, auto P>
163struct IsAutoAssignPrimaryKeyField<
Field<T, P, PrimaryKey::AutoAssign>>: std::true_type {};
166struct IsAutoIncrementPrimaryKeyField: std::false_type {};
168template <
typename T, auto P>
169struct IsAutoIncrementPrimaryKeyField<
Field<T, PrimaryKey::ServerSideAutoIncrement, P>>: std::true_type {};
171template <
typename T, auto P>
172struct IsAutoIncrementPrimaryKeyField<
Field<T, P, PrimaryKey::ServerSideAutoIncrement>>: std::true_type {};
175struct IsFieldType: std::false_type {};
177template <
typename T, auto P1, auto P2>
178struct IsFieldType<
Field<T, P1, P2>>: std::true_type {};
185constexpr bool IsPrimaryKey =
186 detail::IsAutoAssignPrimaryKeyField<T>::value || detail::IsAutoIncrementPrimaryKeyField<T>::value;
190constexpr bool IsAutoIncrementPrimaryKey = detail::IsAutoIncrementPrimaryKeyField<T>::value;
193constexpr bool IsField = detail::IsFieldType<std::remove_cvref_t<T>>::value;
195template <detail::FieldElementType T, auto P1, auto P2>
196template <
typename... S>
197 requires std::constructible_from<T, S...>
199 _value(std::forward<S>(value)...)
203template <detail::FieldElementType T, auto P1, auto P2>
205 requires std::constructible_from<T, S>
208 _value = std::forward<S>(value);
213template <detail::FieldElementType T, auto P1, auto P2>
216 return _value <=> other._value;
219template <detail::FieldElementType T, auto P1, auto P2>
221 requires std::convertible_to<S, T>
224 return _value == value;
227template <detail::FieldElementType T, auto P1, auto P2>
229 requires std::convertible_to<S, T>
232 return _value != value;
235template <detail::FieldElementType T, auto P1, auto P2>
238 if constexpr (std::is_same_v<T, std::string>)
240 std::stringstream result;
241 result << std::quoted(_value,
'\'');
244 else if constexpr (std::is_same_v<T, SqlText>)
246 std::stringstream result;
247 result << std::quoted(_value.value,
'\'');
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(); })
258 if (_value.has_value())
259 return std::format(
"{}", _value.value());
264 return std::format(
"{}", _value);
269template <detail::FieldElementType T, auto P1, auto P2>
275template <detail::FieldElementType T, auto P1, auto P2>
281template <detail::FieldElementType T, auto P1, auto P2>
287template <detail::FieldElementType T, auto P1, auto P2>
293template <detail::FieldElementType T, auto P1, auto P2>
294struct SqlDataBinder<
Field<T, P1, P2>>
298 static constexpr auto ColumnType = SqlDataBinder<T>::ColumnType;
300 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
302 ValueType
const& value,
305 return SqlDataBinder<T>::InputParameter(stmt, column, value.Value(), cb);
308 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN
309 OutputColumn(SQLHSTMT stmt, SQLUSMALLINT column, ValueType* result, SQLLEN* indicator,
SqlDataBinderCallback& cb)
311 return SqlDataBinder<T>::OutputColumn(stmt, column, &result->MutableValue(), indicator, cb);
314 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
320 return SqlDataBinder<T>::GetColumn(stmt, column, &result->emplace(), indicator, cb);
323 static LIGHTWEIGHT_FORCE_INLINE std::string Inspect(ValueType
const& value)
noexcept
325 return value.InspectValue();
329template <detail::FieldElementType T, auto P1, auto P2>
330struct std::formatter<
Field<T, P1, P2>>: std::formatter<T>
332 template <
typename FormatContext>
Represents a single column in a table.
constexpr T & MutableValue() noexcept
std::string InspectValue() const
Returns a string representation of the value, suitable for use in debugging and logging.
constexpr T const & Value() const noexcept
Returns the value of the field.
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.
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.
static constexpr auto IsAutoIncrementPrimaryKey
Indicates if this is a primary key, it also is auto-incremented by the database.
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.
constexpr bool IsModified() const noexcept
Checks if the field has been modified.
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.
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.
static constexpr auto IsOptional
Indicates if the field is optional, i.e., it can be NULL.