5#include "../SqlQueryFormatter.hpp"
9#include <reflection-cpp/reflection.hpp>
11enum class SqlQueryBuilderMode : uint8_t
17struct [[nodiscard]] SqlFieldExpression final
19 std::string expression;
25inline SqlFieldExpression Count(std::string
const& field =
"*") noexcept
27 return SqlFieldExpression { .expression = std::format(
"COUNT(\"{}\")", field) };
32 return SqlFieldExpression { .expression = std::format(R
"(COUNT("{}"."{}"))", field.tableName, field.columnName) };
35inline SqlFieldExpression Sum(std::string
const& field)
noexcept
37 return SqlFieldExpression { .expression = std::format(
"SUM(\"{}\")", field) };
42 return SqlFieldExpression { .expression = std::format(R
"(SUM("{}"."{}"))", field.tableName, field.columnName) };
45inline SqlFieldExpression Avg(std::string
const& field)
noexcept
47 return SqlFieldExpression { .expression = std::format(
"AVG(\"{}\")", field) };
52 return SqlFieldExpression { .expression = std::format(R
"(AVG("{}"."{}"))", field.tableName, field.columnName) };
55inline SqlFieldExpression Min(std::string
const& field)
noexcept
57 return SqlFieldExpression { .expression = std::format(
"MIN(\"{}\")", field) };
62 return SqlFieldExpression { .expression = std::format(R
"(MIN("{}"."{}"))", field.tableName, field.columnName) };
65inline SqlFieldExpression Max(std::string
const& field)
noexcept
67 return SqlFieldExpression { .expression = std::format(
"MAX(\"{}\")", field) };
72 return SqlFieldExpression { .expression = std::format(R
"(MAX("{}"."{}"))", field.tableName, field.columnName) };
83 using SelectType = detail::SelectType;
87 std::string tableAlias)
noexcept:
88 SqlBasicSelectQueryBuilder<SqlSelectQueryBuilder> {},
89 _formatter { formatter }
91 _query.formatter = &formatter;
92 _query.searchCondition.tableName = std::move(table);
93 _query.searchCondition.tableAlias = std::move(tableAlias);
94 _query.fields.reserve(256);
100 _mode = SqlQueryBuilderMode::Varying;
105 template <
typename... MoreFields>
125 std::string_view tableName);
127 LIGHTWEIGHT_API
SqlSelectQueryBuilder& Fields(std::initializer_list<std::string_view>
const& fieldNames,
128 std::string_view tableName);
131 template <
typename FirstRecord,
typename... MoreRecords>
135 [[deprecated(
"Use Field(...).As(\"alias\") instead.")]]
139 [[deprecated(
"Use Field(...).As(\"alias\") instead.")]]
141 std::string_view
const& alias);
143 template <
typename Callable>
147 LIGHTWEIGHT_API ComposedQuery
Count();
150 LIGHTWEIGHT_API ComposedQuery
All();
153 LIGHTWEIGHT_API ComposedQuery
First(
size_t count = 1);
156 LIGHTWEIGHT_API ComposedQuery
Range(std::size_t offset, std::size_t limit);
158 LIGHTWEIGHT_FORCE_INLINE SqlSearchCondition& SearchCondition() noexcept
160 return _query.searchCondition;
163 [[nodiscard]] LIGHTWEIGHT_FORCE_INLINE
SqlQueryFormatter const& Formatter() const noexcept
170 SqlQueryBuilderMode _mode = SqlQueryBuilderMode::Fluent;
171 bool _aliasAllowed =
false;
174template <
typename... MoreFields>
177 using namespace std::string_view_literals;
179 std::ostringstream fragment;
181 if (!_query.fields.empty())
184 fragment <<
'"' << firstField <<
'"';
186 if constexpr (
sizeof...(MoreFields) > 0)
187 ((fragment << R
"(, ")"sv << std::forward<MoreFields>(moreFields) << '"') << ...);
189 _query.fields += fragment.str();
193template <
typename Callable>
194inline LIGHTWEIGHT_FORCE_INLINE
SqlSelectQueryBuilder& SqlSelectQueryBuilder::Build(Callable
const& callable)
200template <
typename FirstRecord,
typename... MoreRecords>
203 if constexpr (
sizeof...(MoreRecords) == 0)
205 Reflection::EnumerateMembers<FirstRecord>(
206 [&]<
size_t FieldIndex,
typename FieldType>() {
Field(FieldNameAt<FieldIndex, FirstRecord>); });
210 Reflection::EnumerateMembers<FirstRecord>([&]<
size_t FieldIndex,
typename FieldType>() {
212 .tableName = RecordTableName<FirstRecord>,
213 .columnName = FieldNameAt<FieldIndex, FirstRecord>,
217 (Reflection::EnumerateMembers<MoreRecords>([&]<
size_t FieldIndex,
typename FieldType>() {
219 .tableName = RecordTableName<MoreRecords>,
220 .columnName = FieldNameAt<FieldIndex, MoreRecords>,
Query builder for building SELECT ... queries.
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.
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.
SqlQualifiedTableColumnName represents a column name qualified with a table name.