@pipeline-builder/pipeline-manager 3.3.28 → 3.3.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -30,18 +30,47 @@ pipeline-manager deploy
30
30
 
31
31
  ## Commands
32
32
 
33
+ ### Project lifecycle
34
+
33
35
  | Command | Purpose |
34
36
  | --- | --- |
35
37
  | `bootstrap` | Scaffold a new pipeline project with `cdk.json` and starter config |
36
38
  | `synth` | Run CDK synth to emit the CloudFormation template for the pipeline |
37
- | `deploy` | Deploy the synthesized pipeline stack to AWS |
39
+ | `deploy` | Deploy the synthesized pipeline stack to AWS (also registers the deployed ARN with the platform) |
38
40
  | `status` | Report the current deployment and execution status |
41
+
42
+ ### Resource management
43
+
44
+ | Command | Purpose |
45
+ | --- | --- |
39
46
  | `create-pipeline` | Register a new pipeline definition with the platform |
40
47
  | `list-pipelines` / `get-pipeline` | Inspect pipelines registered to your organization |
41
48
  | `list-plugins` / `get-plugin` | Browse the plugin catalog and fetch a single plugin spec |
42
49
  | `upload-plugin` | Publish a custom plugin spec + Dockerfile to the platform |
43
- | `setup-events` | Wire CodePipeline events into the platform's reporting stream |
44
- | `login` / `store-token` | Manage the `PLATFORM_TOKEN` used to authenticate API calls |
50
+ | `validate-templates` | Parse and validate `{{ ... }}` templates in a pipeline or plugin spec (local file, registered pipeline by ID, or registered plugin by `name:version`) |
51
+
52
+ ### Auth & infrastructure
53
+
54
+ | Command | Purpose |
55
+ | --- | --- |
56
+ | `login` | Authenticate against the platform and persist the access token |
57
+ | `store-token` | Generate a long-lived JWT and store it in AWS Secrets Manager (used by the events Lambda and CodePipeline synth steps) |
58
+ | `setup-events` | Deploy the EventBridge → SQS → Lambda stack that streams CodePipeline events into the platform's reporting service |
59
+
60
+ ### Operator audits (cron-friendly)
61
+
62
+ These commands report drift and exit non-zero when findings exist. Designed to run on a schedule.
63
+
64
+ | Command | Purpose | Exit codes |
65
+ | --- | --- | --- |
66
+ | `audit-stacks` | Diff CloudFormation stacks tagged `pipeline-builder` against the platform's `pipeline_registry`. Surfaces orphaned stacks (no DB row) and missing stacks (DB row but no live stack). See [docs/aws-deployment.md](https://mwashburn160.github.io/pipeline-builder/docs/aws-deployment.html#drift-detection-audit-stacks). | `0` clean / `1` findings / `2` AWS error |
67
+ | `audit-tokens` | Scan platform tokens stored in AWS Secrets Manager and flag any expiring within `--warn-days` (default 7). Run before tokens lapse to avoid silent reporting outages. | `0` clean / `1` at-risk / `2` AWS error |
68
+
69
+ ### Misc
70
+
71
+ | Command | Purpose |
72
+ | --- | --- |
73
+ | `completions` | Print shell completion script. Source it from your `~/.bashrc` / `~/.zshrc`: `eval "$(pipeline-manager completions bash)"` |
45
74
  | `version` | Print CLI version info |
46
75
 
47
76
  Run `pipeline-manager <command> --help` for the full flag reference on any command.
package/dist/cli.d.ts.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AAkCA;;GAEG;AACH,UAAU,UAAU;IAClB;;;OAGG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IAExB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;IAElB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAkOD;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CAyD5D;AAiDD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,IAAI,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CA0BnD"}
1
+ {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AAmCA;;GAEG;AACH,UAAU,UAAU;IAClB;;;OAGG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IAExB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;IAElB;;;OAGG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB;;;OAGG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAsOD;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CAyD5D;AAiDD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,IAAI,CAAC,OAAO,GAAE,UAAe,GAAG,IAAI,CA0BnD"}
package/dist/cli.js CHANGED
@@ -16,6 +16,7 @@ const get_plugin_1 = require("./commands/get-plugin");
16
16
  const list_pipelines_1 = require("./commands/list-pipelines");
17
17
  const list_plugins_1 = require("./commands/list-plugins");
18
18
  const login_1 = require("./commands/login");
19
+ const register_1 = require("./commands/register");
19
20
  const setup_events_1 = require("./commands/setup-events");
20
21
  const status_1 = require("./commands/status");
21
22
  const store_token_1 = require("./commands/store-token");
@@ -137,6 +138,7 @@ Examples:
137
138
  (0, setup_events_1.setupEvents)(commander_1.program); // Deploy EventBridge event ingestion infrastructure
138
139
  (0, bootstrap_1.bootstrap)(commander_1.program); // Bootstrap CDK toolkit stack
139
140
  (0, deploy_1.deploy)(commander_1.program); // Deploy pipeline with CDK (--app prints boilerplate path)
141
+ (0, register_1.register)(commander_1.program); // Re-register a deployed pipeline ARN + drain pending intents from prior failed deploys
140
142
  // Operator audit commands (cron-friendly: exit 1 on findings)
141
143
  (0, output_utils_1.printDebug)('Registering audit commands');
142
144
  (0, audit_tokens_1.auditTokens)(commander_1.program); // Scan Secrets Manager for expiring platform tokens
@@ -150,12 +152,15 @@ Examples:
150
152
  .description('Generate shell completions (bash, zsh, fish)')
151
153
  .argument('<shell>', 'Shell type: bash, zsh, or fish')
152
154
  .action((shell) => {
155
+ // Pull command names from commander's registered list so completions never
156
+ // drift from the actual CLI surface. Sorted for stable output.
157
+ const commands = commander_1.program.commands.map(c => c.name()).sort().join(' ');
153
158
  switch (shell) {
154
159
  case 'bash':
155
160
  console.log(`# pipeline-manager bash completions
156
161
  _pipeline_manager_completions() {
157
162
  local cur="\${COMP_WORDS[COMP_CWORD]}"
158
- local commands="login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions"
163
+ local commands="${commands}"
159
164
  COMPREPLY=($(compgen -W "\${commands}" -- "\${cur}"))
160
165
  }
161
166
  complete -F _pipeline_manager_completions pipeline-manager`);
@@ -163,14 +168,14 @@ complete -F _pipeline_manager_completions pipeline-manager`);
163
168
  case 'zsh':
164
169
  console.log(`# pipeline-manager zsh completions
165
170
  _pipeline_manager() {
166
- local commands=(login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions)
171
+ local commands=(${commands})
167
172
  _describe 'command' commands
168
173
  }
169
174
  compdef _pipeline_manager pipeline-manager`);
170
175
  break;
171
176
  case 'fish':
172
177
  console.log(`# pipeline-manager fish completions
173
- complete -c pipeline-manager -n '__fish_use_subcommand' -a 'login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions'`);
178
+ complete -c pipeline-manager -n '__fish_use_subcommand' -a '${commands}'`);
174
179
  break;
175
180
  default:
176
181
  console.error(`Unknown shell: ${shell}. Use bash, zsh, or fish.`);
@@ -377,4 +382,4 @@ if (require.main === module) {
377
382
  };
378
383
  main(options);
379
384
  }
380
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { program } from 'commander';\nimport { auditStacks } from './commands/audit-stacks';\nimport { auditTokens } from './commands/audit-tokens';\nimport { bootstrap } from './commands/bootstrap';\nimport { createPipeline } from './commands/create-pipeline';\nimport { deploy } from './commands/deploy';\nimport { getPipeline } from './commands/get-pipeline';\nimport { getPlugin } from './commands/get-plugin';\nimport { listPipelines } from './commands/list-pipelines';\nimport { listPlugins } from './commands/list-plugins';\nimport { login } from './commands/login';\nimport { setupEvents } from './commands/setup-events';\nimport { status } from './commands/status';\nimport { storeToken } from './commands/store-token';\nimport { synth } from './commands/synth';\nimport { uploadPlugin } from './commands/upload-plugin';\nimport { validateTemplatesCommand } from './commands/validate-templates';\nimport { version } from './commands/version';\nimport {\n  APP_NAME,\n  APP_DESCRIPTION,\n  APP_VERSION,\n  ENV_VARS,\n  isDebugMode,\n  generateExecutionId,\n} from './config/cli.constants';\nimport { banner, miniBanner } from './utils/banner';\nimport { ERROR_CODES, handleError } from './utils/error-handler';\nimport { printInfo, printError, printWarning, printDebug, printSection } from './utils/output-utils';\n\n/**\n * CLI initialization options\n */\ninterface CliOptions {\n  /**\n   * Show banner on startup\n   * @default true\n   */\n  showBanner?: boolean;\n\n  /**\n   * Minimal banner (no ASCII art)\n   * @default false\n   */\n  minimalBanner?: boolean;\n\n  /**\n   * Debug mode\n   * @default false\n   */\n  debug?: boolean;\n\n  /**\n   * Verbose output\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Quiet mode (minimal output)\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * No color output\n   * @default false\n   */\n  noColor?: boolean;\n}\n\n/**\n * Check environment and display warnings\n */\nfunction checkEnvironment(): void {\n  const warnings: string[] = [];\n\n  // Check for required environment variables\n  if (!process.env[ENV_VARS.PLATFORM_TOKEN]) {\n    warnings.push('PLATFORM_TOKEN environment variable is not set');\n    warnings.push('Authentication will fail for API operations');\n  }\n\n  // Check Node version\n  const nodeVersion = process.version;\n  const majorVersion = parseInt(nodeVersion.slice(1).split('.')[0] || '0', 10);\n  if (majorVersion < 18) {\n    warnings.push(`Node.js ${nodeVersion} detected - version 18+ recommended`);\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    printSection('Environment Warnings');\n    warnings.forEach(warning => {\n      printWarning(warning);\n    });\n    console.log('');\n  }\n}\n\n/**\n * Display startup information\n */\nfunction displayStartupInfo(options: CliOptions): void {\n  if (options.quiet) return;\n\n  const executionId = generateExecutionId();\n\n  if (options.debug || options.verbose) {\n    printDebug('CLI Configuration', {\n      name: APP_NAME,\n      version: APP_VERSION,\n      executionId,\n      node: process.version,\n      platform: process.platform,\n      arch: process.arch,\n      cwd: process.cwd(),\n      env: {\n        debug: process.env.DEBUG,\n        token: process.env[ENV_VARS.PLATFORM_TOKEN] ? 'set' : 'not set',\n        url: process.env[ENV_VARS.PLATFORM_BASE_URL] || 'default',\n      },\n    });\n  } else {\n    printDebug('Starting CLI', {\n      version: APP_VERSION,\n      executionId,\n    });\n  }\n}\n\n/**\n * Register all CLI commands\n */\nfunction registerCommands(): void {\n  printDebug('Registering commands');\n\n  // Configure program\n  program\n    .name(APP_NAME)\n    .description(APP_DESCRIPTION)\n    .version(APP_VERSION, '-v, --version', 'Show CLI version')\n    .option('--debug', 'Enable debug output with stack traces', false)\n    .option('--verbose', 'Show detailed information', false)\n    .option('--quiet', 'Minimal output (errors only)', false)\n    .option('--no-color', 'Disable colored output', false)\n    .addHelpText('after', `\nEnvironment Variables:\n  ${ENV_VARS.PLATFORM_TOKEN}              Authentication token (required)\n  ${ENV_VARS.PLATFORM_BASE_URL}                 API base URL (optional)\n  ${ENV_VARS.CLI_CONFIG_PATH}              Config file path (optional)\n  ${ENV_VARS.TLS_REJECT_UNAUTHORIZED}      Disable SSL verification if '0'\n  ${ENV_VARS.DEBUG}                        Enable debug mode if 'true'\n\nExamples:\n  $ ${APP_NAME} version\n  $ ${APP_NAME} list-pipelines --project my-app\n  $ ${APP_NAME} get-pipeline --id pipe-123 --format json\n  $ ${APP_NAME} store-token --days 30 --region us-east-1\n  $ ${APP_NAME} bootstrap --account 123456789012 --region us-east-1\n  $ ${APP_NAME} deploy --id pipe-123 --profile production\n\n`);\n\n  // Version command (special handling)\n  version(program);\n\n  // Authentication commands\n  printDebug('Registering authentication commands');\n  login(program); // Login and obtain PLATFORM_TOKEN (also supports --refresh)\n\n  // Query commands\n  printDebug('Registering query commands');\n  getPlugin(program); // Single plugin by ID\n  listPlugins(program); // Multiple plugins with filters\n  getPipeline(program); // Single pipeline by ID\n  listPipelines(program); // Multiple pipelines with filters\n\n  // Create/Upload commands\n  printDebug('Registering create/upload commands');\n  createPipeline(program); // Create pipeline configuration\n  uploadPlugin(program); // Upload and deploy plugin\n\n  // Status command\n  printDebug('Registering status command');\n  status(program); // Show environment and connectivity status\n\n  // Deployment commands\n  printDebug('Registering deployment commands');\n  storeToken(program); // Generate JWT token and store in Secrets Manager\n  setupEvents(program); // Deploy EventBridge event ingestion infrastructure\n  bootstrap(program); // Bootstrap CDK toolkit stack\n  deploy(program); // Deploy pipeline with CDK (--app prints boilerplate path)\n\n  // Operator audit commands (cron-friendly: exit 1 on findings)\n  printDebug('Registering audit commands');\n  auditTokens(program); // Scan Secrets Manager for expiring platform tokens\n  auditStacks(program); // Diff CFN stacks vs pipeline_registry to find drift\n  synth(program); // Run CDK synthesis\n  validateTemplatesCommand(program); // Validate {{ ... }} templates in a pipeline or plugin\n\n  // Shell completions\n  printDebug('Registering completions command');\n  program\n    .command('completions')\n    .description('Generate shell completions (bash, zsh, fish)')\n    .argument('<shell>', 'Shell type: bash, zsh, or fish')\n    .action((shell: string) => {\n      switch (shell) {\n        case 'bash':\n          console.log(`# pipeline-manager bash completions\n_pipeline_manager_completions() {\n  local cur=\"\\${COMP_WORDS[COMP_CWORD]}\"\n  local commands=\"login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions\"\n  COMPREPLY=($(compgen -W \"\\${commands}\" -- \"\\${cur}\"))\n}\ncomplete -F _pipeline_manager_completions pipeline-manager`);\n          break;\n        case 'zsh':\n          console.log(`# pipeline-manager zsh completions\n_pipeline_manager() {\n  local commands=(login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions)\n  _describe 'command' commands\n}\ncompdef _pipeline_manager pipeline-manager`);\n          break;\n        case 'fish':\n          console.log(`# pipeline-manager fish completions\ncomplete -c pipeline-manager -n '__fish_use_subcommand' -a 'login deploy synth upload-plugin create-pipeline status version bootstrap setup-events store-token completions'`);\n          break;\n        default:\n          console.error(`Unknown shell: ${shell}. Use bash, zsh, or fish.`);\n          process.exit(1);\n      }\n    });\n\n  printDebug('All commands registered successfully');\n}\n\n/**\n * Handle program errors\n */\nfunction setupErrorHandlers(): void {\n  // Handle uncaught exceptions\n  process.on('uncaughtException', (error: Error) => {\n    console.error(''); // Empty line\n    printError('Uncaught exception', {\n      error: error.message,\n      name: error.name,\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'uncaughtException',\n      },\n    });\n  });\n\n  // Handle unhandled promise rejections\n  process.on('unhandledRejection', (reason: unknown) => {\n    console.error(''); // Empty line\n    printError('Unhandled promise rejection', {\n      reason: reason instanceof Error ? reason.message : String(reason),\n    });\n\n    const error = reason instanceof Error ? reason : new Error(String(reason));\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'unhandledRejection',\n      },\n    });\n  });\n\n  // Handle SIGINT (Ctrl+C)\n  process.on('SIGINT', () => {\n    console.log(''); // Empty line\n    printWarning('Process interrupted by user (SIGINT)');\n    console.log(''); // Empty line\n    process.exit(130); // Standard exit code for SIGINT\n  });\n\n  // Handle SIGTERM\n  process.on('SIGTERM', () => {\n    console.log(''); // Empty line\n    printWarning('Process terminated (SIGTERM)');\n    console.log(''); // Empty line\n    process.exit(143); // Standard exit code for SIGTERM\n  });\n}\n\n/**\n * Initialize CLI with options\n *\n * @param options - CLI initialization options\n */\nexport function initializeCli(options: CliOptions = {}): void {\n  const {\n    showBanner = true,\n    minimalBanner = false,\n    debug = false,\n    verbose = false,\n    quiet = false,\n    noColor = false,\n  } = options;\n\n  // Handle color output\n  if (noColor) {\n    process.env.NO_COLOR = '1';\n  }\n\n  // Display banner\n  if (showBanner && !quiet) {\n    if (minimalBanner) {\n      miniBanner();\n    } else {\n      banner({\n        minimal: false,\n        showDescription: true,\n        showTimestamp: debug || verbose,\n        showExecutionId: debug,\n      });\n    }\n  }\n\n  // Display startup info\n  displayStartupInfo({ debug, verbose, quiet });\n\n  // Check environment\n  if (!quiet) {\n    checkEnvironment();\n  }\n\n  // Setup error handlers\n  setupErrorHandlers();\n\n  // Register commands\n  try {\n    registerCommands();\n    printDebug('CLI initialization complete');\n  } catch (error) {\n    printError('CLI initialization failed', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.CONFIGURATION, {\n      debug: debug || isDebugMode(),\n      exit: true,\n      context: {\n        stage: 'initialization',\n      },\n    });\n  }\n}\n\n/**\n * Parse command line arguments\n */\nfunction parseArguments(): void {\n  try {\n    program.parse(process.argv);\n\n    // Show help if no command provided\n    const args = process.argv.slice(2);\n    if (args.length === 0) {\n      printInfo('No command specified - displaying help');\n      console.log(''); // Empty line\n      program.outputHelp();\n      process.exit(0);\n    }\n\n    // Check if command exists\n    const command = args[0];\n    if (command && !command.startsWith('-')) {\n      const commandExists = program.commands.some(cmd => cmd.name() === command);\n\n      if (!commandExists) {\n        console.log(''); // Empty line\n        printError(`Unknown command: ${command}`);\n        console.log(''); // Empty line\n        printInfo('Available commands:');\n        program.commands.forEach(cmd => {\n          console.log(`  • ${cmd.name()} - ${cmd.description()}`);\n        });\n        console.log(''); // Empty line\n        process.exit(1);\n      }\n    }\n\n    printDebug('Command line arguments parsed successfully');\n  } catch (error) {\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        stage: 'argument-parsing',\n        argv: process.argv,\n      },\n    });\n  }\n}\n\n/**\n * Main CLI entry point\n *\n * @param options - CLI initialization options\n *\n * @example\n * ```typescript\n * // Standard initialization\n * main();\n *\n * // Minimal mode\n * main({ minimalBanner: true, quiet: true });\n *\n * // Debug mode\n * main({ debug: true, verbose: true });\n * ```\n */\nexport function main(options: CliOptions = {}): void {\n  try {\n    // Initialize CLI\n    initializeCli(options);\n\n    // Parse arguments and execute command\n    parseArguments();\n\n    // If we reach here, command completed successfully\n    printDebug('CLI execution completed successfully');\n  } catch (error) {\n    // Final catch-all error handler\n    console.error(''); // Empty line\n    printError('Fatal CLI error', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(options),\n      exit: true,\n      context: {\n        stage: 'main',\n        options,\n      },\n    });\n  }\n}\n\n/**\n * Run CLI if executed directly\n */\nif (require.main === module) {\n  // Parse CLI options from environment or command line\n  const options: CliOptions = {\n    debug: process.env.DEBUG === 'true',\n    quiet: process.argv.includes('--quiet'),\n    verbose: process.argv.includes('--verbose'),\n    noColor: process.argv.includes('--no-color') || process.env.NO_COLOR === '1',\n  };\n\n  main(options);\n}"]}
385
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { program } from 'commander';\nimport { auditStacks } from './commands/audit-stacks';\nimport { auditTokens } from './commands/audit-tokens';\nimport { bootstrap } from './commands/bootstrap';\nimport { createPipeline } from './commands/create-pipeline';\nimport { deploy } from './commands/deploy';\nimport { getPipeline } from './commands/get-pipeline';\nimport { getPlugin } from './commands/get-plugin';\nimport { listPipelines } from './commands/list-pipelines';\nimport { listPlugins } from './commands/list-plugins';\nimport { login } from './commands/login';\nimport { register } from './commands/register';\nimport { setupEvents } from './commands/setup-events';\nimport { status } from './commands/status';\nimport { storeToken } from './commands/store-token';\nimport { synth } from './commands/synth';\nimport { uploadPlugin } from './commands/upload-plugin';\nimport { validateTemplatesCommand } from './commands/validate-templates';\nimport { version } from './commands/version';\nimport {\n  APP_NAME,\n  APP_DESCRIPTION,\n  APP_VERSION,\n  ENV_VARS,\n  isDebugMode,\n  generateExecutionId,\n} from './config/cli.constants';\nimport { banner, miniBanner } from './utils/banner';\nimport { ERROR_CODES, handleError } from './utils/error-handler';\nimport { printInfo, printError, printWarning, printDebug, printSection } from './utils/output-utils';\n\n/**\n * CLI initialization options\n */\ninterface CliOptions {\n  /**\n   * Show banner on startup\n   * @default true\n   */\n  showBanner?: boolean;\n\n  /**\n   * Minimal banner (no ASCII art)\n   * @default false\n   */\n  minimalBanner?: boolean;\n\n  /**\n   * Debug mode\n   * @default false\n   */\n  debug?: boolean;\n\n  /**\n   * Verbose output\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Quiet mode (minimal output)\n   * @default false\n   */\n  quiet?: boolean;\n\n  /**\n   * No color output\n   * @default false\n   */\n  noColor?: boolean;\n}\n\n/**\n * Check environment and display warnings\n */\nfunction checkEnvironment(): void {\n  const warnings: string[] = [];\n\n  // Check for required environment variables\n  if (!process.env[ENV_VARS.PLATFORM_TOKEN]) {\n    warnings.push('PLATFORM_TOKEN environment variable is not set');\n    warnings.push('Authentication will fail for API operations');\n  }\n\n  // Check Node version\n  const nodeVersion = process.version;\n  const majorVersion = parseInt(nodeVersion.slice(1).split('.')[0] || '0', 10);\n  if (majorVersion < 18) {\n    warnings.push(`Node.js ${nodeVersion} detected - version 18+ recommended`);\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    printSection('Environment Warnings');\n    warnings.forEach(warning => {\n      printWarning(warning);\n    });\n    console.log('');\n  }\n}\n\n/**\n * Display startup information\n */\nfunction displayStartupInfo(options: CliOptions): void {\n  if (options.quiet) return;\n\n  const executionId = generateExecutionId();\n\n  if (options.debug || options.verbose) {\n    printDebug('CLI Configuration', {\n      name: APP_NAME,\n      version: APP_VERSION,\n      executionId,\n      node: process.version,\n      platform: process.platform,\n      arch: process.arch,\n      cwd: process.cwd(),\n      env: {\n        debug: process.env.DEBUG,\n        token: process.env[ENV_VARS.PLATFORM_TOKEN] ? 'set' : 'not set',\n        url: process.env[ENV_VARS.PLATFORM_BASE_URL] || 'default',\n      },\n    });\n  } else {\n    printDebug('Starting CLI', {\n      version: APP_VERSION,\n      executionId,\n    });\n  }\n}\n\n/**\n * Register all CLI commands\n */\nfunction registerCommands(): void {\n  printDebug('Registering commands');\n\n  // Configure program\n  program\n    .name(APP_NAME)\n    .description(APP_DESCRIPTION)\n    .version(APP_VERSION, '-v, --version', 'Show CLI version')\n    .option('--debug', 'Enable debug output with stack traces', false)\n    .option('--verbose', 'Show detailed information', false)\n    .option('--quiet', 'Minimal output (errors only)', false)\n    .option('--no-color', 'Disable colored output', false)\n    .addHelpText('after', `\nEnvironment Variables:\n  ${ENV_VARS.PLATFORM_TOKEN}              Authentication token (required)\n  ${ENV_VARS.PLATFORM_BASE_URL}                 API base URL (optional)\n  ${ENV_VARS.CLI_CONFIG_PATH}              Config file path (optional)\n  ${ENV_VARS.TLS_REJECT_UNAUTHORIZED}      Disable SSL verification if '0'\n  ${ENV_VARS.DEBUG}                        Enable debug mode if 'true'\n\nExamples:\n  $ ${APP_NAME} version\n  $ ${APP_NAME} list-pipelines --project my-app\n  $ ${APP_NAME} get-pipeline --id pipe-123 --format json\n  $ ${APP_NAME} store-token --days 30 --region us-east-1\n  $ ${APP_NAME} bootstrap --account 123456789012 --region us-east-1\n  $ ${APP_NAME} deploy --id pipe-123 --profile production\n\n`);\n\n  // Version command (special handling)\n  version(program);\n\n  // Authentication commands\n  printDebug('Registering authentication commands');\n  login(program); // Login and obtain PLATFORM_TOKEN (also supports --refresh)\n\n  // Query commands\n  printDebug('Registering query commands');\n  getPlugin(program); // Single plugin by ID\n  listPlugins(program); // Multiple plugins with filters\n  getPipeline(program); // Single pipeline by ID\n  listPipelines(program); // Multiple pipelines with filters\n\n  // Create/Upload commands\n  printDebug('Registering create/upload commands');\n  createPipeline(program); // Create pipeline configuration\n  uploadPlugin(program); // Upload and deploy plugin\n\n  // Status command\n  printDebug('Registering status command');\n  status(program); // Show environment and connectivity status\n\n  // Deployment commands\n  printDebug('Registering deployment commands');\n  storeToken(program); // Generate JWT token and store in Secrets Manager\n  setupEvents(program); // Deploy EventBridge event ingestion infrastructure\n  bootstrap(program); // Bootstrap CDK toolkit stack\n  deploy(program); // Deploy pipeline with CDK (--app prints boilerplate path)\n  register(program); // Re-register a deployed pipeline ARN + drain pending intents from prior failed deploys\n\n  // Operator audit commands (cron-friendly: exit 1 on findings)\n  printDebug('Registering audit commands');\n  auditTokens(program); // Scan Secrets Manager for expiring platform tokens\n  auditStacks(program); // Diff CFN stacks vs pipeline_registry to find drift\n  synth(program); // Run CDK synthesis\n  validateTemplatesCommand(program); // Validate {{ ... }} templates in a pipeline or plugin\n\n  // Shell completions\n  printDebug('Registering completions command');\n  program\n    .command('completions')\n    .description('Generate shell completions (bash, zsh, fish)')\n    .argument('<shell>', 'Shell type: bash, zsh, or fish')\n    .action((shell: string) => {\n      // Pull command names from commander's registered list so completions never\n      // drift from the actual CLI surface. Sorted for stable output.\n      const commands = program.commands.map(c => c.name()).sort().join(' ');\n      switch (shell) {\n        case 'bash':\n          console.log(`# pipeline-manager bash completions\n_pipeline_manager_completions() {\n  local cur=\"\\${COMP_WORDS[COMP_CWORD]}\"\n  local commands=\"${commands}\"\n  COMPREPLY=($(compgen -W \"\\${commands}\" -- \"\\${cur}\"))\n}\ncomplete -F _pipeline_manager_completions pipeline-manager`);\n          break;\n        case 'zsh':\n          console.log(`# pipeline-manager zsh completions\n_pipeline_manager() {\n  local commands=(${commands})\n  _describe 'command' commands\n}\ncompdef _pipeline_manager pipeline-manager`);\n          break;\n        case 'fish':\n          console.log(`# pipeline-manager fish completions\ncomplete -c pipeline-manager -n '__fish_use_subcommand' -a '${commands}'`);\n          break;\n        default:\n          console.error(`Unknown shell: ${shell}. Use bash, zsh, or fish.`);\n          process.exit(1);\n      }\n    });\n\n  printDebug('All commands registered successfully');\n}\n\n/**\n * Handle program errors\n */\nfunction setupErrorHandlers(): void {\n  // Handle uncaught exceptions\n  process.on('uncaughtException', (error: Error) => {\n    console.error(''); // Empty line\n    printError('Uncaught exception', {\n      error: error.message,\n      name: error.name,\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'uncaughtException',\n      },\n    });\n  });\n\n  // Handle unhandled promise rejections\n  process.on('unhandledRejection', (reason: unknown) => {\n    console.error(''); // Empty line\n    printError('Unhandled promise rejection', {\n      reason: reason instanceof Error ? reason.message : String(reason),\n    });\n\n    const error = reason instanceof Error ? reason : new Error(String(reason));\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        type: 'unhandledRejection',\n      },\n    });\n  });\n\n  // Handle SIGINT (Ctrl+C)\n  process.on('SIGINT', () => {\n    console.log(''); // Empty line\n    printWarning('Process interrupted by user (SIGINT)');\n    console.log(''); // Empty line\n    process.exit(130); // Standard exit code for SIGINT\n  });\n\n  // Handle SIGTERM\n  process.on('SIGTERM', () => {\n    console.log(''); // Empty line\n    printWarning('Process terminated (SIGTERM)');\n    console.log(''); // Empty line\n    process.exit(143); // Standard exit code for SIGTERM\n  });\n}\n\n/**\n * Initialize CLI with options\n *\n * @param options - CLI initialization options\n */\nexport function initializeCli(options: CliOptions = {}): void {\n  const {\n    showBanner = true,\n    minimalBanner = false,\n    debug = false,\n    verbose = false,\n    quiet = false,\n    noColor = false,\n  } = options;\n\n  // Handle color output\n  if (noColor) {\n    process.env.NO_COLOR = '1';\n  }\n\n  // Display banner\n  if (showBanner && !quiet) {\n    if (minimalBanner) {\n      miniBanner();\n    } else {\n      banner({\n        minimal: false,\n        showDescription: true,\n        showTimestamp: debug || verbose,\n        showExecutionId: debug,\n      });\n    }\n  }\n\n  // Display startup info\n  displayStartupInfo({ debug, verbose, quiet });\n\n  // Check environment\n  if (!quiet) {\n    checkEnvironment();\n  }\n\n  // Setup error handlers\n  setupErrorHandlers();\n\n  // Register commands\n  try {\n    registerCommands();\n    printDebug('CLI initialization complete');\n  } catch (error) {\n    printError('CLI initialization failed', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.CONFIGURATION, {\n      debug: debug || isDebugMode(),\n      exit: true,\n      context: {\n        stage: 'initialization',\n      },\n    });\n  }\n}\n\n/**\n * Parse command line arguments\n */\nfunction parseArguments(): void {\n  try {\n    program.parse(process.argv);\n\n    // Show help if no command provided\n    const args = process.argv.slice(2);\n    if (args.length === 0) {\n      printInfo('No command specified - displaying help');\n      console.log(''); // Empty line\n      program.outputHelp();\n      process.exit(0);\n    }\n\n    // Check if command exists\n    const command = args[0];\n    if (command && !command.startsWith('-')) {\n      const commandExists = program.commands.some(cmd => cmd.name() === command);\n\n      if (!commandExists) {\n        console.log(''); // Empty line\n        printError(`Unknown command: ${command}`);\n        console.log(''); // Empty line\n        printInfo('Available commands:');\n        program.commands.forEach(cmd => {\n          console.log(`  • ${cmd.name()} - ${cmd.description()}`);\n        });\n        console.log(''); // Empty line\n        process.exit(1);\n      }\n    }\n\n    printDebug('Command line arguments parsed successfully');\n  } catch (error) {\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(program.opts()),\n      exit: true,\n      context: {\n        stage: 'argument-parsing',\n        argv: process.argv,\n      },\n    });\n  }\n}\n\n/**\n * Main CLI entry point\n *\n * @param options - CLI initialization options\n *\n * @example\n * ```typescript\n * // Standard initialization\n * main();\n *\n * // Minimal mode\n * main({ minimalBanner: true, quiet: true });\n *\n * // Debug mode\n * main({ debug: true, verbose: true });\n * ```\n */\nexport function main(options: CliOptions = {}): void {\n  try {\n    // Initialize CLI\n    initializeCli(options);\n\n    // Parse arguments and execute command\n    parseArguments();\n\n    // If we reach here, command completed successfully\n    printDebug('CLI execution completed successfully');\n  } catch (error) {\n    // Final catch-all error handler\n    console.error(''); // Empty line\n    printError('Fatal CLI error', {\n      error: error instanceof Error ? error.message : String(error),\n    });\n\n    handleError(error, ERROR_CODES.GENERAL, {\n      debug: isDebugMode(options),\n      exit: true,\n      context: {\n        stage: 'main',\n        options,\n      },\n    });\n  }\n}\n\n/**\n * Run CLI if executed directly\n */\nif (require.main === module) {\n  // Parse CLI options from environment or command line\n  const options: CliOptions = {\n    debug: process.env.DEBUG === 'true',\n    quiet: process.argv.includes('--quiet'),\n    verbose: process.argv.includes('--verbose'),\n    noColor: process.argv.includes('--no-color') || process.env.NO_COLOR === '1',\n  };\n\n  main(options);\n}"]}
@@ -134,4 +134,4 @@ function auditTokens(program) {
134
134
  }
135
135
  });
136
136
  }
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-tokens.js","sourceRoot":"","sources":["../../src/commands/audit-tokens.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { CoreConstants } from '@pipeline-builder/pipeline-core';\nimport { Command } from 'commander';\nimport { decodeTokenPayload } from '../utils/auth-guard';\nimport { getSecretValue, listSecrets } from '../utils/aws-secrets';\nimport { printCommandHeader } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printError, printInfo, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\ninterface TokenAuditEntry {\n  secretName: string;\n  arn: string;\n  expiresAt: Date;\n  daysUntilExpiry: number;\n  status: 'expired' | 'expiring-soon' | 'ok';\n}\n\n/**\n * Registers the `audit-tokens` command with the CLI program.\n *\n * Scans AWS Secrets Manager for stored platform JWTs (matching the\n * `pipeline-builder/<orgId>/platform` naming convention used by `store-token`)\n * and reports which secrets are expired or close to expiring.\n *\n * Designed for cron use:\n *   - Exits 0 when nothing is at-risk.\n *   - Exits 1 when at least one secret is expired or expires within `--warn-days`.\n *   - Exits 2 on AWS errors / scan failures.\n *\n * @example\n * ```bash\n * pipeline-manager audit-tokens --region us-east-1 --warn-days 7\n * pipeline-manager audit-tokens --region us-east-1 --json\n * ```\n */\nexport function auditTokens(program: Command): void {\n  program\n    .command('audit-tokens')\n    .description('Scan stored platform tokens in AWS Secrets Manager and report upcoming expirations')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--warn-days <days>', 'Flag tokens expiring within N days as at-risk', '7')\n    .option('--prefix <prefix>', 'Secrets Manager name prefix to scan', `${CoreConstants.SECRETS_PATH_PREFIX}/`)\n    .option('--json', 'Output results as JSON', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('Audit Tokens');\n      const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n      const warnDays = parseInt(options.warnDays, 10);\n      if (!Number.isFinite(warnDays) || warnDays < 0) {\n        printError('Invalid --warn-days value', { provided: options.warnDays });\n        process.exit(2);\n      }\n\n      try {\n        printInfo('Listing secrets', { region, prefix: options.prefix });\n        const secrets = await listSecrets(options.prefix, { region, profile: options.profile });\n\n        // Filter to ones following the `<prefix>/<orgId>/platform` pattern.\n        const platformSecrets = secrets.filter((s) => s.name.endsWith('/platform'));\n\n        const entries: TokenAuditEntry[] = [];\n        const now = Date.now();\n        const warnCutoff = now + warnDays * 24 * 60 * 60 * 1000;\n\n        for (const s of platformSecrets) {\n          let raw: string;\n          try {\n            raw = await getSecretValue(s.name, { region, profile: options.profile });\n          } catch (err) {\n            printWarning(`Could not read ${s.name}: ${err instanceof Error ? err.message : String(err)}`);\n            continue;\n          }\n\n          let parsed: { accessToken?: string; expiresAt?: string };\n          try {\n            parsed = JSON.parse(raw);\n          } catch {\n            printWarning(`Secret ${s.name} is not valid JSON, skipping`);\n            continue;\n          }\n\n          // Prefer the explicit expiresAt field written by store-token; fall back\n          // to decoding the JWT's exp claim if the field is missing.\n          let expiresAt: Date | undefined;\n          if (parsed.expiresAt) {\n            const d = new Date(parsed.expiresAt);\n            if (!Number.isNaN(d.getTime())) expiresAt = d;\n          }\n          if (!expiresAt && parsed.accessToken) {\n            const payload = decodeTokenPayload(parsed.accessToken);\n            if (payload?.exp && typeof payload.exp === 'number') {\n              expiresAt = new Date(payload.exp * 1000);\n            }\n          }\n          if (!expiresAt) {\n            printWarning(`Secret ${s.name} has no expiry information, skipping`);\n            continue;\n          }\n\n          const daysUntilExpiry = Math.floor((expiresAt.getTime() - now) / (24 * 60 * 60 * 1000));\n          const status: TokenAuditEntry['status'] =\n            expiresAt.getTime() < now ? 'expired'\n              : expiresAt.getTime() < warnCutoff ? 'expiring-soon'\n              : 'ok';\n          entries.push({ secretName: s.name, arn: s.arn, expiresAt, daysUntilExpiry, status });\n        }\n\n        const atRisk = entries.filter((e) => e.status !== 'ok');\n        const exitCode = atRisk.length > 0 ? 1 : 0;\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            scannedAt: new Date().toISOString(),\n            region,\n            warnDays,\n            totalScanned: entries.length,\n            atRiskCount: atRisk.length,\n            entries: entries.map((e) => ({ ...e, expiresAt: e.expiresAt.toISOString() })),\n            executionId,\n          }, null, 2));\n        } else {\n          printSection('Audit Results');\n          printInfo(`Scanned ${entries.length} platform secret${entries.length === 1 ? '' : 's'}`);\n          if (atRisk.length === 0) {\n            printSuccess(`All tokens valid for at least ${warnDays} days`);\n          } else {\n            for (const e of atRisk) {\n              const label = e.status === 'expired'\n                ? `EXPIRED ${Math.abs(e.daysUntilExpiry)} day${e.daysUntilExpiry === -1 ? '' : 's'} ago`\n                : `expires in ${e.daysUntilExpiry} day${e.daysUntilExpiry === 1 ? '' : 's'}`;\n              printWarning(`${e.secretName} — ${label}`);\n            }\n            printError(`${atRisk.length} secret${atRisk.length === 1 ? '' : 's'} need rotation. Run \\`pipeline-manager store-token --days <N>\\` to refresh.`);\n          }\n        }\n        process.exit(exitCode);\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: false,\n          context: { command: 'audit-tokens', executionId },\n        });\n        process.exit(2);\n      }\n    });\n}\n"]}
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit-tokens.js","sourceRoot":"","sources":["../../src/commands/audit-tokens.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { CoreConstants } from '@pipeline-builder/pipeline-core';\nimport { Command } from 'commander';\nimport { decodeTokenPayload } from '../utils/auth-guard';\nimport { getSecretValue, listSecrets } from '../utils/aws-secrets';\nimport { printCommandHeader } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printError, printInfo, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\ninterface TokenAuditEntry {\n  secretName: string;\n  arn: string;\n  expiresAt: Date;\n  daysUntilExpiry: number;\n  status: 'expired' | 'expiring-soon' | 'ok';\n}\n\n/**\n * Registers the `audit-tokens` command with the CLI program.\n *\n * Scans AWS Secrets Manager for stored platform JWTs (matching the\n * `pipeline-builder/<orgId>/platform` naming convention used by `store-token`)\n * and reports which secrets are expired or close to expiring.\n *\n * Designed for cron use:\n *   - Exits 0 when nothing is at-risk.\n *   - Exits 1 when at least one secret is expired or expires within `--warn-days`.\n *   - Exits 2 on AWS errors / scan failures.\n *\n * @example\n * ```bash\n * pipeline-manager audit-tokens --region us-east-1 --warn-days 7\n * pipeline-manager audit-tokens --region us-east-1 --json\n * ```\n */\nexport function auditTokens(program: Command): void {\n  program\n    .command('audit-tokens')\n    .description('Scan stored platform tokens in AWS Secrets Manager and report upcoming expirations')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--warn-days <days>', 'Flag tokens expiring within N days as at-risk', '7')\n    .option('--prefix <prefix>', 'Secrets Manager name prefix to scan', `${CoreConstants.SECRETS_PATH_PREFIX}/`)\n    .option('--json', 'Output results as JSON', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('Audit Tokens');\n      const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n      const warnDays = parseInt(options.warnDays, 10);\n      if (!Number.isFinite(warnDays) || warnDays < 0) {\n        printError('Invalid --warn-days value', { provided: options.warnDays });\n        process.exit(2);\n      }\n\n      try {\n        printInfo('Listing secrets', { region, prefix: options.prefix });\n        const secrets = await listSecrets(options.prefix, { region, profile: options.profile });\n\n        // Filter to ones following the `<prefix>/<orgId>/platform` pattern.\n        const platformSecrets = secrets.filter((s) => s.name.endsWith('/platform'));\n\n        const entries: TokenAuditEntry[] = [];\n        const now = Date.now();\n        const warnCutoff = now + warnDays * 24 * 60 * 60 * 1000;\n\n        for (const s of platformSecrets) {\n          let raw: string;\n          try {\n            raw = await getSecretValue(s.name, { region, profile: options.profile });\n          } catch (err) {\n            printWarning(`Could not read ${s.name}: ${err instanceof Error ? err.message : String(err)}`);\n            continue;\n          }\n\n          let parsed: { accessToken?: string; expiresAt?: string };\n          try {\n            parsed = JSON.parse(raw);\n          } catch {\n            printWarning(`Secret ${s.name} is not valid JSON, skipping`);\n            continue;\n          }\n\n          // Prefer the explicit expiresAt field written by store-token; fall back\n          // to decoding the JWT's exp claim if the field is missing.\n          let expiresAt: Date | undefined;\n          if (parsed.expiresAt) {\n            const d = new Date(parsed.expiresAt);\n            if (!Number.isNaN(d.getTime())) expiresAt = d;\n          }\n          if (!expiresAt && parsed.accessToken) {\n            const payload = decodeTokenPayload(parsed.accessToken);\n            if (payload?.exp && typeof payload.exp === 'number') {\n              expiresAt = new Date(payload.exp * 1000);\n            }\n          }\n          if (!expiresAt) {\n            printWarning(`Secret ${s.name} has no expiry information, skipping`);\n            continue;\n          }\n\n          const daysUntilExpiry = Math.floor((expiresAt.getTime() - now) / (24 * 60 * 60 * 1000));\n          const status: TokenAuditEntry['status'] =\n            expiresAt.getTime() < now ? 'expired'\n              : expiresAt.getTime() < warnCutoff ? 'expiring-soon'\n                : 'ok';\n          entries.push({ secretName: s.name, arn: s.arn, expiresAt, daysUntilExpiry, status });\n        }\n\n        const atRisk = entries.filter((e) => e.status !== 'ok');\n        const exitCode = atRisk.length > 0 ? 1 : 0;\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            scannedAt: new Date().toISOString(),\n            region,\n            warnDays,\n            totalScanned: entries.length,\n            atRiskCount: atRisk.length,\n            entries: entries.map((e) => ({ ...e, expiresAt: e.expiresAt.toISOString() })),\n            executionId,\n          }, null, 2));\n        } else {\n          printSection('Audit Results');\n          printInfo(`Scanned ${entries.length} platform secret${entries.length === 1 ? '' : 's'}`);\n          if (atRisk.length === 0) {\n            printSuccess(`All tokens valid for at least ${warnDays} days`);\n          } else {\n            for (const e of atRisk) {\n              const label = e.status === 'expired'\n                ? `EXPIRED ${Math.abs(e.daysUntilExpiry)} day${e.daysUntilExpiry === -1 ? '' : 's'} ago`\n                : `expires in ${e.daysUntilExpiry} day${e.daysUntilExpiry === 1 ? '' : 's'}`;\n              printWarning(`${e.secretName} — ${label}`);\n            }\n            printError(`${atRisk.length} secret${atRisk.length === 1 ? '' : 's'} need rotation. Run \\`pipeline-manager store-token --days <N>\\` to refresh.`);\n          }\n        }\n        process.exit(exitCode);\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: false,\n          context: { command: 'audit-tokens', executionId },\n        });\n        process.exit(2);\n      }\n    });\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"deploy.d.ts","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAYpC;;;;;;;;;;;GAWG;AACH,wBAAgB,MAAM,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA6O7C"}
1
+ {"version":3,"file":"deploy.d.ts","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAapC;;;;;;;;;;;GAWG;AACH,wBAAgB,MAAM,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAgQ7C"}
@@ -6,8 +6,6 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
6
6
  };
7
7
  Object.defineProperty(exports, "__esModule", { value: true });
8
8
  exports.deploy = deploy;
9
- const crypto_1 = require("crypto");
10
- const client_sts_1 = require("@aws-sdk/client-sts");
11
9
  const picocolors_1 = __importDefault(require("picocolors"));
12
10
  const cli_constants_1 = require("../config/cli.constants");
13
11
  const audit_log_1 = require("../utils/audit-log");
@@ -15,6 +13,7 @@ const cdk_utils_1 = require("../utils/cdk-utils");
15
13
  const command_utils_1 = require("../utils/command-utils");
16
14
  const error_handler_1 = require("../utils/error-handler");
17
15
  const output_utils_1 = require("../utils/output-utils");
16
+ const registry_1 = require("../utils/registry");
18
17
  const { bold, cyan, dim } = picocolors_1.default;
19
18
  /**
20
19
  * Registers the `deploy` command with the CLI program.
@@ -197,32 +196,53 @@ function deploy(program) {
197
196
  (0, output_utils_1.printInfo)('Skipping pipeline registry (local-spec mode)');
198
197
  return;
199
198
  }
199
+ // Build the payload up-front so a registration POST failure can write
200
+ // the same payload to a pending-intent file for `pipeline-manager
201
+ // register` to drain later. We never want to retry STS lookups —
202
+ // they can fail too (e.g. credential rotation) and would compound
203
+ // the issue.
204
+ if (!pipeline.orgId) {
205
+ (0, output_utils_1.printWarning)('Pipeline has no orgId — skipping registration');
206
+ return;
207
+ }
208
+ let payload;
200
209
  try {
201
- const stsClient = new client_sts_1.STSClient({ region: process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION });
202
- const identity = await stsClient.send(new client_sts_1.GetCallerIdentityCommand({}));
203
- const account = identity.Account ?? '';
204
- const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';
205
- const pipelineName = pipeline.pipelineName
206
- || `${pipeline.organization}-${pipeline.project}-pipeline`.toLowerCase();
207
- const hashedAccount = (0, crypto_1.createHash)('sha256').update(account).digest('hex').slice(0, 12);
208
- const pipelineArn = `arn:aws:codepipeline:${region}:${hashedAccount}:${pipelineName}`;
209
- await platformClient.post(`${platformConfig.api.pipelineUrl}/registry`, {
210
- pipelineId: pipeline.id,
210
+ payload = await (0, registry_1.buildRegistryPayload)({
211
+ id: pipeline.id,
211
212
  orgId: pipeline.orgId,
212
- pipelineArn,
213
- pipelineName,
214
- accountId: hashedAccount,
215
- region,
213
+ pipelineName: pipeline.pipelineName,
216
214
  project: pipeline.project,
217
215
  organization: pipeline.organization,
218
- stackName: `${pipeline.project}-${pipeline.organization}`.toLowerCase(),
216
+ }, options.region);
217
+ }
218
+ catch (buildError) {
219
+ (0, output_utils_1.printWarning)('Could not build registry payload — skipping registration', {
220
+ error: buildError instanceof Error ? buildError.message : String(buildError),
219
221
  });
220
- (0, output_utils_1.printSuccess)('Pipeline registered for event reporting', { arn: pipelineArn });
222
+ return;
223
+ }
224
+ try {
225
+ await platformClient.post(`${platformConfig.api.pipelineUrl}/registry`, payload);
226
+ (0, output_utils_1.printSuccess)('Pipeline registered for event reporting', { arn: payload.pipelineArn });
221
227
  }
222
228
  catch (regError) {
223
- (0, output_utils_1.printWarning)('Pipeline registry update failed (reporting may be incomplete)', {
224
- error: regError instanceof Error ? regError.message : String(regError),
225
- });
229
+ // Persist for retry. The user can drain with `pipeline-manager
230
+ // register` (or just re-run that command at any time — it's
231
+ // idempotent). The deploy itself does NOT fail.
232
+ try {
233
+ const intentPath = await (0, registry_1.writePendingIntent)(payload);
234
+ (0, output_utils_1.printWarning)('Pipeline registry update failed; queued for retry', {
235
+ error: regError instanceof Error ? regError.message : String(regError),
236
+ retry: 'pipeline-manager register',
237
+ intent: intentPath,
238
+ });
239
+ }
240
+ catch (writeErr) {
241
+ (0, output_utils_1.printWarning)('Pipeline registry update failed (retry queue also failed)', {
242
+ error: regError instanceof Error ? regError.message : String(regError),
243
+ queueError: writeErr instanceof Error ? writeErr.message : String(writeErr),
244
+ });
245
+ }
226
246
  }
227
247
  }
228
248
  }
@@ -240,4 +260,4 @@ function deploy(program) {
240
260
  }
241
261
  });
242
262
  }
243
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createHash } from 'crypto';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\nimport { Command } from 'commander';\nimport pico from 'picocolors';\nimport { assertShellSafe } from '../config/cli.constants';\nimport { Pipeline, PipelineResponse } from '../types';\nimport { auditLog } from '../utils/audit-log';\nimport { ensureCdkAvailable, executeCdkShellCommand, resolveBoilerplatePath } from '../utils/cdk-utils';\nimport { printCommandHeader, printSslWarning, createAuthenticatedClientAsync } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { ensureOutputDirectory, extractSingleResponse, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\nconst { bold, cyan, dim } = pico;\n\n/**\n * Registers the `deploy` command with the CLI program.\n *\n * Fetches pipeline properties by ID from the platform API, then\n * runs `cdk deploy` to provision the pipeline infrastructure in AWS.\n * For synthesis only, use `pipeline-manager synth`.\n *\n * Requires service credentials to be pre-stored in AWS Secrets Manager.\n * Create them first with: `pipeline-manager store-token`\n *\n * @param program - The root Commander program instance to attach the command to.\n */\nexport function deploy(program: Command): void {\n  program\n    .command('deploy')\n    .description('Deploy pipeline by ID using AWS CDK, or --local-spec to deploy a local pipeline.json without the platform')\n    .option('-i, --id <id>', 'Pipeline ID (fetches config from the platform)')\n    .option('--local-spec <path>', 'Path to a local pipeline.json — deploys without contacting the platform (no auth, no compliance, no plugin lookup)')\n    .option('--profile <profile>', 'AWS profile', 'default')\n    .option('--require-approval <approval>', 'Approval level: never|any-change|broadening', 'never')\n    .option('--output <dir>', 'CDK output directory', 'cdk.out')\n    .option('--store-tokens', 'Authenticate using token from AWS Secrets Manager (requires PLATFORM_SECRET_NAME env var)', false)\n    .option('--region <region>', 'AWS region (for --store-tokens)')\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .option('--show-resolved', 'Print the resolved pipeline config (with {{ ... }} templates expanded) and exit without deploying', false)\n    .action(async (options) => {\n      // Mutually exclusive input sources\n      if (!options.id && !options.localSpec) {\n        throw new Error('Either --id <pipeline-id> or --local-spec <path> is required');\n      }\n      if (options.id && options.localSpec) {\n        throw new Error('--id and --local-spec are mutually exclusive');\n      }\n\n      const executionId = printCommandHeader('Pipeline Deploy');\n\n      try {\n        auditLog('deploy', { executionId, pipelineId: options.id, profile: options.profile });\n\n        printInfo('Deployment parameters', {\n          id: options.id,\n          awsProfile: options.profile,\n          outputDir: options.output,\n          requireApproval: options.requireApproval,\n          verifySsl: options.verifySsl,\n        });\n\n        // Security warning for SSL verification disabled\n        printSslWarning(options.verifySsl);\n\n        // Propagate to process.env so CDK constructs (Lambda, CodeBuild) inherit it\n        if (options.verifySsl === false) {\n          process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';\n        }\n\n        ensureCdkAvailable();\n        printSuccess('AWS CDK is available');\n\n        let pipeline: Pipeline;\n        let propsWithIds: Record<string, unknown>;\n        // Remote-mode registry-post handles — null in --local-spec mode.\n        let platformClient: Awaited<ReturnType<typeof createAuthenticatedClientAsync>> | undefined;\n        let platformConfig: { api: { pipelineUrl: string } } | undefined;\n\n        if (options.localSpec) {\n          // --local-spec: read pipeline.json from disk; no platform contact.\n          // Compliance / quota / plugin-lookup features all require the platform —\n          // this mode is for air-gapped or simple standalone CDK deployments.\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const fsMod = require('fs');\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const pathMod = require('path');\n          const absPath = pathMod.resolve(options.localSpec);\n          printInfo('Loading local pipeline spec', { path: absPath });\n          if (!fsMod.existsSync(absPath)) {\n            throw new Error(`Local spec file not found: ${absPath}`);\n          }\n          const raw = fsMod.readFileSync(absPath, 'utf-8');\n          const parsed = JSON.parse(raw) as Partial<Pipeline> & { props?: Record<string, unknown> };\n          if (!parsed.props) {\n            throw new Error(`Local spec file is missing required 'props' field: ${absPath}`);\n          }\n          pipeline = {\n            id: parsed.id ?? 'local',\n            project: parsed.project ?? 'local-project',\n            organization: parsed.organization ?? 'local-org',\n            orgId: parsed.orgId,\n            isDefault: parsed.isDefault ?? false,\n            isActive: parsed.isActive ?? true,\n            props: parsed.props,\n          } as Pipeline;\n          printSuccess('Local spec loaded');\n          printKeyValue({\n            Source: absPath,\n            Project: pipeline.project,\n            Organization: pipeline.organization,\n          });\n          propsWithIds = {\n            ...pipeline.props,\n            ...(pipeline.orgId && { orgId: pipeline.orgId }),\n            pipelineId: pipeline.id,\n          };\n        } else {\n          // Remote path: fetch config from platform API\n          platformClient = await createAuthenticatedClientAsync(options);\n          platformConfig = platformClient.getConfig() as { api: { pipelineUrl: string } };\n\n          printInfo('Fetching pipeline configuration', { id: options.id });\n          const response = await platformClient.get<PipelineResponse>(\n            `${platformConfig.api.pipelineUrl}/${options.id}`,\n          );\n\n          const fetched = extractSingleResponse<Pipeline>(response, 'pipeline', 'props');\n\n          if (!fetched?.props) {\n            printError('Invalid pipeline response', {\n              id: options.id,\n              hasProps: !!fetched?.props,\n              responseKeys: response ? Object.keys(response) : '(null)',\n            });\n            throw new Error(`Failed to retrieve valid pipeline properties for ID: ${options.id}`);\n          }\n          pipeline = fetched;\n\n          printSuccess('Pipeline configuration retrieved');\n          printKeyValue({\n            'ID': pipeline.id,\n            'Project': pipeline.project,\n            'Organization': pipeline.organization,\n            'Is Default': pipeline.isDefault,\n            'Is Active': pipeline.isActive,\n          });\n\n          propsWithIds = {\n            ...pipeline.props,\n            ...(pipeline.orgId && { orgId: pipeline.orgId }),\n            pipelineId: pipeline.id,\n          };\n        }\n\n        // --show-resolved: print resolved config and exit (no CDK deploy)\n        if ((options as { showResolved?: boolean }).showResolved) {\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const { resolveSelfReferencing } = require('@pipeline-builder/pipeline-core');\n          const propsAny = propsWithIds as Record<string, unknown>;\n          const scope = { metadata: propsAny.metadata ?? {}, vars: propsAny.vars ?? {} };\n          const isTpl = (f: string) => f === 'projectName' || f.startsWith('metadata.') || f.startsWith('vars.');\n          const result = resolveSelfReferencing(propsAny, scope, isTpl, (f: string) => isTpl(f) ? f : null, 'pipeline');\n          if (result.errors.length) {\n            console.error('Resolution errors:');\n            for (const e of result.errors) console.error(`  [${e.field ?? '?'}] ${e.message}`);\n            process.exit(1);\n          }\n          console.log(JSON.stringify(propsWithIds, null, 2));\n          return;\n        }\n\n        const encoded = Buffer.from(JSON.stringify(propsWithIds), 'utf-8').toString('base64');\n        const outputPath = options.output;\n\n        // Ensure output directory exists\n        printInfo('Preparing output directory', { path: outputPath });\n        ensureOutputDirectory(outputPath);\n\n        // Build CDK command (validate inputs that flow into shell)\n        if (options.profile) assertShellSafe(options.profile, 'profile');\n        assertShellSafe(outputPath, 'output');\n\n        const scriptPath = resolveBoilerplatePath(__dirname);\n        const profileArg = options.profile ? `--profile=${options.profile}` : '';\n        const outputArg = `--output=${outputPath}`;\n        const appArg = `--app=\"node ${scriptPath}\"`;\n\n        const command = `cdk deploy ${profileArg} --require-approval=${options.requireApproval} ${outputArg} --notices=false ${appArg}`;\n\n        printSection('CDK Execution');\n        console.log(cyan(bold('Command:')), dim(command.split(' --')[0] + ' ...'));\n        console.log(''); // Empty line\n\n        // Execute CDK command\n        const result = executeCdkShellCommand(command, {\n          debug: program.opts().debug,\n          showOutput: true,\n          env: { PIPELINE_PROPS: encoded },\n        });\n\n        console.log(''); // Empty line\n        printSection('Deployment Complete');\n\n        if (result.success) {\n          printKeyValue({\n            'Execution ID': executionId,\n            'Duration': `${result.duration}ms`,\n            'Output Directory': outputPath,\n            'Status': '✓ Success',\n          });\n\n          // Register pipeline ARN for event reporting (non-blocking).\n          // Skipped in --local-spec mode since there's no platform to register with.\n          if (!platformClient || !platformConfig) {\n            printInfo('Skipping pipeline registry (local-spec mode)');\n            return;\n          }\n          try {\n            const stsClient = new STSClient({ region: process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION });\n            const identity = await stsClient.send(new GetCallerIdentityCommand({}));\n            const account = identity.Account ?? '';\n            const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n\n            const pipelineName = pipeline.pipelineName\n              || `${pipeline.organization}-${pipeline.project}-pipeline`.toLowerCase();\n\n            const hashedAccount = createHash('sha256').update(account).digest('hex').slice(0, 12);\n            const pipelineArn = `arn:aws:codepipeline:${region}:${hashedAccount}:${pipelineName}`;\n\n            await platformClient.post(`${platformConfig.api.pipelineUrl}/registry`, {\n              pipelineId: pipeline.id,\n              orgId: pipeline.orgId,\n              pipelineArn,\n              pipelineName,\n              accountId: hashedAccount,\n              region,\n              project: pipeline.project,\n              organization: pipeline.organization,\n              stackName: `${pipeline.project}-${pipeline.organization}`.toLowerCase(),\n            });\n\n            printSuccess('Pipeline registered for event reporting', { arn: pipelineArn });\n          } catch (regError) {\n            printWarning('Pipeline registry update failed (reporting may be incomplete)', {\n              error: regError instanceof Error ? regError.message : String(regError),\n            });\n          }\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'deploy',\n            executionId,\n            pipelineId: options.id,\n            verifySsl: options.verifySsl,\n          },\n        });\n      }\n    });\n}\n"]}
263
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Command } from 'commander';\nimport pico from 'picocolors';\nimport { assertShellSafe } from '../config/cli.constants';\nimport { Pipeline, PipelineResponse } from '../types';\nimport { auditLog } from '../utils/audit-log';\nimport { ensureCdkAvailable, executeCdkShellCommand, resolveBoilerplatePath } from '../utils/cdk-utils';\nimport { printCommandHeader, printSslWarning, createAuthenticatedClientAsync } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { ensureOutputDirectory, extractSingleResponse, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\nimport { buildRegistryPayload, writePendingIntent } from '../utils/registry';\n\nconst { bold, cyan, dim } = pico;\n\n/**\n * Registers the `deploy` command with the CLI program.\n *\n * Fetches pipeline properties by ID from the platform API, then\n * runs `cdk deploy` to provision the pipeline infrastructure in AWS.\n * For synthesis only, use `pipeline-manager synth`.\n *\n * Requires service credentials to be pre-stored in AWS Secrets Manager.\n * Create them first with: `pipeline-manager store-token`\n *\n * @param program - The root Commander program instance to attach the command to.\n */\nexport function deploy(program: Command): void {\n  program\n    .command('deploy')\n    .description('Deploy pipeline by ID using AWS CDK, or --local-spec to deploy a local pipeline.json without the platform')\n    .option('-i, --id <id>', 'Pipeline ID (fetches config from the platform)')\n    .option('--local-spec <path>', 'Path to a local pipeline.json — deploys without contacting the platform (no auth, no compliance, no plugin lookup)')\n    .option('--profile <profile>', 'AWS profile', 'default')\n    .option('--require-approval <approval>', 'Approval level: never|any-change|broadening', 'never')\n    .option('--output <dir>', 'CDK output directory', 'cdk.out')\n    .option('--store-tokens', 'Authenticate using token from AWS Secrets Manager (requires PLATFORM_SECRET_NAME env var)', false)\n    .option('--region <region>', 'AWS region (for --store-tokens)')\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .option('--show-resolved', 'Print the resolved pipeline config (with {{ ... }} templates expanded) and exit without deploying', false)\n    .action(async (options) => {\n      // Mutually exclusive input sources\n      if (!options.id && !options.localSpec) {\n        throw new Error('Either --id <pipeline-id> or --local-spec <path> is required');\n      }\n      if (options.id && options.localSpec) {\n        throw new Error('--id and --local-spec are mutually exclusive');\n      }\n\n      const executionId = printCommandHeader('Pipeline Deploy');\n\n      try {\n        auditLog('deploy', { executionId, pipelineId: options.id, profile: options.profile });\n\n        printInfo('Deployment parameters', {\n          id: options.id,\n          awsProfile: options.profile,\n          outputDir: options.output,\n          requireApproval: options.requireApproval,\n          verifySsl: options.verifySsl,\n        });\n\n        // Security warning for SSL verification disabled\n        printSslWarning(options.verifySsl);\n\n        // Propagate to process.env so CDK constructs (Lambda, CodeBuild) inherit it\n        if (options.verifySsl === false) {\n          process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';\n        }\n\n        ensureCdkAvailable();\n        printSuccess('AWS CDK is available');\n\n        let pipeline: Pipeline;\n        let propsWithIds: Record<string, unknown>;\n        // Remote-mode registry-post handles — null in --local-spec mode.\n        let platformClient: Awaited<ReturnType<typeof createAuthenticatedClientAsync>> | undefined;\n        let platformConfig: { api: { pipelineUrl: string } } | undefined;\n\n        if (options.localSpec) {\n          // --local-spec: read pipeline.json from disk; no platform contact.\n          // Compliance / quota / plugin-lookup features all require the platform —\n          // this mode is for air-gapped or simple standalone CDK deployments.\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const fsMod = require('fs');\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const pathMod = require('path');\n          const absPath = pathMod.resolve(options.localSpec);\n          printInfo('Loading local pipeline spec', { path: absPath });\n          if (!fsMod.existsSync(absPath)) {\n            throw new Error(`Local spec file not found: ${absPath}`);\n          }\n          const raw = fsMod.readFileSync(absPath, 'utf-8');\n          const parsed = JSON.parse(raw) as Partial<Pipeline> & { props?: Record<string, unknown> };\n          if (!parsed.props) {\n            throw new Error(`Local spec file is missing required 'props' field: ${absPath}`);\n          }\n          pipeline = {\n            id: parsed.id ?? 'local',\n            project: parsed.project ?? 'local-project',\n            organization: parsed.organization ?? 'local-org',\n            orgId: parsed.orgId,\n            isDefault: parsed.isDefault ?? false,\n            isActive: parsed.isActive ?? true,\n            props: parsed.props,\n          } as Pipeline;\n          printSuccess('Local spec loaded');\n          printKeyValue({\n            Source: absPath,\n            Project: pipeline.project,\n            Organization: pipeline.organization,\n          });\n          propsWithIds = {\n            ...pipeline.props,\n            ...(pipeline.orgId && { orgId: pipeline.orgId }),\n            pipelineId: pipeline.id,\n          };\n        } else {\n          // Remote path: fetch config from platform API\n          platformClient = await createAuthenticatedClientAsync(options);\n          platformConfig = platformClient.getConfig() as { api: { pipelineUrl: string } };\n\n          printInfo('Fetching pipeline configuration', { id: options.id });\n          const response = await platformClient.get<PipelineResponse>(\n            `${platformConfig.api.pipelineUrl}/${options.id}`,\n          );\n\n          const fetched = extractSingleResponse<Pipeline>(response, 'pipeline', 'props');\n\n          if (!fetched?.props) {\n            printError('Invalid pipeline response', {\n              id: options.id,\n              hasProps: !!fetched?.props,\n              responseKeys: response ? Object.keys(response) : '(null)',\n            });\n            throw new Error(`Failed to retrieve valid pipeline properties for ID: ${options.id}`);\n          }\n          pipeline = fetched;\n\n          printSuccess('Pipeline configuration retrieved');\n          printKeyValue({\n            'ID': pipeline.id,\n            'Project': pipeline.project,\n            'Organization': pipeline.organization,\n            'Is Default': pipeline.isDefault,\n            'Is Active': pipeline.isActive,\n          });\n\n          propsWithIds = {\n            ...pipeline.props,\n            ...(pipeline.orgId && { orgId: pipeline.orgId }),\n            pipelineId: pipeline.id,\n          };\n        }\n\n        // --show-resolved: print resolved config and exit (no CDK deploy)\n        if ((options as { showResolved?: boolean }).showResolved) {\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          const { resolveSelfReferencing } = require('@pipeline-builder/pipeline-core');\n          const propsAny = propsWithIds as Record<string, unknown>;\n          const scope = { metadata: propsAny.metadata ?? {}, vars: propsAny.vars ?? {} };\n          const isTpl = (f: string) => f === 'projectName' || f.startsWith('metadata.') || f.startsWith('vars.');\n          const result = resolveSelfReferencing(propsAny, scope, isTpl, (f: string) => isTpl(f) ? f : null, 'pipeline');\n          if (result.errors.length) {\n            console.error('Resolution errors:');\n            for (const e of result.errors) console.error(`  [${e.field ?? '?'}] ${e.message}`);\n            process.exit(1);\n          }\n          console.log(JSON.stringify(propsWithIds, null, 2));\n          return;\n        }\n\n        const encoded = Buffer.from(JSON.stringify(propsWithIds), 'utf-8').toString('base64');\n        const outputPath = options.output;\n\n        // Ensure output directory exists\n        printInfo('Preparing output directory', { path: outputPath });\n        ensureOutputDirectory(outputPath);\n\n        // Build CDK command (validate inputs that flow into shell)\n        if (options.profile) assertShellSafe(options.profile, 'profile');\n        assertShellSafe(outputPath, 'output');\n\n        const scriptPath = resolveBoilerplatePath(__dirname);\n        const profileArg = options.profile ? `--profile=${options.profile}` : '';\n        const outputArg = `--output=${outputPath}`;\n        const appArg = `--app=\"node ${scriptPath}\"`;\n\n        const command = `cdk deploy ${profileArg} --require-approval=${options.requireApproval} ${outputArg} --notices=false ${appArg}`;\n\n        printSection('CDK Execution');\n        console.log(cyan(bold('Command:')), dim(command.split(' --')[0] + ' ...'));\n        console.log(''); // Empty line\n\n        // Execute CDK command\n        const result = executeCdkShellCommand(command, {\n          debug: program.opts().debug,\n          showOutput: true,\n          env: { PIPELINE_PROPS: encoded },\n        });\n\n        console.log(''); // Empty line\n        printSection('Deployment Complete');\n\n        if (result.success) {\n          printKeyValue({\n            'Execution ID': executionId,\n            'Duration': `${result.duration}ms`,\n            'Output Directory': outputPath,\n            'Status': '✓ Success',\n          });\n\n          // Register pipeline ARN for event reporting (non-blocking).\n          // Skipped in --local-spec mode since there's no platform to register with.\n          if (!platformClient || !platformConfig) {\n            printInfo('Skipping pipeline registry (local-spec mode)');\n            return;\n          }\n          // Build the payload up-front so a registration POST failure can write\n          // the same payload to a pending-intent file for `pipeline-manager\n          // register` to drain later. We never want to retry STS lookups —\n          // they can fail too (e.g. credential rotation) and would compound\n          // the issue.\n          if (!pipeline.orgId) {\n            printWarning('Pipeline has no orgId — skipping registration');\n            return;\n          }\n          let payload;\n          try {\n            payload = await buildRegistryPayload(\n              {\n                id: pipeline.id,\n                orgId: pipeline.orgId,\n                pipelineName: pipeline.pipelineName,\n                project: pipeline.project,\n                organization: pipeline.organization,\n              },\n              options.region,\n            );\n          } catch (buildError) {\n            printWarning('Could not build registry payload — skipping registration', {\n              error: buildError instanceof Error ? buildError.message : String(buildError),\n            });\n            return;\n          }\n\n          try {\n            await platformClient.post(`${platformConfig.api.pipelineUrl}/registry`, payload);\n            printSuccess('Pipeline registered for event reporting', { arn: payload.pipelineArn });\n          } catch (regError) {\n            // Persist for retry. The user can drain with `pipeline-manager\n            // register` (or just re-run that command at any time — it's\n            // idempotent). The deploy itself does NOT fail.\n            try {\n              const intentPath = await writePendingIntent(payload);\n              printWarning('Pipeline registry update failed; queued for retry', {\n                error: regError instanceof Error ? regError.message : String(regError),\n                retry: 'pipeline-manager register',\n                intent: intentPath,\n              });\n            } catch (writeErr) {\n              printWarning('Pipeline registry update failed (retry queue also failed)', {\n                error: regError instanceof Error ? regError.message : String(regError),\n                queueError: writeErr instanceof Error ? writeErr.message : String(writeErr),\n              });\n            }\n          }\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'deploy',\n            executionId,\n            pipelineId: options.id,\n            verifySsl: options.verifySsl,\n          },\n        });\n      }\n    });\n}\n"]}
@@ -0,0 +1,38 @@
1
+ import { Command } from 'commander';
2
+ /**
3
+ * Registers the `register` command with the CLI program.
4
+ *
5
+ * Two purposes — both targeted at recovering from a deploy whose registration
6
+ * step didn't land:
7
+ *
8
+ * 1. Explicit re-register (`--id <pipelineId>`) — recomputes the ARN from
9
+ * STS + region and POSTs it to the platform. Useful when the platform
10
+ * was unreachable during `deploy`, when the registry row was deleted by
11
+ * mistake, or for any other reason a deployed stack has no registry row.
12
+ *
13
+ * 2. Drain pending intents (every invocation) — `deploy` writes a local
14
+ * file under `~/.pipeline-manager/pending-registrations/` whenever its
15
+ * registration POST fails. Each `register` invocation tries to drain
16
+ * every such file. On success the file is deleted; on failure it stays
17
+ * and surfaces as a warning.
18
+ *
19
+ * Flags:
20
+ * - `--id <pipelineId>` — re-register a specific pipeline by ID
21
+ * - `--region <region>` — AWS region (default: AWS_REGION env / us-east-1)
22
+ * - `--no-drain` — skip draining pending intents
23
+ *
24
+ * Exit codes:
25
+ * - 0: requested registration succeeded AND no pending intents remain
26
+ * - 1: at least one registration (explicit or pending) still failed
27
+ *
28
+ * @example
29
+ * ```bash
30
+ * # Retry a registration that failed during deploy
31
+ * pipeline-manager register --id pipe-123
32
+ *
33
+ * # Just drain pending intents from disk, no specific registration
34
+ * pipeline-manager register
35
+ * ```
36
+ */
37
+ export declare function register(program: Command): void;
38
+ //# sourceMappingURL=register.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"register.d.ts","sourceRoot":"","sources":["../../src/commands/register.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAapC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkCG;AACH,wBAAgB,QAAQ,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA2G/C"}