@launch77/cli 1.5.0 → 1.6.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.
Files changed (79) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/dist/cli.js +16 -14
  3. package/dist/cli.js.map +1 -1
  4. package/dist/modules/app/commands/create-app.d.ts +1 -1
  5. package/dist/modules/app/commands/create-app.js +1 -1
  6. package/dist/modules/app/commands/delete-app.d.ts +1 -1
  7. package/dist/modules/app/commands/delete-app.js +1 -1
  8. package/dist/modules/app/index.d.ts +2 -3
  9. package/dist/modules/app/index.d.ts.map +1 -1
  10. package/dist/modules/app/index.js +2 -3
  11. package/dist/modules/app/index.js.map +1 -1
  12. package/dist/modules/catalog/commands/generate.d.ts +1 -1
  13. package/dist/modules/catalog/commands/generate.d.ts.map +1 -1
  14. package/dist/modules/catalog/commands/generate.js +1 -1
  15. package/dist/modules/catalog/commands/generate.js.map +1 -1
  16. package/dist/modules/catalog/commands/scan.d.ts +1 -1
  17. package/dist/modules/catalog/commands/scan.d.ts.map +1 -1
  18. package/dist/modules/catalog/commands/scan.js +1 -1
  19. package/dist/modules/catalog/commands/scan.js.map +1 -1
  20. package/dist/modules/library/commands/delete-library.d.ts +3 -0
  21. package/dist/modules/library/commands/delete-library.d.ts.map +1 -0
  22. package/dist/modules/library/commands/delete-library.js +111 -0
  23. package/dist/modules/library/commands/delete-library.js.map +1 -0
  24. package/dist/modules/library/errors/library-errors.d.ts +7 -0
  25. package/dist/modules/library/errors/library-errors.d.ts.map +1 -0
  26. package/dist/modules/library/errors/library-errors.js +13 -0
  27. package/dist/modules/library/errors/library-errors.js.map +1 -0
  28. package/dist/modules/library/index.d.ts +5 -0
  29. package/dist/modules/library/index.d.ts.map +1 -0
  30. package/dist/modules/library/index.js +7 -0
  31. package/dist/modules/library/index.js.map +1 -0
  32. package/dist/modules/library/services/library-svc.d.ts +9 -0
  33. package/dist/modules/library/services/library-svc.d.ts.map +1 -0
  34. package/dist/modules/library/services/library-svc.js +38 -0
  35. package/dist/modules/library/services/library-svc.js.map +1 -0
  36. package/dist/modules/library/types/library-types.d.ts +7 -0
  37. package/dist/modules/library/types/library-types.d.ts.map +1 -0
  38. package/dist/modules/library/types/library-types.js +2 -0
  39. package/dist/modules/library/types/library-types.js.map +1 -0
  40. package/dist/modules/plugin/commands/delete-plugin.d.ts +3 -0
  41. package/dist/modules/plugin/commands/delete-plugin.d.ts.map +1 -0
  42. package/dist/modules/plugin/commands/delete-plugin.js +113 -0
  43. package/dist/modules/plugin/commands/delete-plugin.js.map +1 -0
  44. package/dist/modules/plugin/errors/plugin-errors.d.ts +12 -0
  45. package/dist/modules/plugin/errors/plugin-errors.d.ts.map +1 -1
  46. package/dist/modules/plugin/errors/plugin-errors.js +18 -0
  47. package/dist/modules/plugin/errors/plugin-errors.js.map +1 -1
  48. package/dist/modules/plugin/index.d.ts +1 -0
  49. package/dist/modules/plugin/index.d.ts.map +1 -1
  50. package/dist/modules/plugin/index.js +1 -0
  51. package/dist/modules/plugin/index.js.map +1 -1
  52. package/dist/modules/plugin/services/plugin-svc.d.ts +5 -1
  53. package/dist/modules/plugin/services/plugin-svc.d.ts.map +1 -1
  54. package/dist/modules/plugin/services/plugin-svc.js +34 -1
  55. package/dist/modules/plugin/services/plugin-svc.js.map +1 -1
  56. package/dist/modules/plugin/types/plugin-types.d.ts +6 -0
  57. package/dist/modules/plugin/types/plugin-types.d.ts.map +1 -1
  58. package/package.json +1 -1
  59. package/src/cli.ts +16 -14
  60. package/src/modules/app/commands/create-app.ts +1 -1
  61. package/src/modules/app/commands/delete-app.ts +1 -1
  62. package/src/modules/app/index.ts +2 -3
  63. package/src/modules/catalog/commands/generate.ts +1 -1
  64. package/src/modules/catalog/commands/scan.ts +1 -1
  65. package/src/modules/library/commands/delete-library.ts +126 -0
  66. package/src/modules/library/errors/library-errors.ts +13 -0
  67. package/src/modules/library/index.ts +11 -0
  68. package/src/modules/library/services/library-svc.ts +50 -0
  69. package/src/modules/library/types/library-types.ts +7 -0
  70. package/src/modules/plugin/commands/delete-plugin.ts +128 -0
  71. package/src/modules/plugin/errors/plugin-errors.ts +20 -0
  72. package/src/modules/plugin/index.ts +1 -0
  73. package/src/modules/plugin/services/plugin-svc.ts +43 -2
  74. package/src/modules/plugin/types/plugin-types.ts +8 -0
  75. package/dist/modules/app/commands/validate-manifest.d.ts +0 -3
  76. package/dist/modules/app/commands/validate-manifest.d.ts.map +0 -1
  77. package/dist/modules/app/commands/validate-manifest.js +0 -67
  78. package/dist/modules/app/commands/validate-manifest.js.map +0 -1
  79. package/src/modules/app/commands/validate-manifest.ts +0 -77
@@ -0,0 +1,128 @@
1
+ /* eslint-disable no-console */
2
+ import * as path from 'path'
3
+
4
+ import chalk from 'chalk'
5
+ import { Command } from 'commander'
6
+ import inquirer from 'inquirer'
7
+ import ora from 'ora'
8
+
9
+ import { detectLaunch77Context } from '@launch77/plugin-runtime'
10
+ import { PluginService } from '../services/plugin-svc.js'
11
+
12
+ export function pluginDeleteCommand(): Command {
13
+ const command = new Command('plugin:delete')
14
+ .argument('<plugin-name>', 'Name of the plugin to delete')
15
+ .description('Delete a plugin from the workspace (requires confirmation)')
16
+ .action(async (pluginName: string) => {
17
+ try {
18
+ console.log(chalk.yellow(`\n⚠️ WARNING: You are about to delete the plugin '${pluginName}'\n`))
19
+
20
+ // Detect context
21
+ const context = await detectLaunch77Context(process.cwd())
22
+
23
+ // Create plugin service
24
+ const pluginService = new PluginService()
25
+
26
+ // Get plugin path for display (before deletion)
27
+ const pluginPath = path.join(context.workspaceRoot, 'plugins', pluginName)
28
+
29
+ // Display what will be deleted
30
+ console.log(chalk.red('Plugin location:'), pluginPath)
31
+ console.log(chalk.red('\nThis will permanently delete:'))
32
+ console.log(chalk.gray(' • All source code and configuration files'))
33
+ console.log(chalk.gray(' • All node_modules (will be cleaned from monorepo)'))
34
+ console.log(chalk.gray(' • All build artifacts'))
35
+ console.log(chalk.gray(' • Plugin templates and generators'))
36
+
37
+ console.log(chalk.yellow('\n⚠️ Important notes:'))
38
+ console.log(chalk.gray(' • Check if plugin is installed in other packages'))
39
+ console.log(chalk.gray(' • If published to npm, you may want to deprecate/unpublish it manually'))
40
+ console.log(chalk.gray(' • Remove plugin from launch77.installedPlugins in dependent packages manually'))
41
+
42
+ console.log(chalk.red('\n⚠️ THIS ACTION CANNOT BE UNDONE!\n'))
43
+
44
+ // First confirmation: Type plugin name
45
+ const { confirmName } = await inquirer.prompt([
46
+ {
47
+ type: 'input',
48
+ name: 'confirmName',
49
+ message: `Type the plugin name to confirm:`,
50
+ validate: (input: string) => {
51
+ if (input === pluginName) {
52
+ return true
53
+ }
54
+ return `Please type '${pluginName}' exactly to confirm`
55
+ },
56
+ },
57
+ ])
58
+
59
+ if (confirmName !== pluginName) {
60
+ console.log(chalk.gray('\n✖ Deletion cancelled\n'))
61
+ return
62
+ }
63
+
64
+ // Second confirmation: Type DELETE
65
+ const { confirmDelete } = await inquirer.prompt([
66
+ {
67
+ type: 'input',
68
+ name: 'confirmDelete',
69
+ message: 'Type DELETE (all caps) to confirm permanent deletion:',
70
+ validate: (input: string) => {
71
+ if (input === 'DELETE') {
72
+ return true
73
+ }
74
+ return 'Please type DELETE (all caps) to confirm'
75
+ },
76
+ },
77
+ ])
78
+
79
+ if (confirmDelete !== 'DELETE') {
80
+ console.log(chalk.gray('\n✖ Deletion cancelled\n'))
81
+ return
82
+ }
83
+
84
+ // Delete the plugin
85
+ const deleteSpinner = ora('Deleting plugin directory...').start()
86
+ try {
87
+ await pluginService.deletePlugin({ pluginName }, context)
88
+ deleteSpinner.succeed('Plugin directory deleted')
89
+ } catch (error) {
90
+ deleteSpinner.fail('Failed to delete plugin directory')
91
+ throw error
92
+ }
93
+
94
+ // Update dependencies
95
+ const installSpinner = ora('Updating workspace dependencies...').start()
96
+ try {
97
+ // This is already done by plugin service, just show spinner for consistency
98
+ installSpinner.succeed('Dependencies updated (package-lock.json cleaned)')
99
+ } catch (error) {
100
+ installSpinner.fail('Failed to update dependencies')
101
+ console.log(chalk.yellow('\n⚠️ You may need to run "npm install" manually at the workspace root\n'))
102
+ }
103
+
104
+ // Success message
105
+ console.log(chalk.green(`\n✅ Plugin '${pluginName}' has been deleted\n`))
106
+
107
+ // Reminder about cleanup
108
+ console.log(chalk.yellow('📝 Remember to:'))
109
+ console.log(chalk.gray(' • Check packages that had this plugin installed'))
110
+ console.log(chalk.gray(' • Remove plugin from launch77.installedPlugins in package.json files'))
111
+ console.log(chalk.gray(' • Remove any plugin-generated files if needed'))
112
+ console.log(chalk.gray(' • Consider deprecating/unpublishing from npm if published'))
113
+ console.log()
114
+
115
+ // Reminder about Git
116
+ console.log(chalk.yellow('📝 Remember to commit the deletion:'))
117
+ console.log(chalk.gray(' git add -A'))
118
+ console.log(chalk.gray(` git commit -m "Remove ${pluginName} plugin"`))
119
+ console.log()
120
+ } catch (error) {
121
+ const message = error instanceof Error ? error.message : String(error)
122
+ console.error(chalk.red('Error:'), message)
123
+ process.exit(1)
124
+ }
125
+ })
126
+
127
+ return command
128
+ }
@@ -123,3 +123,23 @@ ${cause ? `\nOriginal error: ${cause.message}` : ''}`)
123
123
  this.name = 'NpmInstallationError'
124
124
  }
125
125
  }
126
+
127
+ /**
128
+ * Error when plugin directory is not found for deletion
129
+ */
130
+ export class PluginDirectoryNotFoundError extends Error {
131
+ constructor(pluginName: string, expectedPath: string) {
132
+ super(`Plugin '${pluginName}' does not exist.\n\n` + `Expected location: ${expectedPath}\n\n` + `Available plugins:\n` + ` cd plugins/\n` + ` ls`)
133
+ this.name = 'PluginDirectoryNotFoundError'
134
+ }
135
+ }
136
+
137
+ /**
138
+ * Error when plugin name validation fails
139
+ */
140
+ export class InvalidPluginNameError extends Error {
141
+ constructor(message: string) {
142
+ super(message)
143
+ this.name = 'InvalidPluginNameError'
144
+ }
145
+ }
@@ -11,6 +11,7 @@ export { PluginNotFoundError, InvalidPluginContextError, PluginInstallationError
11
11
  // Commands
12
12
  export { pluginInstallCommand } from './commands/plugin-install.js'
13
13
  export { pluginCreateCommand } from './commands/plugin-create.js'
14
+ export { pluginDeleteCommand } from './commands/delete-plugin.js'
14
15
 
15
16
  // Utilities
16
17
  export { listAvailablePlugins } from './lib/plugin-registry.js'
@@ -3,14 +3,18 @@ import * as fs from 'fs/promises'
3
3
 
4
4
  import chalk from 'chalk'
5
5
  import { execa } from 'execa'
6
+ import fsExtra from 'fs-extra'
6
7
  import { readPluginMetadata } from '@launch77/plugin-runtime'
7
8
 
8
9
  import { downloadNpmPackage } from '../../../infrastructure/npm-package.js'
9
- import { PluginInstallationError, InvalidPluginContextError, createInvalidContextError, MissingPluginTargetsError, createInvalidTargetError, NpmInstallationError, PluginResolutionError } from '../errors/plugin-errors.js'
10
+ import * as npm from '../../../infrastructure/npm.js'
11
+ import { validateWorkspaceContext } from '../../../utils/launch77-validation.js'
12
+ import { validateAppName } from '../../../utils/validation.js'
13
+ import { PluginInstallationError, InvalidPluginContextError, createInvalidContextError, MissingPluginTargetsError, createInvalidTargetError, NpmInstallationError, PluginResolutionError, PluginDirectoryNotFoundError, InvalidPluginNameError } from '../errors/plugin-errors.js'
10
14
  import { PluginResolver } from '../lib/plugin-resolver.js'
11
15
 
12
16
  import type { Launch77Context, Launch77LocationType, Launch77PackageManifest, InstalledPluginMetadata, PluginMetadata } from '@launch77/plugin-runtime'
13
- import type { InstallPluginRequest, InstallPluginResult } from '../types/plugin-types.js'
17
+ import type { InstallPluginRequest, InstallPluginResult, DeletePluginRequest, DeletePluginResult } from '../types/plugin-types.js'
14
18
 
15
19
  /**
16
20
  * Map location type to target string
@@ -259,4 +263,41 @@ export class PluginService {
259
263
  throw new PluginInstallationError(`Failed to write plugin manifest: ${error instanceof Error ? error.message : String(error)}`, error instanceof Error ? error : undefined)
260
264
  }
261
265
  }
266
+
267
+ /**
268
+ * Delete a plugin from the workspace
269
+ */
270
+ async deletePlugin(request: DeletePluginRequest, context: Launch77Context): Promise<DeletePluginResult> {
271
+ const { pluginName } = request
272
+
273
+ // 1. Validate plugin name (reusing app name validation since format is the same)
274
+ const nameValidation = validateAppName(pluginName)
275
+ if (!nameValidation.valid) {
276
+ throw new InvalidPluginNameError(nameValidation.errorMessage || 'Invalid plugin name')
277
+ }
278
+
279
+ // 2. Validate workspace context
280
+ const contextValidation = validateWorkspaceContext(context)
281
+ if (!contextValidation.valid) {
282
+ throw new Error(contextValidation.errorMessage || 'Invalid workspace context')
283
+ }
284
+
285
+ // 3. Determine plugin path
286
+ const pluginPath = path.join(context.workspaceRoot, 'plugins', pluginName)
287
+
288
+ // 4. Check if plugin exists
289
+ if (!(await fsExtra.pathExists(pluginPath))) {
290
+ throw new PluginDirectoryNotFoundError(pluginName, pluginPath)
291
+ }
292
+
293
+ // 5. Delete the plugin directory
294
+ await fsExtra.remove(pluginPath)
295
+
296
+ // 6. Update dependencies
297
+ await npm.install(context.workspaceRoot)
298
+
299
+ return {
300
+ pluginName,
301
+ }
302
+ }
262
303
  }
@@ -19,3 +19,11 @@ export interface HookResult {
19
19
  templateVariables?: Record<string, string>
20
20
  hookContext?: Record<string, string>
21
21
  }
22
+
23
+ export interface DeletePluginRequest {
24
+ pluginName: string
25
+ }
26
+
27
+ export interface DeletePluginResult {
28
+ pluginName: string
29
+ }
@@ -1,3 +0,0 @@
1
- import { Command } from 'commander';
2
- export declare function validateManifestCommand(): Command;
3
- //# sourceMappingURL=validate-manifest.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"validate-manifest.d.ts","sourceRoot":"","sources":["../../../../src/modules/app/commands/validate-manifest.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AAInC,wBAAgB,uBAAuB,IAAI,OAAO,CAoEjD"}
@@ -1,67 +0,0 @@
1
- /* eslint-disable no-console */
2
- import * as path from 'path';
3
- import chalk from 'chalk';
4
- import { Command } from 'commander';
5
- import { ManifestService } from '../services/manifest-svc.js';
6
- export function validateManifestCommand() {
7
- const command = new Command('app:validate-manifest')
8
- .description('Validate a Launch77 manifest file')
9
- .option('--path <folder>', 'Path to search for manifest (defaults to current directory)')
10
- .action(async (options) => {
11
- try {
12
- const searchPath = options.path ? path.resolve(options.path) : process.cwd();
13
- console.log(chalk.blue('🔍 Searching for Launch77 manifest (.launch/app.json)...\n'));
14
- const manifestService = new ManifestService();
15
- const { manifest, manifestPath, appRoot } = await manifestService.loadManifest(searchPath);
16
- console.log(chalk.green('✅ Valid manifest found!\n'));
17
- console.log(chalk.gray('Path:'), manifestPath);
18
- console.log(chalk.gray('App root:'), appRoot);
19
- console.log();
20
- console.log(chalk.cyan('Manifest details:'));
21
- console.log(chalk.gray(' Schema version:'), manifest.schemaVersion);
22
- console.log(chalk.gray(' Package:'), manifest.package);
23
- if ('name' in manifest && manifest.name) {
24
- console.log(chalk.gray(' Name:'), manifest.name);
25
- }
26
- // Check if deployment field exists
27
- if (manifest.deployment) {
28
- console.log(chalk.gray(' Deployment platform:'), manifest.deployment.platform);
29
- if (manifest.deployment.connected) {
30
- console.log(chalk.gray(' Connected:'), 'Yes');
31
- if (manifest.deployment.projectName) {
32
- console.log(chalk.gray(' Project:'), manifest.deployment.projectName);
33
- }
34
- }
35
- }
36
- console.log();
37
- console.log(chalk.green('✨ Manifest validation successful!'));
38
- // Exit with success code
39
- process.exit(0);
40
- }
41
- catch (error) {
42
- console.error(chalk.red('❌ Manifest validation failed:\n'));
43
- if (error instanceof Error) {
44
- // If it's our formatted error with validation issues, print it as-is
45
- if (error.message.includes('Manifest validation failed:')) {
46
- console.error(error.message);
47
- }
48
- else {
49
- console.error(chalk.red(error.message));
50
- }
51
- }
52
- else {
53
- console.error(chalk.red('Unknown error occurred'));
54
- }
55
- console.log();
56
- console.log(chalk.gray('To create a new manifest, use:'));
57
- console.log(chalk.cyan(' launch77 create-app <type> <name>'));
58
- console.log();
59
- console.log(chalk.gray('Or create one manually at .launch/app.json following the schema at:'));
60
- console.log(chalk.cyan(' launch77/cli/schemas/app.launch77.schema.json'));
61
- // Exit with error code
62
- process.exit(1);
63
- }
64
- });
65
- return command;
66
- }
67
- //# sourceMappingURL=validate-manifest.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"validate-manifest.js","sourceRoot":"","sources":["../../../../src/modules/app/commands/validate-manifest.ts"],"names":[],"mappings":"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"}
@@ -1,77 +0,0 @@
1
- /* eslint-disable no-console */
2
- import * as path from 'path'
3
-
4
- import chalk from 'chalk'
5
- import { Command } from 'commander'
6
-
7
- import { ManifestService } from '../services/manifest-svc.js'
8
-
9
- export function validateManifestCommand(): Command {
10
- const command = new Command('app:validate-manifest')
11
- .description('Validate a Launch77 manifest file')
12
- .option('--path <folder>', 'Path to search for manifest (defaults to current directory)')
13
- .action(async (options) => {
14
- try {
15
- const searchPath = options.path ? path.resolve(options.path) : process.cwd()
16
-
17
- console.log(chalk.blue('🔍 Searching for Launch77 manifest (.launch/app.json)...\n'))
18
-
19
- const manifestService = new ManifestService()
20
- const { manifest, manifestPath, appRoot } = await manifestService.loadManifest(searchPath)
21
-
22
- console.log(chalk.green('✅ Valid manifest found!\n'))
23
- console.log(chalk.gray('Path:'), manifestPath)
24
- console.log(chalk.gray('App root:'), appRoot)
25
- console.log()
26
- console.log(chalk.cyan('Manifest details:'))
27
- console.log(chalk.gray(' Schema version:'), manifest.schemaVersion)
28
- console.log(chalk.gray(' Package:'), manifest.package)
29
-
30
- if ('name' in manifest && manifest.name) {
31
- console.log(chalk.gray(' Name:'), manifest.name)
32
- }
33
-
34
- // Check if deployment field exists
35
- if (manifest.deployment) {
36
- console.log(chalk.gray(' Deployment platform:'), manifest.deployment.platform)
37
- if (manifest.deployment.connected) {
38
- console.log(chalk.gray(' Connected:'), 'Yes')
39
- if (manifest.deployment.projectName) {
40
- console.log(chalk.gray(' Project:'), manifest.deployment.projectName)
41
- }
42
- }
43
- }
44
-
45
- console.log()
46
- console.log(chalk.green('✨ Manifest validation successful!'))
47
-
48
- // Exit with success code
49
- process.exit(0)
50
- } catch (error) {
51
- console.error(chalk.red('❌ Manifest validation failed:\n'))
52
-
53
- if (error instanceof Error) {
54
- // If it's our formatted error with validation issues, print it as-is
55
- if (error.message.includes('Manifest validation failed:')) {
56
- console.error(error.message)
57
- } else {
58
- console.error(chalk.red(error.message))
59
- }
60
- } else {
61
- console.error(chalk.red('Unknown error occurred'))
62
- }
63
-
64
- console.log()
65
- console.log(chalk.gray('To create a new manifest, use:'))
66
- console.log(chalk.cyan(' launch77 create-app <type> <name>'))
67
- console.log()
68
- console.log(chalk.gray('Or create one manually at .launch/app.json following the schema at:'))
69
- console.log(chalk.cyan(' launch77/cli/schemas/app.launch77.schema.json'))
70
-
71
- // Exit with error code
72
- process.exit(1)
73
- }
74
- })
75
-
76
- return command
77
- }