@pipeline-builder/pipeline-manager 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (110) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +74 -0
  3. package/cdk.json +91 -0
  4. package/config.yml +94 -0
  5. package/dist/boilerplate.d.ts +3 -0
  6. package/dist/boilerplate.d.ts.map +1 -0
  7. package/dist/boilerplate.js +58 -0
  8. package/dist/cdk.json +91 -0
  9. package/dist/cli.d.ts +62 -0
  10. package/dist/cli.d.ts.map +1 -0
  11. package/dist/cli.js +372 -0
  12. package/dist/commands/bootstrap.d.ts +11 -0
  13. package/dist/commands/bootstrap.d.ts.map +1 -0
  14. package/dist/commands/bootstrap.js +159 -0
  15. package/dist/commands/create-pipeline.d.ts +12 -0
  16. package/dist/commands/create-pipeline.d.ts.map +1 -0
  17. package/dist/commands/create-pipeline.js +291 -0
  18. package/dist/commands/deploy.d.ts +15 -0
  19. package/dist/commands/deploy.d.ts.map +1 -0
  20. package/dist/commands/deploy.js +167 -0
  21. package/dist/commands/get-pipeline.d.ts +13 -0
  22. package/dist/commands/get-pipeline.d.ts.map +1 -0
  23. package/dist/commands/get-pipeline.js +97 -0
  24. package/dist/commands/get-plugin.d.ts +13 -0
  25. package/dist/commands/get-plugin.d.ts.map +1 -0
  26. package/dist/commands/get-plugin.js +98 -0
  27. package/dist/commands/list-pipelines.d.ts +20 -0
  28. package/dist/commands/list-pipelines.d.ts.map +1 -0
  29. package/dist/commands/list-pipelines.js +172 -0
  30. package/dist/commands/list-plugins.d.ts +20 -0
  31. package/dist/commands/list-plugins.d.ts.map +1 -0
  32. package/dist/commands/list-plugins.js +167 -0
  33. package/dist/commands/login.d.ts +21 -0
  34. package/dist/commands/login.d.ts.map +1 -0
  35. package/dist/commands/login.js +179 -0
  36. package/dist/commands/setup-events.d.ts +15 -0
  37. package/dist/commands/setup-events.d.ts.map +1 -0
  38. package/dist/commands/setup-events.js +177 -0
  39. package/dist/commands/status.d.ts +11 -0
  40. package/dist/commands/status.d.ts.map +1 -0
  41. package/dist/commands/status.js +89 -0
  42. package/dist/commands/store-token.d.ts +20 -0
  43. package/dist/commands/store-token.d.ts.map +1 -0
  44. package/dist/commands/store-token.js +233 -0
  45. package/dist/commands/synth.d.ts +21 -0
  46. package/dist/commands/synth.d.ts.map +1 -0
  47. package/dist/commands/synth.js +143 -0
  48. package/dist/commands/upload-plugin.d.ts +21 -0
  49. package/dist/commands/upload-plugin.d.ts.map +1 -0
  50. package/dist/commands/upload-plugin.js +311 -0
  51. package/dist/commands/version.d.ts +12 -0
  52. package/dist/commands/version.d.ts.map +1 -0
  53. package/dist/commands/version.js +223 -0
  54. package/dist/config/cli.constants.d.ts +101 -0
  55. package/dist/config/cli.constants.d.ts.map +1 -0
  56. package/dist/config/cli.constants.js +165 -0
  57. package/dist/config.yml +94 -0
  58. package/dist/templates/events-stack.json +141 -0
  59. package/dist/types/config.d.ts +44 -0
  60. package/dist/types/config.d.ts.map +1 -0
  61. package/dist/types/config.js +5 -0
  62. package/dist/types/error.d.ts +61 -0
  63. package/dist/types/error.d.ts.map +1 -0
  64. package/dist/types/error.js +39 -0
  65. package/dist/types/index.d.ts +8 -0
  66. package/dist/types/index.d.ts.map +1 -0
  67. package/dist/types/index.js +26 -0
  68. package/dist/types/pipeline.d.ts +144 -0
  69. package/dist/types/pipeline.d.ts.map +1 -0
  70. package/dist/types/pipeline.js +5 -0
  71. package/dist/types/plugin.d.ts +160 -0
  72. package/dist/types/plugin.d.ts.map +1 -0
  73. package/dist/types/plugin.js +5 -0
  74. package/dist/utils/api-client.d.ts +26 -0
  75. package/dist/utils/api-client.d.ts.map +1 -0
  76. package/dist/utils/api-client.js +160 -0
  77. package/dist/utils/audit-log.d.ts +8 -0
  78. package/dist/utils/audit-log.d.ts.map +1 -0
  79. package/dist/utils/audit-log.js +53 -0
  80. package/dist/utils/auth-guard.d.ts +16 -0
  81. package/dist/utils/auth-guard.d.ts.map +1 -0
  82. package/dist/utils/auth-guard.js +25 -0
  83. package/dist/utils/aws-secrets.d.ts +21 -0
  84. package/dist/utils/aws-secrets.d.ts.map +1 -0
  85. package/dist/utils/aws-secrets.js +74 -0
  86. package/dist/utils/banner.d.ts +19 -0
  87. package/dist/utils/banner.d.ts.map +1 -0
  88. package/dist/utils/banner.js +59 -0
  89. package/dist/utils/cdk-utils.d.ts +51 -0
  90. package/dist/utils/cdk-utils.d.ts.map +1 -0
  91. package/dist/utils/cdk-utils.js +101 -0
  92. package/dist/utils/command-utils.d.ts +56 -0
  93. package/dist/utils/command-utils.d.ts.map +1 -0
  94. package/dist/utils/command-utils.js +138 -0
  95. package/dist/utils/config-loader.d.ts +27 -0
  96. package/dist/utils/config-loader.d.ts.map +1 -0
  97. package/dist/utils/config-loader.js +166 -0
  98. package/dist/utils/error-handler.d.ts +29 -0
  99. package/dist/utils/error-handler.d.ts.map +1 -0
  100. package/dist/utils/error-handler.js +255 -0
  101. package/dist/utils/list-command-utils.d.ts +23 -0
  102. package/dist/utils/list-command-utils.d.ts.map +1 -0
  103. package/dist/utils/list-command-utils.js +60 -0
  104. package/dist/utils/output-utils.d.ts +60 -0
  105. package/dist/utils/output-utils.d.ts.map +1 -0
  106. package/dist/utils/output-utils.js +320 -0
  107. package/dist/utils/rate-limiter.d.ts +14 -0
  108. package/dist/utils/rate-limiter.d.ts.map +1 -0
  109. package/dist/utils/rate-limiter.js +73 -0
  110. package/package.json +144 -0
@@ -0,0 +1,311 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
5
+ if (k2 === undefined) k2 = k;
6
+ var desc = Object.getOwnPropertyDescriptor(m, k);
7
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
8
+ desc = { enumerable: true, get: function() { return m[k]; } };
9
+ }
10
+ Object.defineProperty(o, k2, desc);
11
+ }) : (function(o, m, k, k2) {
12
+ if (k2 === undefined) k2 = k;
13
+ o[k2] = m[k];
14
+ }));
15
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
16
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
17
+ }) : function(o, v) {
18
+ o["default"] = v;
19
+ });
20
+ var __importStar = (this && this.__importStar) || (function () {
21
+ var ownKeys = function(o) {
22
+ ownKeys = Object.getOwnPropertyNames || function (o) {
23
+ var ar = [];
24
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
25
+ return ar;
26
+ };
27
+ return ownKeys(o);
28
+ };
29
+ return function (mod) {
30
+ if (mod && mod.__esModule) return mod;
31
+ var result = {};
32
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
33
+ __setModuleDefault(result, mod);
34
+ return result;
35
+ };
36
+ })();
37
+ var __importDefault = (this && this.__importDefault) || function (mod) {
38
+ return (mod && mod.__esModule) ? mod : { "default": mod };
39
+ };
40
+ Object.defineProperty(exports, "__esModule", { value: true });
41
+ exports.uploadPlugin = uploadPlugin;
42
+ const fs = __importStar(require("fs"));
43
+ const path = __importStar(require("path"));
44
+ const form_data_1 = __importDefault(require("form-data"));
45
+ const ora_1 = __importDefault(require("ora"));
46
+ const picocolors_1 = __importDefault(require("picocolors"));
47
+ const cli_constants_1 = require("../config/cli.constants");
48
+ const command_utils_1 = require("../utils/command-utils");
49
+ const error_handler_1 = require("../utils/error-handler");
50
+ const output_utils_1 = require("../utils/output-utils");
51
+ const { bold, green } = picocolors_1.default;
52
+ /**
53
+ * Registers the `upload-plugin` command with the CLI program.
54
+ *
55
+ * Validates a local plugin ZIP file (size, extension, readability),
56
+ * builds a multipart form, and uploads it to the platform API.
57
+ * Plugin name and version can be auto-detected from the package
58
+ * or overridden via CLI flags.
59
+ *
60
+ * @param program - The root Commander program instance to attach the command to.
61
+ *
62
+ * @example
63
+ * ```bash
64
+ * cli upload-plugin --file plugin.zip --organization acme
65
+ * cli upload-plugin --file plugin.zip --organization acme --public
66
+ * cli upload-plugin --file plugin.zip --organization acme --name my-plugin --version 1.0.0
67
+ * cli upload-plugin --file plugin.zip --organization acme --no-verify-ssl
68
+ * ```
69
+ */
70
+ function uploadPlugin(program) {
71
+ program
72
+ .command('upload-plugin')
73
+ .description('Upload and deploy a plugin package')
74
+ .requiredOption('-f, --file <file>', 'Path to plugin ZIP file')
75
+ .requiredOption('-o, --organization <organization>', 'Organization name')
76
+ .option('-n, --name <name>', 'Plugin name (optional, extracted from package if not provided)')
77
+ .option('-v, --version <version>', 'Plugin version (optional, extracted from package if not provided)')
78
+ .option('--public', 'Make plugin publicly accessible', false)
79
+ .option('--active', 'Set plugin as active', true)
80
+ .option('--verify-ssl', 'Enable SSL certificate verification')
81
+ .option('--no-verify-ssl', 'Disable SSL certificate verification')
82
+ .option('--dry-run', 'Validate file without uploading', false)
83
+ .action(async (options) => {
84
+ const executionId = (0, command_utils_1.printCommandHeader)('Upload Plugin');
85
+ try {
86
+ // Display parameters
87
+ (0, output_utils_1.printInfo)('Upload parameters', {
88
+ file: options.file,
89
+ organization: options.organization,
90
+ name: options.name || '(auto-detect)',
91
+ version: options.version || '(auto-detect)',
92
+ public: options.public ? 'Yes' : 'No',
93
+ active: options.active ? 'Yes' : 'No',
94
+ dryRun: options.dryRun,
95
+ verifySsl: options.verifySsl,
96
+ });
97
+ // Security warning for SSL verification disabled
98
+ (0, command_utils_1.printSslWarning)(options.verifySsl);
99
+ // Validate organization
100
+ if (!options.organization || typeof options.organization !== 'string' || options.organization.trim().length === 0) {
101
+ (0, output_utils_1.printError)('Invalid organization name', { provided: options.organization });
102
+ throw new error_handler_1.ValidationError('Organization must be a non-empty string', 'organization', options.organization);
103
+ }
104
+ // Validate file path
105
+ if (!options.file || typeof options.file !== 'string' || options.file.trim().length === 0) {
106
+ (0, output_utils_1.printError)('Invalid file path', { provided: options.file });
107
+ throw new error_handler_1.ValidationError('File path must be a non-empty string', 'file', options.file);
108
+ }
109
+ const filePath = path.resolve(options.file);
110
+ // Validate file exists
111
+ (0, output_utils_1.printInfo)('Validating plugin file', { path: filePath });
112
+ if (!(0, output_utils_1.fileExists)(filePath)) {
113
+ (0, output_utils_1.printError)('Plugin file not found', { path: filePath });
114
+ throw new error_handler_1.ValidationError(`Plugin file not found: ${filePath}`, 'file', filePath);
115
+ }
116
+ // Validate file extension
117
+ const fileExt = path.extname(filePath).toLowerCase();
118
+ if (fileExt !== '.zip') {
119
+ (0, output_utils_1.printWarning)('File extension is not .zip', {
120
+ provided: fileExt,
121
+ expected: '.zip',
122
+ });
123
+ throw new error_handler_1.ValidationError('Plugin file must be a ZIP archive', 'file', filePath);
124
+ }
125
+ // Get file stats
126
+ const stats = fs.statSync(filePath);
127
+ const sizeBytes = stats.size;
128
+ const sizeFormatted = (0, cli_constants_1.formatFileSize)(sizeBytes);
129
+ (0, output_utils_1.printSuccess)('Plugin file found', {
130
+ path: filePath,
131
+ size: sizeFormatted,
132
+ modified: new Date(stats.mtime).toLocaleString(),
133
+ });
134
+ // Check file size
135
+ if (sizeBytes > cli_constants_1.FILE_SIZE_LIMITS.PLUGIN) {
136
+ (0, output_utils_1.printError)('Plugin file too large', {
137
+ size: sizeFormatted,
138
+ maximum: (0, cli_constants_1.formatFileSize)(cli_constants_1.FILE_SIZE_LIMITS.PLUGIN),
139
+ exceededBy: (0, cli_constants_1.formatFileSize)(sizeBytes - cli_constants_1.FILE_SIZE_LIMITS.PLUGIN),
140
+ });
141
+ throw new error_handler_1.ValidationError(`Plugin file exceeds maximum size of ${(0, cli_constants_1.formatFileSize)(cli_constants_1.FILE_SIZE_LIMITS.PLUGIN)} (actual: ${sizeFormatted})`, 'file.size', sizeBytes);
142
+ }
143
+ // Check if file is readable
144
+ try {
145
+ fs.accessSync(filePath, fs.constants.R_OK);
146
+ }
147
+ catch (error) {
148
+ (0, output_utils_1.printError)('Cannot read plugin file', {
149
+ path: filePath,
150
+ error: error instanceof Error ? error.message : String(error),
151
+ });
152
+ throw new error_handler_1.ValidationError('Plugin file is not readable', 'file', filePath);
153
+ }
154
+ (0, output_utils_1.printSuccess)('Plugin file validated');
155
+ // Validate version format if provided
156
+ if (options.version) {
157
+ const versionRegex = /^\d+\.\d+\.\d+(-[a-zA-Z0-9.-]+)?$/;
158
+ if (!versionRegex.test(options.version)) {
159
+ (0, output_utils_1.printWarning)('Version format may be invalid', {
160
+ provided: options.version,
161
+ expected: 'semantic version (e.g., 1.0.0, 1.2.3-beta.1)',
162
+ });
163
+ }
164
+ }
165
+ // Validate name format if provided
166
+ if (options.name) {
167
+ const nameRegex = /^[a-z0-9-]+$/;
168
+ if (!nameRegex.test(options.name)) {
169
+ (0, output_utils_1.printWarning)('Plugin name format may be invalid', {
170
+ provided: options.name,
171
+ expected: 'lowercase alphanumeric with dashes (e.g., my-plugin-name)',
172
+ });
173
+ }
174
+ }
175
+ // Dry run mode
176
+ if (options.dryRun) {
177
+ console.log('');
178
+ (0, output_utils_1.printSection)('Dry Run - Validation Complete');
179
+ (0, output_utils_1.printSuccess)('File validation passed - no upload performed');
180
+ (0, output_utils_1.printKeyValue)({
181
+ File: filePath,
182
+ Size: sizeFormatted,
183
+ Organization: options.organization,
184
+ Name: options.name || '(will be auto-detected)',
185
+ Version: options.version || '(will be auto-detected)',
186
+ Public: options.public ? 'Yes' : 'No',
187
+ Active: options.active ? 'Yes' : 'No',
188
+ });
189
+ return;
190
+ }
191
+ // Create authenticated API client
192
+ const client = (0, command_utils_1.createAuthenticatedClient)(options);
193
+ const config = client.getConfig();
194
+ // Create form data
195
+ console.log('');
196
+ (0, output_utils_1.printSection)('Uploading Plugin');
197
+ (0, output_utils_1.printInfo)('Preparing upload', {
198
+ file: path.basename(filePath),
199
+ size: sizeFormatted,
200
+ });
201
+ const formData = new form_data_1.default();
202
+ formData.append('plugin', fs.createReadStream(filePath), {
203
+ filename: path.basename(filePath),
204
+ contentType: 'application/zip',
205
+ });
206
+ formData.append('organization', options.organization);
207
+ if (options.name)
208
+ formData.append('name', options.name);
209
+ if (options.version)
210
+ formData.append('version', options.version);
211
+ formData.append('isPublic', options.public ? 'true' : 'false');
212
+ formData.append('isActive', options.active ? 'true' : 'false');
213
+ // Make API request
214
+ const endpoint = config.api.pluginUploadUrl;
215
+ (0, output_utils_1.printInfo)('Uploading to API', {
216
+ endpoint: `${config.api.baseUrl}${endpoint}`,
217
+ });
218
+ console.log('');
219
+ const spinner = (0, ora_1.default)('Uploading plugin...').start();
220
+ let rawResponse;
221
+ let duration;
222
+ try {
223
+ const startTime = Date.now();
224
+ rawResponse = await client.postForm(endpoint, formData);
225
+ duration = Date.now() - startTime;
226
+ spinner.succeed('Plugin uploaded');
227
+ }
228
+ catch (error) {
229
+ spinner.fail('Upload failed');
230
+ throw error;
231
+ }
232
+ const response = (0, output_utils_1.extractSingleResponse)(rawResponse, 'plugin', 'name');
233
+ if (!response) {
234
+ (0, output_utils_1.printError)('No valid plugin data in response', {
235
+ responseKeys: rawResponse ? Object.keys(rawResponse) : '(null)',
236
+ });
237
+ throw new Error('Upload failed - no valid plugin data received');
238
+ }
239
+ console.log('');
240
+ (0, output_utils_1.printSection)('Plugin Uploaded Successfully');
241
+ // Display plugin information
242
+ (0, output_utils_1.printKeyValue)({
243
+ 'Plugin ID': green(bold(response.id)),
244
+ 'Name': response.name,
245
+ 'Version': response.version,
246
+ 'Organization': response.organization,
247
+ 'Description': response.description || '(not set)',
248
+ });
249
+ console.log('');
250
+ (0, output_utils_1.printKeyValue)({
251
+ 'File URL': response.fileUrl || '(not available)',
252
+ 'File Size': response.fileSize ? `${(response.fileSize / 1024).toFixed(2)} KB` : '(not available)',
253
+ 'Checksum': response.checksum ? `${response.checksum.substring(0, 16)}...` : '(not available)',
254
+ });
255
+ console.log('');
256
+ (0, output_utils_1.printKeyValue)({
257
+ 'Public': response.isPublic ? 'Yes' : 'No',
258
+ 'Active': response.isActive ? 'Yes' : 'No',
259
+ 'Created At': response.createdAt || '(not available)',
260
+ 'Uploaded By': response.uploadedBy || '(not available)',
261
+ });
262
+ console.log('');
263
+ (0, output_utils_1.printKeyValue)({
264
+ 'Execution ID': executionId,
265
+ 'Upload Duration': `${(duration / 1000).toFixed(2)}s`,
266
+ 'Status': green('✓ Success'),
267
+ });
268
+ // Print additional information if available
269
+ if (response.metadata) {
270
+ console.log('');
271
+ (0, output_utils_1.printInfo)('Plugin metadata available', {
272
+ keys: Object.keys(response.metadata).length,
273
+ });
274
+ }
275
+ // Print deployment logs URL if available
276
+ const resp = response;
277
+ const requestId = resp['X-Request-Id'] || resp.requestId;
278
+ if (requestId) {
279
+ console.log('');
280
+ (0, output_utils_1.printInfo)('Deployment logs available', {
281
+ requestId,
282
+ url: `${config.api.baseUrl}/logs/${requestId}`,
283
+ });
284
+ }
285
+ // Next steps
286
+ console.log('');
287
+ (0, output_utils_1.printInfo)('Next steps', {
288
+ view: `Use "get-plugin --id ${response.id}" to view plugin details`,
289
+ list: 'Use "list-plugins" to see all plugins',
290
+ });
291
+ }
292
+ catch (error) {
293
+ (0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.API_REQUEST, {
294
+ debug: program.opts().debug,
295
+ exit: true,
296
+ context: {
297
+ command: 'upload-plugin',
298
+ executionId,
299
+ file: options.file,
300
+ organization: options.organization,
301
+ name: options.name,
302
+ version: options.version,
303
+ public: options.public,
304
+ active: options.active,
305
+ verifySsl: options.verifySsl,
306
+ },
307
+ });
308
+ }
309
+ });
310
+ }
311
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"upload-plugin.js","sourceRoot":"","sources":["../../src/commands/upload-plugin.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { Command } from 'commander';\nimport FormData from 'form-data';\nimport ora from 'ora';\nimport pico from 'picocolors';\nimport { FILE_SIZE_LIMITS, formatFileSize } from '../config/cli.constants';\nimport { Plugin, PluginResponse } from '../types';\nimport { printCommandHeader, printSslWarning, createAuthenticatedClient } from '../utils/command-utils';\nimport { ERROR_CODES, handleError, ValidationError } from '../utils/error-handler';\nimport { extractSingleResponse, fileExists, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\nconst { bold, green } = pico;\n\n/**\n * Registers the `upload-plugin` command with the CLI program.\n *\n * Validates a local plugin ZIP file (size, extension, readability),\n * builds a multipart form, and uploads it to the platform API.\n * Plugin name and version can be auto-detected from the package\n * or overridden via CLI flags.\n *\n * @param program - The root Commander program instance to attach the command to.\n *\n * @example\n * ```bash\n * cli upload-plugin --file plugin.zip --organization acme\n * cli upload-plugin --file plugin.zip --organization acme --public\n * cli upload-plugin --file plugin.zip --organization acme --name my-plugin --version 1.0.0\n * cli upload-plugin --file plugin.zip --organization acme --no-verify-ssl\n * ```\n */\nexport function uploadPlugin(program: Command): void {\n  program\n    .command('upload-plugin')\n    .description('Upload and deploy a plugin package')\n    .requiredOption('-f, --file <file>', 'Path to plugin ZIP file')\n    .requiredOption('-o, --organization <organization>', 'Organization name')\n    .option('-n, --name <name>', 'Plugin name (optional, extracted from package if not provided)')\n    .option('-v, --version <version>', 'Plugin version (optional, extracted from package if not provided)')\n    .option('--public', 'Make plugin publicly accessible', false)\n    .option('--active', 'Set plugin as active', true)\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .option('--dry-run', 'Validate file without uploading', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('Upload Plugin');\n\n      try {\n\n        // Display parameters\n        printInfo('Upload parameters', {\n          file: options.file,\n          organization: options.organization,\n          name: options.name || '(auto-detect)',\n          version: options.version || '(auto-detect)',\n          public: options.public ? 'Yes' : 'No',\n          active: options.active ? 'Yes' : 'No',\n          dryRun: options.dryRun,\n          verifySsl: options.verifySsl,\n        });\n\n        // Security warning for SSL verification disabled\n        printSslWarning(options.verifySsl);\n\n        // Validate organization\n        if (!options.organization || typeof options.organization !== 'string' || options.organization.trim().length === 0) {\n          printError('Invalid organization name', { provided: options.organization });\n          throw new ValidationError('Organization must be a non-empty string', 'organization', options.organization);\n        }\n\n        // Validate file path\n        if (!options.file || typeof options.file !== 'string' || options.file.trim().length === 0) {\n          printError('Invalid file path', { provided: options.file });\n          throw new ValidationError('File path must be a non-empty string', 'file', options.file);\n        }\n\n        const filePath = path.resolve(options.file);\n\n        // Validate file exists\n        printInfo('Validating plugin file', { path: filePath });\n\n        if (!fileExists(filePath)) {\n          printError('Plugin file not found', { path: filePath });\n          throw new ValidationError(`Plugin file not found: ${filePath}`, 'file', filePath);\n        }\n\n        // Validate file extension\n        const fileExt = path.extname(filePath).toLowerCase();\n        if (fileExt !== '.zip') {\n          printWarning('File extension is not .zip', {\n            provided: fileExt,\n            expected: '.zip',\n          });\n          throw new ValidationError('Plugin file must be a ZIP archive', 'file', filePath);\n        }\n\n        // Get file stats\n        const stats = fs.statSync(filePath);\n        const sizeBytes = stats.size;\n        const sizeFormatted = formatFileSize(sizeBytes);\n\n        printSuccess('Plugin file found', {\n          path: filePath,\n          size: sizeFormatted,\n          modified: new Date(stats.mtime).toLocaleString(),\n        });\n\n        // Check file size\n        if (sizeBytes > FILE_SIZE_LIMITS.PLUGIN) {\n          printError('Plugin file too large', {\n            size: sizeFormatted,\n            maximum: formatFileSize(FILE_SIZE_LIMITS.PLUGIN),\n            exceededBy: formatFileSize(sizeBytes - FILE_SIZE_LIMITS.PLUGIN),\n          });\n          throw new ValidationError(\n            `Plugin file exceeds maximum size of ${formatFileSize(FILE_SIZE_LIMITS.PLUGIN)} (actual: ${sizeFormatted})`,\n            'file.size',\n            sizeBytes,\n          );\n        }\n\n        // Check if file is readable\n        try {\n          fs.accessSync(filePath, fs.constants.R_OK);\n        } catch (error) {\n          printError('Cannot read plugin file', {\n            path: filePath,\n            error: error instanceof Error ? error.message : String(error),\n          });\n          throw new ValidationError('Plugin file is not readable', 'file', filePath);\n        }\n\n        printSuccess('Plugin file validated');\n\n        // Validate version format if provided\n        if (options.version) {\n          const versionRegex = /^\\d+\\.\\d+\\.\\d+(-[a-zA-Z0-9.-]+)?$/;\n          if (!versionRegex.test(options.version)) {\n            printWarning('Version format may be invalid', {\n              provided: options.version,\n              expected: 'semantic version (e.g., 1.0.0, 1.2.3-beta.1)',\n            });\n          }\n        }\n\n        // Validate name format if provided\n        if (options.name) {\n          const nameRegex = /^[a-z0-9-]+$/;\n          if (!nameRegex.test(options.name)) {\n            printWarning('Plugin name format may be invalid', {\n              provided: options.name,\n              expected: 'lowercase alphanumeric with dashes (e.g., my-plugin-name)',\n            });\n          }\n        }\n\n        // Dry run mode\n        if (options.dryRun) {\n          console.log('');\n          printSection('Dry Run - Validation Complete');\n          printSuccess('File validation passed - no upload performed');\n\n          printKeyValue({\n            File: filePath,\n            Size: sizeFormatted,\n            Organization: options.organization,\n            Name: options.name || '(will be auto-detected)',\n            Version: options.version || '(will be auto-detected)',\n            Public: options.public ? 'Yes' : 'No',\n            Active: options.active ? 'Yes' : 'No',\n          });\n\n          return;\n        }\n\n        // Create authenticated API client\n        const client = createAuthenticatedClient(options);\n        const config = client.getConfig();\n\n        // Create form data\n        console.log('');\n        printSection('Uploading Plugin');\n        printInfo('Preparing upload', {\n          file: path.basename(filePath),\n          size: sizeFormatted,\n        });\n\n        const formData = new FormData();\n        formData.append('plugin', fs.createReadStream(filePath), {\n          filename: path.basename(filePath),\n          contentType: 'application/zip',\n        });\n        formData.append('organization', options.organization);\n\n        if (options.name) formData.append('name', options.name);\n        if (options.version) formData.append('version', options.version);\n        formData.append('isPublic', options.public ? 'true' : 'false');\n        formData.append('isActive', options.active ? 'true' : 'false');\n\n        // Make API request\n        const endpoint = config.api.pluginUploadUrl;\n        printInfo('Uploading to API', {\n          endpoint: `${config.api.baseUrl}${endpoint}`,\n        });\n\n        console.log('');\n\n        const spinner = ora('Uploading plugin...').start();\n        let rawResponse: PluginResponse;\n        let duration: number;\n        try {\n          const startTime = Date.now();\n          rawResponse = await client.postForm<PluginResponse>(endpoint, formData);\n          duration = Date.now() - startTime;\n          spinner.succeed('Plugin uploaded');\n        } catch (error) {\n          spinner.fail('Upload failed');\n          throw error;\n        }\n\n        const response = extractSingleResponse<Plugin>(rawResponse, 'plugin', 'name');\n\n        if (!response) {\n          printError('No valid plugin data in response', {\n            responseKeys: rawResponse ? Object.keys(rawResponse) : '(null)',\n          });\n          throw new Error('Upload failed - no valid plugin data received');\n        }\n\n        console.log('');\n        printSection('Plugin Uploaded Successfully');\n\n        // Display plugin information\n        printKeyValue({\n          'Plugin ID': green(bold(response.id)),\n          'Name': response.name,\n          'Version': response.version,\n          'Organization': response.organization,\n          'Description': response.description || '(not set)',\n        });\n\n        console.log('');\n        printKeyValue({\n          'File URL': response.fileUrl || '(not available)',\n          'File Size': response.fileSize ? `${(response.fileSize / 1024).toFixed(2)} KB` : '(not available)',\n          'Checksum': response.checksum ? `${response.checksum.substring(0, 16)}...` : '(not available)',\n        });\n\n        console.log('');\n        printKeyValue({\n          'Public': response.isPublic ? 'Yes' : 'No',\n          'Active': response.isActive ? 'Yes' : 'No',\n          'Created At': response.createdAt || '(not available)',\n          'Uploaded By': response.uploadedBy || '(not available)',\n        });\n\n        console.log('');\n        printKeyValue({\n          'Execution ID': executionId,\n          'Upload Duration': `${(duration / 1000).toFixed(2)}s`,\n          'Status': green('✓ Success'),\n        });\n\n        // Print additional information if available\n        if (response.metadata) {\n          console.log('');\n          printInfo('Plugin metadata available', {\n            keys: Object.keys(response.metadata).length,\n          });\n        }\n\n        // Print deployment logs URL if available\n        const resp = response as unknown as Record<string, unknown>;\n        const requestId = resp['X-Request-Id'] || resp.requestId;\n        if (requestId) {\n          console.log('');\n          printInfo('Deployment logs available', {\n            requestId,\n            url: `${config.api.baseUrl}/logs/${requestId}`,\n          });\n        }\n\n        // Next steps\n        console.log('');\n        printInfo('Next steps', {\n          view: `Use \"get-plugin --id ${response.id}\" to view plugin details`,\n          list: 'Use \"list-plugins\" to see all plugins',\n        });\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'upload-plugin',\n            executionId,\n            file: options.file,\n            organization: options.organization,\n            name: options.name,\n            version: options.version,\n            public: options.public,\n            active: options.active,\n            verifySsl: options.verifySsl,\n          },\n        });\n      }\n    });\n}\n"]}
@@ -0,0 +1,12 @@
1
+ import { Command } from 'commander';
2
+ /**
3
+ * Registers the `version` command with the CLI program.
4
+ *
5
+ * Displays the CLI name/version, AWS CDK status, and optionally
6
+ * full system diagnostics (`--verbose`) or configuration validation
7
+ * (`--check-config`). Supports `--json` for machine-readable output.
8
+ *
9
+ * @param program - The root Commander program instance to attach the command to.
10
+ */
11
+ export declare function version(program: Command): void;
12
+ //# sourceMappingURL=version.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"version.d.ts","sourceRoot":"","sources":["../../src/commands/version.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAoFpC;;;;;;;;GAQG;AACH,wBAAgB,OAAO,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA0J9C"}
@@ -0,0 +1,223 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ var __importDefault = (this && this.__importDefault) || function (mod) {
5
+ return (mod && mod.__esModule) ? mod : { "default": mod };
6
+ };
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.version = version;
9
+ const child_process_1 = require("child_process");
10
+ const picocolors_1 = __importDefault(require("picocolors"));
11
+ const cli_constants_1 = require("../config/cli.constants");
12
+ const cdk_utils_1 = require("../utils/cdk-utils");
13
+ const config_loader_1 = require("../utils/config-loader");
14
+ const error_handler_1 = require("../utils/error-handler");
15
+ const output_utils_1 = require("../utils/output-utils");
16
+ const { bold, cyan, dim, green, magenta, red, yellow } = picocolors_1.default;
17
+ /**
18
+ * Collects runtime system information including Node.js/npm versions,
19
+ * platform, architecture, heap memory usage, and process uptime.
20
+ */
21
+ function getSystemInfo() {
22
+ const mem = process.memoryUsage();
23
+ const totalMem = (mem.heapTotal / 1024 / 1024).toFixed(2);
24
+ const availableHeap = ((mem.heapTotal - mem.heapUsed) / 1024 / 1024).toFixed(2);
25
+ const uptime = process.uptime();
26
+ let npm;
27
+ try {
28
+ npm = (0, child_process_1.execSync)('npm --version', { encoding: 'utf-8', stdio: 'pipe' }).trim();
29
+ }
30
+ catch {
31
+ npm = null;
32
+ }
33
+ return {
34
+ nodejs: process.version,
35
+ npm,
36
+ platform: process.platform,
37
+ architecture: process.arch,
38
+ memory: { total: `${totalMem} MB`, availableHeap: `${availableHeap} MB` },
39
+ uptime: `${Math.floor(uptime / 3600)}h ${Math.floor((uptime % 3600) / 60)}m ${Math.floor(uptime % 60)}s`,
40
+ };
41
+ }
42
+ /**
43
+ * Attempts to load the CLI configuration and reports whether it is valid.
44
+ * @returns An object with `valid: true` on success, or `valid: false` with an error message.
45
+ */
46
+ function checkConfiguration() {
47
+ try {
48
+ (0, config_loader_1.getConfig)();
49
+ return { valid: true };
50
+ }
51
+ catch (error) {
52
+ return {
53
+ valid: false,
54
+ error: error instanceof Error ? error.message : String(error),
55
+ };
56
+ }
57
+ }
58
+ /**
59
+ * Checks which key environment variables are currently set.
60
+ * @returns Presence flags for `PLATFORM_TOKEN`, `PLATFORM_BASE_URL`, and `CLI_CONFIG_PATH`.
61
+ */
62
+ function getEnvironmentStatus() {
63
+ return {
64
+ token: !!process.env.PLATFORM_TOKEN,
65
+ url: !!process.env.PLATFORM_BASE_URL,
66
+ configPath: !!process.env.CLI_CONFIG_PATH,
67
+ };
68
+ }
69
+ /**
70
+ * Registers the `version` command with the CLI program.
71
+ *
72
+ * Displays the CLI name/version, AWS CDK status, and optionally
73
+ * full system diagnostics (`--verbose`) or configuration validation
74
+ * (`--check-config`). Supports `--json` for machine-readable output.
75
+ *
76
+ * @param program - The root Commander program instance to attach the command to.
77
+ */
78
+ function version(program) {
79
+ program
80
+ .command('version')
81
+ .description('Display CLI version and environment information')
82
+ .option('--verbose', 'Show detailed environment information', false)
83
+ .option('--check-config', 'Verify configuration status', false)
84
+ .option('--json', 'Output in JSON format', false)
85
+ .action((options) => {
86
+ try {
87
+ const executionId = (0, cli_constants_1.generateExecutionId)();
88
+ const cdkInfo = (0, cdk_utils_1.getCdkInfo)();
89
+ // JSON output
90
+ if (options.json) {
91
+ const systemInfo = getSystemInfo();
92
+ const configStatus = options.checkConfig ? checkConfiguration() : { valid: true };
93
+ const envStatus = getEnvironmentStatus();
94
+ console.log(JSON.stringify({
95
+ cli: { name: cli_constants_1.APP_NAME, version: cli_constants_1.APP_VERSION },
96
+ system: systemInfo,
97
+ cdk: { available: cdkInfo.available, version: cdkInfo.version },
98
+ configuration: { valid: configStatus.valid, error: configStatus.error },
99
+ environment: envStatus,
100
+ executionId,
101
+ timestamp: new Date().toISOString(),
102
+ }, null, 2));
103
+ return;
104
+ }
105
+ // Standard output
106
+ (0, output_utils_1.printSection)('Version Information');
107
+ console.log(`${magenta(`[EXE-${executionId}]`)} ${cyan(bold('CLI Version Check'))}`);
108
+ console.log('');
109
+ (0, output_utils_1.printKeyValue)({
110
+ 'CLI Name': bold(cli_constants_1.APP_NAME),
111
+ 'CLI Version': green(bold(cli_constants_1.APP_VERSION)),
112
+ });
113
+ (0, output_utils_1.printDivider)();
114
+ // Verbose: system environment
115
+ if (options.verbose) {
116
+ (0, output_utils_1.printSection)('System Environment');
117
+ const systemInfo = getSystemInfo();
118
+ (0, output_utils_1.printKeyValue)({
119
+ 'Node.js': systemInfo.nodejs,
120
+ 'npm': systemInfo.npm || yellow('(not available)'),
121
+ 'Platform': systemInfo.platform,
122
+ 'Architecture': systemInfo.architecture,
123
+ 'Memory (Heap)': `${systemInfo.memory.total} total, ${systemInfo.memory.availableHeap} available`,
124
+ 'Process Uptime': systemInfo.uptime,
125
+ });
126
+ (0, output_utils_1.printDivider)();
127
+ (0, output_utils_1.printSection)('AWS CDK');
128
+ }
129
+ // CDK information
130
+ if (cdkInfo.available && cdkInfo.version) {
131
+ if (options.verbose)
132
+ (0, output_utils_1.printSuccess)('AWS CDK is installed');
133
+ (0, output_utils_1.printKeyValue)({
134
+ 'CDK Version': green(cdkInfo.version),
135
+ 'Status': green('✓ Available'),
136
+ });
137
+ }
138
+ else {
139
+ if (options.verbose) {
140
+ (0, output_utils_1.printWarning)('AWS CDK is not installed');
141
+ console.log(dim(' Install CDK with: npm install -g aws-cdk'));
142
+ if (cdkInfo.error)
143
+ console.log(dim(` Error: ${cdkInfo.error}`));
144
+ }
145
+ (0, output_utils_1.printKeyValue)({
146
+ 'AWS CDK': red('✗ Not installed'),
147
+ 'Status': red('✗ Not Available'),
148
+ });
149
+ }
150
+ if (!options.verbose) {
151
+ console.log(dim('\n Run with --verbose for detailed environment information'));
152
+ }
153
+ // Configuration check
154
+ if (options.checkConfig) {
155
+ (0, output_utils_1.printDivider)();
156
+ (0, output_utils_1.printSection)('Configuration Status');
157
+ const configStatus = checkConfiguration();
158
+ const envStatus = getEnvironmentStatus();
159
+ if (configStatus.valid) {
160
+ (0, output_utils_1.printSuccess)('Configuration is valid');
161
+ try {
162
+ const config = (0, config_loader_1.getConfig)();
163
+ (0, output_utils_1.printKeyValue)({
164
+ 'API Base URL': config.api.baseUrl,
165
+ 'SSL Verification': config.api.rejectUnauthorized ? green('Enabled') : yellow('Disabled'),
166
+ 'Timeout': `${config.api.timeout}ms`,
167
+ 'Authenticated': config.auth.token ? green('Yes') : red('No'),
168
+ });
169
+ }
170
+ catch (error) {
171
+ (0, output_utils_1.printWarning)('Could not load full configuration');
172
+ }
173
+ }
174
+ else {
175
+ (0, output_utils_1.printError)('Configuration is invalid');
176
+ if (configStatus.error) {
177
+ console.log(red(` Error: ${configStatus.error}`));
178
+ }
179
+ }
180
+ console.log('');
181
+ (0, output_utils_1.printInfo)('Environment Variables');
182
+ (0, output_utils_1.printKeyValue)({
183
+ PLATFORM_TOKEN: envStatus.token ? green('✓ Set') : red('✗ Not set'),
184
+ PLATFORM_BASE_URL: envStatus.url ? green('✓ Set') : dim('(not set - using default)'),
185
+ CLI_CONFIG_PATH: envStatus.configPath ? green('✓ Set') : dim('(not set - using default)'),
186
+ });
187
+ if (!envStatus.token) {
188
+ console.log('');
189
+ (0, output_utils_1.printWarning)('PLATFORM_TOKEN is required for API operations');
190
+ console.log(dim(' Set it with: export PLATFORM_TOKEN="your-token-here"'));
191
+ }
192
+ }
193
+ else {
194
+ console.log(dim('\n 💡 Tip: Run with --check-config to verify configuration'));
195
+ }
196
+ // Footer
197
+ (0, output_utils_1.printDivider)();
198
+ console.log(dim(` Execution ID: ${executionId}`));
199
+ console.log(dim(` Timestamp: ${new Date().toISOString()}`));
200
+ console.log('');
201
+ }
202
+ catch (error) {
203
+ (0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.GENERAL, {
204
+ debug: program.opts().debug,
205
+ exit: true,
206
+ context: {
207
+ command: 'version',
208
+ options: {
209
+ verbose: options.verbose,
210
+ checkConfig: options.checkConfig,
211
+ json: options.json,
212
+ },
213
+ },
214
+ });
215
+ }
216
+ });
217
+ // Also support -v and --version flags at root level
218
+ program.on('option:version', () => {
219
+ console.log(`${cli_constants_1.APP_NAME} v${cli_constants_1.APP_VERSION}`);
220
+ process.exit(0);
221
+ });
222
+ }
223
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"version.js","sourceRoot":"","sources":["../../src/commands/version.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { execSync } from 'child_process';\nimport { Command } from 'commander';\nimport pico from 'picocolors';\nimport { APP_NAME, APP_VERSION, generateExecutionId } from '../config/cli.constants';\nimport { getCdkInfo } from '../utils/cdk-utils';\nimport { getConfig } from '../utils/config-loader';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printDivider, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\nconst { bold, cyan, dim, green, magenta, red, yellow } = pico;\n\n/**\n * Runtime system information collected for the `version --verbose` output.\n */\ninterface SystemInfo {\n  nodejs: string;\n  npm: string | null;\n  platform: string;\n  architecture: string;\n  memory: {\n    total: string;\n    availableHeap: string;\n  };\n  uptime: string;\n}\n\n/**\n * Collects runtime system information including Node.js/npm versions,\n * platform, architecture, heap memory usage, and process uptime.\n */\nfunction getSystemInfo(): SystemInfo {\n  const mem = process.memoryUsage();\n  const totalMem = (mem.heapTotal / 1024 / 1024).toFixed(2);\n  const availableHeap = ((mem.heapTotal - mem.heapUsed) / 1024 / 1024).toFixed(2);\n  const uptime = process.uptime();\n\n  let npm: string | null;\n  try {\n    npm = execSync('npm --version', { encoding: 'utf-8', stdio: 'pipe' }).trim();\n  } catch {\n    npm = null;\n  }\n\n  return {\n    nodejs: process.version,\n    npm,\n    platform: process.platform,\n    architecture: process.arch,\n    memory: { total: `${totalMem} MB`, availableHeap: `${availableHeap} MB` },\n    uptime: `${Math.floor(uptime / 3600)}h ${Math.floor((uptime % 3600) / 60)}m ${Math.floor(uptime % 60)}s`,\n  };\n}\n\n/**\n * Attempts to load the CLI configuration and reports whether it is valid.\n * @returns An object with `valid: true` on success, or `valid: false` with an error message.\n */\nfunction checkConfiguration(): { valid: boolean; error?: string } {\n  try {\n    getConfig();\n    return { valid: true };\n  } catch (error) {\n    return {\n      valid: false,\n      error: error instanceof Error ? error.message : String(error),\n    };\n  }\n}\n\n/**\n * Checks which key environment variables are currently set.\n * @returns Presence flags for `PLATFORM_TOKEN`, `PLATFORM_BASE_URL`, and `CLI_CONFIG_PATH`.\n */\nfunction getEnvironmentStatus(): {\n  token: boolean;\n  url: boolean;\n  configPath: boolean;\n} {\n  return {\n    token: !!process.env.PLATFORM_TOKEN,\n    url: !!process.env.PLATFORM_BASE_URL,\n    configPath: !!process.env.CLI_CONFIG_PATH,\n  };\n}\n\n/**\n * Registers the `version` command with the CLI program.\n *\n * Displays the CLI name/version, AWS CDK status, and optionally\n * full system diagnostics (`--verbose`) or configuration validation\n * (`--check-config`).  Supports `--json` for machine-readable output.\n *\n * @param program - The root Commander program instance to attach the command to.\n */\nexport function version(program: Command): void {\n  program\n    .command('version')\n    .description('Display CLI version and environment information')\n    .option('--verbose', 'Show detailed environment information', false)\n    .option('--check-config', 'Verify configuration status', false)\n    .option('--json', 'Output in JSON format', false)\n    .action((options) => {\n      try {\n        const executionId = generateExecutionId();\n        const cdkInfo = getCdkInfo();\n\n        // JSON output\n        if (options.json) {\n          const systemInfo = getSystemInfo();\n          const configStatus = options.checkConfig ? checkConfiguration() : { valid: true };\n          const envStatus = getEnvironmentStatus();\n\n          console.log(JSON.stringify({\n            cli: { name: APP_NAME, version: APP_VERSION },\n            system: systemInfo,\n            cdk: { available: cdkInfo.available, version: cdkInfo.version },\n            configuration: { valid: configStatus.valid, error: configStatus.error },\n            environment: envStatus,\n            executionId,\n            timestamp: new Date().toISOString(),\n          }, null, 2));\n          return;\n        }\n\n        // Standard output\n        printSection('Version Information');\n        console.log(`${magenta(`[EXE-${executionId}]`)} ${cyan(bold('CLI Version Check'))}`);\n        console.log('');\n\n        printKeyValue({\n          'CLI Name': bold(APP_NAME),\n          'CLI Version': green(bold(APP_VERSION)),\n        });\n\n        printDivider();\n\n        // Verbose: system environment\n        if (options.verbose) {\n          printSection('System Environment');\n          const systemInfo = getSystemInfo();\n          printKeyValue({\n            'Node.js': systemInfo.nodejs,\n            'npm': systemInfo.npm || yellow('(not available)'),\n            'Platform': systemInfo.platform,\n            'Architecture': systemInfo.architecture,\n            'Memory (Heap)': `${systemInfo.memory.total} total, ${systemInfo.memory.availableHeap} available`,\n            'Process Uptime': systemInfo.uptime,\n          });\n          printDivider();\n          printSection('AWS CDK');\n        }\n\n        // CDK information\n        if (cdkInfo.available && cdkInfo.version) {\n          if (options.verbose) printSuccess('AWS CDK is installed');\n          printKeyValue({\n            'CDK Version': green(cdkInfo.version),\n            'Status': green('✓ Available'),\n          });\n        } else {\n          if (options.verbose) {\n            printWarning('AWS CDK is not installed');\n            console.log(dim('  Install CDK with: npm install -g aws-cdk'));\n            if (cdkInfo.error) console.log(dim(`  Error: ${cdkInfo.error}`));\n          }\n          printKeyValue({\n            'AWS CDK': red('✗ Not installed'),\n            'Status': red('✗ Not Available'),\n          });\n        }\n\n        if (!options.verbose) {\n          console.log(dim('\\n  Run with --verbose for detailed environment information'));\n        }\n\n        // Configuration check\n        if (options.checkConfig) {\n          printDivider();\n          printSection('Configuration Status');\n\n          const configStatus = checkConfiguration();\n          const envStatus = getEnvironmentStatus();\n\n          if (configStatus.valid) {\n            printSuccess('Configuration is valid');\n\n            try {\n              const config = getConfig();\n              printKeyValue({\n                'API Base URL': config.api.baseUrl,\n                'SSL Verification': config.api.rejectUnauthorized ? green('Enabled') : yellow('Disabled'),\n                'Timeout': `${config.api.timeout}ms`,\n                'Authenticated': config.auth.token ? green('Yes') : red('No'),\n              });\n            } catch (error) {\n              printWarning('Could not load full configuration');\n            }\n          } else {\n            printError('Configuration is invalid');\n            if (configStatus.error) {\n              console.log(red(`  Error: ${configStatus.error}`));\n            }\n          }\n\n          console.log('');\n          printInfo('Environment Variables');\n          printKeyValue({\n            PLATFORM_TOKEN: envStatus.token ? green('✓ Set') : red('✗ Not set'),\n            PLATFORM_BASE_URL: envStatus.url ? green('✓ Set') : dim('(not set - using default)'),\n            CLI_CONFIG_PATH: envStatus.configPath ? green('✓ Set') : dim('(not set - using default)'),\n          });\n\n          if (!envStatus.token) {\n            console.log('');\n            printWarning('PLATFORM_TOKEN is required for API operations');\n            console.log(dim('  Set it with: export PLATFORM_TOKEN=\"your-token-here\"'));\n          }\n        } else {\n          console.log(dim('\\n  💡 Tip: Run with --check-config to verify configuration'));\n        }\n\n        // Footer\n        printDivider();\n        console.log(dim(`  Execution ID: ${executionId}`));\n        console.log(dim(`  Timestamp: ${new Date().toISOString()}`));\n        console.log('');\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.GENERAL, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'version',\n            options: {\n              verbose: options.verbose,\n              checkConfig: options.checkConfig,\n              json: options.json,\n            },\n          },\n        });\n      }\n    });\n\n  // Also support -v and --version flags at root level\n  program.on('option:version', () => {\n    console.log(`${APP_NAME} v${APP_VERSION}`);\n    process.exit(0);\n  });\n}\n"]}