@mikeyt23/node-cli-utils 1.4.1 → 2.0.0-beta.2

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.
Files changed (76) hide show
  1. package/README.md +45 -87
  2. package/dist/cjs/GitUtility.d.ts +7 -0
  3. package/dist/cjs/GitUtility.d.ts.map +1 -0
  4. package/dist/cjs/GitUtility.js +49 -0
  5. package/dist/cjs/NodeCliUtilsConfig.d.ts +21 -0
  6. package/dist/cjs/NodeCliUtilsConfig.d.ts.map +1 -0
  7. package/dist/cjs/NodeCliUtilsConfig.js +41 -0
  8. package/dist/cjs/TarballUtility.d.ts +63 -0
  9. package/dist/cjs/TarballUtility.d.ts.map +1 -0
  10. package/dist/cjs/TarballUtility.js +143 -0
  11. package/dist/cjs/certUtils.d.ts +66 -0
  12. package/dist/cjs/certUtils.d.ts.map +1 -0
  13. package/dist/cjs/certUtils.js +283 -0
  14. package/dist/cjs/dbMigrationUtils.d.ts +39 -0
  15. package/dist/cjs/dbMigrationUtils.d.ts.map +1 -0
  16. package/dist/cjs/dbMigrationUtils.js +195 -0
  17. package/dist/cjs/dotnetUtils.d.ts +25 -0
  18. package/dist/cjs/dotnetUtils.d.ts.map +1 -0
  19. package/dist/cjs/dotnetUtils.js +61 -0
  20. package/dist/cjs/esmSpecific.d.mts +7 -0
  21. package/dist/cjs/esmSpecific.d.mts.map +1 -0
  22. package/dist/cjs/esmSpecific.mjs +15 -0
  23. package/dist/cjs/generalUtils.d.ts +565 -0
  24. package/dist/cjs/generalUtils.d.ts.map +1 -0
  25. package/dist/cjs/generalUtils.js +1068 -0
  26. package/dist/cjs/generalUtilsInternal.d.ts +15 -0
  27. package/dist/cjs/generalUtilsInternal.d.ts.map +1 -0
  28. package/dist/cjs/generalUtilsInternal.js +317 -0
  29. package/dist/cjs/hostsUtils.d.ts +16 -0
  30. package/dist/cjs/hostsUtils.d.ts.map +1 -0
  31. package/dist/cjs/hostsUtils.js +82 -0
  32. package/dist/cjs/index.d.ts +4 -0
  33. package/dist/cjs/index.d.ts.map +1 -0
  34. package/dist/cjs/index.js +25 -0
  35. package/dist/cjs/package.json +5 -0
  36. package/dist/cjs/runWhileParentAlive.d.ts +2 -0
  37. package/dist/cjs/runWhileParentAlive.d.ts.map +1 -0
  38. package/dist/cjs/runWhileParentAlive.js +159 -0
  39. package/dist/esm/GitUtility.d.ts +7 -0
  40. package/dist/esm/GitUtility.d.ts.map +1 -0
  41. package/dist/esm/GitUtility.js +43 -0
  42. package/dist/esm/NodeCliUtilsConfig.d.ts +21 -0
  43. package/dist/esm/NodeCliUtilsConfig.d.ts.map +1 -0
  44. package/dist/esm/NodeCliUtilsConfig.js +35 -0
  45. package/dist/esm/TarballUtility.d.ts +63 -0
  46. package/dist/esm/TarballUtility.d.ts.map +1 -0
  47. package/dist/esm/TarballUtility.js +139 -0
  48. package/dist/esm/certUtils.d.ts +66 -0
  49. package/dist/esm/certUtils.d.ts.map +1 -0
  50. package/dist/esm/certUtils.js +271 -0
  51. package/dist/esm/dbMigrationUtils.d.ts +39 -0
  52. package/dist/esm/dbMigrationUtils.d.ts.map +1 -0
  53. package/dist/esm/dbMigrationUtils.js +184 -0
  54. package/dist/esm/dotnetUtils.d.ts +25 -0
  55. package/dist/esm/dotnetUtils.d.ts.map +1 -0
  56. package/dist/esm/dotnetUtils.js +54 -0
  57. package/dist/esm/esmSpecific.d.mts +7 -0
  58. package/dist/esm/esmSpecific.d.mts.map +1 -0
  59. package/dist/esm/esmSpecific.mjs +11 -0
  60. package/dist/esm/generalUtils.d.ts +565 -0
  61. package/dist/esm/generalUtils.d.ts.map +1 -0
  62. package/dist/esm/generalUtils.js +976 -0
  63. package/dist/esm/generalUtilsInternal.d.ts +15 -0
  64. package/dist/esm/generalUtilsInternal.d.ts.map +1 -0
  65. package/dist/esm/generalUtilsInternal.js +278 -0
  66. package/dist/esm/hostsUtils.d.ts +16 -0
  67. package/dist/esm/hostsUtils.d.ts.map +1 -0
  68. package/dist/esm/hostsUtils.js +69 -0
  69. package/dist/esm/index.d.ts +4 -0
  70. package/dist/esm/index.d.ts.map +1 -0
  71. package/dist/esm/index.js +4 -0
  72. package/dist/esm/runWhileParentAlive.d.ts +2 -0
  73. package/dist/esm/runWhileParentAlive.d.ts.map +1 -0
  74. package/dist/esm/runWhileParentAlive.js +151 -0
  75. package/package.json +69 -10
  76. package/index.js +0 -627
@@ -0,0 +1,283 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.linuxInstallCert = exports.winGetPfxInfo = exports.winUninstallCert = exports.winCertIsInstalled = exports.winInstallCert = exports.generateCertWithOpenSsl = void 0;
7
+ const node_fs_1 = __importDefault(require("node:fs"));
8
+ const promises_1 = __importDefault(require("node:fs/promises"));
9
+ const node_path_1 = __importDefault(require("node:path"));
10
+ const generalUtils_js_1 = require("./generalUtils.js");
11
+ const defaultCertLogOptions = {
12
+ logSpawnOutput: false,
13
+ logTraceMessages: false,
14
+ logElevatedPermissionsMessage: true,
15
+ logSuccess: true
16
+ };
17
+ /**
18
+ * Wrapper function for calling openssl to generate a self-signed cert to be used for developing a local website with trusted https.
19
+ * @param url The url to generate a cert for. This will be used as the common name (CN) in the cert as well as the filename for the generated cert files.
20
+ * @param options Options for generating the cert.
21
+ * @returns The path to the generated pfx file.
22
+ */
23
+ async function generateCertWithOpenSsl(url, options) {
24
+ (0, generalUtils_js_1.requireString)('url', url);
25
+ throwIfMaybeBadUrlChars(url);
26
+ const isMac = (0, generalUtils_js_1.isPlatformMac)();
27
+ const mergedOptions = { ...defaultCertLogOptions, outputDirectory: './cert', ...options };
28
+ const spawnArgs = { cwd: mergedOptions.outputDirectory, stdio: mergedOptions.logSpawnOutput ? 'inherit' : 'pipe' };
29
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, 'checking if openssl is installed');
30
+ let brewOpenSslPath = '';
31
+ if (!isMac) {
32
+ const openSslPath = (0, generalUtils_js_1.whichSync)('openssl').location;
33
+ if (!openSslPath) {
34
+ throw Error('openssl is required but was not found');
35
+ }
36
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `using openssl at: ${openSslPath}`);
37
+ }
38
+ else if (isMac) {
39
+ const brewOpenSslDirectory = getBrewOpensslPath();
40
+ if (!brewOpenSslDirectory) {
41
+ throw Error('openssl (brew version) is required but was not found');
42
+ }
43
+ brewOpenSslPath = `${getBrewOpensslPath()}/bin/openssl`;
44
+ if (!node_fs_1.default.existsSync(brewOpenSslPath)) {
45
+ throw Error(`openssl (brew version) is required but was not found at: ${brewOpenSslPath}`);
46
+ }
47
+ else {
48
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `using openssl at: ${brewOpenSslPath}`);
49
+ }
50
+ }
51
+ (0, generalUtils_js_1.ensureDirectory)(mergedOptions.outputDirectory);
52
+ const crtName = url + '.crt';
53
+ const keyName = url + '.key';
54
+ const pfxName = url + '.pfx';
55
+ const sanCnfName = url + '.cnf';
56
+ const filesToCheck = [crtName, keyName, pfxName, sanCnfName];
57
+ for (const file of filesToCheck) {
58
+ const filePath = node_path_1.default.join(mergedOptions.outputDirectory, file);
59
+ if (node_fs_1.default.existsSync(filePath)) {
60
+ throw Error(`${generalUtils_js_1.Emoji.Stop} File ${filePath} already exists. Delete or rename all of the following files from '${mergedOptions.outputDirectory}' if you want to generate a new cert: ${filesToCheck.join(', ')}.`);
61
+ }
62
+ }
63
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `writing ${sanCnfName} file for use with openssl command`);
64
+ const sanCnfContents = getSanCnfFileContents(url);
65
+ const sanCnfPath = node_path_1.default.join(mergedOptions.outputDirectory, sanCnfName);
66
+ await promises_1.default.writeFile(sanCnfPath, sanCnfContents);
67
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `attempting to generate cert ${pfxName}`);
68
+ const genKeyAndCrtArgs = `req -x509 -newkey rsa:4096 -sha256 -days 3650 -nodes -keyout ${keyName} -out ${crtName} -subj /CN=${url} -config ${sanCnfName}`.split(' ');
69
+ const command = isMac ? brewOpenSslPath : 'openssl';
70
+ let result = await (0, generalUtils_js_1.spawnAsync)(command, genKeyAndCrtArgs, spawnArgs);
71
+ throwIfSpawnResultError(result);
72
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, 'converting key and crt to pfx');
73
+ const convertToPfxArgs = `pkcs12 -certpbe AES-256-CBC -export -out ${pfxName} -aes256 -inkey ${keyName} -in ${crtName} -password pass:`.split(' ');
74
+ result = await (0, generalUtils_js_1.spawnAsync)(command, convertToPfxArgs, spawnArgs);
75
+ throwIfSpawnResultError(result);
76
+ const pfxPath = node_path_1.default.join(mergedOptions.outputDirectory, pfxName);
77
+ (0, generalUtils_js_1.logIf)(mergedOptions.logSuccess, `${generalUtils_js_1.Emoji.GreenCheck} Successfully generated cert: ${pfxPath}`);
78
+ return pfxPath;
79
+ }
80
+ exports.generateCertWithOpenSsl = generateCertWithOpenSsl;
81
+ /**
82
+ * Uses Powershell to install a cert to the local machine's trusted root store. Must have elevated permissions.
83
+ * If the cert is already installed, this function will do nothing.
84
+ * @param pfxPath The path to the pfx file to install.
85
+ */
86
+ async function winInstallCert(pfxPath, options) {
87
+ if (!(0, generalUtils_js_1.isPlatformWindows)()) {
88
+ throw Error('winInstallCert is only supported on Windows');
89
+ }
90
+ validatePfxPath(pfxPath);
91
+ const mergedOptions = { ...defaultCertLogOptions, ...options };
92
+ (0, generalUtils_js_1.logIf)(mergedOptions.logElevatedPermissionsMessage, getRequiresElevatedPermissionsMessage(true));
93
+ if (await winCertIsInstalled({ pfxPath }, mergedOptions)) {
94
+ const certInfo = await winGetPfxInfo(pfxPath);
95
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `${generalUtils_js_1.Emoji.Warning} certificate '${pfxPath}' with subject '${certInfo.subject}' is already installed - to install it again, first uninstall it manually or with the winUninstallCert function`);
96
+ return;
97
+ }
98
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `installing cert '${pfxPath}'`);
99
+ const psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Import-PfxCertificate -FilePath '${pfxPath}' -CertStoreLocation Cert:\\LocalMachine\\Root`);
100
+ const result = await (0, generalUtils_js_1.spawnAsync)('powershell', psCommandArgs, { stdio: mergedOptions.logSpawnOutput ? 'inherit' : 'pipe' });
101
+ throwIfSpawnResultError(result);
102
+ (0, generalUtils_js_1.logIf)(mergedOptions.logSuccess, `${generalUtils_js_1.Emoji.GreenCheck} Successfully installed cert: ${pfxPath}`);
103
+ }
104
+ exports.winInstallCert = winInstallCert;
105
+ /**
106
+ * Uses Powershell to check if a cert is already installed to the local machine's trusted root store.
107
+ * Uses the subject of the cert in order to avoid false negatives from regenerating the same self-signed cert
108
+ * with the same subject but different thumbprint. Note that this method is geared towards use with certs generated
109
+ * with the {@link generateCertWithOpenSsl} function, so this may not work using subject if your subject is not precisely "`CN=<url>`".
110
+ * @param identifier The subject or path to the pfx file of the cert to check.
111
+ * @returns `true` if the cert is already installed, `false` otherwise.
112
+ */
113
+ async function winCertIsInstalled(identifier, options) {
114
+ if (!(0, generalUtils_js_1.isPlatformWindows)()) {
115
+ throw new Error('winCertIsInstalled is only supported on Windows');
116
+ }
117
+ const mergedOptions = { ...defaultCertLogOptions, ...options };
118
+ let psCommandArgs;
119
+ // Get the count of certs installed with the same subject as the one we're trying to install
120
+ if (typeof identifier === 'string') {
121
+ (0, generalUtils_js_1.requireString)('subject', identifier);
122
+ validateSubject(identifier);
123
+ const subject = identifier.startsWith('CN=') ? identifier : `CN=${identifier}`;
124
+ psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Write-Host (Get-ChildItem Cert:\\LocalMachine\\Root | Where-Object { $_.Subject -eq '${subject}' } | Measure-Object).Count`);
125
+ }
126
+ else if ('pfxPath' in identifier) {
127
+ validatePfxPath(identifier.pfxPath);
128
+ psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Write-Host (Get-ChildItem Cert:\\LocalMachine\\Root | Where-Object { $_.Subject -eq (Get-PfxCertificate -FilePath '${identifier.pfxPath}').Subject } | Measure-Object).Count`);
129
+ }
130
+ const tracePart = (typeof identifier === 'string') ? `subject ${identifier}` : `pfxPath ${identifier.pfxPath}`;
131
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `checking if cert with ${tracePart} is already installed`);
132
+ const result = await (0, generalUtils_js_1.spawnAsync)('powershell', psCommandArgs, { stdio: 'pipe' });
133
+ throwIfSpawnResultError(result);
134
+ const lines = (0, generalUtils_js_1.stringToNonEmptyLines)(result.stdout);
135
+ if (lines.length !== 1) {
136
+ throw new Error(`Unexpected output from powershell command to check if the cert is already installed: ${result.stdout}`);
137
+ }
138
+ return lines[0].trim() !== '0';
139
+ }
140
+ exports.winCertIsInstalled = winCertIsInstalled;
141
+ /**
142
+ * Uses Powershell to uninstall a cert from the local machine's trusted root store. Must have elevated permissions.
143
+ * @param identifier The subject, thumbprint or path to the pfx file of the cert to uninstall.
144
+ * @param options Options for uninstalling the cert.
145
+ */
146
+ async function winUninstallCert(identifier, options) {
147
+ if (!(0, generalUtils_js_1.isPlatformWindows)()) {
148
+ throw new Error('winUninstallCert is only supported on Windows');
149
+ }
150
+ const mergedOptions = { ...defaultCertLogOptions, ...options };
151
+ (0, generalUtils_js_1.logIf)(mergedOptions.logElevatedPermissionsMessage, getRequiresElevatedPermissionsMessage(false));
152
+ let psCommandArgs;
153
+ if (typeof identifier === 'string') {
154
+ (0, generalUtils_js_1.requireString)('subject', identifier);
155
+ validateSubject(identifier);
156
+ psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Get-ChildItem Cert:\\LocalMachine\\Root | Where-Object { $_.Subject -match '${identifier}' } | Remove-Item`);
157
+ }
158
+ else if ('thumbprint' in identifier) {
159
+ validateNoQuotes('thumbprint', identifier.thumbprint);
160
+ psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Get-ChildItem Cert:\\LocalMachine\\Root | Where-Object { $_.Thumbprint -eq '${identifier.thumbprint}' } | Remove-Item`);
161
+ }
162
+ else if ('pfxPath' in identifier) {
163
+ validatePfxPath(identifier.pfxPath);
164
+ psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`$thumbprint = (Get-PfxCertificate -FilePath '${identifier.pfxPath}').Thumbprint; Get-ChildItem Cert:\\LocalMachine\\Root | Where-Object { $_.Thumbprint -eq $thumbprint } | Remove-Item`);
165
+ }
166
+ const tracePart = typeof identifier === 'string' ? `'${identifier}'` : JSON.stringify(identifier);
167
+ (0, generalUtils_js_1.logIf)(mergedOptions.logTraceMessages, `uninstalling cert ${tracePart}`);
168
+ const result = await (0, generalUtils_js_1.spawnAsync)('powershell', psCommandArgs, { stdio: mergedOptions.logSpawnOutput ? 'inherit' : 'pipe' });
169
+ throwIfSpawnResultError(result);
170
+ (0, generalUtils_js_1.logIf)(mergedOptions.logSuccess, `${generalUtils_js_1.Emoji.GreenCheck} Successfully uninstalled cert`);
171
+ }
172
+ exports.winUninstallCert = winUninstallCert;
173
+ /**
174
+ * Uses Powershell to get info about a cert.
175
+ * @param pfxPath The path to the pfx file to get info for.
176
+ * @returns The subject, thumbprint and pfxPath of the cert.
177
+ */
178
+ async function winGetPfxInfo(pfxPath) {
179
+ if (!(0, generalUtils_js_1.isPlatformWindows)()) {
180
+ throw new Error('winGetPfxInfo is only supported on Windows');
181
+ }
182
+ validatePfxPath(pfxPath);
183
+ const psCommandArgs = (0, generalUtils_js_1.getPowershellHackArgs)(`Get-PfxCertificate -FilePath '${pfxPath}' | Select-Object -Property Subject, Thumbprint, @{Name='PfxPath';Expression={'${pfxPath}'}} | ConvertTo-Json`);
184
+ const result = await (0, generalUtils_js_1.spawnAsync)('powershell', psCommandArgs, { stdio: 'pipe' });
185
+ throwIfSpawnResultError(result);
186
+ const json = result.stdout.trim();
187
+ const parsedJson = JSON.parse(json);
188
+ const certInfo = {
189
+ subject: parsedJson.Subject,
190
+ thumbprint: parsedJson.Thumbprint,
191
+ pfxPath: parsedJson.PfxPath
192
+ };
193
+ return certInfo;
194
+ }
195
+ exports.winGetPfxInfo = winGetPfxInfo;
196
+ /**
197
+ * Does not actually do anything - just outputs the manual instructions for installing a cert for use by chrome on linux.
198
+ */
199
+ function linuxInstallCert() {
200
+ const instructions = `Automated linux cert install not supported (chrome does not use system certs without significant extra configuration).
201
+ Manual Instructions:
202
+ - In Chrome, go to chrome://settings/certificates
203
+ - Select Authorities -> import
204
+ - Select your generated .crt file (in the ./cert/ directory by default - if you haven't generated it, see the generateCertWithOpenSsl function)
205
+ - Check box for "Trust certificate for identifying websites"
206
+ - Click OK
207
+ - Reload site`;
208
+ console.log(instructions);
209
+ }
210
+ exports.linuxInstallCert = linuxInstallCert;
211
+ function throwIfMaybeBadUrlChars(url, varName = 'url') {
212
+ if (url.includes(' ')) {
213
+ throw Error(`${varName} should not contain spaces`);
214
+ }
215
+ if (url.includes('/')) {
216
+ throw Error(`${varName} should not contain forward slashes`);
217
+ }
218
+ if (url.includes('\\')) {
219
+ throw Error(`${varName} should not contain backslashes`);
220
+ }
221
+ }
222
+ function getBrewOpensslPath() {
223
+ const brewResult = (0, generalUtils_js_1.simpleSpawnSync)('brew', ['--prefix', 'openssl']);
224
+ if (brewResult.error) {
225
+ throw Error('error attempting to find openssl installed by brew');
226
+ }
227
+ if (brewResult.stdoutLines.length === 0 || brewResult.stdoutLines.length > 1) {
228
+ throw new Error(`unexpected output from brew command 'brew --prefix openssl': ${brewResult.stdout}`);
229
+ }
230
+ return brewResult.stdoutLines[0];
231
+ }
232
+ function getSanCnfFileContents(url) {
233
+ return sanCnfTemplate.replace(/{{url}}/g, url);
234
+ }
235
+ function validateSubject(subject) {
236
+ if (subject.includes('\\') || subject.includes('/') || subject.endsWith('.pfx')) {
237
+ throw new Error(`The subject appears to be a file path, which is not allowed. Did you mean to pass something like this instead: { pfxPath: '${subject}' } ?`);
238
+ }
239
+ validateNoQuotes('subject', subject);
240
+ }
241
+ function validateNoQuotes(name, value) {
242
+ if (value.includes("'") || value.includes('"')) {
243
+ throw new Error(`The value passed for '${name}' contains a single or double quote, which is not allowed.`);
244
+ }
245
+ }
246
+ function throwIfSpawnResultError(result) {
247
+ if (result.code !== 0) {
248
+ // There won't be any stderr if stdio was set to 'inherit', so we're checking first
249
+ if (result.stderr) {
250
+ console.error((0, generalUtils_js_1.red)('Error:'), result.stderr);
251
+ }
252
+ throw Error(`Spawned command failed with exit code ${result.code}`);
253
+ }
254
+ }
255
+ function validatePfxPath(pfxPath) {
256
+ if (!pfxPath.endsWith('.pfx')) {
257
+ throw new Error('pfxPath must end with .pfx');
258
+ }
259
+ (0, generalUtils_js_1.requireValidPath)('pfxPath', pfxPath);
260
+ validateNoQuotes('pfxPath', pfxPath);
261
+ }
262
+ function getRequiresElevatedPermissionsMessage(isInstall = true) {
263
+ return `${generalUtils_js_1.Emoji.Info} Important: ${isInstall ? '' : 'un'}installing a certificate requires elevated permissions`;
264
+ }
265
+ // Newer cert requirements force the need for "extension info" with DNS and IP info, but openssl v1.x doesn't support that with the
266
+ // CLI option -addext, so we're using a san.cnf file instead and passing this into the CLI command with the -config option.
267
+ const sanCnfTemplate = `[req]
268
+ distinguished_name=req
269
+ x509_extensions = v3_req
270
+ prompt = no
271
+
272
+ [req_distinguished_name]
273
+ CN = {{url}}
274
+
275
+ [v3_req]
276
+ subjectAltName = @alt_names
277
+
278
+ [alt_names]
279
+ DNS.1 = {{url}}
280
+ IP.1 = 127.0.0.1
281
+
282
+ `;
283
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2VydFV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NlcnRVdGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxzREFBd0I7QUFDeEIsZ0VBQWtDO0FBQ2xDLDBEQUE0QjtBQUM1Qix1REFnQjBCO0FBZTFCLE1BQU0scUJBQXFCLEdBQW1CO0lBQzVDLGNBQWMsRUFBRSxLQUFLO0lBQ3JCLGdCQUFnQixFQUFFLEtBQUs7SUFDdkIsNkJBQTZCLEVBQUUsSUFBSTtJQUNuQyxVQUFVLEVBQUUsSUFBSTtDQUNqQixDQUFBO0FBV0Q7Ozs7O0dBS0c7QUFDSSxLQUFLLFVBQVUsdUJBQXVCLENBQUMsR0FBVyxFQUFFLE9BQXNDO0lBQy9GLElBQUEsK0JBQWEsRUFBQyxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUE7SUFDekIsdUJBQXVCLENBQUMsR0FBRyxDQUFDLENBQUE7SUFDNUIsTUFBTSxLQUFLLEdBQUcsSUFBQSwrQkFBYSxHQUFFLENBQUE7SUFFN0IsTUFBTSxhQUFhLEdBQXdCLEVBQUUsR0FBRyxxQkFBcUIsRUFBRSxlQUFlLEVBQUUsUUFBUSxFQUFFLEdBQUcsT0FBTyxFQUFFLENBQUE7SUFFOUcsTUFBTSxTQUFTLEdBQW1DLEVBQUUsR0FBRyxFQUFFLGFBQWEsQ0FBQyxlQUFlLEVBQUUsS0FBSyxFQUFFLGFBQWEsQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUE7SUFFbEosSUFBQSx1QkFBSyxFQUFDLGFBQWEsQ0FBQyxnQkFBZ0IsRUFBRSxrQ0FBa0MsQ0FBQyxDQUFBO0lBRXpFLElBQUksZUFBZSxHQUFXLEVBQUUsQ0FBQTtJQUNoQyxJQUFJLENBQUMsS0FBSyxFQUFFO1FBQ1YsTUFBTSxXQUFXLEdBQUcsSUFBQSwyQkFBUyxFQUFDLFNBQVMsQ0FBQyxDQUFDLFFBQVEsQ0FBQTtRQUNqRCxJQUFJLENBQUMsV0FBVyxFQUFFO1lBQ2hCLE1B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
@@ -0,0 +1,39 @@
1
+ /**
2
+ * Wrapper function for `dotnet ef`. If you don't pass `false` for `noBuild`, be sure the project has already been built by some other means.
3
+ *
4
+ * Docs for "dotnet ef" CLI: https://learn.microsoft.com/en-us/ef/core/cli/dotnet.
5
+ * @param projectPath Path to project that has the DbContext and Migration files used for the `--project` argument
6
+ * @param dbContextName The name of the DbContext class used for the `--context` argument
7
+ * @param args Arguments to pass to the `dotnet ef` CLI
8
+ * @param noBuild If true, the `--no-build` argument will be passed to the `dotnet ef` CLI (default: true)
9
+ */
10
+ export declare function dotnetEfCommand(projectPath: string, dbContextName: string, args: string[], noBuild?: boolean): Promise<number>;
11
+ /**
12
+ * Wrapper function for `dotnet ef migrations list`.
13
+ * @param projectPath The path to the project that contains the DbContext and Migration files
14
+ * @param dbContextName The name of the DbContext class
15
+ */
16
+ export declare function efMigrationsList(projectPath: string, dbContextName: string): Promise<void>;
17
+ /**
18
+ * Wrapper function for `dotnet ef database update <migration_name>`.
19
+ * @param projectPath The path to the project that contains the DbContext and Migration files
20
+ * @param dbContextName The name of the DbContext class
21
+ * @param migrationName The name of the migration to update to (optional). If not provided, all migrations will be applied.
22
+ */
23
+ export declare function efMigrationsUpdate(projectPath: string, dbContextName: string, migrationName?: string): Promise<void>;
24
+ /**
25
+ *
26
+ * @param projectPath The path to the project that contains the DbContext and Migration files
27
+ * @param dbContextName The name of the DbContext class
28
+ * @param migrationName The name of the migration to add
29
+ * @param withBoilerplate If true, boilerplate will be added to the migration C# file and empty Up and Down SQL files will be created
30
+ */
31
+ export declare function efAddMigration(projectPath: string, dbContextName: string, migrationName: string, withBoilerplate?: boolean): Promise<void>;
32
+ /**
33
+ *
34
+ * @param projectPath The path to the project that contains the DbContext and Migration files
35
+ * @param dbContextName The name of the DbContext class
36
+ * @param skipConfirm If `true`, the user will not be prompted to confirm the removal of the last migration
37
+ */
38
+ export declare function efRemoveLastMigration(projectPath: string, dbContextName: string, skipConfirm?: boolean): Promise<void>;
39
+ //# sourceMappingURL=dbMigrationUtils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dbMigrationUtils.d.ts","sourceRoot":"","sources":["../../src/dbMigrationUtils.ts"],"names":[],"mappings":"AAKA;;;;;;;;GAQG;AACH,wBAAsB,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,OAAO,UAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAKjI;AAED;;;;GAIG;AACH,wBAAsB,gBAAgB,CAAC,WAAW,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,iBAEhF;AAED;;;;;GAKG;AACH,wBAAsB,kBAAkB,CAAC,WAAW,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,MAAM,iBAE1G;AAED;;;;;;GAMG;AACH,wBAAsB,cAAc,CAAC,WAAW,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,eAAe,UAAQ,iBAY9H;AAED;;;;;GAKG;AACH,wBAAsB,qBAAqB,CAAC,WAAW,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,EAAE,WAAW,UAAQ,iBAe1G"}
@@ -0,0 +1,195 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.efRemoveLastMigration = exports.efAddMigration = exports.efMigrationsUpdate = exports.efMigrationsList = exports.dotnetEfCommand = void 0;
7
+ const generalUtils_js_1 = require("./generalUtils.js");
8
+ const node_path_1 = __importDefault(require("node:path"));
9
+ const node_fs_1 = __importDefault(require("node:fs"));
10
+ const promises_1 = __importDefault(require("node:fs/promises"));
11
+ /**
12
+ * Wrapper function for `dotnet ef`. If you don't pass `false` for `noBuild`, be sure the project has already been built by some other means.
13
+ *
14
+ * Docs for "dotnet ef" CLI: https://learn.microsoft.com/en-us/ef/core/cli/dotnet.
15
+ * @param projectPath Path to project that has the DbContext and Migration files used for the `--project` argument
16
+ * @param dbContextName The name of the DbContext class used for the `--context` argument
17
+ * @param args Arguments to pass to the `dotnet ef` CLI
18
+ * @param noBuild If true, the `--no-build` argument will be passed to the `dotnet ef` CLI (default: true)
19
+ */
20
+ async function dotnetEfCommand(projectPath, dbContextName, args, noBuild = true) {
21
+ (0, generalUtils_js_1.requireValidPath)('projectPath', projectPath);
22
+ (0, generalUtils_js_1.requireString)('dbContextName', dbContextName);
23
+ const result = await (0, generalUtils_js_1.spawnAsync)('dotnet', ['ef', '--project', projectPath, ...args, '--context', dbContextName, ...(noBuild ? ['--no-build'] : [])]);
24
+ return result.code;
25
+ }
26
+ exports.dotnetEfCommand = dotnetEfCommand;
27
+ /**
28
+ * Wrapper function for `dotnet ef migrations list`.
29
+ * @param projectPath The path to the project that contains the DbContext and Migration files
30
+ * @param dbContextName The name of the DbContext class
31
+ */
32
+ async function efMigrationsList(projectPath, dbContextName) {
33
+ await dotnetEfCommand(projectPath, dbContextName, ['migrations', 'list']);
34
+ }
35
+ exports.efMigrationsList = efMigrationsList;
36
+ /**
37
+ * Wrapper function for `dotnet ef database update <migration_name>`.
38
+ * @param projectPath The path to the project that contains the DbContext and Migration files
39
+ * @param dbContextName The name of the DbContext class
40
+ * @param migrationName The name of the migration to update to (optional). If not provided, all migrations will be applied.
41
+ */
42
+ async function efMigrationsUpdate(projectPath, dbContextName, migrationName) {
43
+ await dotnetEfCommand(projectPath, dbContextName, ['database', 'update', ...(migrationName ? [migrationName] : [])]);
44
+ }
45
+ exports.efMigrationsUpdate = efMigrationsUpdate;
46
+ /**
47
+ *
48
+ * @param projectPath The path to the project that contains the DbContext and Migration files
49
+ * @param dbContextName The name of the DbContext class
50
+ * @param migrationName The name of the migration to add
51
+ * @param withBoilerplate If true, boilerplate will be added to the migration C# file and empty Up and Down SQL files will be created
52
+ */
53
+ async function efAddMigration(projectPath, dbContextName, migrationName, withBoilerplate = false) {
54
+ const projectDirectory = projectPath.endsWith('.csproj') ? projectPath.substring(0, projectPath.lastIndexOf('/')) : projectPath;
55
+ const migrationsOutputDir = getMigrationsProjectRelativePath(dbContextName);
56
+ await dotnetEfCommand(projectPath, dbContextName, ['migrations', 'add', migrationName, '-o', migrationsOutputDir]);
57
+ if (withBoilerplate) {
58
+ try {
59
+ await addDbMigrationBoilerplate(projectDirectory, dbContextName, migrationName);
60
+ }
61
+ catch (error) {
62
+ console.error(error);
63
+ await efRemoveLastMigration(projectPath, dbContextName, true);
64
+ }
65
+ }
66
+ }
67
+ exports.efAddMigration = efAddMigration;
68
+ /**
69
+ *
70
+ * @param projectPath The path to the project that contains the DbContext and Migration files
71
+ * @param dbContextName The name of the DbContext class
72
+ * @param skipConfirm If `true`, the user will not be prompted to confirm the removal of the last migration
73
+ */
74
+ async function efRemoveLastMigration(projectPath, dbContextName, skipConfirm = false) {
75
+ const lastMigrationName = await getLastMigrationName(projectPath, dbContextName);
76
+ if (!skipConfirm && !await (0, generalUtils_js_1.getConfirmation)(`Are you sure you want to remove the last migration: ➡️${lastMigrationName}?`)) {
77
+ return;
78
+ }
79
+ const returnCode = await dotnetEfCommand(projectPath, dbContextName, ['migrations', 'remove']);
80
+ if (returnCode !== 0) {
81
+ throw new Error(`dotnet ef migrations remove returned non-zero exit code: ${returnCode}`);
82
+ }
83
+ (0, generalUtils_js_1.log)(`Removing migration SQL script files for migration if they're empty`);
84
+ await deleteScriptFileIfEmpty(getScriptPath(projectPath, lastMigrationName, true));
85
+ await deleteScriptFileIfEmpty(getScriptPath(projectPath, lastMigrationName, false));
86
+ }
87
+ exports.efRemoveLastMigration = efRemoveLastMigration;
88
+ async function deleteScriptFileIfEmpty(scriptPath) {
89
+ if (node_fs_1.default.existsSync(scriptPath)) {
90
+ const scriptContents = node_fs_1.default.readFileSync(scriptPath, { encoding: 'utf8' });
91
+ if (scriptContents.trim().length === 0) {
92
+ await promises_1.default.unlink(scriptPath);
93
+ }
94
+ else {
95
+ (0, generalUtils_js_1.log)(`${generalUtils_js_1.Emoji.Warning} Skipping deletion of non-empty script file: ${scriptPath}`);
96
+ }
97
+ }
98
+ }
99
+ async function getLastMigrationName(projectPath, dbContextName) {
100
+ const migrationsDirectory = getMigrationsDirectory(projectPath, dbContextName);
101
+ const filenames = node_fs_1.default.readdirSync(migrationsDirectory);
102
+ const migrationNames = filenames.filter(filename => filename.endsWith('.cs') &&
103
+ !filename.endsWith('.Designer.cs') &&
104
+ !filename.endsWith('.ModelSnapshot.cs') &&
105
+ filename.includes('_')).map(filename => filename.substring(0, filename.length - 3));
106
+ const migrationNamesWithTimestamps = migrationNames.map(migrationName => {
107
+ const timestamp = migrationName.substring(0, 14);
108
+ const name = migrationName.substring(15);
109
+ return { timestamp, name };
110
+ });
111
+ (0, generalUtils_js_1.log)(`Found migrations: ${migrationNamesWithTimestamps.map(m => m.name).join(', ')}`);
112
+ (0, generalUtils_js_1.log)(`Found timestamps: ${migrationNamesWithTimestamps.map(m => m.timestamp).join(', ')}`);
113
+ const sortedMigrationNames = [...migrationNamesWithTimestamps].sort((a, b) => a.timestamp.localeCompare(b.timestamp));
114
+ const lastMigrationName = sortedMigrationNames[sortedMigrationNames.length - 1].name;
115
+ return lastMigrationName;
116
+ }
117
+ function getMigrationsProjectRelativePath(dbContextName) {
118
+ return `Migrations/${dbContextName}Migrations`;
119
+ }
120
+ function getMigrationsDirectory(projectDirectory, dbContextName) {
121
+ return node_path_1.default.join(projectDirectory, `Migrations/${dbContextName}Migrations`);
122
+ }
123
+ function getScriptPath(projectDirectory, migrationName, isUp) {
124
+ return node_path_1.default.join(projectDirectory, `Scripts/${migrationName}${isUp ? '' : '_Down'}.sql`);
125
+ }
126
+ async function getCSharpMigrationFilePath(projectDirectory, dbContextName, migrationName) {
127
+ const migrationsOutputDir = getMigrationsDirectory(projectDirectory, dbContextName);
128
+ if (!node_fs_1.default.existsSync(migrationsOutputDir)) {
129
+ throw new Error(`Unable to add migration C# boilerplate - could not find migrations output directory: ${migrationsOutputDir}`);
130
+ }
131
+ (0, generalUtils_js_1.log)(`Checking for generated C# file 📄XXXX_${migrationName}.cs in directory 📁${migrationsOutputDir}`);
132
+ const filenamePattern = `_${migrationName}.cs`;
133
+ const filenames = node_fs_1.default.readdirSync(migrationsOutputDir).filter(filename => filename.endsWith(filenamePattern));
134
+ if (!filenames || filenames.length === 0) {
135
+ throw new Error(`Auto-generated migration file not found - migrations output directory has no C# files ending with : ${filenamePattern}`);
136
+ }
137
+ if (filenames.length > 1) {
138
+ throw new Error(`Auto-generated migration file not found - migrations output directory has multiple C# files with the same migration name: ${filenames.join(', ')}`);
139
+ }
140
+ const filename = filenames[0];
141
+ const filePath = node_path_1.default.join(migrationsOutputDir, filename);
142
+ if (!node_fs_1.default.existsSync(filePath)) {
143
+ throw new Error(`Issue generating file path for migration (bad file path): ${filePath}`);
144
+ }
145
+ return filePath;
146
+ }
147
+ async function addDbMigrationBoilerplate(projectDirectory, dbContextName, migrationName) {
148
+ const filePath = await getCSharpMigrationFilePath(projectDirectory, dbContextName, migrationName);
149
+ (0, generalUtils_js_1.log)(`Replacing file contents with boilerplate for file 📄${filePath}`);
150
+ const newFileContents = cSharpMigrationFileTemplate
151
+ .replaceAll(contextNamePlaceholder, dbContextName)
152
+ .replaceAll(migrationNamePlaceholder, migrationName);
153
+ await promises_1.default.writeFile(filePath, newFileContents, { encoding: 'utf8' });
154
+ (0, generalUtils_js_1.log)(`Updated file with boilerplate - please ensure it is correct: 📄${filePath}`);
155
+ const upScriptPath = node_path_1.default.join(projectDirectory, `Scripts/${migrationName}.sql`);
156
+ const downScriptPath = node_path_1.default.join(projectDirectory, `Scripts/${migrationName}_Down.sql`);
157
+ (0, generalUtils_js_1.log)('\nCreating corresponding empty sql files (no action will be taken if they already exist):');
158
+ (0, generalUtils_js_1.log)(` - 📄${upScriptPath}`);
159
+ (0, generalUtils_js_1.log)(` - 📄${downScriptPath}\n`);
160
+ await writeEmptySqlFileIfNotExists(upScriptPath, 'Up');
161
+ await writeEmptySqlFileIfNotExists(downScriptPath, 'Down');
162
+ }
163
+ async function writeEmptySqlFileIfNotExists(scriptPath, scriptType) {
164
+ if (!node_fs_1.default.existsSync(scriptPath)) {
165
+ await promises_1.default.writeFile(scriptPath, '', { encoding: 'utf8' });
166
+ }
167
+ else {
168
+ (0, generalUtils_js_1.log)(`Skipping ${scriptType} sql script (already exists)`);
169
+ }
170
+ }
171
+ const contextNamePlaceholder = '{{context_name}}';
172
+ const migrationNamePlaceholder = '{{migration_name}}';
173
+ const cSharpMigrationFileTemplate = `using Microsoft.EntityFrameworkCore.Migrations;
174
+ using MikeyT.DbMigrations;
175
+
176
+ #nullable disable
177
+
178
+ namespace DbMigrator.Migrations.${contextNamePlaceholder}Migrations
179
+ {
180
+ public partial class ${migrationNamePlaceholder} : Migration
181
+ {
182
+ protected override void Up(MigrationBuilder migrationBuilder)
183
+ {
184
+ MigrationScriptRunner.RunScript(migrationBuilder, "${migrationNamePlaceholder}.sql");
185
+ }
186
+
187
+ protected override void Down(MigrationBuilder migrationBuilder)
188
+ {
189
+ MigrationScriptRunner.RunScript(migrationBuilder, "${migrationNamePlaceholder}_Down.sql");
190
+ }
191
+ }
192
+ }
193
+
194
+ `;
195
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGJNaWdyYXRpb25VdGlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYk1pZ3JhdGlvblV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBLHVEQUE0RztBQUM1RywwREFBNEI7QUFDNUIsc0RBQXdCO0FBQ3hCLGdFQUFrQztBQUVsQzs7Ozs7Ozs7R0FRRztBQUNJLEtBQUssVUFBVSxlQUFlLENBQUMsV0FBbUIsRUFBRSxhQUFxQixFQUFFLElBQWMsRUFBRSxPQUFPLEdBQUcsSUFBSTtJQUM5RyxJQUFBLGtDQUFnQixFQUFDLGFBQWEsRUFBRSxXQUFXLENBQUMsQ0FBQTtJQUM1QyxJQUFBLCtCQUFhLEVBQUMsZUFBZSxFQUFFLGFBQWEsQ0FBQyxDQUFBO0lBQzdDLE1BQU0sTUFBTSxHQUFHLE1BQU0sSUFBQSw0QkFBVSxFQUFDLFFBQVEsRUFBRSxDQUFDLElBQUksRUFBRSxXQUFXLEVBQUUsV0FBVyxFQUFFLEdBQUcsSUFBSSxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsR0FBRyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxZQUFZLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFBO0lBQ3BKLE9BQU8sTUFBTSxDQUFDLElBQUksQ0FBQTtBQUNwQixDQUFDO0FBTEQsMENBS0M7QUFFRDs7OztHQUlHO0FBQ0ksS0FBSyxVQUFVLGdCQUFnQixDQUFDLFdBQW1CLEVBQUUsYUFBcUI7SUFDL0UsTUFBTSxlQUFlLENBQUMsV0FBVyxFQUFFLGFBQWEsRUFBRSxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsQ0FBRSxDQUFBO0FBQzVFLENBQUM7QUFGRCw0Q0FFQztBQUVEOzs7OztHQUtHO0FBQ0ksS0FBSyxVQUFVLGtCQUFrQixDQUFDLFdBQW1CLEVBQUUsYUFBcUIsRUFBRSxhQUFzQjtJQUN6RyxNQUFNLGVBQWUsQ0FBQyxXQUFXLEVBQUUsYUFBYSxFQUFFLENBQUMsVUFB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
@@ -0,0 +1,25 @@
1
+ /**
2
+ * Runs dotnet build on the specified project.
3
+ * @param projectPath Path to project file (like .csproj) or directory of project to build
4
+ * @throws A {@link SpawnError} if the spawned process exits with a non-zero exit code
5
+ */
6
+ export declare function dotnetBuild(projectPath: string): Promise<void>;
7
+ /**
8
+ * Helper method to spawn a process and run 'dotnet publish'.
9
+ * @param projectPath Path to project file (like .csproj) or directory of project to build
10
+ * @param configuration Build configuration, such as 'Release'
11
+ * @param outputDir The relative or absolute path for the build output
12
+ * @param cwd Optionally run the command from another current working directory
13
+ */
14
+ export declare function dotnetPublish(projectPath?: string, configuration?: string, outputDir?: string, cwd?: string): Promise<void>;
15
+ /**
16
+ * Spawns a process that runs the necessary commands to install or update the dotnet-ef tool globally on the system.
17
+ */
18
+ export declare function installOrUpdateDotnetEfTool(): Promise<void>;
19
+ /**
20
+ * Spawns a process that runs the following commands to clean and re-install the dotnet dev certs:
21
+ * - dotnet dev-certs https --clean
22
+ * - dotnet dev-certs https -t
23
+ */
24
+ export declare function configureDotnetDevCerts(): Promise<void>;
25
+ //# sourceMappingURL=dotnetUtils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dotnetUtils.d.ts","sourceRoot":"","sources":["../../src/dotnetUtils.ts"],"names":[],"mappings":"AAMA;;;;GAIG;AACH,wBAAsB,WAAW,CAAC,WAAW,EAAE,MAAM,iBAGpD;AAED;;;;;;GAMG;AACH,wBAAsB,aAAa,CAAC,WAAW,GAAE,MAAa,EAAE,aAAa,GAAE,MAAkB,EAAE,SAAS,GAAE,MAAkB,EAAE,GAAG,CAAC,EAAE,MAAM,iBAY7I;AAED;;GAEG;AACH,wBAAsB,2BAA2B,kBAShD;AAED;;;;GAIG;AACH,wBAAsB,uBAAuB,kBAG5C"}