@studiosonrai/nestjs-migrations 1.0.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.
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
@@ -0,0 +1,242 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ const typeorm_1 = require("typeorm");
5
+ const path = require("path");
6
+ const migration_core_1 = require("../src/migration-core");
7
+ function printUsage() {
8
+ console.log(`
9
+ @studiosonrai/nestjs-migrations CLI
10
+
11
+ Usage:
12
+ nestjs-migrations run [options]
13
+ nestjs-migrations status [options]
14
+
15
+ Commands:
16
+ run Run all pending migrations
17
+ status Show migration status
18
+
19
+ Options:
20
+ --migrations-dir, -d Path to migrations directory (required)
21
+ --host, -h Database host (default: localhost)
22
+ --port, -p Database port (default: 1433)
23
+ --database, -D Database name (required)
24
+ --username, -u Database username
25
+ --password, -P Database password
26
+ --encrypt Enable encryption (default: false)
27
+ --trust-cert Trust server certificate (default: false)
28
+ --dry-run Show pending migrations without applying
29
+ --verbose, -v Verbose output (default: true)
30
+ --help Show this help message
31
+
32
+ Environment variables:
33
+ DB_HOST Database host
34
+ DB_PORT Database port
35
+ DB_DATABASE Database name
36
+ DB_USERNAME Database username
37
+ DB_PASSWORD Database password
38
+ DB_ENCRYPT Enable encryption
39
+ MIGRATIONS_DIR Path to migrations directory
40
+
41
+ Examples:
42
+ nestjs-migrations run -d ./migrations -D mydb -u sa -P password
43
+ nestjs-migrations status --migrations-dir ./migrations --database mydb
44
+ nestjs-migrations run --dry-run -d ./migrations -D mydb
45
+ `);
46
+ }
47
+ function parseArgs(args) {
48
+ const options = {
49
+ verbose: true,
50
+ };
51
+ let command = '';
52
+ for (let i = 0; i < args.length; i++) {
53
+ const arg = args[i];
54
+ const nextArg = args[i + 1];
55
+ switch (arg) {
56
+ case 'run':
57
+ case 'status':
58
+ command = arg;
59
+ break;
60
+ case '--migrations-dir':
61
+ case '-d':
62
+ options.migrationsDir = nextArg;
63
+ i++;
64
+ break;
65
+ case '--host':
66
+ case '-h':
67
+ options.host = nextArg;
68
+ i++;
69
+ break;
70
+ case '--port':
71
+ case '-p':
72
+ options.port = parseInt(nextArg, 10);
73
+ i++;
74
+ break;
75
+ case '--database':
76
+ case '-D':
77
+ options.database = nextArg;
78
+ i++;
79
+ break;
80
+ case '--username':
81
+ case '-u':
82
+ options.username = nextArg;
83
+ i++;
84
+ break;
85
+ case '--password':
86
+ case '-P':
87
+ options.password = nextArg;
88
+ i++;
89
+ break;
90
+ case '--encrypt':
91
+ options.encrypt = true;
92
+ break;
93
+ case '--trust-cert':
94
+ options.trustServerCertificate = true;
95
+ break;
96
+ case '--dry-run':
97
+ options.dryRun = true;
98
+ break;
99
+ case '--verbose':
100
+ case '-v':
101
+ options.verbose = true;
102
+ break;
103
+ case '--help':
104
+ printUsage();
105
+ process.exit(0);
106
+ }
107
+ }
108
+ return { command, options };
109
+ }
110
+ function resolveOptions(cliOptions) {
111
+ // Load environment variables
112
+ const dotenv = require('dotenv');
113
+ dotenv.config();
114
+ const options = {
115
+ migrationsDir: cliOptions.migrationsDir || process.env.MIGRATIONS_DIR || '',
116
+ host: cliOptions.host || process.env.DB_HOST || 'localhost',
117
+ port: cliOptions.port || parseInt(process.env.DB_PORT || '1433', 10),
118
+ database: cliOptions.database || process.env.DB_DATABASE || '',
119
+ username: cliOptions.username || process.env.DB_USERNAME,
120
+ password: cliOptions.password || process.env.DB_PASSWORD,
121
+ encrypt: cliOptions.encrypt || process.env.DB_ENCRYPT === 'true',
122
+ trustServerCertificate: cliOptions.trustServerCertificate || false,
123
+ dryRun: cliOptions.dryRun || false,
124
+ verbose: cliOptions.verbose ?? true,
125
+ };
126
+ // Resolve relative paths
127
+ if (options.migrationsDir && !path.isAbsolute(options.migrationsDir)) {
128
+ options.migrationsDir = path.resolve(process.cwd(), options.migrationsDir);
129
+ }
130
+ return options;
131
+ }
132
+ function validateOptions(options) {
133
+ if (!options.migrationsDir) {
134
+ console.error('Error: --migrations-dir is required');
135
+ process.exit(1);
136
+ }
137
+ if (!options.database) {
138
+ console.error('Error: --database is required');
139
+ process.exit(1);
140
+ }
141
+ }
142
+ async function createDataSource(options) {
143
+ const dataSource = new typeorm_1.DataSource({
144
+ type: 'mssql',
145
+ host: options.host,
146
+ port: options.port,
147
+ database: options.database,
148
+ username: options.username,
149
+ password: options.password,
150
+ options: {
151
+ encrypt: options.encrypt || false,
152
+ trustServerCertificate: options.trustServerCertificate || false,
153
+ },
154
+ });
155
+ await dataSource.initialize();
156
+ return dataSource;
157
+ }
158
+ async function runCommand(options) {
159
+ console.log('Starting migration runner...');
160
+ console.log(`Migrations directory: ${options.migrationsDir}`);
161
+ console.log(`Database: ${options.database}`);
162
+ console.log(`Host: ${options.host}:${options.port}`);
163
+ console.log('');
164
+ const dataSource = await createDataSource(options);
165
+ console.log('Database connection established');
166
+ try {
167
+ const migrationCore = new migration_core_1.MigrationCore(dataSource, {
168
+ migrationsDir: options.migrationsDir,
169
+ });
170
+ const result = await migrationCore.runMigrations({
171
+ dryRun: options.dryRun,
172
+ verbose: options.verbose,
173
+ });
174
+ if (result.success) {
175
+ console.log('');
176
+ console.log('Migration completed successfully!');
177
+ }
178
+ }
179
+ finally {
180
+ await dataSource.destroy();
181
+ console.log('Database connection closed');
182
+ }
183
+ }
184
+ async function statusCommand(options) {
185
+ console.log('Checking migration status...');
186
+ console.log(`Migrations directory: ${options.migrationsDir}`);
187
+ console.log(`Database: ${options.database}`);
188
+ console.log('');
189
+ const dataSource = await createDataSource(options);
190
+ try {
191
+ const migrationCore = new migration_core_1.MigrationCore(dataSource, {
192
+ migrationsDir: options.migrationsDir,
193
+ });
194
+ const status = await migrationCore.getStatus();
195
+ console.log(`Total migrations: ${status.total}`);
196
+ console.log(`Applied: ${status.applied}`);
197
+ console.log(`Pending: ${status.pending}`);
198
+ if (status.pendingFiles.length > 0) {
199
+ console.log('');
200
+ console.log('Pending migrations:');
201
+ status.pendingFiles.forEach((file) => console.log(` - ${file}`));
202
+ }
203
+ }
204
+ finally {
205
+ await dataSource.destroy();
206
+ }
207
+ }
208
+ async function main() {
209
+ const args = process.argv.slice(2);
210
+ if (args.length === 0 || args.includes('--help')) {
211
+ printUsage();
212
+ process.exit(0);
213
+ }
214
+ const { command, options: cliOptions } = parseArgs(args);
215
+ const options = resolveOptions(cliOptions);
216
+ if (!command) {
217
+ console.error('Error: No command specified. Use "run" or "status".');
218
+ printUsage();
219
+ process.exit(1);
220
+ }
221
+ validateOptions(options);
222
+ try {
223
+ switch (command) {
224
+ case 'run':
225
+ await runCommand(options);
226
+ break;
227
+ case 'status':
228
+ await statusCommand(options);
229
+ break;
230
+ default:
231
+ console.error(`Unknown command: ${command}`);
232
+ printUsage();
233
+ process.exit(1);
234
+ }
235
+ }
236
+ catch (error) {
237
+ console.error('Migration failed:', error);
238
+ process.exit(1);
239
+ }
240
+ }
241
+ 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"]}
@@ -0,0 +1,4 @@
1
+ export { MigrationModule } from './migration.module';
2
+ export { MigrationService } from './migration.service';
3
+ export { MigrationCore } from './migration-core';
4
+ export { MigrationModuleOptions, MigrationModuleAsyncOptions, MigrationModuleOptionsFactory, MigrationResult, MigrationRunOptions, MigrationCoreOptions, MIGRATION_MODULE_OPTIONS, } from './types';
@@ -0,0 +1,16 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MIGRATION_MODULE_OPTIONS = exports.MigrationCore = exports.MigrationService = exports.MigrationModule = void 0;
4
+ // Module
5
+ var migration_module_1 = require("./migration.module");
6
+ Object.defineProperty(exports, "MigrationModule", { enumerable: true, get: function () { return migration_module_1.MigrationModule; } });
7
+ // Service
8
+ var migration_service_1 = require("./migration.service");
9
+ Object.defineProperty(exports, "MigrationService", { enumerable: true, get: function () { return migration_service_1.MigrationService; } });
10
+ // Core (for standalone/CLI usage)
11
+ var migration_core_1 = require("./migration-core");
12
+ Object.defineProperty(exports, "MigrationCore", { enumerable: true, get: function () { return migration_core_1.MigrationCore; } });
13
+ // Types
14
+ var types_1 = require("./types");
15
+ Object.defineProperty(exports, "MIGRATION_MODULE_OPTIONS", { enumerable: true, get: function () { return types_1.MIGRATION_MODULE_OPTIONS; } });
16
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsU0FBUztBQUNULHVEQUFxRDtBQUE1QyxtSEFBQSxlQUFlLE9BQUE7QUFFeEIsVUFBVTtBQUNWLHlEQUF1RDtBQUE5QyxxSEFBQSxnQkFBZ0IsT0FBQTtBQUV6QixrQ0FBa0M7QUFDbEMsbURBQWlEO0FBQXhDLCtHQUFBLGFBQWEsT0FBQTtBQUV0QixRQUFRO0FBQ1IsaUNBUWlCO0FBRGYsaUhBQUEsd0JBQXdCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBNb2R1bGVcbmV4cG9ydCB7IE1pZ3JhdGlvbk1vZHVsZSB9IGZyb20gJy4vbWlncmF0aW9uLm1vZHVsZSc7XG5cbi8vIFNlcnZpY2VcbmV4cG9ydCB7IE1pZ3JhdGlvblNlcnZpY2UgfSBmcm9tICcuL21pZ3JhdGlvbi5zZXJ2aWNlJztcblxuLy8gQ29yZSAoZm9yIHN0YW5kYWxvbmUvQ0xJIHVzYWdlKVxuZXhwb3J0IHsgTWlncmF0aW9uQ29yZSB9IGZyb20gJy4vbWlncmF0aW9uLWNvcmUnO1xuXG4vLyBUeXBlc1xuZXhwb3J0IHtcbiAgTWlncmF0aW9uTW9kdWxlT3B0aW9ucyxcbiAgTWlncmF0aW9uTW9kdWxlQXN5bmNPcHRpb25zLFxuICBNaWdyYXRpb25Nb2R1bGVPcHRpb25zRmFjdG9yeSxcbiAgTWlncmF0aW9uUmVzdWx0LFxuICBNaWdyYXRpb25SdW5PcHRpb25zLFxuICBNaWdyYXRpb25Db3JlT3B0aW9ucyxcbiAgTUlHUkFUSU9OX01PRFVMRV9PUFRJT05TLFxufSBmcm9tICcuL3R5cGVzJztcbiJdfQ==
@@ -0,0 +1,49 @@
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
+ */
8
+ export declare class MigrationCore {
9
+ private readonly dataSource;
10
+ private readonly migrationsDir;
11
+ 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
+ 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
+ */
25
+ 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
+ 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
+ runMigrations(options?: MigrationRunOptions): Promise<MigrationResult>;
36
+ /**
37
+ * Gets the list of pending migrations without applying them.
38
+ */
39
+ getPendingMigrations(): Promise<string[]>;
40
+ /**
41
+ * Gets migration status information.
42
+ */
43
+ getStatus(): Promise<{
44
+ total: number;
45
+ applied: number;
46
+ pending: number;
47
+ pendingFiles: string[];
48
+ }>;
49
+ }
@@ -0,0 +1,184 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MigrationCore = void 0;
4
+ const fs = require("fs");
5
+ const path = require("path");
6
+ 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
+ class MigrationCore {
13
+ dataSource;
14
+ migrationsDir;
15
+ constructor(dataSource, options) {
16
+ this.dataSource = dataSource;
17
+ this.migrationsDir = options.migrationsDir;
18
+ }
19
+ /**
20
+ * Ensures the migrations tracking table exists in the database.
21
+ * Creates it if it doesn't exist.
22
+ */
23
+ async ensureMigrationsTable() {
24
+ const tableExists = await this.dataSource.query(`
25
+ SELECT COUNT(*) as count FROM sysobjects WHERE name='${MIGRATION_TABLE}' AND xtype='U'
26
+ `);
27
+ if (tableExists[0].count === 0) {
28
+ await this.dataSource.query(`
29
+ CREATE TABLE ${MIGRATION_TABLE} (
30
+ id INT IDENTITY(1,1) PRIMARY KEY,
31
+ name NVARCHAR(255) UNIQUE NOT NULL,
32
+ run_on DATETIME DEFAULT GETDATE() NOT NULL
33
+ )
34
+ `);
35
+ }
36
+ }
37
+ /**
38
+ * Gets the list of migrations that have already been applied.
39
+ */
40
+ async getAppliedMigrations() {
41
+ const rows = await this.dataSource.query(`SELECT name FROM ${MIGRATION_TABLE}`);
42
+ return rows.map((row) => row.name);
43
+ }
44
+ /**
45
+ * Gets all SQL migration files from the migrations directory.
46
+ * Files are sorted alphabetically (by convention, use numeric prefixes).
47
+ */
48
+ async getMigrationFiles() {
49
+ if (!fs.existsSync(this.migrationsDir)) {
50
+ throw new Error(`Migrations directory not found: ${this.migrationsDir}`);
51
+ }
52
+ return fs
53
+ .readdirSync(this.migrationsDir)
54
+ .filter((f) => f.endsWith('.sql'))
55
+ .sort();
56
+ }
57
+ /**
58
+ * Runs a single migration file within the provided query runner.
59
+ * Handles SQL Server's GO batch separator.
60
+ */
61
+ 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
65
+ const batches = sql
66
+ .split(/^\s*GO\s*$/im)
67
+ .map((batch) => batch.trim())
68
+ .filter((batch) => batch.length > 0);
69
+ // Run each batch separately
70
+ for (const batch of batches) {
71
+ await queryRunner.query(batch);
72
+ }
73
+ // Track the migration
74
+ await queryRunner.query(`INSERT INTO ${MIGRATION_TABLE} (name, run_on) VALUES (@0, GETDATE())`, [file]);
75
+ }
76
+ /**
77
+ * Runs all pending migrations.
78
+ * Each migration runs in its own transaction for isolation.
79
+ */
80
+ async runMigrations(options) {
81
+ const result = {
82
+ success: false,
83
+ totalMigrations: 0,
84
+ appliedMigrations: 0,
85
+ pendingMigrations: 0,
86
+ };
87
+ try {
88
+ await this.ensureMigrationsTable();
89
+ const files = await this.getMigrationFiles();
90
+ const applied = await this.getAppliedMigrations();
91
+ result.totalMigrations = files.length;
92
+ result.appliedMigrations = applied.length;
93
+ if (options?.verbose) {
94
+ console.log(`Found ${files.length} migration files`);
95
+ console.log(`${applied.length} migrations already applied`);
96
+ }
97
+ const pendingMigrations = files.filter((file) => !applied.includes(file));
98
+ result.pendingMigrations = pendingMigrations.length;
99
+ if (pendingMigrations.length === 0) {
100
+ if (options?.verbose) {
101
+ console.log('No pending migrations to apply');
102
+ }
103
+ result.success = true;
104
+ return result;
105
+ }
106
+ if (options?.dryRun) {
107
+ console.log('Dry run mode - would apply:');
108
+ pendingMigrations.forEach((file) => console.log(` - ${file}`));
109
+ result.success = true;
110
+ return result;
111
+ }
112
+ if (options?.verbose) {
113
+ console.log(`Running ${pendingMigrations.length} pending migrations...`);
114
+ }
115
+ for (const file of pendingMigrations) {
116
+ if (options?.verbose) {
117
+ console.log(`Applying migration: ${file}`);
118
+ }
119
+ // Use a transaction for each migration
120
+ const queryRunner = this.dataSource.createQueryRunner();
121
+ await queryRunner.connect();
122
+ await queryRunner.startTransaction();
123
+ try {
124
+ await this.runSingleMigration(file, queryRunner);
125
+ await queryRunner.commitTransaction();
126
+ if (options?.verbose) {
127
+ console.log(`✓ Applied migration: ${file}`);
128
+ }
129
+ }
130
+ catch (error) {
131
+ await queryRunner.rollbackTransaction();
132
+ if (options?.verbose) {
133
+ console.error(`Failed to apply migration ${file}:`, error);
134
+ }
135
+ result.failedMigration = file;
136
+ result.error = error;
137
+ throw error;
138
+ }
139
+ finally {
140
+ await queryRunner.release();
141
+ }
142
+ }
143
+ result.success = true;
144
+ result.appliedMigrations += pendingMigrations.length;
145
+ if (options?.verbose) {
146
+ console.log('All migrations completed successfully!');
147
+ }
148
+ return result;
149
+ }
150
+ catch (error) {
151
+ result.error = error;
152
+ if (options?.verbose) {
153
+ console.error('Migration failed:', error);
154
+ }
155
+ throw error;
156
+ }
157
+ }
158
+ /**
159
+ * Gets the list of pending migrations without applying them.
160
+ */
161
+ async getPendingMigrations() {
162
+ await this.ensureMigrationsTable();
163
+ const files = await this.getMigrationFiles();
164
+ const applied = await this.getAppliedMigrations();
165
+ return files.filter((file) => !applied.includes(file));
166
+ }
167
+ /**
168
+ * Gets migration status information.
169
+ */
170
+ async getStatus() {
171
+ await this.ensureMigrationsTable();
172
+ const files = await this.getMigrationFiles();
173
+ const applied = await this.getAppliedMigrations();
174
+ const pendingFiles = files.filter((file) => !applied.includes(file));
175
+ return {
176
+ total: files.length,
177
+ applied: applied.length,
178
+ pending: pendingFiles.length,
179
+ pendingFiles,
180
+ };
181
+ }
182
+ }
183
+ 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"]}
@@ -0,0 +1,37 @@
1
+ import { DynamicModule } from '@nestjs/common';
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
+ export declare class MigrationModule {
27
+ /**
28
+ * Configure the migration module with static options.
29
+ */
30
+ 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
+ static forRootAsync(options: MigrationModuleAsyncOptions): DynamicModule;
36
+ private static createAsyncProviders;
37
+ }
@@ -0,0 +1,108 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var MigrationModule_1;
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.MigrationModule = void 0;
11
+ const common_1 = require("@nestjs/common");
12
+ const migration_service_1 = require("./migration.service");
13
+ const types_1 = require("./types");
14
+ /**
15
+ * NestJS module for SQL Server migrations.
16
+ *
17
+ * @example
18
+ * // Synchronous configuration
19
+ * MigrationModule.forRoot({
20
+ * migrationsDir: path.join(__dirname, '../migrations'),
21
+ * autoRun: true,
22
+ * verbose: true,
23
+ * })
24
+ *
25
+ * @example
26
+ * // Asynchronous configuration with ConfigService
27
+ * MigrationModule.forRootAsync({
28
+ * imports: [ConfigModule],
29
+ * inject: [ConfigService],
30
+ * useFactory: (configService: ConfigService) => ({
31
+ * migrationsDir: path.join(__dirname, '../migrations'),
32
+ * autoRun: configService.get<boolean>('AUTO_RUN_MIGRATIONS', false),
33
+ * verbose: true,
34
+ * }),
35
+ * })
36
+ */
37
+ let MigrationModule = MigrationModule_1 = class MigrationModule {
38
+ /**
39
+ * Configure the migration module with static options.
40
+ */
41
+ static forRoot(options) {
42
+ return {
43
+ module: MigrationModule_1,
44
+ providers: [
45
+ {
46
+ provide: types_1.MIGRATION_MODULE_OPTIONS,
47
+ useValue: options,
48
+ },
49
+ migration_service_1.MigrationService,
50
+ ],
51
+ exports: [migration_service_1.MigrationService],
52
+ };
53
+ }
54
+ /**
55
+ * Configure the migration module with async options.
56
+ * Useful when options depend on ConfigService or other async providers.
57
+ */
58
+ static forRootAsync(options) {
59
+ return {
60
+ module: MigrationModule_1,
61
+ imports: options.imports || [],
62
+ providers: [
63
+ ...this.createAsyncProviders(options),
64
+ migration_service_1.MigrationService,
65
+ ],
66
+ exports: [migration_service_1.MigrationService],
67
+ };
68
+ }
69
+ static createAsyncProviders(options) {
70
+ if (options.useFactory) {
71
+ return [
72
+ {
73
+ provide: types_1.MIGRATION_MODULE_OPTIONS,
74
+ useFactory: options.useFactory,
75
+ inject: options.inject || [],
76
+ },
77
+ ];
78
+ }
79
+ if (options.useClass) {
80
+ return [
81
+ {
82
+ provide: options.useClass,
83
+ useClass: options.useClass,
84
+ },
85
+ {
86
+ provide: types_1.MIGRATION_MODULE_OPTIONS,
87
+ useFactory: async (optionsFactory) => optionsFactory.createMigrationOptions(),
88
+ inject: [options.useClass],
89
+ },
90
+ ];
91
+ }
92
+ if (options.useExisting) {
93
+ return [
94
+ {
95
+ provide: types_1.MIGRATION_MODULE_OPTIONS,
96
+ useFactory: async (optionsFactory) => optionsFactory.createMigrationOptions(),
97
+ inject: [options.useExisting],
98
+ },
99
+ ];
100
+ }
101
+ return [];
102
+ }
103
+ };
104
+ exports.MigrationModule = MigrationModule;
105
+ exports.MigrationModule = MigrationModule = MigrationModule_1 = __decorate([
106
+ (0, common_1.Module)({})
107
+ ], MigrationModule);
108
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlncmF0aW9uLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9taWdyYXRpb24ubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7QUFBQSwyQ0FBaUU7QUFDakUsMkRBQXVEO0FBQ3ZELG1DQUtpQjtBQUVqQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXNCRztBQUVJLElBQU0sZUFBZSx1QkFBckIsTUFBTSxlQUFlO0lBQzFCOztPQUVHO0lBQ0gsTUFBTSxDQUFDLE9BQU8sQ0FBQyxPQUErQjtRQUM1QyxPQUFPO1lBQ0wsTUFBTSxFQUFFLGlCQUFlO1lBQ3ZCLFNBQVMsRUFBRTtnQkFDVDtvQkFDRSxPQUFPLEVBQUUsZ0NBQXdCO29CQUNqQyxRQUFRLEVBQUUsT0FBTztpQkFDbEI7Z0JBQ0Qsb0NBQWdCO2FBQ2pCO1lBQ0QsT0FBTyxFQUFFLENBQUMsb0NBQWdCLENBQUM7U0FDNUIsQ0FBQztJQUNKLENBQUM7SUFFRDs7O09BR0c7SUFDSCxNQUFNLENBQUMsWUFBWSxDQUFDLE9BQW9DO1FBQ3RELE9BQU87WUFDTCxNQUFNLEVBQUUsaUJBQWU7WUFDdkIsT0FBTyxFQUFFLE9BQU8sQ0FBQyxPQUFPLElBQUksRUFBRTtZQUM5QixTQUFTLEVBQUU7Z0JBQ1QsR0FBRyxJQUFJLENBQUMsb0JBQW9CLENBQUMsT0FBTyxDQUFDO2dCQUNyQyxvQ0FBZ0I7YUFDakI7WUFDRCxPQUFPLEVBQUUsQ0FBQyxvQ0FBZ0IsQ0FBQztTQUM1QixDQUFDO0lBQ0osQ0FBQztJQUVPLE1BQU0sQ0FBQyxvQkFBb0IsQ0FDakMsT0FBb0M7UUFFcEMsSUFBSSxPQUFPLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDdkIsT0FBTztnQkFDTDtvQkFDRSxPQUFPLEVBQUUsZ0NBQXdCO29CQUNqQyxVQUFVLEVBQUUsT0FBTyxDQUFDLFVBQVU7b0JBQzlCLE1BQU0sRUFBRSxPQUFPLENBQUMsTUFBTSxJQUFJLEVBQUU7aUJBQzdCO2FBQ0YsQ0FBQztRQUNKLENBQUM7UUFFRCxJQUFJLE9BQU8sQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNyQixPQUFPO2dCQUNMO29CQUNFLE9BQU8sRUFBRSxPQUFPLENBQUMsUUFBUTtvQkFDekIsUUFBUSxFQUFFLE9BQU8sQ0FBQyxRQUFRO2lCQUMzQjtnQkFDRDtvQkFDRSxPQUFPLEVBQUUsZ0NBQXdCO29CQUNqQyxVQUFVLEVBQUUsS0FBSyxFQUFFLGNBQTZDLEVBQUUsRUFBRSxDQUNsRSxjQUFjLENBQUMsc0JBQXNCLEVBQUU7b0JBQ3pDLE1BQU0sRUFBRSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUM7aUJBQzNCO2FBQ0YsQ0FBQztRQUNKLENBQUM7UUFFRCxJQUFJLE9BQU8sQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUN4QixPQUFPO2dCQUNMO29CQUNFLE9BQU8sRUFBRSxnQ0FBd0I7b0JBQ2pDLFVBQVUsRUFBRSxLQUFLLEVBQUUsY0FBNkMsRUFBRSxFQUFFLENBQ2xFLGNBQWMsQ0FBQyxzQkFBc0IsRUFBRTtvQkFDekMsTUFBTSxFQUFFLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQztpQkFDOUI7YUFDRixDQUFDO1FBQ0osQ0FBQztRQUVELE9BQU8sRUFBRSxDQUFDO0lBQ1osQ0FBQztDQUNGLENBQUE7QUEzRVksMENBQWU7MEJBQWYsZUFBZTtJQUQzQixJQUFBLGVBQU0sRUFBQyxFQUFFLENBQUM7R0FDRSxlQUFlLENBMkUzQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IER5bmFtaWNNb2R1bGUsIE1vZHVsZSwgUHJvdmlkZXIgfSBmcm9tICdAbmVzdGpzL2NvbW1vbic7XG5pbXBvcnQgeyBNaWdyYXRpb25TZXJ2aWNlIH0gZnJvbSAnLi9taWdyYXRpb24uc2VydmljZSc7XG5pbXBvcnQge1xuICBNSUdSQVRJT05fTU9EVUxFX09QVElPTlMsXG4gIE1pZ3JhdGlvbk1vZHVsZUFzeW5jT3B0aW9ucyxcbiAgTWlncmF0aW9uTW9kdWxlT3B0aW9ucyxcbiAgTWlncmF0aW9uTW9kdWxlT3B0aW9uc0ZhY3RvcnksXG59IGZyb20gJy4vdHlwZXMnO1xuXG4vKipcbiAqIE5lc3RKUyBtb2R1bGUgZm9yIFNRTCBTZXJ2ZXIgbWlncmF0aW9ucy5cbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gU3luY2hyb25vdXMgY29uZmlndXJhdGlvblxuICogTWlncmF0aW9uTW9kdWxlLmZvclJvb3Qoe1xuICogICBtaWdyYXRpb25zRGlyOiBwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vbWlncmF0aW9ucycpLFxuICogICBhdXRvUnVuOiB0cnVlLFxuICogICB2ZXJib3NlOiB0cnVlLFxuICogfSlcbiAqXG4gKiBAZXhhbXBsZVxuICogLy8gQXN5bmNocm9ub3VzIGNvbmZpZ3VyYXRpb24gd2l0aCBDb25maWdTZXJ2aWNlXG4gKiBNaWdyYXRpb25Nb2R1bGUuZm9yUm9vdEFzeW5jKHtcbiAqICAgaW1wb3J0czogW0NvbmZpZ01vZHVsZV0sXG4gKiAgIGluamVjdDogW0NvbmZpZ1NlcnZpY2VdLFxuICogICB1c2VGYWN0b3J5OiAoY29uZmlnU2VydmljZTogQ29uZmlnU2VydmljZSkgPT4gKHtcbiAqICAgICBtaWdyYXRpb25zRGlyOiBwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vbWlncmF0aW9ucycpLFxuICogICAgIGF1dG9SdW46IGNvbmZpZ1NlcnZpY2UuZ2V0PGJvb2xlYW4+KCdBVVRPX1JVTl9NSUdSQVRJT05TJywgZmFsc2UpLFxuICogICAgIHZlcmJvc2U6IHRydWUsXG4gKiAgIH0pLFxuICogfSlcbiAqL1xuQE1vZHVsZSh7fSlcbmV4cG9ydCBjbGFzcyBNaWdyYXRpb25Nb2R1bGUge1xuICAvKipcbiAgICogQ29uZmlndXJlIHRoZSBtaWdyYXRpb24gbW9kdWxlIHdpdGggc3RhdGljIG9wdGlvbnMuXG4gICAqL1xuICBzdGF0aWMgZm9yUm9vdChvcHRpb25zOiBNaWdyYXRpb25Nb2R1bGVPcHRpb25zKTogRHluYW1pY01vZHVsZSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIG1vZHVsZTogTWlncmF0aW9uTW9kdWxlLFxuICAgICAgcHJvdmlkZXJzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBNSUdSQVRJT05fTU9EVUxFX09QVElPTlMsXG4gICAgICAgICAgdXNlVmFsdWU6IG9wdGlvbnMsXG4gICAgICAgIH0sXG4gICAgICAgIE1pZ3JhdGlvblNlcnZpY2UsXG4gICAgICBdLFxuICAgICAgZXhwb3J0czogW01pZ3JhdGlvblNlcnZpY2VdLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ29uZmlndXJlIHRoZSBtaWdyYXRpb24gbW9kdWxlIHdpdGggYXN5bmMgb3B0aW9ucy5cbiAgICogVXNlZnVsIHdoZW4gb3B0aW9ucyBkZXBlbmQgb24gQ29uZmlnU2VydmljZSBvciBvdGhlciBhc3luYyBwcm92aWRlcnMuXG4gICAqL1xuICBzdGF0aWMgZm9yUm9vdEFzeW5jKG9wdGlvbnM6IE1pZ3JhdGlvbk1vZHVsZUFzeW5jT3B0aW9ucyk6IER5bmFtaWNNb2R1bGUge1xuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IE1pZ3JhdGlvbk1vZHVsZSxcbiAgICAgIGltcG9ydHM6IG9wdGlvbnMuaW1wb3J0cyB8fCBbXSxcbiAgICAgIHByb3ZpZGVyczogW1xuICAgICAgICAuLi50aGlzLmNyZWF0ZUFzeW5jUHJvdmlkZXJzKG9wdGlvbnMpLFxuICAgICAgICBNaWdyYXRpb25TZXJ2aWNlLFxuICAgICAgXSxcbiAgICAgIGV4cG9ydHM6IFtNaWdyYXRpb25TZXJ2aWNlXSxcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQXN5bmNQcm92aWRlcnMoXG4gICAgb3B0aW9uczogTWlncmF0aW9uTW9kdWxlQXN5bmNPcHRpb25zLFxuICApOiBQcm92aWRlcltdIHtcbiAgICBpZiAob3B0aW9ucy51c2VGYWN0b3J5KSB7XG4gICAgICByZXR1cm4gW1xuICAgICAgICB7XG4gICAgICAgICAgcHJvdmlkZTogTUlHUkFUSU9OX01PRFVMRV9PUFRJT05TLFxuICAgICAgICAgIHVzZUZhY3Rvcnk6IG9wdGlvbnMudXNlRmFjdG9yeSxcbiAgICAgICAgICBpbmplY3Q6IG9wdGlvbnMuaW5qZWN0IHx8IFtdLFxuICAgICAgICB9LFxuICAgICAgXTtcbiAgICB9XG5cbiAgICBpZiAob3B0aW9ucy51c2VDbGFzcykge1xuICAgICAgcmV0dXJuIFtcbiAgICAgICAge1xuICAgICAgICAgIHByb3ZpZGU6IG9wdGlvbnMudXNlQ2xhc3MsXG4gICAgICAgICAgdXNlQ2xhc3M6IG9wdGlvbnMudXNlQ2xhc3MsXG4gICAgICAgIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBNSUdSQVRJT05fTU9EVUxFX09QVElPTlMsXG4gICAgICAgICAgdXNlRmFjdG9yeTogYXN5bmMgKG9wdGlvbnNGYWN0b3J5OiBNaWdyYXRpb25Nb2R1bGVPcHRpb25zRmFjdG9yeSkgPT5cbiAgICAgICAgICAgIG9wdGlvbnNGYWN0b3J5LmNyZWF0ZU1pZ3JhdGlvbk9wdGlvbnMoKSxcbiAgICAgICAgICBpbmplY3Q6IFtvcHRpb25zLnVzZUNsYXNzXSxcbiAgICAgICAgfSxcbiAgICAgIF07XG4gICAgfVxuXG4gICAgaWYgKG9wdGlvbnMudXNlRXhpc3RpbmcpIHtcbiAgICAgIHJldHVybiBbXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBNSUdSQVRJT05fTU9EVUxFX09QVElPTlMsXG4gICAgICAgICAgdXNlRmFjdG9yeTogYXN5bmMgKG9wdGlvbnNGYWN0b3J5OiBNaWdyYXRpb25Nb2R1bGVPcHRpb25zRmFjdG9yeSkgPT5cbiAgICAgICAgICAgIG9wdGlvbnNGYWN0b3J5LmNyZWF0ZU1pZ3JhdGlvbk9wdGlvbnMoKSxcbiAgICAgICAgICBpbmplY3Q6IFtvcHRpb25zLnVzZUV4aXN0aW5nXSxcbiAgICAgICAgfSxcbiAgICAgIF07XG4gICAgfVxuXG4gICAgcmV0dXJuIFtdO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,32 @@
1
+ import { OnModuleInit } from '@nestjs/common';
2
+ import { DataSource } from 'typeorm';
3
+ import { MigrationModuleOptions, MigrationResult, MigrationRunOptions } from './types';
4
+ /**
5
+ * NestJS service for running SQL Server migrations.
6
+ * Automatically runs migrations on module initialization if autoRun is enabled.
7
+ */
8
+ export declare class MigrationService implements OnModuleInit {
9
+ private readonly options;
10
+ private readonly migrationCore;
11
+ private readonly autoRun;
12
+ private readonly verbose;
13
+ constructor(options: MigrationModuleOptions, dataSource: DataSource);
14
+ onModuleInit(): Promise<void>;
15
+ /**
16
+ * Run all pending migrations.
17
+ */
18
+ runMigrations(options?: MigrationRunOptions): Promise<MigrationResult>;
19
+ /**
20
+ * Get list of pending migration files.
21
+ */
22
+ getPendingMigrations(): Promise<string[]>;
23
+ /**
24
+ * Get migration status information.
25
+ */
26
+ getStatus(): Promise<{
27
+ total: number;
28
+ applied: number;
29
+ pending: number;
30
+ pendingFiles: string[];
31
+ }>;
32
+ }
@@ -0,0 +1,77 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ var __param = (this && this.__param) || function (paramIndex, decorator) {
12
+ return function (target, key) { decorator(target, key, paramIndex); }
13
+ };
14
+ Object.defineProperty(exports, "__esModule", { value: true });
15
+ exports.MigrationService = void 0;
16
+ const common_1 = require("@nestjs/common");
17
+ const typeorm_1 = require("typeorm");
18
+ const migration_core_1 = require("./migration-core");
19
+ const types_1 = require("./types");
20
+ /**
21
+ * NestJS service for running SQL Server migrations.
22
+ * Automatically runs migrations on module initialization if autoRun is enabled.
23
+ */
24
+ let MigrationService = class MigrationService {
25
+ options;
26
+ migrationCore;
27
+ autoRun;
28
+ verbose;
29
+ constructor(options, dataSource) {
30
+ this.options = options;
31
+ this.migrationCore = new migration_core_1.MigrationCore(dataSource, {
32
+ migrationsDir: options.migrationsDir,
33
+ });
34
+ this.autoRun = options.autoRun ?? false;
35
+ this.verbose = options.verbose ?? true;
36
+ }
37
+ async onModuleInit() {
38
+ if (!this.autoRun) {
39
+ if (this.verbose) {
40
+ console.log('Auto migrations are disabled');
41
+ }
42
+ return;
43
+ }
44
+ if (this.verbose) {
45
+ console.log('Auto-running migrations...');
46
+ }
47
+ await this.runMigrations();
48
+ }
49
+ /**
50
+ * Run all pending migrations.
51
+ */
52
+ async runMigrations(options) {
53
+ return this.migrationCore.runMigrations({
54
+ verbose: this.verbose,
55
+ ...options,
56
+ });
57
+ }
58
+ /**
59
+ * Get list of pending migration files.
60
+ */
61
+ async getPendingMigrations() {
62
+ return this.migrationCore.getPendingMigrations();
63
+ }
64
+ /**
65
+ * Get migration status information.
66
+ */
67
+ async getStatus() {
68
+ return this.migrationCore.getStatus();
69
+ }
70
+ };
71
+ exports.MigrationService = MigrationService;
72
+ exports.MigrationService = MigrationService = __decorate([
73
+ (0, common_1.Injectable)(),
74
+ __param(0, (0, common_1.Inject)(types_1.MIGRATION_MODULE_OPTIONS)),
75
+ __metadata("design:paramtypes", [Object, typeorm_1.DataSource])
76
+ ], MigrationService);
77
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlncmF0aW9uLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbWlncmF0aW9uLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsMkNBQWtFO0FBQ2xFLHFDQUFxQztBQUNyQyxxREFBaUQ7QUFDakQsbUNBS2lCO0FBRWpCOzs7R0FHRztBQUVJLElBQU0sZ0JBQWdCLEdBQXRCLE1BQU0sZ0JBQWdCO0lBT1I7SUFORixhQUFhLENBQWdCO0lBQzdCLE9BQU8sQ0FBVTtJQUNqQixPQUFPLENBQVU7SUFFbEMsWUFFbUIsT0FBK0IsRUFDaEQsVUFBc0I7UUFETCxZQUFPLEdBQVAsT0FBTyxDQUF3QjtRQUdoRCxJQUFJLENBQUMsYUFBYSxHQUFHLElBQUksOEJBQWEsQ0FBQyxVQUFVLEVBQUU7WUFDakQsYUFBYSxFQUFFLE9BQU8sQ0FBQyxhQUFhO1NBQ3JDLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sSUFBSSxLQUFLLENBQUM7UUFDeEMsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQztJQUN6QyxDQUFDO0lBRUQsS0FBSyxDQUFDLFlBQVk7UUFDaEIsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNsQixJQUFJLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztnQkFDakIsT0FBTyxDQUFDLEdBQUcsQ0FBQyw4QkFBOEIsQ0FBQyxDQUFDO1lBQzlDLENBQUM7WUFDRCxPQUFPO1FBQ1QsQ0FBQztRQUVELElBQUksSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ2pCLE9BQU8sQ0FBQyxHQUFHLENBQUMsNEJBQTRCLENBQUMsQ0FBQztRQUM1QyxDQUFDO1FBRUQsTUFBTSxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7SUFDN0IsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSyxDQUFDLGFBQWEsQ0FBQyxPQUE2QjtRQUMvQyxPQUFPLElBQUksQ0FBQyxhQUFhLENBQUMsYUFBYSxDQUFDO1lBQ3RDLE9BQU8sRUFBRSxJQUFJLENBQUMsT0FBTztZQUNyQixHQUFHLE9BQU87U0FDWCxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsb0JBQW9CO1FBQ3hCLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQyxvQkFBb0IsRUFBRSxDQUFDO0lBQ25ELENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxTQUFTO1FBTWIsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDLFNBQVMsRUFBRSxDQUFDO0lBQ3hDLENBQUM7Q0FDRixDQUFBO0FBNURZLDRDQUFnQjsyQkFBaEIsZ0JBQWdCO0lBRDVCLElBQUEsbUJBQVUsR0FBRTtJQU9SLFdBQUEsSUFBQSxlQUFNLEVBQUMsZ0NBQXdCLENBQUMsQ0FBQTs2Q0FFckIsb0JBQVU7R0FSYixnQkFBZ0IsQ0E0RDVCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0LCBJbmplY3RhYmxlLCBPbk1vZHVsZUluaXQgfSBmcm9tICdAbmVzdGpzL2NvbW1vbic7XG5pbXBvcnQgeyBEYXRhU291cmNlIH0gZnJvbSAndHlwZW9ybSc7XG5pbXBvcnQgeyBNaWdyYXRpb25Db3JlIH0gZnJvbSAnLi9taWdyYXRpb24tY29yZSc7XG5pbXBvcnQge1xuICBNSUdSQVRJT05fTU9EVUxFX09QVElPTlMsXG4gIE1pZ3JhdGlvbk1vZHVsZU9wdGlvbnMsXG4gIE1pZ3JhdGlvblJlc3VsdCxcbiAgTWlncmF0aW9uUnVuT3B0aW9ucyxcbn0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogTmVzdEpTIHNlcnZpY2UgZm9yIHJ1bm5pbmcgU1FMIFNlcnZlciBtaWdyYXRpb25zLlxuICogQXV0b21hdGljYWxseSBydW5zIG1pZ3JhdGlvbnMgb24gbW9kdWxlIGluaXRpYWxpemF0aW9uIGlmIGF1dG9SdW4gaXMgZW5hYmxlZC5cbiAqL1xuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIE1pZ3JhdGlvblNlcnZpY2UgaW1wbGVtZW50cyBPbk1vZHVsZUluaXQge1xuICBwcml2YXRlIHJlYWRvbmx5IG1pZ3JhdGlvbkNvcmU6IE1pZ3JhdGlvbkNvcmU7XG4gIHByaXZhdGUgcmVhZG9ubHkgYXV0b1J1bjogYm9vbGVhbjtcbiAgcHJpdmF0ZSByZWFkb25seSB2ZXJib3NlOiBib29sZWFuO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIEBJbmplY3QoTUlHUkFUSU9OX01PRFVMRV9PUFRJT05TKVxuICAgIHByaXZhdGUgcmVhZG9ubHkgb3B0aW9uczogTWlncmF0aW9uTW9kdWxlT3B0aW9ucyxcbiAgICBkYXRhU291cmNlOiBEYXRhU291cmNlLFxuICApIHtcbiAgICB0aGlzLm1pZ3JhdGlvbkNvcmUgPSBuZXcgTWlncmF0aW9uQ29yZShkYXRhU291cmNlLCB7XG4gICAgICBtaWdyYXRpb25zRGlyOiBvcHRpb25zLm1pZ3JhdGlvbnNEaXIsXG4gICAgfSk7XG4gICAgdGhpcy5hdXRvUnVuID0gb3B0aW9ucy5hdXRvUnVuID8/IGZhbHNlO1xuICAgIHRoaXMudmVyYm9zZSA9IG9wdGlvbnMudmVyYm9zZSA/PyB0cnVlO1xuICB9XG5cbiAgYXN5bmMgb25Nb2R1bGVJbml0KCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGlmICghdGhpcy5hdXRvUnVuKSB7XG4gICAgICBpZiAodGhpcy52ZXJib3NlKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKCdBdXRvIG1pZ3JhdGlvbnMgYXJlIGRpc2FibGVkJyk7XG4gICAgICB9XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKHRoaXMudmVyYm9zZSkge1xuICAgICAgY29uc29sZS5sb2coJ0F1dG8tcnVubmluZyBtaWdyYXRpb25zLi4uJyk7XG4gICAgfVxuXG4gICAgYXdhaXQgdGhpcy5ydW5NaWdyYXRpb25zKCk7XG4gIH1cblxuICAvKipcbiAgICogUnVuIGFsbCBwZW5kaW5nIG1pZ3JhdGlvbnMuXG4gICAqL1xuICBhc3luYyBydW5NaWdyYXRpb25zKG9wdGlvbnM/OiBNaWdyYXRpb25SdW5PcHRpb25zKTogUHJvbWlzZTxNaWdyYXRpb25SZXN1bHQ+IHtcbiAgICByZXR1cm4gdGhpcy5taWdyYXRpb25Db3JlLnJ1bk1pZ3JhdGlvbnMoe1xuICAgICAgdmVyYm9zZTogdGhpcy52ZXJib3NlLFxuICAgICAgLi4ub3B0aW9ucyxcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgbGlzdCBvZiBwZW5kaW5nIG1pZ3JhdGlvbiBmaWxlcy5cbiAgICovXG4gIGFzeW5jIGdldFBlbmRpbmdNaWdyYXRpb25zKCk6IFByb21pc2U8c3RyaW5nW10+IHtcbiAgICByZXR1cm4gdGhpcy5taWdyYXRpb25Db3JlLmdldFBlbmRpbmdNaWdyYXRpb25zKCk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IG1pZ3JhdGlvbiBzdGF0dXMgaW5mb3JtYXRpb24uXG4gICAqL1xuICBhc3luYyBnZXRTdGF0dXMoKTogUHJvbWlzZTx7XG4gICAgdG90YWw6IG51bWJlcjtcbiAgICBhcHBsaWVkOiBudW1iZXI7XG4gICAgcGVuZGluZzogbnVtYmVyO1xuICAgIHBlbmRpbmdGaWxlczogc3RyaW5nW107XG4gIH0+IHtcbiAgICByZXR1cm4gdGhpcy5taWdyYXRpb25Db3JlLmdldFN0YXR1cygpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,83 @@
1
+ import { ModuleMetadata, Type } from '@nestjs/common';
2
+ export interface MigrationModuleOptions {
3
+ /**
4
+ * Path to the directory containing SQL migration files
5
+ */
6
+ migrationsDir: string;
7
+ /**
8
+ * Whether to automatically run migrations on module initialization
9
+ * @default false
10
+ */
11
+ autoRun?: boolean;
12
+ /**
13
+ * Whether to log verbose output during migration execution
14
+ * @default true
15
+ */
16
+ verbose?: boolean;
17
+ }
18
+ export interface MigrationModuleAsyncOptions extends Pick<ModuleMetadata, 'imports'> {
19
+ /**
20
+ * Factory function to create MigrationModuleOptions
21
+ */
22
+ useFactory: (...args: any[]) => Promise<MigrationModuleOptions> | MigrationModuleOptions;
23
+ /**
24
+ * Dependencies to inject into the factory function
25
+ */
26
+ inject?: any[];
27
+ /**
28
+ * Optional class to use for creating options
29
+ */
30
+ useClass?: Type<MigrationModuleOptionsFactory>;
31
+ /**
32
+ * Optional existing instance to use for options
33
+ */
34
+ useExisting?: Type<MigrationModuleOptionsFactory>;
35
+ }
36
+ export interface MigrationModuleOptionsFactory {
37
+ createMigrationOptions(): Promise<MigrationModuleOptions> | MigrationModuleOptions;
38
+ }
39
+ export interface MigrationResult {
40
+ /**
41
+ * Whether the migration run was successful
42
+ */
43
+ success: boolean;
44
+ /**
45
+ * Total number of migration files found
46
+ */
47
+ totalMigrations: number;
48
+ /**
49
+ * Number of migrations already applied
50
+ */
51
+ appliedMigrations: number;
52
+ /**
53
+ * Number of pending migrations to apply
54
+ */
55
+ pendingMigrations: number;
56
+ /**
57
+ * Name of the failed migration file (if any)
58
+ */
59
+ failedMigration?: string;
60
+ /**
61
+ * Error object if migration failed
62
+ */
63
+ error?: any;
64
+ }
65
+ export interface MigrationRunOptions {
66
+ /**
67
+ * If true, only list pending migrations without applying them
68
+ * @default false
69
+ */
70
+ dryRun?: boolean;
71
+ /**
72
+ * Whether to log verbose output
73
+ * @default false
74
+ */
75
+ verbose?: boolean;
76
+ }
77
+ export interface MigrationCoreOptions {
78
+ /**
79
+ * Path to the directory containing SQL migration files
80
+ */
81
+ migrationsDir: string;
82
+ }
83
+ export declare const MIGRATION_MODULE_OPTIONS: unique symbol;
@@ -0,0 +1,5 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MIGRATION_MODULE_OPTIONS = void 0;
4
+ exports.MIGRATION_MODULE_OPTIONS = Symbol('MIGRATION_MODULE_OPTIONS');
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBeUdhLFFBQUEsd0JBQXdCLEdBQUcsTUFBTSxDQUFDLDBCQUEwQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNb2R1bGVNZXRhZGF0YSwgVHlwZSB9IGZyb20gJ0BuZXN0anMvY29tbW9uJztcblxuZXhwb3J0IGludGVyZmFjZSBNaWdyYXRpb25Nb2R1bGVPcHRpb25zIHtcbiAgLyoqXG4gICAqIFBhdGggdG8gdGhlIGRpcmVjdG9yeSBjb250YWluaW5nIFNRTCBtaWdyYXRpb24gZmlsZXNcbiAgICovXG4gIG1pZ3JhdGlvbnNEaXI6IHN0cmluZztcblxuICAvKipcbiAgICogV2hldGhlciB0byBhdXRvbWF0aWNhbGx5IHJ1biBtaWdyYXRpb25zIG9uIG1vZHVsZSBpbml0aWFsaXphdGlvblxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgYXV0b1J1bj86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgdG8gbG9nIHZlcmJvc2Ugb3V0cHV0IGR1cmluZyBtaWdyYXRpb24gZXhlY3V0aW9uXG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICovXG4gIHZlcmJvc2U/OiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1pZ3JhdGlvbk1vZHVsZUFzeW5jT3B0aW9uc1xuICBleHRlbmRzIFBpY2s8TW9kdWxlTWV0YWRhdGEsICdpbXBvcnRzJz4ge1xuICAvKipcbiAgICogRmFjdG9yeSBmdW5jdGlvbiB0byBjcmVhdGUgTWlncmF0aW9uTW9kdWxlT3B0aW9uc1xuICAgKi9cbiAgdXNlRmFjdG9yeTogKFxuICAgIC4uLmFyZ3M6IGFueVtdXG4gICkgPT4gUHJvbWlzZTxNaWdyYXRpb25Nb2R1bGVPcHRpb25zPiB8IE1pZ3JhdGlvbk1vZHVsZU9wdGlvbnM7XG5cbiAgLyoqXG4gICAqIERlcGVuZGVuY2llcyB0byBpbmplY3QgaW50byB0aGUgZmFjdG9yeSBmdW5jdGlvblxuICAgKi9cbiAgaW5qZWN0PzogYW55W107XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGNsYXNzIHRvIHVzZSBmb3IgY3JlYXRpbmcgb3B0aW9uc1xuICAgKi9cbiAgdXNlQ2xhc3M/OiBUeXBlPE1pZ3JhdGlvbk1vZHVsZU9wdGlvbnNGYWN0b3J5PjtcblxuICAvKipcbiAgICogT3B0aW9uYWwgZXhpc3RpbmcgaW5zdGFuY2UgdG8gdXNlIGZvciBvcHRpb25zXG4gICAqL1xuICB1c2VFeGlzdGluZz86IFR5cGU8TWlncmF0aW9uTW9kdWxlT3B0aW9uc0ZhY3Rvcnk+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE1pZ3JhdGlvbk1vZHVsZU9wdGlvbnNGYWN0b3J5IHtcbiAgY3JlYXRlTWlncmF0aW9uT3B0aW9ucygpOlxuICAgIHwgUHJvbWlzZTxNaWdyYXRpb25Nb2R1bGVPcHRpb25zPlxuICAgIHwgTWlncmF0aW9uTW9kdWxlT3B0aW9ucztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNaWdyYXRpb25SZXN1bHQge1xuICAvKipcbiAgICogV2hldGhlciB0aGUgbWlncmF0aW9uIHJ1biB3YXMgc3VjY2Vzc2Z1bFxuICAgKi9cbiAgc3VjY2VzczogYm9vbGVhbjtcblxuICAvKipcbiAgICogVG90YWwgbnVtYmVyIG9mIG1pZ3JhdGlvbiBmaWxlcyBmb3VuZFxuICAgKi9cbiAgdG90YWxNaWdyYXRpb25zOiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIE51bWJlciBvZiBtaWdyYXRpb25zIGFscmVhZHkgYXBwbGllZFxuICAgKi9cbiAgYXBwbGllZE1pZ3JhdGlvbnM6IG51bWJlcjtcblxuICAvKipcbiAgICogTnVtYmVyIG9mIHBlbmRpbmcgbWlncmF0aW9ucyB0byBhcHBseVxuICAgKi9cbiAgcGVuZGluZ01pZ3JhdGlvbnM6IG51bWJlcjtcblxuICAvKipcbiAgICogTmFtZSBvZiB0aGUgZmFpbGVkIG1pZ3JhdGlvbiBmaWxlIChpZiBhbnkpXG4gICAqL1xuICBmYWlsZWRNaWdyYXRpb24/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEVycm9yIG9iamVjdCBpZiBtaWdyYXRpb24gZmFpbGVkXG4gICAqL1xuICBlcnJvcj86IGFueTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNaWdyYXRpb25SdW5PcHRpb25zIHtcbiAgLyoqXG4gICAqIElmIHRydWUsIG9ubHkgbGlzdCBwZW5kaW5nIG1pZ3JhdGlvbnMgd2l0aG91dCBhcHBseWluZyB0aGVtXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICBkcnlSdW4/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIGxvZyB2ZXJib3NlIG91dHB1dFxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgdmVyYm9zZT86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTWlncmF0aW9uQ29yZU9wdGlvbnMge1xuICAvKipcbiAgICogUGF0aCB0byB0aGUgZGlyZWN0b3J5IGNvbnRhaW5pbmcgU1FMIG1pZ3JhdGlvbiBmaWxlc1xuICAgKi9cbiAgbWlncmF0aW9uc0Rpcjogc3RyaW5nO1xufVxuXG5leHBvcnQgY29uc3QgTUlHUkFUSU9OX01PRFVMRV9PUFRJT05TID0gU3ltYm9sKCdNSUdSQVRJT05fTU9EVUxFX09QVElPTlMnKTtcbiJdfQ==
package/package.json ADDED
@@ -0,0 +1,49 @@
1
+ {
2
+ "name": "@studiosonrai/nestjs-migrations",
3
+ "version": "1.0.0",
4
+ "description": "SQL Server migrations module for NestJS applications",
5
+ "main": "dist/src/index.js",
6
+ "types": "dist/src/index.d.ts",
7
+ "bin": {
8
+ "nestjs-migrations": "./dist/bin/cli.js"
9
+ },
10
+ "files": [
11
+ "dist"
12
+ ],
13
+ "scripts": {
14
+ "build": "rm -rf dist && tsc",
15
+ "prepublishOnly": "npm run build",
16
+ "clean": "rm -rf dist"
17
+ },
18
+ "keywords": [
19
+ "nestjs",
20
+ "migrations",
21
+ "sql-server",
22
+ "mssql",
23
+ "typeorm"
24
+ ],
25
+ "author": "",
26
+ "license": "UNLICENSED",
27
+ "peerDependencies": {
28
+ "@nestjs/common": "^11.0.0",
29
+ "@nestjs/config": "^4.0.0",
30
+ "@nestjs/core": "^11.0.0",
31
+ "typeorm": "^0.3.0"
32
+ },
33
+ "dependencies": {
34
+ "dotenv": "^17.2.1"
35
+ },
36
+ "devDependencies": {
37
+ "@nestjs/common": "^11.0.1",
38
+ "@nestjs/config": "^4.0.2",
39
+ "@nestjs/core": "^11.0.1",
40
+ "@types/node": "^22.10.7",
41
+ "reflect-metadata": "^0.2.2",
42
+ "rxjs": "^7.8.1",
43
+ "typeorm": "^0.3.25",
44
+ "typescript": "^5.7.3"
45
+ },
46
+ "publishConfig": {
47
+ "access": "public"
48
+ }
49
+ }