velocious 1.0.352 → 1.0.354

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,28 @@
1
1
  export default class VelociousDatabaseConnectionDriversSqliteSqlAlterTable extends AlterTableBase {
2
2
  logger: Logger;
3
+ /**
4
+ * Merges the current schema with the alter request to produce the desired final schema
5
+ * and the column copy plan.
6
+ * @param {TableData} currentTableData - Current schema as introspected from the database.
7
+ * @param {TableData} alterTableData - Alter request: new columns (`isNewColumn`), renames (`newName`), drops (`dropColumn`), modifies, and new foreign keys.
8
+ * @returns {{targetTableData: TableData, columnPairs: Array<[string, string]>}} - The merged target schema and the [oldName, newName] pairs for INSERT...SELECT.
9
+ */
10
+ _buildTargetSchema(currentTableData: TableData, alterTableData: TableData): {
11
+ targetTableData: TableData;
12
+ columnPairs: Array<[string, string]>;
13
+ };
14
+ /**
15
+ * Returns the foreign key with its column name updated when the column was renamed in the
16
+ * alter request. SQLite re-creates the constraint inside the rebuilt CREATE TABLE, so a
17
+ * stale column name there would reference a column that no longer exists.
18
+ * @param {TableForeignKey} foreignKey - Foreign key to evaluate.
19
+ * @param {Map<string, string>} columnRenames - Map of old → new column names from the alter request.
20
+ * @returns {TableForeignKey} - The original foreign key, or a fresh instance with the renamed column.
21
+ */
22
+ _renameForeignKeyColumn(foreignKey: TableForeignKey, columnRenames: Map<string, string>): TableForeignKey;
3
23
  }
4
24
  import AlterTableBase from "../../../query/alter-table-base.js";
5
25
  import Logger from "../../../../logger.js";
26
+ import TableData from "../../../table-data/index.js";
27
+ import TableForeignKey from "../../../table-data/table-foreign-key.js";
6
28
  //# sourceMappingURL=alter-table.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"alter-table.d.ts","sourceRoot":"","sources":["../../../../../../src/database/drivers/sqlite/sql/alter-table.js"],"names":[],"mappings":"AAQA;IAYI,eAA8B;CA4IjC;2BA9J0B,oCAAoC;mBAE5C,uBAAuB"}
1
+ {"version":3,"file":"alter-table.d.ts","sourceRoot":"","sources":["../../../../../../src/database/drivers/sqlite/sql/alter-table.js"],"names":[],"mappings":"AAUA;IAYI,eAA8B;IA8ChC;;;;;;OAMG;IACH,qCAJW,SAAS,kBACT,SAAS,GACP;QAAC,eAAe,EAAE,SAAS,CAAC;QAAC,WAAW,EAAE,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAA;KAAC,CAmF9E;IAED;;;;;;;OAOG;IACH,oCAJW,eAAe,iBACf,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GACjB,eAAe,CAe3B;CACF;2BAlL0B,oCAAoC;mBAC5C,uBAAuB;sBAEpB,8BAA8B;4BACxB,0CAA0C"}
@@ -1,9 +1,11 @@
1
1
  // @ts-check
2
2
  import AlterTableBase from "../../../query/alter-table-base.js";
3
- import CreateIndexBase from "../../../query/create-index-base.js";
4
3
  import Logger from "../../../../logger.js";
5
4
  import restArgsError from "../../../../utils/rest-args-error.js";
6
5
  import TableData from "../../../table-data/index.js";
6
+ import TableForeignKey from "../../../table-data/table-foreign-key.js";
7
+ import TableIndex from "../../../table-data/table-index.js";
8
+ import TableRebuilder from "../table-rebuilder.js";
7
9
  export default class VelociousDatabaseConnectionDriversSqliteSqlAlterTable extends AlterTableBase {
8
10
  /**
9
11
  * @param {object} args - Options object.
@@ -22,109 +24,138 @@ export default class VelociousDatabaseConnectionDriversSqliteSqlAlterTable exten
22
24
  * @returns {Promise<string[]>} - Resolves with SQL statements.
23
25
  */
24
26
  async toSQLs() {
25
- const { tableData } = this;
26
- const table = await this.getDriver().getTableByName(tableData.getName());
27
+ const driver = this.getDriver();
28
+ const { tableData: alterTableData } = this;
29
+ const tableName = alterTableData.getName();
30
+ const table = await driver.getTableByName(tableName);
27
31
  if (!table)
28
- throw new Error(`Table ${tableData.getName()} does not exist`);
32
+ throw new Error(`Table ${tableName} does not exist`);
29
33
  const currentTableData = await table.getTableData();
30
- const options = this.getOptions();
31
- const tableName = tableData.getName();
32
- const tempTableName = `${tableData.getName()}AlterTableTemp`;
33
- const newColumnNames = currentTableData.getColumns()
34
- .filter((column) => !column.isNewColumn())
35
- .map((column) => {
36
- const newTableColumn = tableData.getColumns().find((tableColumn) => tableColumn.getName() == column.getName());
37
- return newTableColumn?.getNewName() || newTableColumn?.getName() || column.getNewName() || column.getName();
34
+ const { targetTableData, columnPairs } = this._buildTargetSchema(currentTableData, alterTableData);
35
+ const rebuilder = new TableRebuilder({
36
+ columnPairs,
37
+ driver,
38
+ originalTableName: tableName,
39
+ targetTableData
38
40
  });
39
- const oldColumnNames = currentTableData.getColumns().filter((column) => !column.isNewColumn()).map((column) => column.getName());
40
- const newColumnsSQL = newColumnNames.map((name) => options.quoteColumnName(name)).join(", ");
41
- const oldColumnsSQL = oldColumnNames.map((name) => options.quoteColumnName(name)).join(", ");
42
- tableData.setName(tempTableName);
43
- const newTableData = new TableData(tempTableName);
44
- for (const tableDataColumn of currentTableData.getColumns()) {
45
- let newTableDataColumn = tableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == tableDataColumn.getName());
46
- if (newTableDataColumn && newTableDataColumn.isNewColumn()) {
47
- newTableDataColumn = undefined;
41
+ const rebuildSQLs = await rebuilder.toSQLs();
42
+ const sqls = [];
43
+ // PRAGMA foreign_keys can only be toggled outside an active transaction; when the
44
+ // caller is already inside one these become no-ops (matching prior behavior). Outside
45
+ // a transaction they protect the rebuild from cross-table FK enforcement during the
46
+ // DROP/RENAME swap. Capture the prior state so we restore it instead of unconditionally
47
+ // forcing ON — callers that deliberately disabled FK enforcement (e.g. bulk data fixes)
48
+ // shouldn't be silently flipped back on by a migration.
49
+ const priorState = await driver.query("PRAGMA foreign_keys");
50
+ const wasEnabled = priorState[0]?.foreign_keys == 1;
51
+ sqls.push("PRAGMA foreign_keys = OFF");
52
+ for (const sql of rebuildSQLs)
53
+ sqls.push(sql);
54
+ sqls.push(`PRAGMA foreign_keys = ${wasEnabled ? "ON" : "OFF"}`);
55
+ return sqls;
56
+ }
57
+ /**
58
+ * Merges the current schema with the alter request to produce the desired final schema
59
+ * and the column copy plan.
60
+ * @param {TableData} currentTableData - Current schema as introspected from the database.
61
+ * @param {TableData} alterTableData - Alter request: new columns (`isNewColumn`), renames (`newName`), drops (`dropColumn`), modifies, and new foreign keys.
62
+ * @returns {{targetTableData: TableData, columnPairs: Array<[string, string]>}} - The merged target schema and the [oldName, newName] pairs for INSERT...SELECT.
63
+ */
64
+ _buildTargetSchema(currentTableData, alterTableData) {
65
+ const targetTableData = new TableData(currentTableData.getName());
66
+ /** @type {Array<[string, string]>} */
67
+ const columnPairs = [];
68
+ const alterColumns = alterTableData.getColumns();
69
+ const existingNames = new Set(currentTableData.getColumns().map((column) => column.getName()));
70
+ /** @type {Map<string, string>} */
71
+ const columnRenames = new Map();
72
+ for (const alterColumn of alterColumns) {
73
+ if (alterColumn.isNewColumn())
74
+ continue;
75
+ const newName = alterColumn.getNewName();
76
+ if (newName)
77
+ columnRenames.set(alterColumn.getName(), newName);
78
+ }
79
+ for (const currentColumn of currentTableData.getColumns()) {
80
+ const alterColumn = alterColumns.find((column) => column.getName() == currentColumn.getName() && !column.isNewColumn());
81
+ if (alterColumn?.getDropColumn())
82
+ continue;
83
+ let targetColumn;
84
+ if (alterColumn) {
85
+ // The alter request supplies a partial column spec (e.g. just a rename or a type change);
86
+ // inherit unset properties from the current column so we don't lose existing definitions.
87
+ alterColumn.setAutoIncrement(alterColumn.getAutoIncrement() || currentColumn.getAutoIncrement());
88
+ if (alterColumn.getDefault() === undefined)
89
+ alterColumn.setDefault(currentColumn.getDefault());
90
+ if (!alterColumn.getIndex())
91
+ alterColumn.setIndex(currentColumn.getIndex());
92
+ if (!alterColumn.getForeignKey())
93
+ alterColumn.setForeignKey(currentColumn.getForeignKey());
94
+ if (alterColumn.getMaxLength() === undefined)
95
+ alterColumn.setMaxLength(currentColumn.getMaxLength());
96
+ alterColumn.setPrimaryKey(alterColumn.getPrimaryKey() || currentColumn.getPrimaryKey());
97
+ if (!alterColumn.getType())
98
+ alterColumn.setType(currentColumn.getType());
99
+ targetColumn = alterColumn;
48
100
  }
49
- if (newTableDataColumn) {
50
- newTableDataColumn.setAutoIncrement(tableDataColumn.getAutoIncrement());
51
- newTableDataColumn.setDefault(tableDataColumn.getDefault());
52
- newTableDataColumn.setIndex(tableDataColumn.getIndex());
53
- newTableDataColumn.setForeignKey(tableDataColumn.getForeignKey());
54
- newTableDataColumn.setMaxLength(tableDataColumn.getMaxLength());
55
- newTableDataColumn.setPrimaryKey(tableDataColumn.getPrimaryKey());
56
- newTableDataColumn.setType(tableDataColumn.getType());
101
+ else {
102
+ targetColumn = currentColumn;
57
103
  }
58
- newTableData.addColumn(newTableDataColumn || tableDataColumn);
104
+ targetTableData.addColumn(targetColumn);
105
+ columnPairs.push([currentColumn.getName(), targetColumn.getNewName() || targetColumn.getName()]);
59
106
  }
60
- // SQLite rebuilds use existing columns plus new ones; avoid duplicating a column that already exists.
61
- const existingColumnNames = new Set(currentTableData.getColumns().map((column) => column.getName()));
62
- for (const tableDataColumn of tableData.getColumns()) {
63
- if (!tableDataColumn.isNewColumn())
107
+ for (const alterColumn of alterColumns) {
108
+ if (!alterColumn.isNewColumn())
64
109
  continue;
65
- if (existingColumnNames.has(tableDataColumn.getName()))
110
+ if (existingNames.has(alterColumn.getName()))
66
111
  continue;
67
- newTableData.addColumn(tableDataColumn);
112
+ targetTableData.addColumn(alterColumn);
68
113
  }
69
- const foundForeignKeys = [];
70
- for (const tableDataForeignKey of currentTableData.getForeignKeys()) {
71
- const newTableDataForeignKey = newTableData.getForeignKeys().find((newTableDataForeignKey) => newTableDataForeignKey.getName() == tableDataForeignKey.getName());
72
- if (newTableDataForeignKey)
73
- foundForeignKeys.push(newTableDataForeignKey.getName());
74
- const actualTableDataForeignKey = newTableDataForeignKey || tableDataForeignKey;
75
- // Register foreign key on the table
76
- newTableData.addForeignKey(actualTableDataForeignKey);
77
- // Register foreign key on the column
78
- const tableDataColumn = newTableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == actualTableDataForeignKey.getColumnName());
79
- if (!tableDataColumn)
80
- throw new Error(`Couldn't find column for foreign key: ${actualTableDataForeignKey.getName()}`);
81
- this.logger.debugLowLevel(() => `Setting foreign key on column ${tableDataColumn.getName()}`);
82
- tableDataColumn.setForeignKey(actualTableDataForeignKey);
114
+ const seenForeignKeyNames = new Set();
115
+ for (const currentForeignKey of currentTableData.getForeignKeys()) {
116
+ const alterForeignKey = alterTableData.getForeignKeys().find((foreignKey) => foreignKey.getName() == currentForeignKey.getName());
117
+ const finalForeignKey = this._renameForeignKeyColumn(alterForeignKey || currentForeignKey, columnRenames);
118
+ seenForeignKeyNames.add(finalForeignKey.getName());
119
+ targetTableData.addForeignKey(finalForeignKey);
83
120
  }
84
- for (const foreignKey of tableData.getForeignKeys()) {
85
- if (foundForeignKeys.includes(foreignKey.getName()))
121
+ for (const alterForeignKey of alterTableData.getForeignKeys()) {
122
+ if (seenForeignKeyNames.has(alterForeignKey.getName()))
86
123
  continue;
87
- // Register foreign key on the table
88
- newTableData.addForeignKey(foreignKey);
89
- // Register foreign key on the column
90
- const tableDataColumn = newTableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == foreignKey.getColumnName());
91
- if (!tableDataColumn)
92
- throw new Error(`Couldn't find column for foreign key: ${foreignKey.getName()}`);
93
- this.logger.debugLowLevel(() => `Setting foreign key on column ${tableDataColumn.getName()}`);
94
- tableDataColumn.setForeignKey(foreignKey);
124
+ targetTableData.addForeignKey(this._renameForeignKeyColumn(alterForeignKey, columnRenames));
95
125
  }
96
- const createNewTableSQL = await this.getDriver().createTableSql(newTableData);
97
- const insertSQL = `INSERT INTO ${options.quoteTableName(tempTableName)} (${newColumnsSQL}) SELECT ${oldColumnsSQL} FROM ${options.quoteTableName(tableName)}`;
98
- const dropTableSQLs = `DROP TABLE ${options.quoteTableName(tableName)}`;
99
- const renameTableSQL = `ALTER TABLE ${options.quoteTableName(tempTableName)} RENAME TO ${options.quoteTableName(tableName)}`;
100
- const sqls = [];
101
- for (const sql of createNewTableSQL) {
102
- sqls.push(sql);
103
- }
104
- sqls.push(insertSQL);
105
- sqls.push(dropTableSQLs);
106
- sqls.push(renameTableSQL);
107
- for (const tableDataIndex of currentTableData.getIndexes()) {
108
- const newTableDataIndex = newTableData.getIndexes().find((newTableDataIndex) => newTableDataIndex.getName() == tableDataIndex.getName());
109
- const actualTableIndex = newTableDataIndex || tableDataIndex;
110
- newTableData.addIndex(actualTableIndex);
111
- const columnNames = actualTableIndex.getColumns().map((columnName) => {
112
- const newTableColumn = tableData.getColumns().find((tableColumn) => tableColumn.getName() == columnName);
113
- return newTableColumn?.getNewName() || newTableColumn?.getName() || columnName;
126
+ for (const currentIndex of currentTableData.getIndexes()) {
127
+ const renamedColumns = currentIndex.getColumns().map((columnName) => {
128
+ if (typeof columnName != "string")
129
+ return columnName;
130
+ return columnRenames.get(columnName) || columnName;
114
131
  });
115
- const createIndexArgs = {
116
- columns: columnNames,
117
- driver: this.getDriver(),
118
- name: actualTableIndex.getName(),
119
- tableName,
120
- unique: actualTableIndex.getUnique()
121
- };
122
- const createIndexSQLs = await new CreateIndexBase(createIndexArgs).toSQLs();
123
- for (const createIndexSQL of createIndexSQLs) {
124
- sqls.push(createIndexSQL);
125
- }
132
+ targetTableData.addIndex(new TableIndex(renamedColumns, {
133
+ name: currentIndex.getName(),
134
+ unique: currentIndex.getUnique()
135
+ }));
126
136
  }
127
- return sqls;
137
+ return { targetTableData, columnPairs };
138
+ }
139
+ /**
140
+ * Returns the foreign key with its column name updated when the column was renamed in the
141
+ * alter request. SQLite re-creates the constraint inside the rebuilt CREATE TABLE, so a
142
+ * stale column name there would reference a column that no longer exists.
143
+ * @param {TableForeignKey} foreignKey - Foreign key to evaluate.
144
+ * @param {Map<string, string>} columnRenames - Map of old → new column names from the alter request.
145
+ * @returns {TableForeignKey} - The original foreign key, or a fresh instance with the renamed column.
146
+ */
147
+ _renameForeignKeyColumn(foreignKey, columnRenames) {
148
+ const renamed = columnRenames.get(foreignKey.getColumnName());
149
+ if (!renamed)
150
+ return foreignKey;
151
+ return new TableForeignKey({
152
+ columnName: renamed,
153
+ isNewForeignKey: foreignKey.getIsNewForeignKey(),
154
+ name: foreignKey.getName(),
155
+ referencedColumnName: foreignKey.getReferencedColumnName(),
156
+ referencedTableName: foreignKey.getReferencedTableName(),
157
+ tableName: foreignKey.getTableName()
158
+ });
128
159
  }
129
160
  }
130
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alter-table.js","sourceRoot":"","sources":["../../../../../../src/database/drivers/sqlite/sql/alter-table.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport AlterTableBase from \"../../../query/alter-table-base.js\"\nimport CreateIndexBase from \"../../../query/create-index-base.js\"\nimport Logger from \"../../../../logger.js\"\nimport restArgsError from \"../../../../utils/rest-args-error.js\"\nimport TableData from \"../../../table-data/index.js\"\n\nexport default class VelociousDatabaseConnectionDriversSqliteSqlAlterTable extends AlterTableBase {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../../base.js\").default} args.driver - Database driver instance.\n   * @param {import(\"../../../table-data/index.js\").default} args.tableData - Table data.\n   */\n  constructor({driver, tableData, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!(tableData instanceof TableData)) throw new Error(\"Invalid table data was given\")\n\n    super({driver, tableData})\n    this.logger = new Logger(this)\n    this.tableData = tableData\n  }\n\n  /**\n   * @returns {Promise<string[]>} - Resolves with SQL statements.\n   */\n  async toSQLs() {\n    const {tableData} = this\n    const table = await this.getDriver().getTableByName(tableData.getName())\n\n    if (!table) throw new Error(`Table ${tableData.getName()} does not exist`)\n\n    const currentTableData = await table.getTableData()\n    const options = this.getOptions()\n    const tableName = tableData.getName()\n    const tempTableName = `${tableData.getName()}AlterTableTemp`\n    const newColumnNames = currentTableData.getColumns()\n      .filter((column) => !column.isNewColumn())\n      .map((column) => {\n        const newTableColumn = tableData.getColumns().find((tableColumn) => tableColumn.getName() == column.getName())\n\n        return newTableColumn?.getNewName() || newTableColumn?.getName() || column.getNewName() || column.getName()\n      })\n    const oldColumnNames = currentTableData.getColumns().filter((column) => !column.isNewColumn()).map((column) => column.getName())\n    const newColumnsSQL = newColumnNames.map((name) => options.quoteColumnName(name)).join(\", \")\n    const oldColumnsSQL = oldColumnNames.map((name) => options.quoteColumnName(name)).join(\", \")\n\n    tableData.setName(tempTableName)\n\n    const newTableData = new TableData(tempTableName)\n\n    for (const tableDataColumn of currentTableData.getColumns()) {\n      let newTableDataColumn = tableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == tableDataColumn.getName())\n\n      if (newTableDataColumn && newTableDataColumn.isNewColumn()) {\n        newTableDataColumn = undefined\n      }\n\n      if (newTableDataColumn) {\n        newTableDataColumn.setAutoIncrement(tableDataColumn.getAutoIncrement())\n        newTableDataColumn.setDefault(tableDataColumn.getDefault())\n        newTableDataColumn.setIndex(tableDataColumn.getIndex())\n        newTableDataColumn.setForeignKey(tableDataColumn.getForeignKey())\n        newTableDataColumn.setMaxLength(tableDataColumn.getMaxLength())\n        newTableDataColumn.setPrimaryKey(tableDataColumn.getPrimaryKey())\n        newTableDataColumn.setType(tableDataColumn.getType())\n      }\n\n      newTableData.addColumn(newTableDataColumn || tableDataColumn)\n    }\n\n    // SQLite rebuilds use existing columns plus new ones; avoid duplicating a column that already exists.\n    const existingColumnNames = new Set(currentTableData.getColumns().map((column) => column.getName()))\n\n    for (const tableDataColumn of tableData.getColumns()) {\n      if (!tableDataColumn.isNewColumn()) continue\n      if (existingColumnNames.has(tableDataColumn.getName())) continue\n\n      newTableData.addColumn(tableDataColumn)\n    }\n\n    const foundForeignKeys = []\n\n    for (const tableDataForeignKey of currentTableData.getForeignKeys()) {\n      const newTableDataForeignKey = newTableData.getForeignKeys().find((newTableDataForeignKey) => newTableDataForeignKey.getName() == tableDataForeignKey.getName())\n\n      if (newTableDataForeignKey) foundForeignKeys.push(newTableDataForeignKey.getName())\n\n      const actualTableDataForeignKey = newTableDataForeignKey || tableDataForeignKey\n\n      // Register foreign key on the table\n      newTableData.addForeignKey(actualTableDataForeignKey)\n\n      // Register foreign key on the column\n      const tableDataColumn = newTableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == actualTableDataForeignKey.getColumnName())\n\n      if (!tableDataColumn) throw new Error(`Couldn't find column for foreign key: ${actualTableDataForeignKey.getName()}`)\n\n      this.logger.debugLowLevel(() => `Setting foreign key on column ${tableDataColumn.getName()}`)\n      tableDataColumn.setForeignKey(actualTableDataForeignKey)\n    }\n\n    for (const foreignKey of tableData.getForeignKeys()) {\n      if (foundForeignKeys.includes(foreignKey.getName())) continue\n\n      // Register foreign key on the table\n      newTableData.addForeignKey(foreignKey)\n\n      // Register foreign key on the column\n      const tableDataColumn = newTableData.getColumns().find((newTableDataColumn) => newTableDataColumn.getName() == foreignKey.getColumnName())\n\n      if (!tableDataColumn) throw new Error(`Couldn't find column for foreign key: ${foreignKey.getName()}`)\n\n      this.logger.debugLowLevel(() => `Setting foreign key on column ${tableDataColumn.getName()}`)\n      tableDataColumn.setForeignKey(foreignKey)\n    }\n\n    const createNewTableSQL = await this.getDriver().createTableSql(newTableData)\n    const insertSQL = `INSERT INTO ${options.quoteTableName(tempTableName)} (${newColumnsSQL}) SELECT ${oldColumnsSQL} FROM ${options.quoteTableName(tableName)}`\n    const dropTableSQLs = `DROP TABLE ${options.quoteTableName(tableName)}`\n    const renameTableSQL = `ALTER TABLE ${options.quoteTableName(tempTableName)} RENAME TO ${options.quoteTableName(tableName)}`\n    const sqls = []\n\n    for (const sql of createNewTableSQL) {\n      sqls.push(sql)\n    }\n\n    sqls.push(insertSQL)\n    sqls.push(dropTableSQLs)\n    sqls.push(renameTableSQL)\n\n    for (const tableDataIndex of currentTableData.getIndexes()) {\n      const newTableDataIndex = newTableData.getIndexes().find((newTableDataIndex) => newTableDataIndex.getName() == tableDataIndex.getName())\n      const actualTableIndex = newTableDataIndex || tableDataIndex\n\n      newTableData.addIndex(actualTableIndex)\n\n      const columnNames = actualTableIndex.getColumns().map((columnName) => {\n        const newTableColumn = tableData.getColumns().find((tableColumn) => tableColumn.getName() == columnName)\n\n        return newTableColumn?.getNewName() || newTableColumn?.getName() || columnName\n      })\n\n      const createIndexArgs = {\n        columns: columnNames,\n        driver: this.getDriver(),\n        name: actualTableIndex.getName(),\n        tableName,\n        unique: actualTableIndex.getUnique()\n      }\n      const createIndexSQLs = await new CreateIndexBase(createIndexArgs).toSQLs()\n\n      for (const createIndexSQL of createIndexSQLs) {\n        sqls.push(createIndexSQL)\n      }\n    }\n\n    return sqls\n  }\n}\n"]}
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alter-table.js","sourceRoot":"","sources":["../../../../../../src/database/drivers/sqlite/sql/alter-table.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport AlterTableBase from \"../../../query/alter-table-base.js\"\nimport Logger from \"../../../../logger.js\"\nimport restArgsError from \"../../../../utils/rest-args-error.js\"\nimport TableData from \"../../../table-data/index.js\"\nimport TableForeignKey from \"../../../table-data/table-foreign-key.js\"\nimport TableIndex from \"../../../table-data/table-index.js\"\nimport TableRebuilder from \"../table-rebuilder.js\"\n\nexport default class VelociousDatabaseConnectionDriversSqliteSqlAlterTable extends AlterTableBase {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../../base.js\").default} args.driver - Database driver instance.\n   * @param {import(\"../../../table-data/index.js\").default} args.tableData - Table data.\n   */\n  constructor({driver, tableData, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!(tableData instanceof TableData)) throw new Error(\"Invalid table data was given\")\n\n    super({driver, tableData})\n    this.logger = new Logger(this)\n    this.tableData = tableData\n  }\n\n  /**\n   * @returns {Promise<string[]>} - Resolves with SQL statements.\n   */\n  async toSQLs() {\n    const driver = this.getDriver()\n    const {tableData: alterTableData} = this\n    const tableName = alterTableData.getName()\n    const table = await driver.getTableByName(tableName)\n\n    if (!table) throw new Error(`Table ${tableName} does not exist`)\n\n    const currentTableData = await table.getTableData()\n    const {targetTableData, columnPairs} = this._buildTargetSchema(currentTableData, alterTableData)\n\n    const rebuilder = new TableRebuilder({\n      columnPairs,\n      driver,\n      originalTableName: tableName,\n      targetTableData\n    })\n\n    const rebuildSQLs = await rebuilder.toSQLs()\n    const sqls = []\n\n    // PRAGMA foreign_keys can only be toggled outside an active transaction; when the\n    // caller is already inside one these become no-ops (matching prior behavior). Outside\n    // a transaction they protect the rebuild from cross-table FK enforcement during the\n    // DROP/RENAME swap. Capture the prior state so we restore it instead of unconditionally\n    // forcing ON — callers that deliberately disabled FK enforcement (e.g. bulk data fixes)\n    // shouldn't be silently flipped back on by a migration.\n    const priorState = await driver.query(\"PRAGMA foreign_keys\")\n    const wasEnabled = priorState[0]?.foreign_keys == 1\n\n    sqls.push(\"PRAGMA foreign_keys = OFF\")\n\n    for (const sql of rebuildSQLs) sqls.push(sql)\n\n    sqls.push(`PRAGMA foreign_keys = ${wasEnabled ? \"ON\" : \"OFF\"}`)\n\n    return sqls\n  }\n\n  /**\n   * Merges the current schema with the alter request to produce the desired final schema\n   * and the column copy plan.\n   * @param {TableData} currentTableData - Current schema as introspected from the database.\n   * @param {TableData} alterTableData - Alter request: new columns (`isNewColumn`), renames (`newName`), drops (`dropColumn`), modifies, and new foreign keys.\n   * @returns {{targetTableData: TableData, columnPairs: Array<[string, string]>}} - The merged target schema and the [oldName, newName] pairs for INSERT...SELECT.\n   */\n  _buildTargetSchema(currentTableData, alterTableData) {\n    const targetTableData = new TableData(currentTableData.getName())\n    /** @type {Array<[string, string]>} */\n    const columnPairs = []\n    const alterColumns = alterTableData.getColumns()\n    const existingNames = new Set(currentTableData.getColumns().map((column) => column.getName()))\n    /** @type {Map<string, string>} */\n    const columnRenames = new Map()\n\n    for (const alterColumn of alterColumns) {\n      if (alterColumn.isNewColumn()) continue\n\n      const newName = alterColumn.getNewName()\n\n      if (newName) columnRenames.set(alterColumn.getName(), newName)\n    }\n\n    for (const currentColumn of currentTableData.getColumns()) {\n      const alterColumn = alterColumns.find((column) => column.getName() == currentColumn.getName() && !column.isNewColumn())\n\n      if (alterColumn?.getDropColumn()) continue\n\n      let targetColumn\n\n      if (alterColumn) {\n        // The alter request supplies a partial column spec (e.g. just a rename or a type change);\n        // inherit unset properties from the current column so we don't lose existing definitions.\n        alterColumn.setAutoIncrement(alterColumn.getAutoIncrement() || currentColumn.getAutoIncrement())\n        if (alterColumn.getDefault() === undefined) alterColumn.setDefault(currentColumn.getDefault())\n        if (!alterColumn.getIndex()) alterColumn.setIndex(currentColumn.getIndex())\n        if (!alterColumn.getForeignKey()) alterColumn.setForeignKey(currentColumn.getForeignKey())\n        if (alterColumn.getMaxLength() === undefined) alterColumn.setMaxLength(currentColumn.getMaxLength())\n        alterColumn.setPrimaryKey(alterColumn.getPrimaryKey() || currentColumn.getPrimaryKey())\n        if (!alterColumn.getType()) alterColumn.setType(currentColumn.getType())\n\n        targetColumn = alterColumn\n      } else {\n        targetColumn = currentColumn\n      }\n\n      targetTableData.addColumn(targetColumn)\n      columnPairs.push([currentColumn.getName(), targetColumn.getNewName() || targetColumn.getName()])\n    }\n\n    for (const alterColumn of alterColumns) {\n      if (!alterColumn.isNewColumn()) continue\n      if (existingNames.has(alterColumn.getName())) continue\n\n      targetTableData.addColumn(alterColumn)\n    }\n\n    const seenForeignKeyNames = new Set()\n\n    for (const currentForeignKey of currentTableData.getForeignKeys()) {\n      const alterForeignKey = alterTableData.getForeignKeys().find((foreignKey) => foreignKey.getName() == currentForeignKey.getName())\n      const finalForeignKey = this._renameForeignKeyColumn(alterForeignKey || currentForeignKey, columnRenames)\n\n      seenForeignKeyNames.add(finalForeignKey.getName())\n      targetTableData.addForeignKey(finalForeignKey)\n    }\n\n    for (const alterForeignKey of alterTableData.getForeignKeys()) {\n      if (seenForeignKeyNames.has(alterForeignKey.getName())) continue\n\n      targetTableData.addForeignKey(this._renameForeignKeyColumn(alterForeignKey, columnRenames))\n    }\n\n    for (const currentIndex of currentTableData.getIndexes()) {\n      const renamedColumns = currentIndex.getColumns().map((columnName) => {\n        if (typeof columnName != \"string\") return columnName\n\n        return columnRenames.get(columnName) || columnName\n      })\n\n      targetTableData.addIndex(new TableIndex(renamedColumns, {\n        name: currentIndex.getName(),\n        unique: currentIndex.getUnique()\n      }))\n    }\n\n    return {targetTableData, columnPairs}\n  }\n\n  /**\n   * Returns the foreign key with its column name updated when the column was renamed in the\n   * alter request. SQLite re-creates the constraint inside the rebuilt CREATE TABLE, so a\n   * stale column name there would reference a column that no longer exists.\n   * @param {TableForeignKey} foreignKey - Foreign key to evaluate.\n   * @param {Map<string, string>} columnRenames - Map of old → new column names from the alter request.\n   * @returns {TableForeignKey} - The original foreign key, or a fresh instance with the renamed column.\n   */\n  _renameForeignKeyColumn(foreignKey, columnRenames) {\n    const renamed = columnRenames.get(foreignKey.getColumnName())\n\n    if (!renamed) return foreignKey\n\n    return new TableForeignKey({\n      columnName: renamed,\n      isNewForeignKey: foreignKey.getIsNewForeignKey(),\n      name: foreignKey.getName(),\n      referencedColumnName: foreignKey.getReferencedColumnName(),\n      referencedTableName: foreignKey.getReferencedTableName(),\n      tableName: foreignKey.getTableName()\n    })\n  }\n}\n"]}
@@ -0,0 +1,41 @@
1
+ /**
2
+ * Emits the SQL sequence for SQLite's "rebuild" approach to schema changes.
3
+ *
4
+ * SQLite cannot add/drop foreign-key constraints, drop columns on older
5
+ * versions, change column types, or add CHECK constraints via ALTER TABLE.
6
+ * The standard workaround (https://sqlite.org/lang_altertable.html) is to
7
+ * create a new table with the desired schema, copy rows over, drop the
8
+ * original, and rename the replacement.
9
+ *
10
+ * Caller passes the desired final schema; this class handles the mechanical
11
+ * sequence (CREATE temp / INSERT...SELECT / DROP / RENAME / recreate
12
+ * indexes). Caller is responsible for any FK toggling or transaction setup
13
+ * around the returned SQL — `PRAGMA foreign_keys` is connection-scoped and
14
+ * cannot be flipped inside a transaction, so wrapping policy is left to the
15
+ * caller (see `sql/alter-table.js`).
16
+ */
17
+ export default class VelociousDatabaseDriversSqliteTableRebuilder {
18
+ /**
19
+ * @param {object} args - Options object.
20
+ * @param {import("../base.js").default} args.driver - Database driver instance.
21
+ * @param {string} args.originalTableName - Name of the existing table to rebuild.
22
+ * @param {TableData} args.targetTableData - Desired final schema (columns + foreign keys + indexes). The instance's name is overwritten internally during emission.
23
+ * @param {Array<[string, string]>} args.columnPairs - Pairs of [oldColumnName, newColumnName] describing how rows from the original table should populate the rebuilt table.
24
+ */
25
+ constructor({ driver, originalTableName, targetTableData, columnPairs, ...restArgs }: {
26
+ driver: import("../base.js").default;
27
+ originalTableName: string;
28
+ targetTableData: TableData;
29
+ columnPairs: Array<[string, string]>;
30
+ });
31
+ driver: import("../base.js").default;
32
+ originalTableName: string;
33
+ targetTableData: TableData;
34
+ columnPairs: [string, string][];
35
+ /**
36
+ * @returns {Promise<string[]>} - Resolves with SQL statements to execute in order.
37
+ */
38
+ toSQLs(): Promise<string[]>;
39
+ }
40
+ import TableData from "../../table-data/index.js";
41
+ //# sourceMappingURL=table-rebuilder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"table-rebuilder.d.ts","sourceRoot":"","sources":["../../../../../src/database/drivers/sqlite/table-rebuilder.js"],"names":[],"mappings":"AAMA;;;;;;;;;;;;;;;GAeG;AACH;IACE;;;;;;OAMG;IACH,sFALG;QAA2C,MAAM,EAAzC,OAAO,YAAY,EAAE,OAAO;QACf,iBAAiB,EAA9B,MAAM;QACU,eAAe,EAA/B,SAAS;QACqB,WAAW,EAAzC,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;KACjC,EAUA;IAJC,qCAAoB;IACpB,0BAA0C;IAC1C,2BAAsC;IACtC,gCAA8B;IAGhC;;OAEG;IACH,UAFa,OAAO,CAAC,MAAM,EAAE,CAAC,CAkD7B;CACF;sBAzFqB,2BAA2B"}
@@ -0,0 +1,81 @@
1
+ // @ts-check
2
+ import CreateIndexBase from "../../query/create-index-base.js";
3
+ import restArgsError from "../../../utils/rest-args-error.js";
4
+ import TableData from "../../table-data/index.js";
5
+ /**
6
+ * Emits the SQL sequence for SQLite's "rebuild" approach to schema changes.
7
+ *
8
+ * SQLite cannot add/drop foreign-key constraints, drop columns on older
9
+ * versions, change column types, or add CHECK constraints via ALTER TABLE.
10
+ * The standard workaround (https://sqlite.org/lang_altertable.html) is to
11
+ * create a new table with the desired schema, copy rows over, drop the
12
+ * original, and rename the replacement.
13
+ *
14
+ * Caller passes the desired final schema; this class handles the mechanical
15
+ * sequence (CREATE temp / INSERT...SELECT / DROP / RENAME / recreate
16
+ * indexes). Caller is responsible for any FK toggling or transaction setup
17
+ * around the returned SQL — `PRAGMA foreign_keys` is connection-scoped and
18
+ * cannot be flipped inside a transaction, so wrapping policy is left to the
19
+ * caller (see `sql/alter-table.js`).
20
+ */
21
+ export default class VelociousDatabaseDriversSqliteTableRebuilder {
22
+ /**
23
+ * @param {object} args - Options object.
24
+ * @param {import("../base.js").default} args.driver - Database driver instance.
25
+ * @param {string} args.originalTableName - Name of the existing table to rebuild.
26
+ * @param {TableData} args.targetTableData - Desired final schema (columns + foreign keys + indexes). The instance's name is overwritten internally during emission.
27
+ * @param {Array<[string, string]>} args.columnPairs - Pairs of [oldColumnName, newColumnName] describing how rows from the original table should populate the rebuilt table.
28
+ */
29
+ constructor({ driver, originalTableName, targetTableData, columnPairs, ...restArgs }) {
30
+ restArgsError(restArgs);
31
+ if (!(targetTableData instanceof TableData))
32
+ throw new Error("Invalid target table data was given");
33
+ this.driver = driver;
34
+ this.originalTableName = originalTableName;
35
+ this.targetTableData = targetTableData;
36
+ this.columnPairs = columnPairs;
37
+ }
38
+ /**
39
+ * @returns {Promise<string[]>} - Resolves with SQL statements to execute in order.
40
+ */
41
+ async toSQLs() {
42
+ const driver = this.driver;
43
+ const options = driver.options();
44
+ const originalTableName = this.originalTableName;
45
+ const tempTableName = `${originalTableName}_velocious_rebuild`;
46
+ const targetTableData = this.targetTableData;
47
+ const previousTargetName = targetTableData.getName();
48
+ targetTableData.setName(tempTableName);
49
+ let createTableSQLs;
50
+ try {
51
+ createTableSQLs = await driver.createTableSql(targetTableData);
52
+ }
53
+ finally {
54
+ targetTableData.setName(previousTargetName);
55
+ }
56
+ const newColumnsSQL = this.columnPairs.map(([, newName]) => options.quoteColumnName(newName)).join(", ");
57
+ const oldColumnsSQL = this.columnPairs.map(([oldName]) => options.quoteColumnName(oldName)).join(", ");
58
+ const sqls = [];
59
+ for (const sql of createTableSQLs)
60
+ sqls.push(sql);
61
+ if (this.columnPairs.length > 0) {
62
+ sqls.push(`INSERT INTO ${options.quoteTableName(tempTableName)} (${newColumnsSQL}) ` +
63
+ `SELECT ${oldColumnsSQL} FROM ${options.quoteTableName(originalTableName)}`);
64
+ }
65
+ sqls.push(`DROP TABLE ${options.quoteTableName(originalTableName)}`);
66
+ sqls.push(`ALTER TABLE ${options.quoteTableName(tempTableName)} RENAME TO ${options.quoteTableName(originalTableName)}`);
67
+ for (const tableDataIndex of targetTableData.getIndexes()) {
68
+ const createIndexSQLs = await new CreateIndexBase({
69
+ columns: tableDataIndex.getColumns(),
70
+ driver,
71
+ name: tableDataIndex.getName(),
72
+ tableName: originalTableName,
73
+ unique: tableDataIndex.getUnique()
74
+ }).toSQLs();
75
+ for (const sql of createIndexSQLs)
76
+ sqls.push(sql);
77
+ }
78
+ return sqls;
79
+ }
80
+ }
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-rebuilder.js","sourceRoot":"","sources":["../../../../../src/database/drivers/sqlite/table-rebuilder.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport CreateIndexBase from \"../../query/create-index-base.js\"\nimport restArgsError from \"../../../utils/rest-args-error.js\"\nimport TableData from \"../../table-data/index.js\"\n\n/**\n * Emits the SQL sequence for SQLite's \"rebuild\" approach to schema changes.\n *\n * SQLite cannot add/drop foreign-key constraints, drop columns on older\n * versions, change column types, or add CHECK constraints via ALTER TABLE.\n * The standard workaround (https://sqlite.org/lang_altertable.html) is to\n * create a new table with the desired schema, copy rows over, drop the\n * original, and rename the replacement.\n *\n * Caller passes the desired final schema; this class handles the mechanical\n * sequence (CREATE temp / INSERT...SELECT / DROP / RENAME / recreate\n * indexes). Caller is responsible for any FK toggling or transaction setup\n * around the returned SQL — `PRAGMA foreign_keys` is connection-scoped and\n * cannot be flipped inside a transaction, so wrapping policy is left to the\n * caller (see `sql/alter-table.js`).\n */\nexport default class VelociousDatabaseDriversSqliteTableRebuilder {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../base.js\").default} args.driver - Database driver instance.\n   * @param {string} args.originalTableName - Name of the existing table to rebuild.\n   * @param {TableData} args.targetTableData - Desired final schema (columns + foreign keys + indexes). The instance's name is overwritten internally during emission.\n   * @param {Array<[string, string]>} args.columnPairs - Pairs of [oldColumnName, newColumnName] describing how rows from the original table should populate the rebuilt table.\n   */\n  constructor({driver, originalTableName, targetTableData, columnPairs, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!(targetTableData instanceof TableData)) throw new Error(\"Invalid target table data was given\")\n\n    this.driver = driver\n    this.originalTableName = originalTableName\n    this.targetTableData = targetTableData\n    this.columnPairs = columnPairs\n  }\n\n  /**\n   * @returns {Promise<string[]>} - Resolves with SQL statements to execute in order.\n   */\n  async toSQLs() {\n    const driver = this.driver\n    const options = driver.options()\n    const originalTableName = this.originalTableName\n    const tempTableName = `${originalTableName}_velocious_rebuild`\n    const targetTableData = this.targetTableData\n    const previousTargetName = targetTableData.getName()\n\n    targetTableData.setName(tempTableName)\n\n    let createTableSQLs\n\n    try {\n      createTableSQLs = await driver.createTableSql(targetTableData)\n    } finally {\n      targetTableData.setName(previousTargetName)\n    }\n\n    const newColumnsSQL = this.columnPairs.map(([, newName]) => options.quoteColumnName(newName)).join(\", \")\n    const oldColumnsSQL = this.columnPairs.map(([oldName]) => options.quoteColumnName(oldName)).join(\", \")\n\n    const sqls = []\n\n    for (const sql of createTableSQLs) sqls.push(sql)\n\n    if (this.columnPairs.length > 0) {\n      sqls.push(\n        `INSERT INTO ${options.quoteTableName(tempTableName)} (${newColumnsSQL}) ` +\n        `SELECT ${oldColumnsSQL} FROM ${options.quoteTableName(originalTableName)}`\n      )\n    }\n\n    sqls.push(`DROP TABLE ${options.quoteTableName(originalTableName)}`)\n    sqls.push(`ALTER TABLE ${options.quoteTableName(tempTableName)} RENAME TO ${options.quoteTableName(originalTableName)}`)\n\n    for (const tableDataIndex of targetTableData.getIndexes()) {\n      const createIndexSQLs = await new CreateIndexBase({\n        columns: tableDataIndex.getColumns(),\n        driver,\n        name: tableDataIndex.getName(),\n        tableName: originalTableName,\n        unique: tableDataIndex.getUnique()\n      }).toSQLs()\n\n      for (const sql of createIndexSQLs) sqls.push(sql)\n    }\n\n    return sqls\n  }\n}\n"]}
@@ -73,11 +73,40 @@ export default class VelociousDatabaseMigration {
73
73
  unique?: boolean | undefined;
74
74
  }): Promise<void>;
75
75
  /**
76
- * @param {string} tableName - Table name.
77
- * @param {string} referenceName - Reference name.
76
+ * @typedef {object} AddForeignKeyArgsType
77
+ * @property {string} [columnName] - Override the derived FK column name (default: `${reference_underscored}_id`).
78
+ * @property {string} [name] - Override the derived constraint name (default: `fk_${tableName}_${referenceName}`).
79
+ * @property {string} [referencedColumnName] - Override the referenced column name (default: `id`).
80
+ * @property {string} [referencedTableName] - Override the derived referenced table (default: pluralized `referenceName`).
81
+ */
82
+ /**
83
+ * @param {string} tableName - Table the FK lives on.
84
+ * @param {string} referenceName - Singular reference name. Defaults derive
85
+ * the FK column as `${reference}_id`, the referenced table by pluralizing
86
+ * the reference, the referenced column as `id`, and the constraint name
87
+ * as `fk_${tableName}_${referenceName}`. Override any of those via `args`
88
+ * when the schema doesn't follow the convention.
89
+ * @param {AddForeignKeyArgsType} [args] - Optional overrides.
78
90
  * @returns {Promise<void>} - Resolves when complete.
79
91
  */
80
- addForeignKey(tableName: string, referenceName: string): Promise<void>;
92
+ addForeignKey(tableName: string, referenceName: string, args?: {
93
+ /**
94
+ * - Override the derived FK column name (default: `${reference_underscored}_id`).
95
+ */
96
+ columnName?: string | undefined;
97
+ /**
98
+ * - Override the derived constraint name (default: `fk_${tableName}_${referenceName}`).
99
+ */
100
+ name?: string | undefined;
101
+ /**
102
+ * - Override the referenced column name (default: `id`).
103
+ */
104
+ referencedColumnName?: string | undefined;
105
+ /**
106
+ * - Override the derived referenced table (default: pluralized `referenceName`).
107
+ */
108
+ referencedTableName?: string | undefined;
109
+ }): Promise<void>;
81
110
  /**
82
111
  * @param {string} tableName - Table name.
83
112
  * @param {string} referenceName - Reference name.
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/database/migration/index.js"],"names":[],"mappings":"AA+BA;IACE;;;OAGG;IACH,wCAHW,MAAM,EAAE,GACN,IAAI,CAIhB;IAED,mEAAmE;IACnE,iCADc,MAAM,EAAE,GAAG,SAAS,CAGjC;IAED;;;;;OAKG;IACH,uDAJG;QAAuD,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC3B,kBAAkB,EAA/B,MAAM;QACqC,EAAE,EAA7C,OAAO,oBAAoB,EAAE,OAAO;KAC9C,EAQA;IAHC,wDAAkC;IAClC,4BAA6C;IAC7C,0CAAa;IAGf,iCAIC;IAED,qEAAqE;IACrE,aADc,OAAO,oBAAoB,EAAE,OAAO,CACnB;IAC/B,mDAAwC;IAExC,wBAEC;IAED,oBAEC;IAED,sBAEC;IAED;;;OAGG;IACH,aAHW,MAAM,GACJ,OAAO,CAAC,OAAO,oBAAoB,EAAE,eAAe,CAAC,CAIjE;IAED;;;;;;OAMG;IACH,qBANW,MAAM,cACN,MAAM,cACN,MAAM,SACN,iBAAiB,GACf,OAAO,CAAC,IAAI,CAAC,CAezB;IAED;;;;OAIG;IACH,wBAJW,MAAM,cACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;;;;OAKG;IACH;;;;;OAKG;IACH,oBALW,MAAM,WACN,MAAM,GAAG,KAAK,CAAC,MAAM,GAAG,OAAO,+BAA+B,EAAE,OAAO,CAAC;;;;;;;;;;;;;QAEtE,OAAO,CAAC,IAAI,CAAC,CAgBzB;IAED;;;;OAIG;IACH,yBAJW,MAAM,iBACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAkBzB;IAED;;;;;;;;;OASG;IACH,wBATW,MAAM,iBACN,MAAM,QAEd;QAAuB,UAAU;QACV,IAAI;QACL,IAAI;QACH,MAAM;KAC7B,GAAU,OAAO,CAAC,IAAI,CAAC,CAiBzB;IAED;;;;OAIG;IACH,2BAJW,MAAM,iBACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAMzB;IAED;;;;;OAKG;IACH,4BALW,MAAM,cACN,MAAM,YACN,OAAO,GACL,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;;;OAIG;IACH,wBAJW,MAAM,cACN,MAAM,GACJ,OAAO,CAAC,OAAO,CAAC,CAc5B;;;;;;;IAGE,uBACQ,MAAM,YACN,uBAAuB,GACrB,OAAO,CAAC,IAAI,CAAC,CACzB;;;;;;;;IAEE,uBACQ,MAAM,QACN,mBAAmB,YACnB,uBAAuB,GACrB,OAAO,CAAC,IAAI,CAAC,CACzB;IA0ED;;;OAGG;IACH,qBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;;;OAKG;IACH,wBALW,MAAM,iBACN,MAAM,iBACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;OAGG;IACH,uBAHW,MAAM,GACJ,OAAO,CAAC,OAAO,CAAC,CAM5B;CACF;;;;;cA7Wa,GAAG;;;;;;;;;gBAEgB,OAAO;;;;;;;;;;;;;;;;;;;cAO1B,GAAG;;;;;;;;;;;;sCAQJ,CAAC,KAAK,EAAE,SAAS,KAAK,IAAI;AAMvC;CAA0C;sBADpB,wBAAwB"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/database/migration/index.js"],"names":[],"mappings":"AA+BA;IACE;;;OAGG;IACH,wCAHW,MAAM,EAAE,GACN,IAAI,CAIhB;IAED,mEAAmE;IACnE,iCADc,MAAM,EAAE,GAAG,SAAS,CAGjC;IAED;;;;;OAKG;IACH,uDAJG;QAAuD,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC3B,kBAAkB,EAA/B,MAAM;QACqC,EAAE,EAA7C,OAAO,oBAAoB,EAAE,OAAO;KAC9C,EAQA;IAHC,wDAAkC;IAClC,4BAA6C;IAC7C,0CAAa;IAGf,iCAIC;IAED,qEAAqE;IACrE,aADc,OAAO,oBAAoB,EAAE,OAAO,CACnB;IAC/B,mDAAwC;IAExC,wBAEC;IAED,oBAEC;IAED,sBAEC;IAED;;;OAGG;IACH,aAHW,MAAM,GACJ,OAAO,CAAC,OAAO,oBAAoB,EAAE,eAAe,CAAC,CAIjE;IAED;;;;;;OAMG;IACH,qBANW,MAAM,cACN,MAAM,cACN,MAAM,SACN,iBAAiB,GACf,OAAO,CAAC,IAAI,CAAC,CAezB;IAED;;;;OAIG;IACH,wBAJW,MAAM,cACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;;;;OAKG;IACH;;;;;OAKG;IACH,oBALW,MAAM,WACN,MAAM,GAAG,KAAK,CAAC,MAAM,GAAG,OAAO,+BAA+B,EAAE,OAAO,CAAC;;;;;;;;;;;;;QAEtE,OAAO,CAAC,IAAI,CAAC,CAgBzB;IAED;;;;;;OAMG;IACH;;;;;;;;;OASG;IACH,yBATW,MAAM,iBACN,MAAM;;;;;;;;;;;;;;;;;QAMJ,OAAO,CAAC,IAAI,CAAC,CAuBzB;IAED;;;;;;;;;OASG;IACH,wBATW,MAAM,iBACN,MAAM,QAEd;QAAuB,UAAU;QACV,IAAI;QACL,IAAI;QACH,MAAM;KAC7B,GAAU,OAAO,CAAC,IAAI,CAAC,CAiBzB;IAED;;;;OAIG;IACH,2BAJW,MAAM,iBACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAMzB;IAED;;;;;OAKG;IACH,4BALW,MAAM,cACN,MAAM,YACN,OAAO,GACL,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;;;OAIG;IACH,wBAJW,MAAM,cACN,MAAM,GACJ,OAAO,CAAC,OAAO,CAAC,CAc5B;;;;;;;IAGE,uBACQ,MAAM,YACN,uBAAuB,GACrB,OAAO,CAAC,IAAI,CAAC,CACzB;;;;;;;;IAEE,uBACQ,MAAM,QACN,mBAAmB,YACnB,uBAAuB,GACrB,OAAO,CAAC,IAAI,CAAC,CACzB;IA0ED;;;OAGG;IACH,qBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;;;OAKG;IACH,wBALW,MAAM,iBACN,MAAM,iBACN,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;OAGG;IACH,uBAHW,MAAM,GACJ,OAAO,CAAC,OAAO,CAAC,CAM5B;CACF;;;;;cA9Xa,GAAG;;;;;;;;;gBAEgB,OAAO;;;;;;;;;;;;;;;;;;;cAO1B,GAAG;;;;;;;;;;;;sCAQJ,CAAC,KAAK,EAAE,SAAS,KAAK,IAAI;AAMvC;CAA0C;sBADpB,wBAAwB"}