Lightweight 0.1.0
Loading...
Searching...
No Matches
MigrationPlan.hpp
1// SPDX-License-Identifier: Apache-2.0
2
3#pragma once
4
5#include "../Api.hpp"
6#include "../SqlColumnTypeDefinitions.hpp"
7#include "../SqlDataBinder.hpp"
8#include "../Utils.hpp"
9
10#include <reflection-cpp/reflection.hpp>
11
12#include <string>
13#include <string_view>
14#include <variant>
15#include <vector>
16
18
19namespace detail
20{
21
22template <typename T>
24{
25 static_assert(AlwaysFalse<T>, "Unsupported type for SQL column definition.");
26};
27
28template <>
30{
31 static constexpr auto value = SqlColumnTypeDefinitions::Bool {};
32};
33
34template <>
36{
37 static constexpr auto value = SqlColumnTypeDefinitions::Char { 1 };
38};
39
40template <>
42{
43 static constexpr auto value = SqlColumnTypeDefinitions::Date {};
44};
45
46template <>
48{
49 static constexpr auto value = SqlColumnTypeDefinitions::DateTime {};
50};
51
52template <>
54{
55 static constexpr auto value = SqlColumnTypeDefinitions::Time {};
56};
57
58template <size_t Precision, size_t Scale>
59struct SqlColumnTypeDefinitionOf<SqlNumeric<Precision, Scale>>
60{
61 static constexpr auto value = SqlColumnTypeDefinitions::Decimal { .precision = Precision, .scale = Scale };
62};
63
64template <>
66{
67 static constexpr auto value = SqlColumnTypeDefinitions::Guid {};
68};
69
70template <typename T>
71 requires(detail::OneOf<T, int16_t, uint16_t>)
72struct SqlColumnTypeDefinitionOf<T>
73{
74 static constexpr auto value = SqlColumnTypeDefinitions::Smallint {};
75};
76
77template <typename T>
78 requires(detail::OneOf<T, int32_t, uint32_t>)
79struct SqlColumnTypeDefinitionOf<T>
80{
81 static constexpr auto value = SqlColumnTypeDefinitions::Integer {};
82};
83
84template <typename T>
85 requires(detail::OneOf<T, int64_t, uint64_t>)
86struct SqlColumnTypeDefinitionOf<T>
87{
88 static constexpr auto value = SqlColumnTypeDefinitions::Bigint {};
89};
90
91template <typename T>
92 requires(detail::OneOf<T, float, double>)
93struct SqlColumnTypeDefinitionOf<T>
94{
95 static constexpr auto value = SqlColumnTypeDefinitions::Real {};
96};
97
98template <size_t N, typename CharT>
99 requires(detail::OneOf<CharT, char>)
101{
102 static constexpr auto value = SqlColumnTypeDefinitions::Varchar { N };
103};
104
105template <size_t N, typename CharT>
106 requires(detail::OneOf<CharT, char16_t, char32_t, wchar_t>)
108{
109 static constexpr auto value = SqlColumnTypeDefinitions::NVarchar { N };
110};
111
112template <size_t N, typename CharT>
113 requires(detail::OneOf<CharT, char>)
114struct SqlColumnTypeDefinitionOf<SqlFixedString<N, CharT, SqlFixedStringMode::FIXED_SIZE>>
115{
116 static constexpr auto value = SqlColumnTypeDefinitions::Char { N };
117};
118
119template <size_t N, typename CharT>
120 requires(detail::OneOf<CharT, char16_t, char32_t, wchar_t>)
121struct SqlColumnTypeDefinitionOf<SqlFixedString<N, CharT, SqlFixedStringMode::FIXED_SIZE>>
122{
123 static constexpr auto value = SqlColumnTypeDefinitions::NChar { N };
124};
125
126template <size_t N, typename CharT>
127struct SqlColumnTypeDefinitionOf<SqlFixedString<N, CharT, SqlFixedStringMode::FIXED_SIZE_RIGHT_TRIMMED>>
128{
129 static constexpr auto value = SqlColumnTypeDefinitions::Char { N };
130};
131
132template <size_t N, typename CharT>
133 requires(detail::OneOf<CharT, char16_t, char32_t, wchar_t>)
135{
136 static constexpr auto value = SqlColumnTypeDefinitions::NChar { N };
137};
138
139template <size_t N, typename CharT>
140 requires(detail::OneOf<CharT, char>)
141struct SqlColumnTypeDefinitionOf<SqlDynamicString<N, CharT>>
142{
143 static constexpr auto value = SqlColumnTypeDefinitions::Varchar { N };
144};
145
146template <size_t N, typename CharT>
147 requires(detail::OneOf<CharT, char8_t, char16_t, char32_t, wchar_t>)
148struct SqlColumnTypeDefinitionOf<SqlDynamicString<N, CharT>>
149{
150 static constexpr auto value = SqlColumnTypeDefinitions::NVarchar { N };
151};
152
153template <typename T>
154struct SqlColumnTypeDefinitionOf<std::optional<T>>
155{
156 static constexpr auto value = SqlColumnTypeDefinitionOf<T>::value;
157};
158
159} // namespace detail
160
161/// @brief Represents a SQL column type definition of T.
162///
163/// @ingroup QueryBuilder
164template <typename T>
165constexpr auto SqlColumnTypeDefinitionOf = detail::SqlColumnTypeDefinitionOf<T>::value;
166
167/// @brief Represents a primary key type.
168///
169/// This enumeration represents the primary key type of a column.
170///
171/// @ingroup QueryBuilder
172enum class SqlPrimaryKeyType : uint8_t
173{
174 NONE,
175 MANUAL,
176 AUTO_INCREMENT,
177 GUID,
178};
179
180/// @brief Represents a foreign key reference definition.
181///
182/// @ingroup QueryBuilder
184{
185 /// The table name that the foreign key references.
186 std::string tableName;
187
188 /// The column name that the foreign key references.
189 std::string columnName;
190};
191
192/// @brief Represents a SQL column declaration.
193///
194/// @ingroup QueryBuilder
196{
197 /// The name of the column.
198 std::string name;
199
200 /// The type of the column.
201 SqlColumnTypeDefinition type;
202
203 /// The primary key type of the column.
204 SqlPrimaryKeyType primaryKey { SqlPrimaryKeyType::NONE };
205
206 /// The foreign key reference definition of the column.
207 std::optional<SqlForeignKeyReferenceDefinition> foreignKey {};
208
209 /// Indicates if the column is required (non-nullable).
210 bool required { false };
211
212 /// Indicates if the column is unique.
213 bool unique { false };
214
215 /// Indicates if the column is indexed.
216 bool index { false };
217};
218
219struct SqlCreateTablePlan
220{
221 std::string_view tableName;
222 std::vector<SqlColumnDeclaration> columns;
223};
224
225enum class SqlNullable : uint8_t
226{
227 NotNull,
228 Null,
229};
230
231namespace SqlAlterTableCommands
232{
233
234struct RenameTable
235{
236 std::string_view newTableName;
237};
238
239struct AddColumn
240{
241 std::string columnName;
242 SqlColumnTypeDefinition columnType;
243 SqlNullable nullable = SqlNullable::Null;
244};
245
246struct AlterColumn
247{
248 std::string columnName;
249 SqlColumnTypeDefinition columnType;
250 SqlNullable nullable = SqlNullable::Null;
251};
252
253struct AddIndex
254{
255 std::string_view columnName;
256 bool unique = false;
257};
258
259struct RenameColumn
260{
261 std::string_view oldColumnName;
262 std::string_view newColumnName;
263};
264
265struct DropColumn
266{
267 std::string_view columnName;
268};
269
270struct DropIndex
271{
272 std::string_view columnName;
273};
274
275struct AddForeignKey
276{
277 std::string columnName;
278 SqlForeignKeyReferenceDefinition referencedColumn;
279};
280
281struct DropForeignKey
282{
283 std::string columnName;
284};
285
286} // namespace SqlAlterTableCommands
287
288/// @brief Represents a single SQL ALTER TABLE command.
289///
290/// @ingroup QueryBuilder
291using SqlAlterTableCommand = std::variant<SqlAlterTableCommands::RenameTable,
292 SqlAlterTableCommands::AddColumn,
293 SqlAlterTableCommands::AlterColumn,
294 SqlAlterTableCommands::AddIndex,
295 SqlAlterTableCommands::RenameColumn,
296 SqlAlterTableCommands::DropColumn,
297 SqlAlterTableCommands::DropIndex,
298 SqlAlterTableCommands::AddForeignKey,
299 SqlAlterTableCommands::DropForeignKey>;
300
301/// @brief Represents a SQL ALTER TABLE plan on a given table.
302///
303/// @ingroup QueryBuilder
305{
306 /// The name of the table to alter.
307 std::string_view tableName;
308
309 /// The list of commands to execute on the table.
310 std::vector<SqlAlterTableCommand> commands;
311};
312
313/// @brief Represents a SQL DROP TABLE plan.
314///
315/// @ingroup QueryBuilder
317{
318 /// The name of the table to drop.
319 std::string_view tableName;
320};
321
322// clang-format off
323
324/// @brief Represents a single SQL migration plan element.
325///
326/// This variant represents a single SQL migration plan element.
327///
328/// @ingroup QueryBuilder
329using SqlMigrationPlanElement = std::variant<
330 SqlCreateTablePlan,
333>;
334
335// clang-format on
336
337/// Formats the given SQL migration plan element as a list of SQL statements.
338///
339/// @param formatter The SQL query formatter to use.
340/// @param element The SQL migration plan element to format.
341///
342/// @return A list of SQL statements.
343///
344/// @ingroup QueryBuilder
345[[nodiscard]] LIGHTWEIGHT_API std::vector<std::string> ToSql(SqlQueryFormatter const& formatter,
346 SqlMigrationPlanElement const& element);
347
348/// @brief Represents a SQL migration plan.
349///
350/// This structure represents a SQL migration plan that can be executed on a database.
351///
352/// @ingroup QueryBuilder
353struct [[nodiscard]] SqlMigrationPlan
354{
355 SqlQueryFormatter const& formatter;
356 std::vector<SqlMigrationPlanElement> steps {};
357
358 [[nodiscard]] LIGHTWEIGHT_API std::vector<std::string> ToSql() const;
359};
API to format SQL queries for different SQL dialects.
std::variant< SqlCreateTablePlan, SqlAlterTablePlan, SqlDropTablePlan > SqlMigrationPlanElement
Represents a single SQL migration plan element.
constexpr auto SqlColumnTypeDefinitionOf
Represents a SQL column type definition of T.
LIGHTWEIGHT_API std::vector< std::string > ToSql(SqlQueryFormatter const &formatter, SqlMigrationPlanElement const &element)
SqlPrimaryKeyType
Represents a primary key type.
std::variant< SqlAlterTableCommands::RenameTable, SqlAlterTableCommands::AddColumn, SqlAlterTableCommands::AlterColumn, SqlAlterTableCommands::AddIndex, SqlAlterTableCommands::RenameColumn, SqlAlterTableCommands::DropColumn, SqlAlterTableCommands::DropIndex, SqlAlterTableCommands::AddForeignKey, SqlAlterTableCommands::DropForeignKey > SqlAlterTableCommand
Represents a single SQL ALTER TABLE command.
Represents a SQL ALTER TABLE plan on a given table.
std::string_view tableName
The name of the table to alter.
std::vector< SqlAlterTableCommand > commands
The list of commands to execute on the table.
Represents a SQL column declaration.
bool index
Indicates if the column is indexed.
SqlColumnTypeDefinition type
The type of the column.
bool required
Indicates if the column is required (non-nullable).
std::optional< SqlForeignKeyReferenceDefinition > foreignKey
The foreign key reference definition of the column.
SqlPrimaryKeyType primaryKey
The primary key type of the column.
bool unique
Indicates if the column is unique.
std::string name
The name of the column.
Represents a SQL DROP TABLE plan.
std::string_view tableName
The name of the table to drop.
Represents a foreign key reference definition.
std::string columnName
The column name that the foreign key references.
std::string tableName
The table name that the foreign key references.
Represents a SQL migration plan.