Lightweight 0.1.0
Loading...
Searching...
No Matches
Select.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../SqlQueryFormatter.hpp"
6#include "../Utils.hpp"
7#include "Core.hpp"
8
9#include <reflection-cpp/reflection.hpp>
10
11enum class SqlQueryBuilderMode : uint8_t
12{
13 Fluent,
14 Varying
15};
16
17struct [[nodiscard]] SqlFieldExpression final
18{
19 std::string expression;
20};
21
22namespace Aggregate
23{
24
25inline SqlFieldExpression Count(std::string const& field = "*") noexcept
26{
27 return SqlFieldExpression { .expression = std::format("COUNT(\"{}\")", field) };
28}
29
30inline SqlFieldExpression Count(SqlQualifiedTableColumnName const& field) noexcept
31{
32 return SqlFieldExpression { .expression = std::format(R"(COUNT("{}"."{}"))", field.tableName, field.columnName) };
33}
34
35inline SqlFieldExpression Sum(std::string const& field) noexcept
36{
37 return SqlFieldExpression { .expression = std::format("SUM(\"{}\")", field) };
38}
39
40inline SqlFieldExpression Sum(SqlQualifiedTableColumnName const& field) noexcept
41{
42 return SqlFieldExpression { .expression = std::format(R"(SUM("{}"."{}"))", field.tableName, field.columnName) };
43}
44
45inline SqlFieldExpression Avg(std::string const& field) noexcept
46{
47 return SqlFieldExpression { .expression = std::format("AVG(\"{}\")", field) };
48}
49
50inline SqlFieldExpression Avg(SqlQualifiedTableColumnName const& field) noexcept
51{
52 return SqlFieldExpression { .expression = std::format(R"(AVG("{}"."{}"))", field.tableName, field.columnName) };
53}
54
55inline SqlFieldExpression Min(std::string const& field) noexcept
56{
57 return SqlFieldExpression { .expression = std::format("MIN(\"{}\")", field) };
58}
59
60inline SqlFieldExpression Min(SqlQualifiedTableColumnName const& field) noexcept
61{
62 return SqlFieldExpression { .expression = std::format(R"(MIN("{}"."{}"))", field.tableName, field.columnName) };
63}
64
65inline SqlFieldExpression Max(std::string const& field) noexcept
66{
67 return SqlFieldExpression { .expression = std::format("MAX(\"{}\")", field) };
68}
69
70inline SqlFieldExpression Max(SqlQualifiedTableColumnName const& field) noexcept
71{
72 return SqlFieldExpression { .expression = std::format(R"(MAX("{}"."{}"))", field.tableName, field.columnName) };
73}
74
75} // namespace Aggregate
76
77/// @brief Query builder for building SELECT ... queries.
78///
79/// @see SqlQueryBuilder
80class [[nodiscard]] SqlSelectQueryBuilder final: public SqlBasicSelectQueryBuilder<SqlSelectQueryBuilder>
81{
82 public:
83 using SelectType = detail::SelectType;
84
85 explicit SqlSelectQueryBuilder(SqlQueryFormatter const& formatter,
86 std::string table,
87 std::string tableAlias) noexcept:
88 SqlBasicSelectQueryBuilder<SqlSelectQueryBuilder> {},
89 _formatter { formatter }
90 {
91 _query.formatter = &formatter;
92 _query.searchCondition.tableName = std::move(table);
93 _query.searchCondition.tableAlias = std::move(tableAlias);
94 _query.fields.reserve(256);
95 }
96
97 /// Sets the builder mode to Varying, allowing varying final query types.
98 constexpr LIGHTWEIGHT_FORCE_INLINE SqlSelectQueryBuilder& Varying() noexcept
99 {
100 _mode = SqlQueryBuilderMode::Varying;
101 return *this;
102 }
103
104 /// Adds a sequence of columns to the SELECT clause.
105 template <typename... MoreFields>
106 SqlSelectQueryBuilder& Fields(std::string_view const& firstField, MoreFields&&... moreFields);
107
108 /// Adds a single column to the SELECT clause.
109 LIGHTWEIGHT_API SqlSelectQueryBuilder& Field(std::string_view const& fieldName);
110
111 /// Adds a single column to the SELECT clause.
112 LIGHTWEIGHT_API SqlSelectQueryBuilder& Field(SqlQualifiedTableColumnName const& fieldName);
113
114 /// Adds an aggregate function call to the SELECT clause.
115 LIGHTWEIGHT_API SqlSelectQueryBuilder& Field(SqlFieldExpression const& fieldExpression);
116
117 /// Aliases the last added field (a column or an aggregate call) in the SELECT clause.
118 LIGHTWEIGHT_API SqlSelectQueryBuilder& As(std::string_view alias);
119
120 /// Adds a sequence of columns to the SELECT clause.
121 LIGHTWEIGHT_API SqlSelectQueryBuilder& Fields(std::vector<std::string_view> const& fieldNames);
122
123 /// Adds a sequence of columns from the given table to the SELECT clause.
124 LIGHTWEIGHT_API SqlSelectQueryBuilder& Fields(std::vector<std::string_view> const& fieldNames,
125 std::string_view tableName);
126
127 LIGHTWEIGHT_API SqlSelectQueryBuilder& Fields(std::initializer_list<std::string_view> const& fieldNames,
128 std::string_view tableName);
129
130 /// Adds a sequence of columns from the given tables to the SELECT clause.
131 template <typename FirstRecord, typename... MoreRecords>
133
134 /// Adds a single column with an alias to the SELECT clause.
135 [[deprecated("Use Field(...).As(\"alias\") instead.")]]
136 LIGHTWEIGHT_API SqlSelectQueryBuilder& FieldAs(std::string_view const& fieldName, std::string_view const& alias);
137
138 /// Adds a single column with an alias to the SELECT clause.
139 [[deprecated("Use Field(...).As(\"alias\") instead.")]]
141 std::string_view const& alias);
142
143 template <typename Callable>
144 SqlSelectQueryBuilder& Build(Callable const& callable);
145
146 /// Finalizes building the query as SELECT COUNT(*) ... query.
147 LIGHTWEIGHT_API ComposedQuery Count();
148
149 /// Finalizes building the query as SELECT field names FROM ... query.
150 LIGHTWEIGHT_API ComposedQuery All();
151
152 /// Finalizes building the query as SELECT TOP n field names FROM ... query.
153 LIGHTWEIGHT_API ComposedQuery First(size_t count = 1);
154
155 /// Finalizes building the query as SELECT field names FROM ... query with a range.
156 LIGHTWEIGHT_API ComposedQuery Range(std::size_t offset, std::size_t limit);
157
158 LIGHTWEIGHT_FORCE_INLINE SqlSearchCondition& SearchCondition() noexcept
159 {
160 return _query.searchCondition;
161 }
162
163 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE SqlQueryFormatter const& Formatter() const noexcept
164 {
165 return _formatter;
166 }
167
168 private:
169 SqlQueryFormatter const& _formatter;
170 SqlQueryBuilderMode _mode = SqlQueryBuilderMode::Fluent;
171 bool _aliasAllowed = false;
172};
173
174template <typename... MoreFields>
175SqlSelectQueryBuilder& SqlSelectQueryBuilder::Fields(std::string_view const& firstField, MoreFields&&... moreFields)
176{
177 using namespace std::string_view_literals;
178
179 std::ostringstream fragment;
180
181 if (!_query.fields.empty())
182 fragment << ", "sv;
183
184 fragment << '"' << firstField << '"';
185
186 if constexpr (sizeof...(MoreFields) > 0)
187 ((fragment << R"(, ")"sv << std::forward<MoreFields>(moreFields) << '"') << ...);
188
189 _query.fields += fragment.str();
190 return *this;
191}
192
193template <typename Callable>
194inline LIGHTWEIGHT_FORCE_INLINE SqlSelectQueryBuilder& SqlSelectQueryBuilder::Build(Callable const& callable)
195{
196 callable(*this);
197 return *this;
198}
199
200template <typename FirstRecord, typename... MoreRecords>
201inline LIGHTWEIGHT_FORCE_INLINE SqlSelectQueryBuilder& SqlSelectQueryBuilder::Fields()
202{
203 if constexpr (sizeof...(MoreRecords) == 0)
204 {
205 Reflection::EnumerateMembers<FirstRecord>(
206 [&]<size_t FieldIndex, typename FieldType>() { Field(FieldNameAt<FieldIndex, FirstRecord>); });
207 }
208 else
209 {
210 Reflection::EnumerateMembers<FirstRecord>([&]<size_t FieldIndex, typename FieldType>() {
212 .tableName = RecordTableName<FirstRecord>,
213 .columnName = FieldNameAt<FieldIndex, FirstRecord>,
214 });
215 });
216
217 (Reflection::EnumerateMembers<MoreRecords>([&]<size_t FieldIndex, typename FieldType>() {
219 .tableName = RecordTableName<MoreRecords>,
220 .columnName = FieldNameAt<FieldIndex, MoreRecords>,
221 });
222 }),
223 ...);
224 }
225 return *this;
226}
API to format SQL queries for different SQL dialects.
Query builder for building SELECT ... queries.
Definition Select.hpp:81
LIGHTWEIGHT_API SqlSelectQueryBuilder & Fields(std::vector< std::string_view > const &fieldNames, std::string_view tableName)
Adds a sequence of columns from the given table to the SELECT clause.
LIGHTWEIGHT_API ComposedQuery Count()
Finalizes building the query as SELECT COUNT(*) ... query.
LIGHTWEIGHT_API SqlSelectQueryBuilder & As(std::string_view alias)
Aliases the last added field (a column or an aggregate call) in the SELECT clause.
LIGHTWEIGHT_API SqlSelectQueryBuilder & Fields(std::vector< std::string_view > const &fieldNames)
Adds a sequence of columns to the SELECT clause.
LIGHTWEIGHT_API SqlSelectQueryBuilder & Field(SqlQualifiedTableColumnName const &fieldName)
Adds a single column to the SELECT clause.
LIGHTWEIGHT_API SqlSelectQueryBuilder & FieldAs(std::string_view const &fieldName, std::string_view const &alias)
Adds a single column with an alias to the SELECT clause.
LIGHTWEIGHT_API ComposedQuery Range(std::size_t offset, std::size_t limit)
Finalizes building the query as SELECT field names FROM ... query with a range.
constexpr LIGHTWEIGHT_FORCE_INLINE SqlSelectQueryBuilder & Varying() noexcept
Sets the builder mode to Varying, allowing varying final query types.
Definition Select.hpp:98
LIGHTWEIGHT_API ComposedQuery All()
Finalizes building the query as SELECT field names FROM ... query.
LIGHTWEIGHT_API SqlSelectQueryBuilder & Field(std::string_view const &fieldName)
Adds a single column to the SELECT clause.
LIGHTWEIGHT_API SqlSelectQueryBuilder & Field(SqlFieldExpression const &fieldExpression)
Adds an aggregate function call to the SELECT clause.
SqlSelectQueryBuilder & Fields()
Adds a sequence of columns from the given tables to the SELECT clause.
LIGHTWEIGHT_API ComposedQuery First(size_t count=1)
Finalizes building the query as SELECT TOP n field names FROM ... query.
LIGHTWEIGHT_API SqlSelectQueryBuilder & FieldAs(SqlQualifiedTableColumnName const &fieldName, std::string_view const &alias)
Adds a single column with an alias to the SELECT clause.
Represents a single column in a table.
Definition Field.hpp:71
SqlQualifiedTableColumnName represents a column name qualified with a table name.
Definition Core.hpp:40