Lightweight 0.1.0
Loading...
Searching...
No Matches
HasOneThrough.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlStatement.hpp"
6#include "../Utils.hpp"
7#include "Error.hpp"
8
9#include <compare>
10#include <memory>
11#include <type_traits>
12
13/// @brief Represents a one-to-one relationship through a join table.
14///
15/// The `OtherField` parameter is the field in the join table that references the other record.
16/// The `ThroughField` parameter is the field in the join table that references the current record.
17///
18/// @ingroup DataMapper
19template <typename OtherTable, typename ThroughTable>
21{
22 public:
23 /// The record type of the "through" side of the relationship.
24 using ThroughRecord = ThroughTable;
25
26 /// The record type of the "Other" side of the relationship.
27 using ReferencedRecord = OtherTable;
28
29 // clang-format off
30
31 /// Emplaces the given record into this relationship.
32 LIGHTWEIGHT_FORCE_INLINE constexpr void EmplaceRecord(std::shared_ptr<ReferencedRecord> record) { _record = std::move(record); }
33
34 /// Retrieves the record in this relationship.
35 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord& Record() noexcept { RequireLoaded(); return *_record.get(); }
36
37 /// Retrieves the record in this relationship.
38 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const& Record() const noexcept { RequireLoaded(); return *_record.get(); }
39
40 /// Checks if the record is loaded.
41 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr bool IsLoaded() const noexcept { return _record.get() != nullptr; }
42
43 /// Unloads the record from memory.
44 LIGHTWEIGHT_FORCE_INLINE void Unload() noexcept { _record = std::nullopt; }
45
46 /// @brief Retrieves the record in this relationship.
47 /// @note On-demand loads the record if it is not already loaded.
48 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord& operator*() noexcept { RequireLoaded(); return *_record; }
49
50 /// @brief Retrieves the record in this relationship.
51 /// @note On-demand loads the record if it is not already loaded.
52 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const& operator*() const noexcept { RequireLoaded(); return *_record; }
53
54 /// @brief Retrieves the record in this relationship.
55 /// @note On-demand loads the record if it is not already loaded.
56 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord* operator->() noexcept { RequireLoaded(); return &_record.get(); }
57
58 /// @brief Retrieves the record in this relationship.
59 /// @note On-demand loads the record if it is not already loaded.
60 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const* operator->() const noexcept { RequireLoaded(); return &_record.get(); }
61 // clang-format on
62
63 std::weak_ordering operator<=>(HasOneThrough const& other) const noexcept = default;
64
65 struct Loader
66 {
67 std::function<void()> loadReference {};
68 };
69
70 /// Used internally to configure on-demand loading of the record.
71 void SetAutoLoader(Loader loader)
72 {
73 _loader = std::move(loader);
74 }
75
76 private:
77 void RequireLoaded() const
78 {
79 if (IsLoaded())
80 return;
81
82 if (_loader.loadReference)
83 _loader.loadReference();
84
85 if (!IsLoaded())
86 throw SqlRequireLoadedError { Reflection::TypeNameOf<std::remove_cvref_t<decltype(*this)>> };
87 }
88
89 Loader _loader {};
90
91 // We use shared_ptr to not require ReferencedRecord to be declared before HasOneThrough.
92 std::shared_ptr<ReferencedRecord> _record {};
93};
94
95namespace detail
96{
97template <typename T>
98struct IsHasOneThrough: std::false_type
99{
100};
101
102template <typename OtherTable, typename ThroughTable>
103struct IsHasOneThrough<HasOneThrough<OtherTable, ThroughTable>>: std::true_type
104{
105};
106} // namespace detail
107
108template <typename T>
109constexpr bool IsHasOneThrough = detail::IsHasOneThrough<std::remove_cvref_t<T>>::value;
Represents a one-to-one relationship through a join table.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & operator*() noexcept
Retrieves the record in this relationship.
OtherTable ReferencedRecord
The record type of the "Other" side of the relationship.
ThroughTable ThroughRecord
The record type of the "through" side of the relationship.
void SetAutoLoader(Loader loader)
Used internally to configure on-demand loading of the record.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord * operator->() noexcept
Retrieves the record in this relationship.
LIGHTWEIGHT_FORCE_INLINE void Unload() noexcept
Unloads the record from memory.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & operator*() const noexcept
Retrieves the record in this relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const * operator->() const noexcept
Retrieves the record in this relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord const & Record() const noexcept
Retrieves the record in this relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr ReferencedRecord & Record() noexcept
Retrieves the record in this relationship.
LIGHTWEIGHT_FORCE_INLINE constexpr bool IsLoaded() const noexcept
Checks if the record is loaded.
LIGHTWEIGHT_FORCE_INLINE constexpr void EmplaceRecord(std::shared_ptr< ReferencedRecord > record)
Emplaces the given record into this relationship.
Represents an error when a record is required to be loaded but is not.
Definition Error.hpp:13