@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.
- package/dist/bin/cli.d.ts +2 -0
- package/dist/bin/cli.js +242 -0
- package/dist/src/index.d.ts +4 -0
- package/dist/src/index.js +16 -0
- package/dist/src/migration-core.d.ts +49 -0
- package/dist/src/migration-core.js +184 -0
- package/dist/src/migration.module.d.ts +37 -0
- package/dist/src/migration.module.js +108 -0
- package/dist/src/migration.service.d.ts +32 -0
- package/dist/src/migration.service.js +77 -0
- package/dist/src/types.d.ts +83 -0
- package/dist/src/types.js +5 -0
- package/package.json +49 -0
package/dist/bin/cli.js
ADDED
|
@@ -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
|
+
}
|