Lightweight 0.1.0
Loading...
Searching...
No Matches
HasMany.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 "BelongsTo.hpp"
9#include "Field.hpp"
10#include "Record.hpp"
11
12#include <reflection-cpp/reflection.hpp>
13
14#include <compare>
15#include <memory>
16#include <optional>
17#include <type_traits>
18#include <vector>
19
20/// @brief This HasMany<OtherRecord> represents a simple one-to-many relationship between two records.
21///
22/// The HasMany<OtherRecord> is a member of the "one" side of the relationship.
23///
24/// This implemenation of `HasMany<OtherRecord>` must have only one `BelongsTo` member
25/// that points back to this "one" side.
26///
27/// @see DataMapper, Field, HasManyThrough
28/// @ingroup DataMapper
29template <typename OtherRecord>
31{
32 public:
33 /// The record type of the "many" side of the relationship.
34 using ReferencedRecord = OtherRecord;
35
36 /// The list of records on the "many" side of the relationship.
37 using ReferencedRecordList = std::vector<std::shared_ptr<OtherRecord>>;
38
39 /// Record type of the "many" side of the relationship.
40 using value_type = OtherRecord;
41
42 /// Iterator type for the list of records.
43 using iterator = typename ReferencedRecordList::iterator;
44
45 /// Const iterator type for the list of records.
46 using const_iterator = typename ReferencedRecordList::const_iterator;
47
48 /// Retrieves the list of loaded records.
49 [[nodiscard]] ReferencedRecordList const& All() const noexcept;
50
51 /// Retrieves the list of records as mutable reference.
52 [[nodiscard]] ReferencedRecordList& All() noexcept;
53
54 /// @brief Iterates over the list of records and calls the given callable for each record.
55 ///
56 /// @note Use this method if you want to iterate over all records but do not need to store them all in memory, e.g.
57 /// because the full data set wuold be too large.
58 template <typename Callable>
59 void Each(Callable const& callable);
60
61 /// Emplaces the given list of records.
63
64 /// Retrieves the number of records in this 1-to-many relationship.
65 [[nodiscard]] std::size_t Count() const noexcept;
66
67 /// Checks if this 1-to-many relationship is empty.
68 [[nodiscard]] bool IsEmpty() const noexcept;
69
70 /// @brief Retrieves the record at the given index.
71 ///
72 /// @param index The index of the record to retrieve.
73 /// @note This method will on-demand load the records if they are not already loaded.
74 /// @note This method will throw if the index is out of bounds.
75 [[nodiscard]] OtherRecord const& At(std::size_t index) const;
76
77 /// @brief Retrieves the record at the given index.
78 ///
79 /// @param index The index of the record to retrieve.
80 /// @note This method will on-demand load the records if they are not already loaded.
81 /// @note This method will throw if the index is out of bounds.
82 [[nodiscard]] OtherRecord& At(std::size_t index);
83
84 /// @brief Retrieves the record at the given index.
85 ///
86 /// @param index The index of the record to retrieve.
87 /// @note This method will on-demand load the records if they are not already loaded.
88 /// @note This method will NOT throw if the index is out of bounds. The behaviour is undefined.
89 [[nodiscard]] OtherRecord const& operator[](std::size_t index) const;
90
91 /// @brief Retrieves the record at the given index.
92 ///
93 /// @param index The index of the record to retrieve.
94 /// @note This method will on-demand load the records if they are not already loaded.
95 /// @note This method will NOT throw if the index is out of bounds. The behaviour is undefined.
96 [[nodiscard]] OtherRecord& operator[](std::size_t index);
97
98 iterator begin() noexcept;
99 iterator end() noexcept;
100 const_iterator begin() const noexcept;
101 const_iterator end() const noexcept;
102
103 constexpr std::weak_ordering operator<=>(HasMany<OtherRecord> const& other) const noexcept = default;
104 constexpr bool operator==(HasMany<OtherRecord> const& other) const noexcept = default;
105 constexpr bool operator!=(HasMany<OtherRecord> const& other) const noexcept = default;
106
107 struct Loader
108 {
109 std::function<size_t()> count {};
110 std::function<void()> all {};
111 std::function<void(std::function<void(ReferencedRecord const&)>)> each {};
112 };
113
114 /// Used internally to configure on-demand loading of the records.
115 void SetAutoLoader(Loader loader) noexcept;
116
117 private:
118 void RequireLoaded();
119
120 Loader _loader;
121 std::optional<ReferencedRecordList> _records;
122 std::optional<size_t> _count;
123};
124
125template <typename T>
126constexpr bool IsHasMany = IsSpecializationOf<HasMany, T>;
127
128template <typename OtherRecord>
129inline LIGHTWEIGHT_FORCE_INLINE void HasMany<OtherRecord>::SetAutoLoader(Loader loader) noexcept
130{
131 _loader = std::move(loader);
132}
133
134template <typename OtherRecord>
135inline LIGHTWEIGHT_FORCE_INLINE void HasMany<OtherRecord>::RequireLoaded()
136{
137 if (!_records)
138 _loader.all();
139}
140
141template <typename OtherRecord>
143 ReferencedRecordList&& records) noexcept
144{
145 _records = { std::move(records) };
146 return *_records;
147}
148
149template <typename OtherRecord>
151{
152 RequireLoaded();
153 return *_records;
154}
155
156template <typename OtherRecord>
157template <typename Callable>
158void HasMany<OtherRecord>::Each(Callable const& callable)
159{
160 if (!_records && _loader.each)
161 {
162 _loader.each(callable);
163 return;
164 }
165
166 for (auto const& record: All())
167 callable(*record);
168}
169
170template <typename OtherRecord>
172 const noexcept
173{
174 RequireLoaded();
175 return *_records;
176}
177
178template <typename OtherRecord>
179inline LIGHTWEIGHT_FORCE_INLINE std::size_t HasMany<OtherRecord>::Count() const noexcept
180{
181 if (_records)
182 return _records->size();
183
184 if (!_count)
185 const_cast<HasMany<OtherRecord>*>(this)->_count = _loader.count();
186
187 return *_count;
188}
189
190template <typename OtherRecord>
191inline LIGHTWEIGHT_FORCE_INLINE bool HasMany<OtherRecord>::IsEmpty() const noexcept
192{
193 return Count() == 0;
194}
195
196template <typename OtherRecord>
197inline LIGHTWEIGHT_FORCE_INLINE OtherRecord const& HasMany<OtherRecord>::At(std::size_t index) const
198{
199 RequireLoaded();
200 return *_records->at(index);
201}
202
203template <typename OtherRecord>
204inline LIGHTWEIGHT_FORCE_INLINE OtherRecord& HasMany<OtherRecord>::At(std::size_t index)
205{
206 RequireLoaded();
207 return *_records->at(index);
208}
209
210template <typename OtherRecord>
211inline LIGHTWEIGHT_FORCE_INLINE OtherRecord const& HasMany<OtherRecord>::operator[](std::size_t index) const
212{
213 RequireLoaded();
214 return *(*_records)[index];
215}
216
217template <typename OtherRecord>
218inline LIGHTWEIGHT_FORCE_INLINE OtherRecord& HasMany<OtherRecord>::operator[](std::size_t index)
219{
220 RequireLoaded();
221 return *(*_records)[index];
222}
223
224template <typename OtherRecord>
225inline LIGHTWEIGHT_FORCE_INLINE HasMany<OtherRecord>::iterator HasMany<OtherRecord>::begin() noexcept
226{
227 RequireLoaded();
228 return _records->begin();
229}
230
231template <typename OtherRecord>
232inline LIGHTWEIGHT_FORCE_INLINE HasMany<OtherRecord>::iterator HasMany<OtherRecord>::end() noexcept
233{
234 RequireLoaded();
235 return _records->end();
236}
237
238template <typename OtherRecord>
239inline LIGHTWEIGHT_FORCE_INLINE HasMany<OtherRecord>::const_iterator HasMany<OtherRecord>::begin() const noexcept
240{
241 RequireLoaded();
242 return _records->begin();
243}
244
245template <typename OtherRecord>
246inline LIGHTWEIGHT_FORCE_INLINE HasMany<OtherRecord>::const_iterator HasMany<OtherRecord>::end() const noexcept
247{
248 RequireLoaded();
249 return _records->end();
250}
This HasMany<OtherRecord> represents a simple one-to-many relationship between two records.
Definition HasMany.hpp:31
typename ReferencedRecordList::iterator iterator
Iterator type for the list of records.
Definition HasMany.hpp:43
ReferencedRecordList const & All() const noexcept
Retrieves the list of loaded records.
Definition HasMany.hpp:171
OtherRecord value_type
Record type of the "many" side of the relationship.
Definition HasMany.hpp:40
bool IsEmpty() const noexcept
Checks if this 1-to-many relationship is empty.
Definition HasMany.hpp:191
std::size_t Count() const noexcept
Retrieves the number of records in this 1-to-many relationship.
Definition HasMany.hpp:179
void Each(Callable const &callable)
Iterates over the list of records and calls the given callable for each record.
Definition HasMany.hpp:158
void SetAutoLoader(Loader loader) noexcept
Used internally to configure on-demand loading of the records.
Definition HasMany.hpp:129
std::vector< std::shared_ptr< OtherRecord > > ReferencedRecordList
The list of records on the "many" side of the relationship.
Definition HasMany.hpp:37
ReferencedRecordList & Emplace(ReferencedRecordList &&records) noexcept
Emplaces the given list of records.
Definition HasMany.hpp:142
typename ReferencedRecordList::const_iterator const_iterator
Const iterator type for the list of records.
Definition HasMany.hpp:46
OtherRecord ReferencedRecord
The record type of the "many" side of the relationship.
Definition HasMany.hpp:34
OtherRecord const & At(std::size_t index) const
Retrieves the record at the given index.
Definition HasMany.hpp:197
OtherRecord const & operator[](std::size_t index) const
Retrieves the record at the given index.
Definition HasMany.hpp:211