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>
19enum class IndexType : std::uint8_t
54 std::string columnName, SqlColumnTypeDefinition columnType = SqlColumnTypeDefinitions::Bigint {});
58 SqlColumnTypeDefinition columnType,
63 SqlColumnTypeDefinition columnType,
68 std::string referencedTableName,
69 std::vector<std::string> referencedColumns);
81 SqlCreateTablePlan& _plan;
107 template <auto MemberPo
inter>
110 return AddColumn(std::string(FieldNameOf<MemberPointer>),
116 SqlColumnTypeDefinition columnType);
123 template <auto MemberPo
inter>
126 return AddNotRequiredColumn(std::string(FieldNameOf<MemberPointer>),
150 SqlColumnTypeDefinition columnType,
151 SqlNullable nullable);
169 SqlColumnTypeDefinition columnType);
173 SqlColumnTypeDefinition columnType);
225 SqlColumnTypeDefinition columnType,
245 std::string referencedTableName,
246 std::vector<std::string> referencedColumns);
254 template <
typename Record>
259#if defined(LIGHTWEIGHT_CXX26_REFLECTION)
260 constexpr auto ctx = std::meta::access_context::current();
261 template for (
constexpr auto el: define_static_array(nonstatic_data_members_of(^^Record, ctx)))
263 using FieldType =
typename[:std::meta::type_of(el):];
266 if constexpr (IsAutoIncrementPrimaryKey<FieldType>)
267 builder.PrimaryKeyWithAutoIncrement(
268 std::string(FieldNameOf<el>),
269 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
270 else if constexpr (FieldType::IsPrimaryKey)
271 builder.
PrimaryKey(std::string(FieldNameOf<el>),
272 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
273 else if constexpr (IsBelongsTo<FieldType>)
275 constexpr size_t referencedFieldIndex = []()
constexpr ->
size_t {
276 auto index = size_t(-1);
277 Reflection::EnumerateMembers<typename FieldType::ReferencedRecord>(
278 [&index]<
size_t J,
typename ReferencedFieldType>()
constexpr ->
void {
279 if constexpr (IsField<ReferencedFieldType>)
280 if constexpr (ReferencedFieldType::IsPrimaryKey)
286 std::string(FieldNameOf<el>),
287 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value,
288 SqlForeignKeyReferenceDefinition {
289 .tableName = std::string { RecordTableName<typename FieldType::ReferencedRecord> },
290 .columnName = std::string { FieldNameOf<FieldType::ReferencedField> } });
292 else if constexpr (FieldType::IsMandatory)
294 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
296 builder.
Column(std::string(FieldNameOf<el>),
297 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
301 Reflection::EnumerateMembers<Record>([&builder]<
size_t I,
typename FieldType>() {
302 if constexpr (FieldWithStorage<FieldType>)
304 if constexpr (IsAutoIncrementPrimaryKey<FieldType>)
305 builder.PrimaryKeyWithAutoIncrement(
306 std::string(FieldNameAt<I, Record>()),
307 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
308 else if constexpr (FieldType::IsPrimaryKey)
309 builder.
PrimaryKey(std::string(FieldNameAt<I, Record>()),
310 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
311 else if constexpr (IsBelongsTo<FieldType>)
313 constexpr size_t referencedFieldIndex = []()
constexpr ->
size_t {
314 auto index = size_t(-1);
315 Reflection::EnumerateMembers<typename FieldType::ReferencedRecord>(
316 [&index]<
size_t J,
typename ReferencedFieldType>()
constexpr ->
void {
317 if constexpr (IsField<ReferencedFieldType>)
318 if constexpr (ReferencedFieldType::IsPrimaryKey)
324 std::string(FieldNameAt<I, Record>()),
325 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value,
326 SqlForeignKeyReferenceDefinition {
327 .tableName = std::string { RecordTableName<typename FieldType::ReferencedRecord> },
329 std::string { FieldNameAt<referencedFieldIndex, typename FieldType::ReferencedRecord>() } });
331 else if constexpr (FieldType::IsMandatory)
333 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
335 builder.
Column(std::string(FieldNameAt<I, Record>()),
336 detail::SqlColumnTypeDefinitionOf<typename FieldType::ValueType>::value);
356 template <
typename T>
359 _plan.columns.emplace_back(std::move(columnName),
SqlVariant(value));
380 template <
typename T>
383 _plan.setColumns.emplace_back(std::move(columnName),
SqlVariant(value));
388 template <
typename T>
391 _plan.whereColumn = std::move(columnName);
392 _plan.whereOp = std::move(op);
414 template <
typename T>
417 _plan.whereColumn = std::move(columnName);
418 _plan.whereOp = std::move(op);
433 _formatter { formatter },
434 _migrationPlan { .formatter = formatter }
471 template <
typename Record>
476 auto builder = CreateTable(RecordTableName<Record>);
477 detail::PopulateCreateTableBuilder<Record>(builder);
482 template <
typename Record>
486 return AlterTable(RecordTableName<Record>);
505 std::string tableName,
506 std::vector<std::string> columns,
507 bool unique =
false);
515 std::string tableName,
516 std::vector<std::string> columns);
532 std::vector<std::string> columns,
533 IndexType type = IndexType::NonUnique);
563 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 & 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.
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 & 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.
Query builder for building INSERT queries in migrations.
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 & 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)
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.
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.
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.