Lightweight 0.1.0
Loading...
Searching...
No Matches
BelongsTo.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../DataBinder/Core.hpp"
6#include "../DataBinder/SqlNullValue.hpp"
7#include "../SqlStatement.hpp"
8#include "../Utils.hpp"
9#include "Error.hpp"
10#include "Field.hpp"
11
12#include <compare>
13#include <optional>
14#include <string_view>
15#include <type_traits>
16
17/// @brief Represents a one-to-one relationship.
18///
19/// The `TheReferencedField` parameter is the field in the other record that references the current record,
20/// in the form of `&OtherRecord::Field`.
21/// Other Field must be a primary key.
22///
23/// @ingroup DataMapper
24template <auto TheReferencedField, auto ColumnNameOverrideString = std::nullopt>
26{
27 public:
28 /// The field in the other record that references the current record.
29 static constexpr auto ReferencedField = TheReferencedField;
30
31 /// If not an empty string, this value will be used as the column name in the database.
32 static constexpr std::string_view ColumnNameOverride = []() consteval {
33 if constexpr (!std::same_as<decltype(ColumnNameOverrideString), std::nullopt_t>)
34 return std::string_view{ColumnNameOverrideString};
35 else
36 return std::string_view {};
37 }();
38
39 /// Represents the record type of the other field.
40 using ReferencedRecord = MemberClassType<decltype(TheReferencedField)>;
41
42 static_assert(std::remove_cvref_t<decltype(std::declval<ReferencedRecord>().*ReferencedField)>::IsPrimaryKey,
43 "The referenced field must be a primary key.");
44
45 /// Represents the column type of the foreign key, matching the primary key of the other record.
46 using ValueType =
47 typename std::remove_cvref_t<decltype(std::declval<ReferencedRecord>().*ReferencedField)>::ValueType;
48
49 static constexpr auto IsOptional = true;
50 static constexpr auto IsMandatory = !IsOptional;
51 static constexpr auto IsPrimaryKey = false;
52 static constexpr auto IsAutoIncrementPrimaryKey = false;
53
54 template <typename... S>
55 requires std::constructible_from<ValueType, S...>
56 constexpr BelongsTo(S&&... value) noexcept:
57 _referencedFieldValue(std::forward<S>(value)...)
58 {
59 }
60
61 constexpr BelongsTo(ReferencedRecord const& other) noexcept:
62 _referencedFieldValue { (other.*ReferencedField).Value() },
63 _loaded { true },
64 _record { other }
65 {
66 }
67
68 BelongsTo& operator=(SqlNullType /*nullValue*/) noexcept
69 {
70 if (!_referencedFieldValue)
71 return *this;
72 _loaded = false;
73 _record = std::nullopt;
74 _referencedFieldValue = {};
75 _modified = true;
76 return *this;
77 }
78
79 BelongsTo& operator=(ReferencedRecord& other)
80 {
81 if (_referencedFieldValue == (other.*ReferencedField).Value())
82 return *this;
83 _loaded = true;
84 _record.emplace(other);
85 _referencedFieldValue = (other.*ReferencedField).Value();
86 _modified = true;
87 return *this;
88 }
89
90 // clang-format off
91
92 /// Marks the field as modified or unmodified.
93 LIGHTWEIGHT_FORCE_INLINE constexpr void SetModified(bool value) noexcept { _modified = value; }
94
95 /// Checks if the field is modified.
96 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr bool IsModified() const noexcept { return _modified; }
97
98 /// Retrieves the reference to the value of the field.
99 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ValueType const& Value() const noexcept { return _referencedFieldValue; }
100
101 /// Retrieves the mutable reference to the value of the field.
102 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ValueType& MutableValue() noexcept { return _referencedFieldValue; }
103
104 /// Retrieves a record from the relationship.
105 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord& Record() noexcept { RequireLoaded(); return _record.value(); }
106
107 /// Retrieves an immutable reference to the record from the relationship.
108 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const& Record() const noexcept { RequireLoaded(); return _record.value(); }
109
110 /// Checks if the record is loaded into memory.
111 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr bool IsLoaded() const noexcept { return _loaded; }
112
113 /// Unloads the record from memory.
114 LIGHTWEIGHT_FORCE_INLINE void Unload() noexcept { _record = std::nullopt; _loaded = false; }
115
116 /// Retrieves the record from the relationship.
117 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord& operator*() noexcept { RequireLoaded(); return _record.value(); }
118
119 /// Retrieves the record from the relationship.
120 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const& operator*() const noexcept { RequireLoaded(); return _record.value(); }
121
122 /// Retrieves the record from the relationship.
123 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord* operator->() noexcept { RequireLoaded(); return &_record.value(); }
124
125 /// Retrieves the record from the relationship.
126 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const* operator->() const noexcept { RequireLoaded(); return &_record.value(); }
127
128 /// Checks if the field value is NULL.
129 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr bool operator!() const noexcept { return !_referencedFieldValue; }
130
131 /// Checks if the field value is not NULL.
132 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr explicit operator bool() const noexcept { return static_cast<bool>(_referencedFieldValue); }
133
134 /// Emplaces a record into the relationship. This will mark the relationship as loaded.
135 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord& EmplaceRecord() { _loaded = true; return _record.emplace(); }
136
137 LIGHTWEIGHT_FORCE_INLINE void BindOutputColumn(SQLSMALLINT outputIndex, SqlStatement& stmt) { stmt.BindOutputColumn(outputIndex, &_referencedFieldValue); }
138 // clang-format on
139
140 template <auto OtherReferencedField>
141 std::weak_ordering operator<=>(BelongsTo<OtherReferencedField> const& other) const noexcept
142 {
143 return _referencedFieldValue <=> other.Value();
144 }
145
146 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
147 std::weak_ordering operator<=>(Field<T, IsPrimaryKeyValue> const& other) const noexcept
148 {
149 return _referencedFieldValue <=> other.Value();
150 }
151
152 template <auto OtherReferencedField>
153 bool operator==(BelongsTo<OtherReferencedField> const& other) const noexcept
154 {
155 return (_referencedFieldValue <=> other.Value()) == std::weak_ordering::equivalent;
156 }
157
158 template <auto OtherReferencedField>
159 bool operator!=(BelongsTo<OtherReferencedField> const& other) const noexcept
160 {
161 return (_referencedFieldValue <=> other.Value()) != std::weak_ordering::equivalent;
162 }
163
164 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
165 bool operator==(Field<T, IsPrimaryKeyValue> const& other) const noexcept
166 {
167 return (_referencedFieldValue <=> other.Value()) == std::weak_ordering::equivalent;
168 }
169
170 template <detail::FieldElementType T, PrimaryKey IsPrimaryKeyValue = PrimaryKey::No>
171 bool operator!=(Field<T, IsPrimaryKeyValue> const& other) const noexcept
172 {
173 return (_referencedFieldValue <=> other.Value()) != std::weak_ordering::equivalent;
174 }
175
176 struct Loader
177 {
178 std::function<void()> loadReference {};
179 };
180
181 /// Used internally to configure on-demand loading of the record.
182 void SetAutoLoader(Loader loader) noexcept
183 {
184 _loader = std::move(loader);
185 }
186
187 private:
188 void RequireLoaded()
189 {
190 if (_loaded)
191 return;
192
193 if (_loader.loadReference)
194 _loader.loadReference();
195
196 if (!_loaded)
197 throw SqlRequireLoadedError(Reflection::TypeNameOf<std::remove_cvref_t<decltype(*this)>>);
198 }
199
200 ValueType _referencedFieldValue {};
201 Loader _loader {};
202 bool _loaded = false;
203 bool _modified = false;
204 std::optional<ReferencedRecord> _record {};
205};
206
207template <auto ReferencedField>
208std::ostream& operator<<(std::ostream& os, BelongsTo<ReferencedField> const& belongsTo)
209{
210 return os << belongsTo.Value();
211}
212
213namespace detail
214{
215template <typename T>
216struct IsBelongsTo: std::false_type
217{
218};
219
220template <auto ReferencedField, auto ColumnNameOverrideString>
221struct IsBelongsTo<BelongsTo<ReferencedField, ColumnNameOverrideString>>: std::true_type
222{
223};
224
225} // namespace detail
226
227template <typename T>
228constexpr bool IsBelongsTo = detail::IsBelongsTo<std::remove_cvref_t<T>>::value;
229
230template <auto ReferencedField>
231struct SqlDataBinder<BelongsTo<ReferencedField>>
232{
233 using SelfType = BelongsTo<ReferencedField>;
234 using InnerType = typename SelfType::ValueType;
235
236 static constexpr auto ColumnType = SqlDataBinder<InnerType>::ColumnType;
237
238 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN InputParameter(SQLHSTMT stmt,
239 SQLUSMALLINT column,
240 SelfType const& value,
242 {
243 return SqlDataBinder<InnerType>::InputParameter(stmt, column, value.Value(), cb);
244 }
245
246 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN
247 OutputColumn(SQLHSTMT stmt, SQLUSMALLINT column, SelfType* result, SQLLEN* indicator, SqlDataBinderCallback& cb)
248 {
249 auto const sqlReturn =
250 SqlDataBinder<InnerType>::OutputColumn(stmt, column, &result->MutableValue(), indicator, cb);
251 cb.PlanPostProcessOutputColumn([result]() { result->SetModified(true); });
252 return sqlReturn;
253 }
254
255 static LIGHTWEIGHT_FORCE_INLINE SQLRETURN GetColumn(SQLHSTMT stmt,
256 SQLUSMALLINT column,
257 SelfType* result,
258 SQLLEN* indicator,
259 SqlDataBinderCallback const& cb) noexcept
260 {
261 auto const sqlReturn = SqlDataBinder<InnerType>::GetColumn(stmt, column, &result->emplace(), indicator, cb);
262 if (SQL_SUCCEEDED(sqlReturn))
263 result->SetModified(true);
264 return sqlReturn;
265 }
266};
Represents a one-to-one relationship.
Definition BelongsTo.hpp:26
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & Record() noexcept
Retrieves a record from the relationship.
static constexpr std::string_view ColumnNameOverride
If not an empty string, this value will be used as the column name in the database.
Definition BelongsTo.hpp:32
LIGHTWEIGHT_FORCE_INLINE constexpr ValueType & MutableValue() noexcept
Retrieves the mutable reference to the value of the field.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & operator*() noexcept
Retrieves the record from the relationship.
MemberClassType< decltype(TheReferencedField)> ReferencedRecord
Represents the record type of the other field.
Definition BelongsTo.hpp:40
void SetAutoLoader(Loader loader) noexcept
Used internally to configure on-demand loading of the record.
static constexpr auto ReferencedField
The field in the other record that references the current record.
Definition BelongsTo.hpp:29
LIGHTWEIGHT_FORCE_INLINE constexpr void SetModified(bool value) noexcept
Marks the field as modified or unmodified.
Definition BelongsTo.hpp:93
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & EmplaceRecord()
Emplaces a record into the relationship. This will mark the relationship as loaded.
LIGHTWEIGHT_FORCE_INLINE void Unload() noexcept
Unloads the record from memory.
LIGHTWEIGHT_FORCE_INLINE constexpr ValueType const & Value() const noexcept
Retrieves the reference to the value of the field.
Definition BelongsTo.hpp:99
typename std::remove_cvref_t< decltype(std::declval< ReferencedRecord >().*ReferencedField)>::ValueType ValueType
Represents the column type of the foreign key, matching the primary key of the other record.
Definition BelongsTo.hpp:47
LIGHTWEIGHT_FORCE_INLINE constexpr bool IsLoaded() const noexcept
Checks if the record is loaded into memory.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & operator*() const noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr bool operator!() const noexcept
Checks if the field value is NULL.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord * operator->() noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr bool IsModified() const noexcept
Checks if the field is modified.
Definition BelongsTo.hpp:96
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const * operator->() const noexcept
Retrieves the record from the relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & Record() const noexcept
Retrieves an immutable reference to the record from the relationship.
Represents an error when a record is required to be loaded but is not.
Definition Error.hpp:13
High level API for (prepared) raw SQL statements.
Represents a single column in a table.
Definition Field.hpp:71