5#include "../DataMapper/BelongsTo.hpp"
6#include "../DataMapper/Field.hpp"
7#include "../DataMapper/Record.hpp"
10#include "MigrationPlan.hpp"
12#include <reflection-cpp/reflection.hpp>
56 std::string columnName, SqlColumnTypeDefinition columnType = SqlColumnTypeDefinitions::Bigint {});
60 SqlColumnTypeDefinition columnType,
65 SqlColumnTypeDefinition columnType,
70 std::string referencedTableName,
71 std::vector<std::string> referencedColumns);
83 SqlCreateTablePlan& _plan;
110 template <auto MemberPo
inter>
113 return AddColumn(std::string(FieldNameOf<MemberPointer>),
119 SqlColumnTypeDefinition columnType);
126 template <auto MemberPo
inter>
129 return AddNotRequiredColumn(std::string(FieldNameOf<MemberPointer>),
153 SqlColumnTypeDefinition columnType,
154 SqlNullable nullable);
172 SqlColumnTypeDefinition columnType);
176 SqlColumnTypeDefinition columnType);
228 SqlColumnTypeDefinition columnType,
248 std::string referencedTableName,
249 std::vector<std::string> referencedColumns);
257 template <
typename Record>
262#if defined(LIGHTWEIGHT_CXX26_REFLECTION)
263 constexpr auto ctx = std::meta::access_context::current();
264 template for (
constexpr auto el: define_static_array(nonstatic_data_members_of(^^Record, ctx)))
266 using FieldType =
typename[:std::meta::type_of(el):];
269 if constexpr (IsAutoIncrementPrimaryKey<FieldType>)
271 std::string(FieldNameOf<el>),
272 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
273 else if constexpr (FieldType::IsPrimaryKey)
274 builder.
PrimaryKey(std::string(FieldNameOf<el>),
275 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
276 else if constexpr (IsBelongsTo<FieldType>)
278 constexpr size_t referencedFieldIndex = []()
constexpr ->
size_t {
279 auto index = size_t(-1);
280 Reflection::EnumerateMembers<typename FieldType::ReferencedRecord>(
281 [&index]<
size_t J,
typename ReferencedFieldType>()
constexpr ->
void {
282 if constexpr (IsField<ReferencedFieldType>)
283 if constexpr (ReferencedFieldType::IsPrimaryKey)
289 std::string(FieldNameOf<el>),
290 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value,
291 SqlForeignKeyReferenceDefinition {
292 .tableName = std::string { RecordTableName<typename FieldType::ReferencedRecord> },
293 .columnName = std::string { FieldNameOf<FieldType::ReferencedField> } });
295 else if constexpr (FieldType::IsMandatory)
297 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
299 builder.
Column(std::string(FieldNameOf<el>),
300 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
304 Reflection::EnumerateMembers<Record>([&builder]<
size_t I,
typename FieldType>() {
305 if constexpr (FieldWithStorage<FieldType>)
307 if constexpr (IsAutoIncrementPrimaryKey<FieldType>)
309 std::string(FieldNameAt<I, Record>()),
310 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
311 else if constexpr (FieldType::IsPrimaryKey)
312 builder.
PrimaryKey(std::string(FieldNameAt<I, Record>()),
313 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
314 else if constexpr (IsBelongsTo<FieldType>)
316 constexpr size_t referencedFieldIndex = []()
constexpr ->
size_t {
317 auto index = size_t(-1);
318 Reflection::EnumerateMembers<typename FieldType::ReferencedRecord>(
319 [&index]<
size_t J,
typename ReferencedFieldType>()
constexpr ->
void {
320 if constexpr (IsField<ReferencedFieldType>)
321 if constexpr (ReferencedFieldType::IsPrimaryKey)
327 std::string(FieldNameAt<I, Record>()),
328 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value,
329 SqlForeignKeyReferenceDefinition {
330 .tableName = std::string { RecordTableName<typename FieldType::ReferencedRecord> },
332 std::string { FieldNameAt<referencedFieldIndex, typename FieldType::ReferencedRecord>() } });
334 else if constexpr (FieldType::IsMandatory)
336 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
338 builder.
Column(std::string(FieldNameAt<I, Record>()),
339 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
360 template <
typename T>
363 _plan.columns.emplace_back(std::move(columnName),
SqlVariant(value));
385 template <
typename T>
388 _plan.setColumns.emplace_back(std::move(columnName),
SqlVariant(value));
393 template <
typename T>
396 _plan.whereColumn = std::move(columnName);
397 _plan.whereOp = std::move(op);
420 template <
typename T>
423 _plan.whereColumn = std::move(columnName);
424 _plan.whereOp = std::move(op);
440 _formatter { formatter },
441 _migrationPlan { .formatter = formatter }
479 template <
typename Record>
484 auto builder = CreateTable(RecordTableName<Record>);
485 detail::PopulateCreateTableBuilder<Record>(builder);
490 template <
typename Record>
494 return AlterTable(RecordTableName<Record>);
513 std::string tableName,
514 std::vector<std::string> columns,
515 bool unique =
false);
523 std::string tableName,
524 std::vector<std::string> columns);
540 std::vector<std::string> columns,
571 std::
string _schemaName;
Query builder for building ALTER TABLE queries.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & DropIndex(std::string_view columnName)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & DropIndexIfExists(std::string_view columnName)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & DropColumn(std::string_view columnName)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddColumn(std::string columnName, SqlColumnTypeDefinition columnType)
Adds a new column to the table that is non-nullable.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & RenameColumn(std::string_view oldColumnName, std::string_view newColumnName)
SqlAlterTableQueryBuilder & AddColumn()
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & RenameTo(std::string_view newTableName)
Renames the table.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddIndex(std::string_view columnName)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddNotRequiredForeignKeyColumn(std::string columnName, SqlColumnTypeDefinition columnType, SqlForeignKeyReferenceDefinition referencedColumn)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddNotRequiredColumn(std::string columnName, SqlColumnTypeDefinition columnType)
Adds a new column to the table that is nullable.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddUniqueIndex(std::string_view columnName)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & DropColumnIfExists(std::string_view columnName)
Drops a column from the table only if it exists.
SqlAlterTableQueryBuilder(SqlAlterTablePlan &plan)
Constructs an ALTER TABLE query builder.
SqlAlterTableQueryBuilder & AddNotRequiredColumn()
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & DropForeignKey(std::string columnName)
Drops a foreign key for the column columnName from the table.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AlterColumn(std::string columnName, SqlColumnTypeDefinition columnType, SqlNullable nullable)
Alters the column to have a new non-nullable type.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddForeignKey(std::string columnName, SqlForeignKeyReferenceDefinition referencedColumn)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddColumnIfNotExists(std::string columnName, SqlColumnTypeDefinition columnType)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddNotRequiredColumnIfNotExists(std::string columnName, SqlColumnTypeDefinition columnType)
Adds a new nullable column to the table only if it does not already exist.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddForeignKeyColumn(std::string columnName, SqlColumnTypeDefinition columnType, SqlForeignKeyReferenceDefinition referencedColumn)
LIGHTWEIGHT_API SqlAlterTableQueryBuilder & AddCompositeForeignKey(std::vector< std::string > columns, std::string referencedTableName, std::vector< std::string > referencedColumns)
Represents a connection to a SQL database.
Query builder for building CREATE TABLE queries.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & ForeignKey(std::vector< std::string > columns, std::string referencedTableName, std::vector< std::string > referencedColumns)
Adds a composite foreign key constraint.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & Timestamps()
Adds the created_at and updated_at columns to the table.
SqlCreateTableQueryBuilder(SqlCreateTablePlan &plan)
Constructs a CREATE TABLE query builder.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & RequiredColumn(std::string columnName, SqlColumnTypeDefinition columnType)
Creates a new column that is non-nullable.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & RequiredForeignKey(std::string columnName, SqlColumnTypeDefinition columnType, SqlForeignKeyReferenceDefinition foreignKey)
Creates a new non-nullable foreign key column.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & ForeignKey(std::string columnName, SqlColumnTypeDefinition columnType, SqlForeignKeyReferenceDefinition foreignKey)
Creates a new nullable foreign key column.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & Index()
Enables the INDEX constraint on the last declared column.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & UniqueIndex()
Enables the UNIQUE and INDEX constraint on the last declared column and makes it an index.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & PrimaryKey(std::string columnName, SqlColumnTypeDefinition columnType)
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & Column(SqlColumnDeclaration column)
Adds a new column to the table.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & Column(std::string columnName, SqlColumnTypeDefinition columnType)
Creates a new nullable column.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & PrimaryKeyWithAutoIncrement(std::string columnName, SqlColumnTypeDefinition columnType=SqlColumnTypeDefinitions::Bigint {})
Creates a new primary key column with auto-increment.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder & Unique()
Enables the UNIQUE constraint on the last declared column.
Query builder for building DELETE queries in migrations.
SqlMigrationDeleteBuilder & Where(std::string columnName, std::string op, T const &value)
Adds a WHERE condition to the DELETE.
SqlMigrationDeleteBuilder(SqlDeleteDataPlan &plan)
Constructs a migration DELETE builder.
Query builder for building INSERT queries in migrations.
SqlMigrationInsertBuilder(SqlInsertDataPlan &plan)
Constructs a migration INSERT builder.
SqlMigrationInsertBuilder & Set(std::string columnName, T const &value)
Sets a column value for the INSERT.
Query builder for building SQL migration queries.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder CreateTable(std::string_view tableName)
Creates a new table.
SqlCreateTableQueryBuilder CreateTable()
Creates a new table for the given record type.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & CommitTransaction()
Commits a transaction.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & RawSql(std::string_view sql)
Executes raw SQL.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & CreateIndex(std::string indexName, std::string tableName, std::vector< std::string > columns, bool unique=false)
LIGHTWEIGHT_API SqlMigrationDeleteBuilder Delete(std::string_view tableName)
Creates a DELETE statement for the migration.
LIGHTWEIGHT_API SqlAlterTableQueryBuilder AlterTable(std::string_view tableName)
Alters an existing table.
LIGHTWEIGHT_API SqlCreateTableQueryBuilder CreateTableIfNotExists(std::string_view tableName)
LIGHTWEIGHT_API SqlMigrationPlan const & GetPlan() const &
Gets the migration plan.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & DropTable(std::string_view tableName)
Drops a table.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & CreateIndex(std::string tableName, std::vector< std::string > columns, IndexType type=IndexType::NonUnique)
LIGHTWEIGHT_API SqlMigrationUpdateBuilder Update(std::string_view tableName)
Creates an UPDATE statement for the migration.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & WithSchema(std::string schemaName)
Sets the schema name for the migration.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & CreateUniqueIndex(std::string indexName, std::string tableName, std::vector< std::string > columns)
LIGHTWEIGHT_API SqlMigrationInsertBuilder Insert(std::string_view tableName)
Creates an INSERT statement for the migration.
SqlAlterTableQueryBuilder AlterTable()
Alters an existing table.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & BeginTransaction()
Starts a transaction.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & DropTableCascade(std::string_view tableName)
SqlMigrationQueryBuilder(SqlQueryFormatter const &formatter)
Constructs a migration query builder.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & DropDatabase(std::string_view databaseName)
Drops a database.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & Native(std::function< std::string(SqlConnection &)> callback)
Executes SQL interactively via a callback.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & DropTableIfExists(std::string_view tableName)
Drops a table if it exists.
LIGHTWEIGHT_API SqlMigrationQueryBuilder & CreateDatabase(std::string_view databaseName)
Creates a new database.
Query builder for building UPDATE queries in migrations.
SqlMigrationUpdateBuilder & Where(std::string columnName, std::string op, T const &value)
Adds a WHERE condition to the UPDATE.
SqlMigrationUpdateBuilder & Set(std::string columnName, T const &value)
Sets a column value for the UPDATE.
SqlMigrationUpdateBuilder(SqlUpdateDataPlan &plan)
Constructs a migration UPDATE builder.
Represents a record type that can be used with the DataMapper.
std::remove_cvref_t< decltype(std::declval< MemberClassType< decltype(Field)> >().*Field)>::ValueType ReferencedFieldTypeOf
Retrieves the type of a member field in a record.
IndexType
Index type for simplified CreateIndex API.
constexpr auto SqlColumnTypeDefinitionOf
Represents a SQL column type definition of T.
@ NonUnique
Regular (non-unique) index.
Represents a SQL ALTER TABLE plan on a given table.
Represents a SQL column declaration.
Represents a SQL DELETE data plan for migrations.
Represents a foreign key reference definition.
Represents a SQL INSERT data plan for migrations.
Represents a SQL migration plan.
Represents a SQL UPDATE data plan for migrations.
Represents a value that can be any of the supported SQL data types.