@studiosonrai/nestjs-migrations 1.0.0 → 1.1.0

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.
package/README.md ADDED
@@ -0,0 +1,56 @@
1
+ # @studiosonrai/nestjs-migrations
2
+
3
+ SQL Server migrations module for NestJS. Runs TypeScript migrations on startup or via CLI.
4
+
5
+ ## Installation
6
+
7
+ ```bash
8
+ npm install @studiosonrai/nestjs-migrations
9
+ ```
10
+
11
+ ## Usage
12
+
13
+ ```typescript
14
+ import { MigrationModule } from '@studiosonrai/nestjs-migrations';
15
+
16
+ @Module({
17
+ imports: [
18
+ MigrationModule.forRootAsync({
19
+ inject: [ConfigService],
20
+ useFactory: (config: ConfigService) => ({
21
+ migrationsDir: path.join(__dirname, '../migrations'),
22
+ autoRun: config.get('AUTO_RUN_MIGRATIONS') === 'true',
23
+ }),
24
+ }),
25
+ ],
26
+ })
27
+ export class AppModule {}
28
+ ```
29
+
30
+ ## CLI
31
+
32
+ ```bash
33
+ npx nestjs-migrations --dir ./migrations --connection-string "your-connection-string"
34
+ ```
35
+
36
+ ## Migration Files
37
+
38
+ Name format: `001_description.sql`, `002_description.sql`, etc.
39
+
40
+ Place SQL files in your migrations directory. They will be executed in alphabetical order.
41
+
42
+ ## Hash Integrity Checking
43
+
44
+ Migration files are tracked with SHA256 hashes. If a migration file is modified after being applied, the migration runner will fail with a `HashMismatchError`. This prevents accidental changes to already-applied migrations.
45
+
46
+ ```typescript
47
+ import { HashMismatchError } from '@studiosonrai/nestjs-migrations';
48
+
49
+ try {
50
+ await migrationService.runMigrations();
51
+ } catch (error) {
52
+ if (error instanceof HashMismatchError) {
53
+ console.error(`Migration ${error.migrationName} was modified!`);
54
+ }
55
+ }
56
+ ```
package/dist/bin/cli.js CHANGED
@@ -1,8 +1,41 @@
1
1
  #!/usr/bin/env node
2
2
  "use strict";
3
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
+ if (k2 === undefined) k2 = k;
5
+ var desc = Object.getOwnPropertyDescriptor(m, k);
6
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
7
+ desc = { enumerable: true, get: function() { return m[k]; } };
8
+ }
9
+ Object.defineProperty(o, k2, desc);
10
+ }) : (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ o[k2] = m[k];
13
+ }));
14
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
15
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
16
+ }) : function(o, v) {
17
+ o["default"] = v;
18
+ });
19
+ var __importStar = (this && this.__importStar) || (function () {
20
+ var ownKeys = function(o) {
21
+ ownKeys = Object.getOwnPropertyNames || function (o) {
22
+ var ar = [];
23
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
24
+ return ar;
25
+ };
26
+ return ownKeys(o);
27
+ };
28
+ return function (mod) {
29
+ if (mod && mod.__esModule) return mod;
30
+ var result = {};
31
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
32
+ __setModuleDefault(result, mod);
33
+ return result;
34
+ };
35
+ })();
3
36
  Object.defineProperty(exports, "__esModule", { value: true });
4
37
  const typeorm_1 = require("typeorm");
5
- const path = require("path");
38
+ const path = __importStar(require("path"));
6
39
  const migration_core_1 = require("../src/migration-core");
7
40
  function printUsage() {
8
41
  console.log(`
@@ -108,7 +141,6 @@ function parseArgs(args) {
108
141
  return { command, options };
109
142
  }
110
143
  function resolveOptions(cliOptions) {
111
- // Load environment variables
112
144
  const dotenv = require('dotenv');
113
145
  dotenv.config();
114
146
  const options = {
@@ -123,7 +155,6 @@ function resolveOptions(cliOptions) {
123
155
  dryRun: cliOptions.dryRun || false,
124
156
  verbose: cliOptions.verbose ?? true,
125
157
  };
126
- // Resolve relative paths
127
158
  if (options.migrationsDir && !path.isAbsolute(options.migrationsDir)) {
128
159
  options.migrationsDir = path.resolve(process.cwd(), options.migrationsDir);
129
160
  }
@@ -239,4 +270,4 @@ async function main() {
239
270
  }
240
271
  }
241
272
  main();
242
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../../bin/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n\nimport { DataSource } from 'typeorm';\nimport * as path from 'path';\nimport { MigrationCore } from '../src/migration-core';\n\ninterface CliOptions {\n  migrationsDir: string;\n  host: string;\n  port: number;\n  database: string;\n  username?: string;\n  password?: string;\n  encrypt?: boolean;\n  trustServerCertificate?: boolean;\n  dryRun?: boolean;\n  verbose?: boolean;\n}\n\nfunction printUsage(): void {\n  console.log(`\n@studiosonrai/nestjs-migrations CLI\n\nUsage:\n  nestjs-migrations run [options]\n  nestjs-migrations status [options]\n\nCommands:\n  run       Run all pending migrations\n  status    Show migration status\n\nOptions:\n  --migrations-dir, -d    Path to migrations directory (required)\n  --host, -h              Database host (default: localhost)\n  --port, -p              Database port (default: 1433)\n  --database, -D          Database name (required)\n  --username, -u          Database username\n  --password, -P          Database password\n  --encrypt               Enable encryption (default: false)\n  --trust-cert            Trust server certificate (default: false)\n  --dry-run               Show pending migrations without applying\n  --verbose, -v           Verbose output (default: true)\n  --help                  Show this help message\n\nEnvironment variables:\n  DB_HOST                 Database host\n  DB_PORT                 Database port\n  DB_DATABASE             Database name\n  DB_USERNAME             Database username\n  DB_PASSWORD             Database password\n  DB_ENCRYPT              Enable encryption\n  MIGRATIONS_DIR          Path to migrations directory\n\nExamples:\n  nestjs-migrations run -d ./migrations -D mydb -u sa -P password\n  nestjs-migrations status --migrations-dir ./migrations --database mydb\n  nestjs-migrations run --dry-run -d ./migrations -D mydb\n`);\n}\n\nfunction parseArgs(args: string[]): { command: string; options: Partial<CliOptions> } {\n  const options: Partial<CliOptions> = {\n    verbose: true,\n  };\n  let command = '';\n\n  for (let i = 0; i < args.length; i++) {\n    const arg = args[i];\n    const nextArg = args[i + 1];\n\n    switch (arg) {\n      case 'run':\n      case 'status':\n        command = arg;\n        break;\n      case '--migrations-dir':\n      case '-d':\n        options.migrationsDir = nextArg;\n        i++;\n        break;\n      case '--host':\n      case '-h':\n        options.host = nextArg;\n        i++;\n        break;\n      case '--port':\n      case '-p':\n        options.port = parseInt(nextArg, 10);\n        i++;\n        break;\n      case '--database':\n      case '-D':\n        options.database = nextArg;\n        i++;\n        break;\n      case '--username':\n      case '-u':\n        options.username = nextArg;\n        i++;\n        break;\n      case '--password':\n      case '-P':\n        options.password = nextArg;\n        i++;\n        break;\n      case '--encrypt':\n        options.encrypt = true;\n        break;\n      case '--trust-cert':\n        options.trustServerCertificate = true;\n        break;\n      case '--dry-run':\n        options.dryRun = true;\n        break;\n      case '--verbose':\n      case '-v':\n        options.verbose = true;\n        break;\n      case '--help':\n        printUsage();\n        process.exit(0);\n    }\n  }\n\n  return { command, options };\n}\n\nfunction resolveOptions(cliOptions: Partial<CliOptions>): CliOptions {\n  // Load environment variables\n  const dotenv = require('dotenv');\n  dotenv.config();\n\n  const options: CliOptions = {\n    migrationsDir: cliOptions.migrationsDir || process.env.MIGRATIONS_DIR || '',\n    host: cliOptions.host || process.env.DB_HOST || 'localhost',\n    port: cliOptions.port || parseInt(process.env.DB_PORT || '1433', 10),\n    database: cliOptions.database || process.env.DB_DATABASE || '',\n    username: cliOptions.username || process.env.DB_USERNAME,\n    password: cliOptions.password || process.env.DB_PASSWORD,\n    encrypt: cliOptions.encrypt || process.env.DB_ENCRYPT === 'true',\n    trustServerCertificate: cliOptions.trustServerCertificate || false,\n    dryRun: cliOptions.dryRun || false,\n    verbose: cliOptions.verbose ?? true,\n  };\n\n  // Resolve relative paths\n  if (options.migrationsDir && !path.isAbsolute(options.migrationsDir)) {\n    options.migrationsDir = path.resolve(process.cwd(), options.migrationsDir);\n  }\n\n  return options;\n}\n\nfunction validateOptions(options: CliOptions): void {\n  if (!options.migrationsDir) {\n    console.error('Error: --migrations-dir is required');\n    process.exit(1);\n  }\n\n  if (!options.database) {\n    console.error('Error: --database is required');\n    process.exit(1);\n  }\n}\n\nasync function createDataSource(options: CliOptions): Promise<DataSource> {\n  const dataSource = new DataSource({\n    type: 'mssql',\n    host: options.host,\n    port: options.port,\n    database: options.database,\n    username: options.username,\n    password: options.password,\n    options: {\n      encrypt: options.encrypt || false,\n      trustServerCertificate: options.trustServerCertificate || false,\n    },\n  });\n\n  await dataSource.initialize();\n  return dataSource;\n}\n\nasync function runCommand(options: CliOptions): Promise<void> {\n  console.log('Starting migration runner...');\n  console.log(`Migrations directory: ${options.migrationsDir}`);\n  console.log(`Database: ${options.database}`);\n  console.log(`Host: ${options.host}:${options.port}`);\n  console.log('');\n\n  const dataSource = await createDataSource(options);\n  console.log('Database connection established');\n\n  try {\n    const migrationCore = new MigrationCore(dataSource, {\n      migrationsDir: options.migrationsDir,\n    });\n\n    const result = await migrationCore.runMigrations({\n      dryRun: options.dryRun,\n      verbose: options.verbose,\n    });\n\n    if (result.success) {\n      console.log('');\n      console.log('Migration completed successfully!');\n    }\n  } finally {\n    await dataSource.destroy();\n    console.log('Database connection closed');\n  }\n}\n\nasync function statusCommand(options: CliOptions): Promise<void> {\n  console.log('Checking migration status...');\n  console.log(`Migrations directory: ${options.migrationsDir}`);\n  console.log(`Database: ${options.database}`);\n  console.log('');\n\n  const dataSource = await createDataSource(options);\n\n  try {\n    const migrationCore = new MigrationCore(dataSource, {\n      migrationsDir: options.migrationsDir,\n    });\n\n    const status = await migrationCore.getStatus();\n\n    console.log(`Total migrations: ${status.total}`);\n    console.log(`Applied: ${status.applied}`);\n    console.log(`Pending: ${status.pending}`);\n\n    if (status.pendingFiles.length > 0) {\n      console.log('');\n      console.log('Pending migrations:');\n      status.pendingFiles.forEach((file) => console.log(`  - ${file}`));\n    }\n  } finally {\n    await dataSource.destroy();\n  }\n}\n\nasync function main(): Promise<void> {\n  const args = process.argv.slice(2);\n\n  if (args.length === 0 || args.includes('--help')) {\n    printUsage();\n    process.exit(0);\n  }\n\n  const { command, options: cliOptions } = parseArgs(args);\n  const options = resolveOptions(cliOptions);\n\n  if (!command) {\n    console.error('Error: No command specified. Use \"run\" or \"status\".');\n    printUsage();\n    process.exit(1);\n  }\n\n  validateOptions(options);\n\n  try {\n    switch (command) {\n      case 'run':\n        await runCommand(options);\n        break;\n      case 'status':\n        await statusCommand(options);\n        break;\n      default:\n        console.error(`Unknown command: ${command}`);\n        printUsage();\n        process.exit(1);\n    }\n  } catch (error) {\n    console.error('Migration failed:', error);\n    process.exit(1);\n  }\n}\n\nmain();\n"]}
273
+ //# sourceMappingURL=cli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.js","sourceRoot":"","sources":["../../bin/cli.ts"],"names":[],"mappings":"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"}
@@ -1,4 +1,4 @@
1
1
  export { MigrationModule } from './migration.module';
2
2
  export { MigrationService } from './migration.service';
3
3
  export { MigrationCore } from './migration-core';
4
- export { MigrationModuleOptions, MigrationModuleAsyncOptions, MigrationModuleOptionsFactory, MigrationResult, MigrationRunOptions, MigrationCoreOptions, MIGRATION_MODULE_OPTIONS, } from './types';
4
+ export { MigrationModuleOptions, MigrationModuleAsyncOptions, MigrationModuleOptionsFactory, MigrationResult, MigrationRunOptions, MigrationCoreOptions, AppliedMigration, HashMismatchError, MIGRATION_MODULE_OPTIONS, } from './types';
package/dist/src/index.js CHANGED
@@ -1,16 +1,13 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.MIGRATION_MODULE_OPTIONS = exports.MigrationCore = exports.MigrationService = exports.MigrationModule = void 0;
4
- // Module
3
+ exports.MIGRATION_MODULE_OPTIONS = exports.HashMismatchError = exports.MigrationCore = exports.MigrationService = exports.MigrationModule = void 0;
5
4
  var migration_module_1 = require("./migration.module");
6
5
  Object.defineProperty(exports, "MigrationModule", { enumerable: true, get: function () { return migration_module_1.MigrationModule; } });
7
- // Service
8
6
  var migration_service_1 = require("./migration.service");
9
7
  Object.defineProperty(exports, "MigrationService", { enumerable: true, get: function () { return migration_service_1.MigrationService; } });
10
- // Core (for standalone/CLI usage)
11
8
  var migration_core_1 = require("./migration-core");
12
9
  Object.defineProperty(exports, "MigrationCore", { enumerable: true, get: function () { return migration_core_1.MigrationCore; } });
13
- // Types
14
10
  var types_1 = require("./types");
11
+ Object.defineProperty(exports, "HashMismatchError", { enumerable: true, get: function () { return types_1.HashMismatchError; } });
15
12
  Object.defineProperty(exports, "MIGRATION_MODULE_OPTIONS", { enumerable: true, get: function () { return types_1.MIGRATION_MODULE_OPTIONS; } });
16
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsU0FBUztBQUNULHVEQUFxRDtBQUE1QyxtSEFBQSxlQUFlLE9BQUE7QUFFeEIsVUFBVTtBQUNWLHlEQUF1RDtBQUE5QyxxSEFBQSxnQkFBZ0IsT0FBQTtBQUV6QixrQ0FBa0M7QUFDbEMsbURBQWlEO0FBQXhDLCtHQUFBLGFBQWEsT0FBQTtBQUV0QixRQUFRO0FBQ1IsaUNBUWlCO0FBRGYsaUhBQUEsd0JBQXdCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBNb2R1bGVcbmV4cG9ydCB7IE1pZ3JhdGlvbk1vZHVsZSB9IGZyb20gJy4vbWlncmF0aW9uLm1vZHVsZSc7XG5cbi8vIFNlcnZpY2VcbmV4cG9ydCB7IE1pZ3JhdGlvblNlcnZpY2UgfSBmcm9tICcuL21pZ3JhdGlvbi5zZXJ2aWNlJztcblxuLy8gQ29yZSAoZm9yIHN0YW5kYWxvbmUvQ0xJIHVzYWdlKVxuZXhwb3J0IHsgTWlncmF0aW9uQ29yZSB9IGZyb20gJy4vbWlncmF0aW9uLWNvcmUnO1xuXG4vLyBUeXBlc1xuZXhwb3J0IHtcbiAgTWlncmF0aW9uTW9kdWxlT3B0aW9ucyxcbiAgTWlncmF0aW9uTW9kdWxlQXN5bmNPcHRpb25zLFxuICBNaWdyYXRpb25Nb2R1bGVPcHRpb25zRmFjdG9yeSxcbiAgTWlncmF0aW9uUmVzdWx0LFxuICBNaWdyYXRpb25SdW5PcHRpb25zLFxuICBNaWdyYXRpb25Db3JlT3B0aW9ucyxcbiAgTUlHUkFUSU9OX01PRFVMRV9PUFRJT05TLFxufSBmcm9tICcuL3R5cGVzJztcbiJdfQ==
13
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":";;;AACA,uDAAqD;AAA5C,mHAAA,eAAe,OAAA;AAGxB,yDAAuD;AAA9C,qHAAA,gBAAgB,OAAA;AAGzB,mDAAiD;AAAxC,+GAAA,aAAa,OAAA;AAGtB,iCAUiB;AAFf,0GAAA,iBAAiB,OAAA;AACjB,iHAAA,wBAAwB,OAAA"}
@@ -1,49 +1,28 @@
1
1
  import { DataSource, QueryRunner } from 'typeorm';
2
- import { MigrationCoreOptions, MigrationResult, MigrationRunOptions } from './types';
3
- /**
4
- * Core migration runner for SQL Server databases.
5
- * Handles reading SQL migration files, tracking applied migrations,
6
- * and executing pending migrations within transactions.
7
- */
2
+ import { MigrationCoreOptions, MigrationResult, MigrationRunOptions, AppliedMigration } from './types';
8
3
  export declare class MigrationCore {
9
4
  private readonly dataSource;
10
5
  private readonly migrationsDir;
11
6
  constructor(dataSource: DataSource, options: MigrationCoreOptions);
12
- /**
13
- * Ensures the migrations tracking table exists in the database.
14
- * Creates it if it doesn't exist.
15
- */
16
7
  ensureMigrationsTable(): Promise<void>;
17
- /**
18
- * Gets the list of migrations that have already been applied.
19
- */
20
- getAppliedMigrations(): Promise<string[]>;
21
- /**
22
- * Gets all SQL migration files from the migrations directory.
23
- * Files are sorted alphabetically (by convention, use numeric prefixes).
24
- */
8
+ calculateFileHash(filePath: string): string;
9
+ getAppliedMigrationNames(): Promise<string[]>;
10
+ getAppliedMigrations(): Promise<AppliedMigration[]>;
11
+ verifyMigrationHashes(): Promise<void>;
25
12
  getMigrationFiles(): Promise<string[]>;
26
- /**
27
- * Runs a single migration file within the provided query runner.
28
- * Handles SQL Server's GO batch separator.
29
- */
30
13
  runSingleMigration(file: string, queryRunner: QueryRunner): Promise<void>;
31
- /**
32
- * Runs all pending migrations.
33
- * Each migration runs in its own transaction for isolation.
34
- */
35
14
  runMigrations(options?: MigrationRunOptions): Promise<MigrationResult>;
36
- /**
37
- * Gets the list of pending migrations without applying them.
38
- */
39
15
  getPendingMigrations(): Promise<string[]>;
40
- /**
41
- * Gets migration status information.
42
- */
43
16
  getStatus(): Promise<{
44
17
  total: number;
45
18
  applied: number;
46
19
  pending: number;
47
20
  pendingFiles: string[];
21
+ hashMismatches: Array<{
22
+ name: string;
23
+ expectedHash: string;
24
+ actualHash: string;
25
+ }>;
26
+ missingFiles: string[];
48
27
  }>;
49
28
  }
@@ -1,14 +1,44 @@
1
1
  "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
2
35
  Object.defineProperty(exports, "__esModule", { value: true });
3
36
  exports.MigrationCore = void 0;
4
- const fs = require("fs");
5
- const path = require("path");
37
+ const fs = __importStar(require("fs"));
38
+ const path = __importStar(require("path"));
39
+ const crypto = __importStar(require("crypto"));
40
+ const types_1 = require("./types");
6
41
  const MIGRATION_TABLE = 'migrations';
7
- /**
8
- * Core migration runner for SQL Server databases.
9
- * Handles reading SQL migration files, tracking applied migrations,
10
- * and executing pending migrations within transactions.
11
- */
12
42
  class MigrationCore {
13
43
  dataSource;
14
44
  migrationsDir;
@@ -16,10 +46,6 @@ class MigrationCore {
16
46
  this.dataSource = dataSource;
17
47
  this.migrationsDir = options.migrationsDir;
18
48
  }
19
- /**
20
- * Ensures the migrations tracking table exists in the database.
21
- * Creates it if it doesn't exist.
22
- */
23
49
  async ensureMigrationsTable() {
24
50
  const tableExists = await this.dataSource.query(`
25
51
  SELECT COUNT(*) as count FROM sysobjects WHERE name='${MIGRATION_TABLE}' AND xtype='U'
@@ -29,22 +55,40 @@ class MigrationCore {
29
55
  CREATE TABLE ${MIGRATION_TABLE} (
30
56
  id INT IDENTITY(1,1) PRIMARY KEY,
31
57
  name NVARCHAR(255) UNIQUE NOT NULL,
58
+ content_hash NVARCHAR(64) NOT NULL,
32
59
  run_on DATETIME DEFAULT GETDATE() NOT NULL
33
60
  )
34
61
  `);
35
62
  }
36
63
  }
37
- /**
38
- * Gets the list of migrations that have already been applied.
39
- */
40
- async getAppliedMigrations() {
64
+ calculateFileHash(filePath) {
65
+ const content = fs.readFileSync(filePath, 'utf8');
66
+ return crypto.createHash('sha256').update(content, 'utf8').digest('hex');
67
+ }
68
+ async getAppliedMigrationNames() {
41
69
  const rows = await this.dataSource.query(`SELECT name FROM ${MIGRATION_TABLE}`);
42
70
  return rows.map((row) => row.name);
43
71
  }
44
- /**
45
- * Gets all SQL migration files from the migrations directory.
46
- * Files are sorted alphabetically (by convention, use numeric prefixes).
47
- */
72
+ async getAppliedMigrations() {
73
+ const rows = await this.dataSource.query(`SELECT name, content_hash, run_on FROM ${MIGRATION_TABLE}`);
74
+ return rows;
75
+ }
76
+ async verifyMigrationHashes() {
77
+ const appliedMigrations = await this.getAppliedMigrations();
78
+ for (const migration of appliedMigrations) {
79
+ if (!migration.content_hash) {
80
+ continue;
81
+ }
82
+ const filePath = path.join(this.migrationsDir, migration.name);
83
+ if (!fs.existsSync(filePath)) {
84
+ throw new Error(`Migration file "${migration.name}" was applied but no longer exists in the migrations directory.`);
85
+ }
86
+ const currentHash = this.calculateFileHash(filePath);
87
+ if (currentHash !== migration.content_hash) {
88
+ throw new types_1.HashMismatchError(migration.name, migration.content_hash, currentHash);
89
+ }
90
+ }
91
+ }
48
92
  async getMigrationFiles() {
49
93
  if (!fs.existsSync(this.migrationsDir)) {
50
94
  throw new Error(`Migrations directory not found: ${this.migrationsDir}`);
@@ -54,29 +98,19 @@ class MigrationCore {
54
98
  .filter((f) => f.endsWith('.sql'))
55
99
  .sort();
56
100
  }
57
- /**
58
- * Runs a single migration file within the provided query runner.
59
- * Handles SQL Server's GO batch separator.
60
- */
61
101
  async runSingleMigration(file, queryRunner) {
62
- const sql = fs.readFileSync(path.join(this.migrationsDir, file), 'utf8');
63
- // Split SQL by GO statements (batch separator for SQL Server)
64
- // GO is not part of T-SQL and must be handled by the client
102
+ const filePath = path.join(this.migrationsDir, file);
103
+ const sql = fs.readFileSync(filePath, 'utf8');
104
+ const contentHash = this.calculateFileHash(filePath);
65
105
  const batches = sql
66
106
  .split(/^\s*GO\s*$/im)
67
107
  .map((batch) => batch.trim())
68
108
  .filter((batch) => batch.length > 0);
69
- // Run each batch separately
70
109
  for (const batch of batches) {
71
110
  await queryRunner.query(batch);
72
111
  }
73
- // Track the migration
74
- await queryRunner.query(`INSERT INTO ${MIGRATION_TABLE} (name, run_on) VALUES (@0, GETDATE())`, [file]);
112
+ await queryRunner.query(`INSERT INTO ${MIGRATION_TABLE} (name, content_hash, run_on) VALUES (@0, @1, GETDATE())`, [file, contentHash]);
75
113
  }
76
- /**
77
- * Runs all pending migrations.
78
- * Each migration runs in its own transaction for isolation.
79
- */
80
114
  async runMigrations(options) {
81
115
  const result = {
82
116
  success: false,
@@ -86,15 +120,22 @@ class MigrationCore {
86
120
  };
87
121
  try {
88
122
  await this.ensureMigrationsTable();
123
+ if (options?.verbose) {
124
+ console.log('Verifying integrity of applied migrations...');
125
+ }
126
+ await this.verifyMigrationHashes();
127
+ if (options?.verbose) {
128
+ console.log('✓ All applied migrations have valid hashes');
129
+ }
89
130
  const files = await this.getMigrationFiles();
90
- const applied = await this.getAppliedMigrations();
131
+ const appliedNames = await this.getAppliedMigrationNames();
91
132
  result.totalMigrations = files.length;
92
- result.appliedMigrations = applied.length;
133
+ result.appliedMigrations = appliedNames.length;
93
134
  if (options?.verbose) {
94
135
  console.log(`Found ${files.length} migration files`);
95
- console.log(`${applied.length} migrations already applied`);
136
+ console.log(`${appliedNames.length} migrations already applied`);
96
137
  }
97
- const pendingMigrations = files.filter((file) => !applied.includes(file));
138
+ const pendingMigrations = files.filter((file) => !appliedNames.includes(file));
98
139
  result.pendingMigrations = pendingMigrations.length;
99
140
  if (pendingMigrations.length === 0) {
100
141
  if (options?.verbose) {
@@ -116,7 +157,6 @@ class MigrationCore {
116
157
  if (options?.verbose) {
117
158
  console.log(`Applying migration: ${file}`);
118
159
  }
119
- // Use a transaction for each migration
120
160
  const queryRunner = this.dataSource.createQueryRunner();
121
161
  await queryRunner.connect();
122
162
  await queryRunner.startTransaction();
@@ -155,30 +195,46 @@ class MigrationCore {
155
195
  throw error;
156
196
  }
157
197
  }
158
- /**
159
- * Gets the list of pending migrations without applying them.
160
- */
161
198
  async getPendingMigrations() {
162
199
  await this.ensureMigrationsTable();
163
200
  const files = await this.getMigrationFiles();
164
- const applied = await this.getAppliedMigrations();
165
- return files.filter((file) => !applied.includes(file));
201
+ const appliedNames = await this.getAppliedMigrationNames();
202
+ return files.filter((file) => !appliedNames.includes(file));
166
203
  }
167
- /**
168
- * Gets migration status information.
169
- */
170
204
  async getStatus() {
171
205
  await this.ensureMigrationsTable();
172
206
  const files = await this.getMigrationFiles();
173
- const applied = await this.getAppliedMigrations();
174
- const pendingFiles = files.filter((file) => !applied.includes(file));
207
+ const appliedMigrations = await this.getAppliedMigrations();
208
+ const appliedNames = appliedMigrations.map((m) => m.name);
209
+ const pendingFiles = files.filter((file) => !appliedNames.includes(file));
210
+ const hashMismatches = [];
211
+ const missingFiles = [];
212
+ for (const migration of appliedMigrations) {
213
+ const filePath = path.join(this.migrationsDir, migration.name);
214
+ if (!fs.existsSync(filePath)) {
215
+ missingFiles.push(migration.name);
216
+ continue;
217
+ }
218
+ if (migration.content_hash) {
219
+ const currentHash = this.calculateFileHash(filePath);
220
+ if (currentHash !== migration.content_hash) {
221
+ hashMismatches.push({
222
+ name: migration.name,
223
+ expectedHash: migration.content_hash,
224
+ actualHash: currentHash,
225
+ });
226
+ }
227
+ }
228
+ }
175
229
  return {
176
230
  total: files.length,
177
- applied: applied.length,
231
+ applied: appliedNames.length,
178
232
  pending: pendingFiles.length,
179
233
  pendingFiles,
234
+ hashMismatches,
235
+ missingFiles,
180
236
  };
181
237
  }
182
238
  }
183
239
  exports.MigrationCore = MigrationCore;
184
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migration-core.js","sourceRoot":"","sources":["../../src/migration-core.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, QueryRunner } from 'typeorm';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { MigrationCoreOptions, MigrationResult, MigrationRunOptions } from './types';\n\nconst MIGRATION_TABLE = 'migrations';\n\n/**\n * Core migration runner for SQL Server databases.\n * Handles reading SQL migration files, tracking applied migrations,\n * and executing pending migrations within transactions.\n */\nexport class MigrationCore {\n  private readonly migrationsDir: string;\n\n  constructor(\n    private readonly dataSource: DataSource,\n    options: MigrationCoreOptions,\n  ) {\n    this.migrationsDir = options.migrationsDir;\n  }\n\n  /**\n   * Ensures the migrations tracking table exists in the database.\n   * Creates it if it doesn't exist.\n   */\n  async ensureMigrationsTable(): Promise<void> {\n    const tableExists = await this.dataSource.query(`\n      SELECT COUNT(*) as count FROM sysobjects WHERE name='${MIGRATION_TABLE}' AND xtype='U'\n    `);\n\n    if (tableExists[0].count === 0) {\n      await this.dataSource.query(`\n        CREATE TABLE ${MIGRATION_TABLE} (\n          id INT IDENTITY(1,1) PRIMARY KEY,\n          name NVARCHAR(255) UNIQUE NOT NULL,\n          run_on DATETIME DEFAULT GETDATE() NOT NULL\n        )\n      `);\n    }\n  }\n\n  /**\n   * Gets the list of migrations that have already been applied.\n   */\n  async getAppliedMigrations(): Promise<string[]> {\n    const rows = await this.dataSource.query<{ name: string }[]>(\n      `SELECT name FROM ${MIGRATION_TABLE}`,\n    );\n    return rows.map((row) => row.name);\n  }\n\n  /**\n   * Gets all SQL migration files from the migrations directory.\n   * Files are sorted alphabetically (by convention, use numeric prefixes).\n   */\n  async getMigrationFiles(): Promise<string[]> {\n    if (!fs.existsSync(this.migrationsDir)) {\n      throw new Error(`Migrations directory not found: ${this.migrationsDir}`);\n    }\n\n    return fs\n      .readdirSync(this.migrationsDir)\n      .filter((f) => f.endsWith('.sql'))\n      .sort();\n  }\n\n  /**\n   * Runs a single migration file within the provided query runner.\n   * Handles SQL Server's GO batch separator.\n   */\n  async runSingleMigration(\n    file: string,\n    queryRunner: QueryRunner,\n  ): Promise<void> {\n    const sql = fs.readFileSync(path.join(this.migrationsDir, file), 'utf8');\n\n    // Split SQL by GO statements (batch separator for SQL Server)\n    // GO is not part of T-SQL and must be handled by the client\n    const batches = sql\n      .split(/^\\s*GO\\s*$/im)\n      .map((batch) => batch.trim())\n      .filter((batch) => batch.length > 0);\n\n    // Run each batch separately\n    for (const batch of batches) {\n      await queryRunner.query(batch);\n    }\n\n    // Track the migration\n    await queryRunner.query(\n      `INSERT INTO ${MIGRATION_TABLE} (name, run_on) VALUES (@0, GETDATE())`,\n      [file],\n    );\n  }\n\n  /**\n   * Runs all pending migrations.\n   * Each migration runs in its own transaction for isolation.\n   */\n  async runMigrations(options?: MigrationRunOptions): Promise<MigrationResult> {\n    const result: MigrationResult = {\n      success: false,\n      totalMigrations: 0,\n      appliedMigrations: 0,\n      pendingMigrations: 0,\n    };\n\n    try {\n      await this.ensureMigrationsTable();\n\n      const files = await this.getMigrationFiles();\n      const applied = await this.getAppliedMigrations();\n\n      result.totalMigrations = files.length;\n      result.appliedMigrations = applied.length;\n\n      if (options?.verbose) {\n        console.log(`Found ${files.length} migration files`);\n        console.log(`${applied.length} migrations already applied`);\n      }\n\n      const pendingMigrations = files.filter((file) => !applied.includes(file));\n      result.pendingMigrations = pendingMigrations.length;\n\n      if (pendingMigrations.length === 0) {\n        if (options?.verbose) {\n          console.log('No pending migrations to apply');\n        }\n        result.success = true;\n        return result;\n      }\n\n      if (options?.dryRun) {\n        console.log('Dry run mode - would apply:');\n        pendingMigrations.forEach((file) => console.log(`  - ${file}`));\n        result.success = true;\n        return result;\n      }\n\n      if (options?.verbose) {\n        console.log(\n          `Running ${pendingMigrations.length} pending migrations...`,\n        );\n      }\n\n      for (const file of pendingMigrations) {\n        if (options?.verbose) {\n          console.log(`Applying migration: ${file}`);\n        }\n\n        // Use a transaction for each migration\n        const queryRunner = this.dataSource.createQueryRunner();\n        await queryRunner.connect();\n        await queryRunner.startTransaction();\n\n        try {\n          await this.runSingleMigration(file, queryRunner);\n          await queryRunner.commitTransaction();\n\n          if (options?.verbose) {\n            console.log(`✓ Applied migration: ${file}`);\n          }\n        } catch (error) {\n          await queryRunner.rollbackTransaction();\n\n          if (options?.verbose) {\n            console.error(`Failed to apply migration ${file}:`, error);\n          }\n\n          result.failedMigration = file;\n          result.error = error;\n          throw error;\n        } finally {\n          await queryRunner.release();\n        }\n      }\n\n      result.success = true;\n      result.appliedMigrations += pendingMigrations.length;\n\n      if (options?.verbose) {\n        console.log('All migrations completed successfully!');\n      }\n\n      return result;\n    } catch (error) {\n      result.error = error;\n\n      if (options?.verbose) {\n        console.error('Migration failed:', error);\n      }\n\n      throw error;\n    }\n  }\n\n  /**\n   * Gets the list of pending migrations without applying them.\n   */\n  async getPendingMigrations(): Promise<string[]> {\n    await this.ensureMigrationsTable();\n    const files = await this.getMigrationFiles();\n    const applied = await this.getAppliedMigrations();\n    return files.filter((file) => !applied.includes(file));\n  }\n\n  /**\n   * Gets migration status information.\n   */\n  async getStatus(): Promise<{\n    total: number;\n    applied: number;\n    pending: number;\n    pendingFiles: string[];\n  }> {\n    await this.ensureMigrationsTable();\n    const files = await this.getMigrationFiles();\n    const applied = await this.getAppliedMigrations();\n    const pendingFiles = files.filter((file) => !applied.includes(file));\n\n    return {\n      total: files.length,\n      applied: applied.length,\n      pending: pendingFiles.length,\n      pendingFiles,\n    };\n  }\n}\n"]}
240
+ //# sourceMappingURL=migration-core.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migration-core.js","sourceRoot":"","sources":["../../src/migration-core.ts"],"names":[],"mappings":"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"}
@@ -1,37 +1,7 @@
1
1
  import { DynamicModule } from '@nestjs/common';
2
2
  import { MigrationModuleAsyncOptions, MigrationModuleOptions } from './types';
3
- /**
4
- * NestJS module for SQL Server migrations.
5
- *
6
- * @example
7
- * // Synchronous configuration
8
- * MigrationModule.forRoot({
9
- * migrationsDir: path.join(__dirname, '../migrations'),
10
- * autoRun: true,
11
- * verbose: true,
12
- * })
13
- *
14
- * @example
15
- * // Asynchronous configuration with ConfigService
16
- * MigrationModule.forRootAsync({
17
- * imports: [ConfigModule],
18
- * inject: [ConfigService],
19
- * useFactory: (configService: ConfigService) => ({
20
- * migrationsDir: path.join(__dirname, '../migrations'),
21
- * autoRun: configService.get<boolean>('AUTO_RUN_MIGRATIONS', false),
22
- * verbose: true,
23
- * }),
24
- * })
25
- */
26
3
  export declare class MigrationModule {
27
- /**
28
- * Configure the migration module with static options.
29
- */
30
4
  static forRoot(options: MigrationModuleOptions): DynamicModule;
31
- /**
32
- * Configure the migration module with async options.
33
- * Useful when options depend on ConfigService or other async providers.
34
- */
35
5
  static forRootAsync(options: MigrationModuleAsyncOptions): DynamicModule;
36
6
  private static createAsyncProviders;
37
7
  }