Lightweight 0.1.0
Loading...
Searching...
No Matches
Core.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../Api.hpp"
6#include "../SqlQueryFormatter.hpp"
7
8#include <concepts>
9#include <ranges>
10
11/// @defgroup QueryBuilder Query Builder
12///
13/// @brief The query builder is a high level API for building SQL queries using high level C++ syntax.
14
15/// @brief SqlWildcardType is a placeholder for an explicit wildcard input parameter in a SQL query.
16///
17/// Use this in the SqlQueryBuilder::Where method to insert a '?' placeholder for a wildcard.
18///
19/// @ingroup QueryBuilder
21{
22};
23
24/// @brief SqlWildcard is a placeholder for an explicit wildcard input parameter in a SQL query.
25static constexpr inline auto SqlWildcard = SqlWildcardType {};
26
27namespace detail
28{
29
30struct RawSqlCondition
31{
32 std::string condition;
33};
34
35} // namespace detail
36
37/// @brief SqlQualifiedTableColumnName represents a column name qualified with a table name.
38/// @ingroup QueryBuilder
40{
41 std::string_view tableName;
42 std::string_view columnName;
43};
44
45namespace detail
46{
47
48template <typename ColumnName>
49std::string MakeSqlColumnName(ColumnName const& columnName)
50{
51 using namespace std::string_view_literals;
52 std::string output;
53
54 if constexpr (std::is_same_v<ColumnName, SqlQualifiedTableColumnName>)
55 {
56 output.reserve(columnName.tableName.size() + columnName.columnName.size() + 5);
57 output += '"';
58 output += columnName.tableName;
59 output += R"(".")"sv;
60 output += columnName.columnName;
61 output += '"';
62 }
63 else if constexpr (std::is_same_v<ColumnName, SqlRawColumnNameView>)
64 {
65 output += columnName.value;
66 }
67 else if constexpr (std::is_same_v<ColumnName, SqlWildcardType>)
68 {
69 output += '?';
70 }
71 else
72 {
73 output += '"';
74 output += columnName;
75 output += '"';
76 }
77 return output;
78}
79
80} // namespace detail
81
82struct [[nodiscard]] SqlSearchCondition
83{
84 std::string tableName;
85 std::string tableAlias;
86 std::string tableJoins;
87 std::string condition;
88 std::vector<SqlVariant>* inputBindings = nullptr;
89};
90
91/// @brief Query builder for building JOIN conditions.
92/// @ingroup QueryBuilder
94{
95 public:
96 explicit SqlJoinConditionBuilder(std::string_view referenceTable, std::string* condition) noexcept:
97 _referenceTable { referenceTable },
98 _condition { *condition }
99 {
100 }
101
102 SqlJoinConditionBuilder& On(std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
103 {
104 return Operator(joinColumnName, onOtherColumn, "AND");
105 }
106
107 SqlJoinConditionBuilder& OrOn(std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
108 {
109 return Operator(joinColumnName, onOtherColumn, "OR");
110 }
111
112 SqlJoinConditionBuilder& Operator(std::string_view joinColumnName,
113 SqlQualifiedTableColumnName onOtherColumn,
114 std::string_view op)
115 {
116 if (_firstCall)
117 _firstCall = !_firstCall;
118 else
119 _condition += std::format(" {} ", op);
120
121 _condition += '"';
122 _condition += _referenceTable;
123 _condition += "\".\"";
124 _condition += joinColumnName;
125 _condition += "\" = \"";
126 _condition += onOtherColumn.tableName;
127 _condition += "\".\"";
128 _condition += onOtherColumn.columnName;
129 _condition += '"';
130
131 return *this;
132 }
133
134 private:
135 std::string_view _referenceTable;
136 std::string& _condition;
137 bool _firstCall = true;
138};
139
140/// Helper CRTP-based class for building WHERE clauses.
141///
142/// This class is inherited by the SqlSelectQueryBuilder, SqlUpdateQueryBuilder, and SqlDeleteQueryBuilder
143///
144/// @ingroup QueryBuilder
145template <typename Derived>
146class [[nodiscard]] SqlWhereClauseBuilder
147{
148 public:
149 /// Indicates, that the next WHERE clause should be AND-ed (default).
150 [[nodiscard]] Derived& And() noexcept;
151
152 /// Indicates, that the next WHERE clause should be OR-ed.
153 [[nodiscard]] Derived& Or() noexcept;
154
155 /// Indicates, that the next WHERE clause should be negated.
156 [[nodiscard]] Derived& Not() noexcept;
157
158 /// Constructs or extends a raw WHERE clause.
159 [[nodiscard]] Derived& WhereRaw(std::string_view sqlConditionExpression);
160
161 /// Constructs or extends a WHERE clause to test for a binary operation.
162 template <typename ColumnName, typename T>
163 [[nodiscard]] Derived& Where(ColumnName const& columnName, std::string_view binaryOp, T const& value);
164
165 /// Constructs or extends a WHERE clause to test for a binary operation for RHS as sub-select query.
166 template <typename ColumnName, typename SubSelectQuery>
167 requires(std::is_invocable_r_v<std::string, decltype(&SubSelectQuery::ToSql), SubSelectQuery const&>)
168 [[nodiscard]] Derived& Where(ColumnName const& columnName, std::string_view binaryOp, SubSelectQuery const& value);
169
170 /// Constructs or extends a WHERE/OR clause to test for a binary operation.
171 template <typename ColumnName, typename T>
172 [[nodiscard]] Derived& OrWhere(ColumnName const& columnName, std::string_view binaryOp, T const& value);
173
174 /// Constructs or extends a WHERE clause to test for a binary operation for RHS as string literal.
175 template <typename ColumnName, std::size_t N>
176 Derived& Where(ColumnName const& columnName, std::string_view binaryOp, char const (&value)[N]);
177
178 /// Constructs or extends a WHERE clause to test for equality.
179 template <typename ColumnName, typename T>
180 [[nodiscard]] Derived& Where(ColumnName const& columnName, T const& value);
181
182 /// Constructs or extends an WHERE/OR clause to test for equality.
183 template <typename ColumnName, typename T>
184 [[nodiscard]] Derived& OrWhere(ColumnName const& columnName, T const& value);
185
186 /// Constructs or extends a WHERE/AND clause to test for a group of values.
187 template <typename Callable>
188 requires std::invocable<Callable, SqlWhereClauseBuilder<Derived>&>
189 [[nodiscard]] Derived& Where(Callable const& callable);
190
191 /// Constructs or extends an WHERE/OR clause to test for a group of values.
192 template <typename Callable>
193 requires std::invocable<Callable, SqlWhereClauseBuilder<Derived>&>
194 [[nodiscard]] Derived& OrWhere(Callable const& callable);
195
196 /// Constructs or extends an WHERE/OR clause to test for a value, satisfying std::ranges::input_range.
197 template <typename ColumnName, std::ranges::input_range InputRange>
198 [[nodiscard]] Derived& WhereIn(ColumnName const& columnName, InputRange const& values);
199
200 /// Constructs or extends an WHERE/OR clause to test for a value, satisfying std::initializer_list.
201 template <typename ColumnName, typename T>
202 [[nodiscard]] Derived& WhereIn(ColumnName const& columnName, std::initializer_list<T> const& values);
203
204 /// Constructs or extends an WHERE/OR clause to test for a value, satisfying a sub-select query.
205 template <typename ColumnName, typename SubSelectQuery>
206 requires(std::is_invocable_r_v<std::string, decltype(&SubSelectQuery::ToSql), SubSelectQuery const&>)
207 [[nodiscard]] Derived& WhereIn(ColumnName const& columnName, SubSelectQuery const& subSelectQuery);
208
209 /// Constructs or extends an WHERE/OR clause to test for a value to be NULL.
210 template <typename ColumnName>
211 [[nodiscard]] Derived& WhereNull(ColumnName const& columnName);
212
213 /// Constructs or extends a WHERE clause to test for a value being not null.
214 template <typename ColumnName>
215 [[nodiscard]] Derived& WhereNotNull(ColumnName const& columnName);
216
217 /// Constructs or extends a WHERE clause to test for a value being equal to another column.
218 template <typename ColumnName, typename T>
219 [[nodiscard]] Derived& WhereNotEqual(ColumnName const& columnName, T const& value);
220
221 /// Constructs or extends a WHERE clause to test for a value being true.
222 template <typename ColumnName>
223 [[nodiscard]] Derived& WhereTrue(ColumnName const& columnName);
224
225 /// Constructs or extends a WHERE clause to test for a value being false.
226 template <typename ColumnName>
227 [[nodiscard]] Derived& WhereFalse(ColumnName const& columnName);
228
229 /// Construts or extends a WHERE clause to test for a binary operation between two columns.
230 template <typename LeftColumn, typename RightColumn>
231 [[nodiscard]] Derived& WhereColumn(LeftColumn const& left, std::string_view binaryOp, RightColumn const& right);
232
233 /// Constructs an INNER JOIN clause.
234 [[nodiscard]] Derived& InnerJoin(std::string_view joinTable,
235 std::string_view joinColumnName,
236 SqlQualifiedTableColumnName onOtherColumn);
237
238 /// Constructs an INNER JOIN clause.
239 [[nodiscard]] Derived& InnerJoin(std::string_view joinTable,
240 std::string_view joinColumnName,
241 std::string_view onMainTableColumn);
242
243 /// Constructs an INNER JOIN clause with a custom ON clause.
244 template <typename OnChainCallable>
245 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
246 [[nodiscard]] Derived& InnerJoin(std::string_view joinTable, OnChainCallable const& onClauseBuilder);
247
248 /// Constructs an LEFT OUTER JOIN clause.
249 [[nodiscard]] Derived& LeftOuterJoin(std::string_view joinTable,
250 std::string_view joinColumnName,
251 SqlQualifiedTableColumnName onOtherColumn);
252
253 /// Constructs an LEFT OUTER JOIN clause.
254 [[nodiscard]] Derived& LeftOuterJoin(std::string_view joinTable,
255 std::string_view joinColumnName,
256 std::string_view onMainTableColumn);
257
258 /// Constructs an LEFT OUTER JOIN clause with a custom ON clause.
259 template <typename OnChainCallable>
260 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
261 [[nodiscard]] Derived& LeftOuterJoin(std::string_view joinTable, OnChainCallable const& onClauseBuilder);
262
263 /// Constructs an RIGHT OUTER JOIN clause.
264 [[nodiscard]] Derived& RightOuterJoin(std::string_view joinTable,
265 std::string_view joinColumnName,
266 SqlQualifiedTableColumnName onOtherColumn);
267
268 /// Constructs an RIGHT OUTER JOIN clause.
269 [[nodiscard]] Derived& RightOuterJoin(std::string_view joinTable,
270 std::string_view joinColumnName,
271 std::string_view onMainTableColumn);
272
273 /// Constructs an RIGHT OUTER JOIN clause with a custom ON clause.
274 template <typename OnChainCallable>
275 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
276 [[nodiscard]] Derived& RightOuterJoin(std::string_view joinTable, OnChainCallable const& onClauseBuilder);
277
278 /// Constructs an FULL OUTER JOIN clause.
279 [[nodiscard]] Derived& FullOuterJoin(std::string_view joinTable,
280 std::string_view joinColumnName,
281 SqlQualifiedTableColumnName onOtherColumn);
282
283 /// Constructs an FULL OUTER JOIN clause.
284 [[nodiscard]] Derived& FullOuterJoin(std::string_view joinTable,
285 std::string_view joinColumnName,
286 std::string_view onMainTableColumn);
287
288 /// Constructs an FULL OUTER JOIN clause with a custom ON clause.
289 template <typename OnChainCallable>
290 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
291 [[nodiscard]] Derived& FullOuterJoin(std::string_view joinTable, OnChainCallable const& onClauseBuilder);
292
293 private:
294 SqlSearchCondition& SearchCondition() noexcept;
295 [[nodiscard]] SqlQueryFormatter const& Formatter() const noexcept;
296
297 enum class WhereJunctor : uint8_t
298 {
299 Null,
300 Where,
301 And,
302 Or,
303 };
304
305 WhereJunctor m_nextWhereJunctor = WhereJunctor::Where;
306 bool m_nextIsNot = false;
307
308 void AppendWhereJunctor();
309
310 template <typename ColumnName>
311 requires(std::same_as<ColumnName, SqlQualifiedTableColumnName>
312 || std::convertible_to<ColumnName, std::string_view> || std::same_as<ColumnName, SqlRawColumnNameView>
313 || std::convertible_to<ColumnName, std::string>)
314 void AppendColumnName(ColumnName const& columnName);
315
316 template <typename LiteralType>
317 void AppendLiteralValue(LiteralType const& value);
318
319 enum class JoinType : uint8_t
320 {
321 INNER,
322 LEFT,
323 RIGHT,
324 FULL
325 };
326
327 /// Constructs a JOIN clause.
328 [[nodiscard]] Derived& Join(JoinType joinType,
329 std::string_view joinTable,
330 std::string_view joinColumnName,
331 SqlQualifiedTableColumnName onOtherColumn);
332
333 /// Constructs a JOIN clause.
334 [[nodiscard]] Derived& Join(JoinType joinType,
335 std::string_view joinTable,
336 std::string_view joinColumnName,
337 std::string_view onMainTableColumn);
338
339 /// Constructs a JOIN clause.
340 template <typename OnChainCallable>
341 [[nodiscard]] Derived& Join(JoinType joinType, std::string_view joinTable, OnChainCallable const& onClauseBuilder);
342};
343
344enum class SqlResultOrdering : uint8_t
345{
346 ASCENDING,
347 DESCENDING
348};
349
350namespace detail
351{
352enum class SelectType : std::uint8_t
353{
354 Undefined,
355 Count,
356 All,
357 First,
358 Range
359};
360
361struct ComposedQuery
362{
363 SelectType selectType = SelectType::Undefined;
364 SqlQueryFormatter const* formatter = nullptr;
365
366 bool distinct = false;
367 SqlSearchCondition searchCondition {};
368
369 std::string fields;
370
371 std::string orderBy;
372 std::string groupBy;
373
374 size_t offset = 0;
375 size_t limit = (std::numeric_limits<size_t>::max)();
376
377 [[nodiscard]] LIGHTWEIGHT_API std::string ToSql() const;
378};
379} // namespace detail
380
381template <typename Derived>
382class [[nodiscard]] SqlBasicSelectQueryBuilder: public SqlWhereClauseBuilder<Derived>
383{
384 public:
385 /// Adds a DISTINCT clause to the SELECT query.
386 Derived& Distinct() noexcept;
387
388 /// Constructs or extends a ORDER BY clause.
389 Derived& OrderBy(SqlQualifiedTableColumnName const& columnName,
390 SqlResultOrdering ordering = SqlResultOrdering::ASCENDING);
391
392 /// Constructs or extends a ORDER BY clause.
393 Derived& OrderBy(std::string_view columnName, SqlResultOrdering ordering = SqlResultOrdering::ASCENDING);
394
395 /// Constructs or extends a GROUP BY clause.
396 Derived& GroupBy(std::string_view columnName);
397
398 using ComposedQuery = detail::ComposedQuery;
399
400 protected:
401 ComposedQuery _query {}; // NOLINT(cppcoreguidelines-non-private-member-variables-in-classes)
402};
403
404template <typename Derived>
405inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlBasicSelectQueryBuilder<Derived>::Distinct() noexcept
406{
407 _query.distinct = true;
408 return static_cast<Derived&>(*this);
409}
410
411template <typename Derived>
412inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlBasicSelectQueryBuilder<Derived>::OrderBy(std::string_view columnName,
413 SqlResultOrdering ordering)
414{
415 if (_query.orderBy.empty())
416 _query.orderBy += "\n ORDER BY ";
417 else
418 _query.orderBy += ", ";
419
420 _query.orderBy += '"';
421 _query.orderBy += columnName;
422 _query.orderBy += '"';
423
424 if (ordering == SqlResultOrdering::DESCENDING)
425 _query.orderBy += " DESC";
426 else if (ordering == SqlResultOrdering::ASCENDING)
427 _query.orderBy += " ASC";
428
429 return static_cast<Derived&>(*this);
430}
431
432template <typename Derived>
433inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlBasicSelectQueryBuilder<Derived>::OrderBy(
434 SqlQualifiedTableColumnName const& columnName, SqlResultOrdering ordering)
435{
436 if (_query.orderBy.empty())
437 _query.orderBy += "\n ORDER BY ";
438 else
439 _query.orderBy += ", ";
440
441 _query.orderBy += '"';
442 _query.orderBy += columnName.tableName;
443 _query.orderBy += "\".\"";
444 _query.orderBy += columnName.columnName;
445 _query.orderBy += '"';
446
447 if (ordering == SqlResultOrdering::DESCENDING)
448 _query.orderBy += " DESC";
449 else if (ordering == SqlResultOrdering::ASCENDING)
450 _query.orderBy += " ASC";
451
452 return static_cast<Derived&>(*this);
453}
454
455template <typename Derived>
456inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlBasicSelectQueryBuilder<Derived>::GroupBy(std::string_view columnName)
457{
458 if (_query.groupBy.empty())
459 _query.groupBy += "\n GROUP BY ";
460 else
461 _query.groupBy += ", ";
462
463 _query.groupBy += '"';
464 _query.groupBy += columnName;
465 _query.groupBy += '"';
466
467 return static_cast<Derived&>(*this);
468}
469
470template <typename Derived>
471inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::And() noexcept
472{
473 m_nextWhereJunctor = WhereJunctor::And;
474 return static_cast<Derived&>(*this);
475}
476
477template <typename Derived>
478inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Or() noexcept
479{
480 m_nextWhereJunctor = WhereJunctor::Or;
481 return static_cast<Derived&>(*this);
482}
483
484template <typename Derived>
485inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Not() noexcept
486{
487 m_nextIsNot = !m_nextIsNot;
488 return static_cast<Derived&>(*this);
489}
490
491template <typename Derived>
492template <typename ColumnName, typename T>
493inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Where(ColumnName const& columnName,
494 T const& value)
495{
496 if constexpr (detail::OneOf<T, SqlNullType, std::nullopt_t>)
497 {
498 if (m_nextIsNot)
499 {
500 m_nextIsNot = false;
501 return Where(columnName, "IS NOT", value);
502 }
503 else
504 return Where(columnName, "IS", value);
505 }
506 else
507 return Where(columnName, "=", value);
508}
509
510template <typename Derived>
511template <typename ColumnName, typename T>
512inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::OrWhere(ColumnName const& columnName,
513 T const& value)
514{
515 return Or().Where(columnName, value);
516}
517
518template <typename Derived>
519template <typename Callable>
520 requires std::invocable<Callable, SqlWhereClauseBuilder<Derived>&>
521inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::OrWhere(Callable const& callable)
522{
523 return Or().Where(callable);
524}
525
526template <typename Derived>
527template <typename Callable>
528 requires std::invocable<Callable, SqlWhereClauseBuilder<Derived>&>
529inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Where(Callable const& callable)
530{
531 auto& condition = SearchCondition().condition;
532
533 auto const originalSize = condition.size();
534
535 AppendWhereJunctor();
536 m_nextWhereJunctor = WhereJunctor::Null;
537 condition += '(';
538
539 auto const sizeBeforeCallable = condition.size();
540
541 (void) callable(*this);
542
543 if (condition.size() == sizeBeforeCallable)
544 condition.resize(originalSize);
545 else
546 condition += ')';
547
548 return static_cast<Derived&>(*this);
549}
550
551namespace detail
552{
553
554inline LIGHTWEIGHT_FORCE_INLINE RawSqlCondition PopulateSqlSetExpression(auto const& values)
555{
556 using namespace std::string_view_literals;
557 std::ostringstream fragment;
558 fragment << '(';
559 for (auto const&& [index, value]: values | std::views::enumerate)
560 {
561 if (index > 0)
562 fragment << ", "sv;
563 fragment << value;
564 }
565 fragment << ')';
566 return RawSqlCondition { fragment.str() };
567}
568
569} // namespace detail
570
571template <typename Derived>
572template <typename ColumnName, std::ranges::input_range InputRange>
573inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereIn(ColumnName const& columnName,
574 InputRange const& values)
575{
576 return Where(columnName, "IN", detail::PopulateSqlSetExpression(values));
577}
578
579template <typename Derived>
580template <typename ColumnName, typename T>
581inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereIn(ColumnName const& columnName,
582 std::initializer_list<T> const& values)
583{
584 return Where(columnName, "IN", detail::PopulateSqlSetExpression(values));
585}
586
587template <typename Derived>
588template <typename ColumnName, typename SubSelectQuery>
589 requires(std::is_invocable_r_v<std::string, decltype(&SubSelectQuery::ToSql), SubSelectQuery const&>)
590inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereIn(ColumnName const& columnName,
591 SubSelectQuery const& subSelectQuery)
592{
593 return Where(columnName, "IN", detail::RawSqlCondition { "(" + subSelectQuery.ToSql() + ")" });
594}
595
596template <typename Derived>
597template <typename ColumnName>
598inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereNotNull(ColumnName const& columnName)
599{
600 return Where(columnName, "IS NOT", detail::RawSqlCondition { "NULL" });
601}
602
603template <typename Derived>
604template <typename ColumnName>
605inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereNull(ColumnName const& columnName)
606{
607 return Where(columnName, "IS", detail::RawSqlCondition { "NULL" });
608}
609
610template <typename Derived>
611template <typename ColumnName, typename T>
612inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereNotEqual(ColumnName const& columnName,
613 T const& value)
614{
615 if constexpr (detail::OneOf<T, SqlNullType, std::nullopt_t>)
616 return Where(columnName, "IS NOT", value);
617 else
618 return Where(columnName, "!=", value);
619}
620
621template <typename Derived>
622template <typename ColumnName>
623inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereTrue(ColumnName const& columnName)
624{
625 return Where(columnName, "=", true);
626}
627
628template <typename Derived>
629template <typename ColumnName>
630inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereFalse(ColumnName const& columnName)
631{
632 return Where(columnName, "=", false);
633}
634
635template <typename Derived>
636template <typename LeftColumn, typename RightColumn>
637inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereColumn(LeftColumn const& left,
638 std::string_view binaryOp,
639 RightColumn const& right)
640{
641 AppendWhereJunctor();
642
643 AppendColumnName(left);
644 SearchCondition().condition += ' ';
645 SearchCondition().condition += binaryOp;
646 SearchCondition().condition += ' ';
647 AppendColumnName(right);
648
649 return static_cast<Derived&>(*this);
650}
651
652template <typename T>
653struct WhereConditionLiteralType
654{
655 constexpr static bool needsQuotes = !std::is_integral_v<T> && !std::is_floating_point_v<T> && !std::same_as<T, bool>
656 && !std::same_as<T, SqlWildcardType>;
657};
658
659template <typename Derived>
660template <typename ColumnName, std::size_t N>
661inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Where(ColumnName const& columnName,
662 std::string_view binaryOp,
663 char const (&value)[N])
664{
665 return Where(columnName, binaryOp, std::string_view { value, N - 1 });
666}
667
668template <typename Derived>
669template <typename ColumnName, typename T>
670inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Where(ColumnName const& columnName,
671 std::string_view binaryOp,
672 T const& value)
673{
674 auto& searchCondition = SearchCondition();
675
676 AppendWhereJunctor();
677 AppendColumnName(columnName);
678 searchCondition.condition += ' ';
679 searchCondition.condition += binaryOp;
680 searchCondition.condition += ' ';
681 AppendLiteralValue(value);
682
683 return static_cast<Derived&>(*this);
684}
685
686template <typename Derived>
687template <typename ColumnName, typename SubSelectQuery>
688 requires(std::is_invocable_r_v<std::string, decltype(&SubSelectQuery::ToSql), SubSelectQuery const&>)
689inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Where(ColumnName const& columnName,
690 std::string_view binaryOp,
691 SubSelectQuery const& value)
692{
693 return Where(columnName, binaryOp, detail::RawSqlCondition { "(" + value.ToSql() + ")" });
694}
695
696template <typename Derived>
697template <typename ColumnName, typename T>
698inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::OrWhere(ColumnName const& columnName,
699 std::string_view binaryOp,
700 T const& value)
701{
702 return Or().Where(columnName, binaryOp, value);
703}
704
705template <typename Derived>
706inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::InnerJoin(
707 std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
708{
709 return Join(JoinType::INNER, joinTable, joinColumnName, onOtherColumn);
710}
711
712template <typename Derived>
713inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::InnerJoin(std::string_view joinTable,
714 std::string_view joinColumnName,
715 std::string_view onMainTableColumn)
716{
717 return Join(JoinType::INNER, joinTable, joinColumnName, onMainTableColumn);
718}
719
720template <typename Derived>
721template <typename OnChainCallable>
722 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
723Derived& SqlWhereClauseBuilder<Derived>::InnerJoin(std::string_view joinTable, OnChainCallable const& onClauseBuilder)
724{
725 return Join(JoinType::INNER, joinTable, onClauseBuilder);
726}
727
728template <typename Derived>
729inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::LeftOuterJoin(
730 std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
731{
732 return Join(JoinType::LEFT, joinTable, joinColumnName, onOtherColumn);
733}
734
735template <typename Derived>
736inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::LeftOuterJoin(
737 std::string_view joinTable, std::string_view joinColumnName, std::string_view onMainTableColumn)
738{
739 return Join(JoinType::LEFT, joinTable, joinColumnName, onMainTableColumn);
740}
741
742template <typename Derived>
743template <typename OnChainCallable>
744 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
745Derived& SqlWhereClauseBuilder<Derived>::LeftOuterJoin(std::string_view joinTable,
746 OnChainCallable const& onClauseBuilder)
747{
748 return Join(JoinType::LEFT, joinTable, onClauseBuilder);
749}
750
751template <typename Derived>
752inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::RightOuterJoin(
753 std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
754{
755 return Join(JoinType::RIGHT, joinTable, joinColumnName, onOtherColumn);
756}
757
758template <typename Derived>
759inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::RightOuterJoin(
760 std::string_view joinTable, std::string_view joinColumnName, std::string_view onMainTableColumn)
761{
762 return Join(JoinType::RIGHT, joinTable, joinColumnName, onMainTableColumn);
763}
764
765template <typename Derived>
766template <typename OnChainCallable>
767 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
768Derived& SqlWhereClauseBuilder<Derived>::RightOuterJoin(std::string_view joinTable,
769 OnChainCallable const& onClauseBuilder)
770{
771 return Join(JoinType::RIGHT, joinTable, onClauseBuilder);
772}
773
774template <typename Derived>
775inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::FullOuterJoin(
776 std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
777{
778 return Join(JoinType::FULL, joinTable, joinColumnName, onOtherColumn);
779}
780
781template <typename Derived>
782inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::FullOuterJoin(
783 std::string_view joinTable, std::string_view joinColumnName, std::string_view onMainTableColumn)
784{
785 return Join(JoinType::FULL, joinTable, joinColumnName, onMainTableColumn);
786}
787
788template <typename Derived>
789template <typename OnChainCallable>
790 requires std::invocable<OnChainCallable, SqlJoinConditionBuilder>
791Derived& SqlWhereClauseBuilder<Derived>::FullOuterJoin(std::string_view joinTable,
792 OnChainCallable const& onClauseBuilder)
793{
794 return Join(JoinType::FULL, joinTable, onClauseBuilder);
795}
796
797template <typename Derived>
798inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::WhereRaw(
799 std::string_view sqlConditionExpression)
800{
801 AppendWhereJunctor();
802
803 auto& condition = SearchCondition().condition;
804 condition += sqlConditionExpression;
805
806 return static_cast<Derived&>(*this);
807}
808
809template <typename Derived>
810inline LIGHTWEIGHT_FORCE_INLINE SqlSearchCondition& SqlWhereClauseBuilder<Derived>::SearchCondition() noexcept
811{
812 return static_cast<Derived*>(this)->SearchCondition();
813}
814
815template <typename Derived>
816inline LIGHTWEIGHT_FORCE_INLINE SqlQueryFormatter const& SqlWhereClauseBuilder<Derived>::Formatter() const noexcept
817{
818 return static_cast<Derived const*>(this)->Formatter();
819}
820
821template <typename Derived>
822inline LIGHTWEIGHT_FORCE_INLINE void SqlWhereClauseBuilder<Derived>::AppendWhereJunctor()
823{
824 using namespace std::string_view_literals;
825
826 auto& condition = SearchCondition().condition;
827
828 switch (m_nextWhereJunctor)
829 {
830 case WhereJunctor::Null:
831 break;
832 case WhereJunctor::Where:
833 condition += "\n WHERE "sv;
834 break;
835 case WhereJunctor::And:
836 condition += " AND "sv;
837 break;
838 case WhereJunctor::Or:
839 condition += " OR "sv;
840 break;
841 }
842
843 if (m_nextIsNot)
844 {
845 condition += "NOT "sv;
846 m_nextIsNot = false;
847 }
848
849 m_nextWhereJunctor = WhereJunctor::And;
850}
851
852template <typename Derived>
853template <typename ColumnName>
854 requires(std::same_as<ColumnName, SqlQualifiedTableColumnName> || std::convertible_to<ColumnName, std::string_view>
855 || std::same_as<ColumnName, SqlRawColumnNameView> || std::convertible_to<ColumnName, std::string>)
856inline LIGHTWEIGHT_FORCE_INLINE void SqlWhereClauseBuilder<Derived>::AppendColumnName(ColumnName const& columnName)
857{
858 SearchCondition().condition += detail::MakeSqlColumnName(columnName);
859}
860
861template <typename Derived>
862template <typename LiteralType>
863inline LIGHTWEIGHT_FORCE_INLINE void SqlWhereClauseBuilder<Derived>::AppendLiteralValue(LiteralType const& value)
864{
865 auto& searchCondition = SearchCondition();
866
867 if constexpr (std::is_same_v<LiteralType, SqlQualifiedTableColumnName>)
868 {
869 searchCondition.condition += '"';
870 searchCondition.condition += value.tableName;
871 searchCondition.condition += "\".\"";
872 searchCondition.condition += value.columnName;
873 searchCondition.condition += '"';
874 }
875 else if constexpr (detail::OneOf<LiteralType, SqlNullType, std::nullopt_t>)
876 {
877 searchCondition.condition += "NULL";
878 }
879 else if constexpr (std::is_same_v<LiteralType, SqlWildcardType>)
880 {
881 searchCondition.condition += '?';
882 }
883 else if constexpr (std::is_same_v<LiteralType, detail::RawSqlCondition>)
884 {
885 searchCondition.condition += value.condition;
886 }
887 else if (searchCondition.inputBindings)
888 {
889 searchCondition.condition += '?';
890 searchCondition.inputBindings->emplace_back(value);
891 }
892 else if constexpr (std::is_same_v<LiteralType, bool>)
893 {
894 searchCondition.condition += Formatter().BooleanLiteral(value);
895 }
896 else if constexpr (!WhereConditionLiteralType<LiteralType>::needsQuotes)
897 {
898 searchCondition.condition += std::format("{}", value);
899 }
900 else
901 {
902 // TODO: Escape single quotes
903 searchCondition.condition += '\'';
904 searchCondition.condition += std::format("{}", value);
905 searchCondition.condition += '\'';
906 }
907}
908
909template <typename Derived>
910inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Join(JoinType joinType,
911 std::string_view joinTable,
912 std::string_view joinColumnName,
913 SqlQualifiedTableColumnName onOtherColumn)
914{
915 static constexpr std::array<std::string_view, 4> JoinTypeStrings = {
916 "INNER",
917 "LEFT OUTER",
918 "RIGHT OUTER",
919 "FULL OUTER",
920 };
921
922 SearchCondition().tableJoins += std::format("\n"
923 R"( {0} JOIN "{1}" ON "{1}"."{2}" = "{3}"."{4}")",
924 JoinTypeStrings[static_cast<std::size_t>(joinType)],
925 joinTable,
926 joinColumnName,
927 onOtherColumn.tableName,
928 onOtherColumn.columnName);
929 return static_cast<Derived&>(*this);
930}
931
932template <typename Derived>
933inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Join(JoinType joinType,
934 std::string_view joinTable,
935 std::string_view joinColumnName,
936 std::string_view onMainTableColumn)
937{
938 return Join(
939 joinType,
940 joinTable,
941 joinColumnName,
942 SqlQualifiedTableColumnName { .tableName = SearchCondition().tableName, .columnName = onMainTableColumn });
943}
944
945template <typename Derived>
946template <typename Callable>
947inline LIGHTWEIGHT_FORCE_INLINE Derived& SqlWhereClauseBuilder<Derived>::Join(JoinType joinType,
948 std::string_view joinTable,
949 Callable const& onClauseBuilder)
950{
951 static constexpr std::array<std::string_view, 4> JoinTypeStrings = {
952 "INNER",
953 "LEFT OUTER",
954 "RIGHT OUTER",
955 "FULL OUTER",
956 };
957
958 size_t const originalSize = SearchCondition().tableJoins.size();
959 SearchCondition().tableJoins +=
960 std::format("\n {0} JOIN \"{1}\" ON ", JoinTypeStrings[static_cast<std::size_t>(joinType)], joinTable);
961 size_t const sizeBefore = SearchCondition().tableJoins.size();
962 onClauseBuilder(SqlJoinConditionBuilder { joinTable, &SearchCondition().tableJoins });
963 size_t const sizeAfter = SearchCondition().tableJoins.size();
964 if (sizeBefore == sizeAfter)
965 SearchCondition().tableJoins.resize(originalSize);
966
967 return static_cast<Derived&>(*this);
968}
Query builder for building JOIN conditions.
Definition Core.hpp:94
API to format SQL queries for different SQL dialects.
Derived & Where(ColumnName const &columnName, std::string_view binaryOp, T const &value)
Constructs or extends a WHERE clause to test for a binary operation.
Derived & LeftOuterJoin(std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
Constructs an LEFT OUTER JOIN clause.
Definition Core.hpp:729
Derived & InnerJoin(std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
Constructs an INNER JOIN clause.
Definition Core.hpp:706
Derived & Not() noexcept
Indicates, that the next WHERE clause should be negated.
Definition Core.hpp:485
Derived & FullOuterJoin(std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
Constructs an FULL OUTER JOIN clause.
Definition Core.hpp:775
Derived & OrWhere(ColumnName const &columnName, std::string_view binaryOp, T const &value)
Constructs or extends a WHERE/OR clause to test for a binary operation.
Derived & WhereNull(ColumnName const &columnName)
Constructs or extends an WHERE/OR clause to test for a value to be NULL.
Derived & WhereTrue(ColumnName const &columnName)
Constructs or extends a WHERE clause to test for a value being true.
Derived & WhereNotNull(ColumnName const &columnName)
Constructs or extends a WHERE clause to test for a value being not null.
Derived & WhereFalse(ColumnName const &columnName)
Constructs or extends a WHERE clause to test for a value being false.
Derived & WhereIn(ColumnName const &columnName, InputRange const &values)
Constructs or extends an WHERE/OR clause to test for a value, satisfying std::ranges::input_range.
Derived & Or() noexcept
Indicates, that the next WHERE clause should be OR-ed.
Definition Core.hpp:478
Derived & RightOuterJoin(std::string_view joinTable, std::string_view joinColumnName, SqlQualifiedTableColumnName onOtherColumn)
Constructs an RIGHT OUTER JOIN clause.
Definition Core.hpp:752
Derived & And() noexcept
Indicates, that the next WHERE clause should be AND-ed (default).
Definition Core.hpp:471
Derived & WhereColumn(LeftColumn const &left, std::string_view binaryOp, RightColumn const &right)
Construts or extends a WHERE clause to test for a binary operation between two columns.
Derived & WhereRaw(std::string_view sqlConditionExpression)
Constructs or extends a raw WHERE clause.
Definition Core.hpp:798
Derived & WhereNotEqual(ColumnName const &columnName, T const &value)
Constructs or extends a WHERE clause to test for a value being equal to another column.
LIGHTWEIGHT_API std::vector< std::string > ToSql(SqlQueryFormatter const &formatter, SqlMigrationPlanElement const &element)
SqlQualifiedTableColumnName represents a column name qualified with a table name.
Definition Core.hpp:40
SqlWildcardType is a placeholder for an explicit wildcard input parameter in a SQL query.
Definition Core.hpp:21