Lightweight 0.1.0
Loading...
Searching...
No Matches
Record.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../Utils.hpp"
6#include "Field.hpp"
7
8#include <reflection-cpp/reflection.hpp>
9
10#include <concepts>
11#include <limits>
12
13/// @brief Represents a record type that can be used with the DataMapper.
14///
15/// The record type must be an aggregate type.
16///
17/// @see DataMapper, Field, BelongsTo, HasMany, HasManyThrough, HasOneThrough
18/// @ingroup DataMapper
19template <typename Record>
20concept DataMapperRecord = std::is_aggregate_v<Record>;
21
22namespace detail
23{
24
25template <std::size_t I, typename Record>
26constexpr std::optional<size_t> FindPrimaryKeyIndex()
27{
28 static_assert(DataMapperRecord<Record>, "Record must satisfy DataMapperRecord");
29 if constexpr (I < Reflection::CountMembers<Record>)
30 {
31 if constexpr (IsPrimaryKey<Reflection::MemberTypeOf<I, Record>>)
32 return { I };
33 else
34 return FindPrimaryKeyIndex<I + 1, Record>();
35 }
36 return std::nullopt;
37}
38
39} // namespace detail
40
41/// Declare RecordPrimaryKeyIndex<Record> to retrieve the primary key index of the given record.
42template <typename Record>
43constexpr size_t RecordPrimaryKeyIndex =
44 detail::FindPrimaryKeyIndex<0, Record>().value_or((std::numeric_limits<size_t>::max)());
45
46/// Retrieves a reference to the given record's primary key.
47template <typename Record>
48decltype(auto) RecordPrimaryKeyOf(Record&& record)
49{
50 // static_assert(DataMapperRecord<Record>, "Record must satisfy DataMapperRecord");
51 // static_assert(RecordPrimaryKeyIndex<Record> != static_cast<size_t>(-1), "Record must have a primary key");
52 return Reflection::GetMemberAt<RecordPrimaryKeyIndex<std::remove_cvref_t<Record>>>(std::forward<Record>(record));
53}
54
55namespace details
56{
57
58template <typename Record>
59struct RecordPrimaryKeyTypeHelper
60{
61 using type = void;
62};
63
64template <typename Record>
65 requires(RecordPrimaryKeyIndex<Record> < Reflection::CountMembers<Record>)
66struct RecordPrimaryKeyTypeHelper<Record>
67{
68 using type = typename Reflection::MemberTypeOf<RecordPrimaryKeyIndex<Record>, Record>::ValueType;
69};
70
71} // namespace details
72
73/// Reflects the primary key type of the given record.
74template <typename Record>
75using RecordPrimaryKeyType = typename details::RecordPrimaryKeyTypeHelper<Record>::type;
76
77/// @brief Maps the fields of the given record to the target that supports the operator[].
78template <typename Record, typename TargetMappable>
79void MapFromRecordFields(Record&& record, TargetMappable& target)
80{
81 Reflection::EnumerateMembers(std::forward<Record>(record), [&]<std::size_t I>(auto const& field) {
82 using MemberType = Reflection::MemberTypeOf<I, Record>;
83 static_assert(IsField<MemberType>, "Record member must be a Field<> type");
84 static_assert(std::is_assignable_v<decltype(target[I]), decltype(field.Value())>,
85 "Target must support operator[] with the field type");
86 target[I] = field.Value();
87 });
88}
Represents a record type that can be used with the DataMapper.
Definition Record.hpp:20