@pipeline-builder/pipeline-manager 3.3.32 → 3.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -82,7 +82,6 @@ Run `pipeline-manager <command> --help` for the full flag reference on any comma
82
82
  | `PLATFORM_TOKEN` | Yes (for API ops) | Auth token for the Pipeline Builder platform |
83
83
  | `PLATFORM_URL` | Yes (for API ops) | Base URL of your platform deployment |
84
84
  | `AWS_REGION` | Yes (for deploy) | Target AWS region for `synth` / `deploy` |
85
- | `RESOLVED_SYNTH_PLUGIN` | No | `true` inside CodePipeline so the synth step resolves plugins via the platform; defaults to `false` for local CLI runs |
86
85
 
87
86
  Full reference: [Environment Variables](https://mwashburn160.github.io/pipeline-builder/docs/environment-variables).
88
87
 
@@ -1 +1 @@
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"}
1
+ {"version":3,"file":"deploy.d.ts","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAcpC;;;;;;;;;;;GAWG;AACH,wBAAgB,MAAM,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA2Q7C"}
@@ -13,6 +13,7 @@ const cdk_utils_1 = require("../utils/cdk-utils");
13
13
  const command_utils_1 = require("../utils/command-utils");
14
14
  const error_handler_1 = require("../utils/error-handler");
15
15
  const output_utils_1 = require("../utils/output-utils");
16
+ const plugin_resolver_1 = require("../utils/plugin-resolver");
16
17
  const registry_1 = require("../utils/registry");
17
18
  const { bold, cyan, dim } = picocolors_1.default;
18
19
  /**
@@ -140,6 +141,16 @@ function deploy(program) {
140
141
  ...(pipeline.orgId && { orgId: pipeline.orgId }),
141
142
  pipelineId: pipeline.id,
142
143
  };
144
+ // Pre-resolve plugins from the platform API so the synthesized
145
+ // template ships with real CodeBuild image URIs baked in. Without
146
+ // this, CDK falls back to standard:7.0 (the deploy-time custom
147
+ // resource attribute is unresolvable at synth time). Skipped in
148
+ // --local-spec mode where there's no platform client.
149
+ const resolvedPlugins = await (0, plugin_resolver_1.resolvePluginsForProps)(platformClient, propsWithIds);
150
+ if (Object.keys(resolvedPlugins).length > 0) {
151
+ propsWithIds.resolvedPlugins = resolvedPlugins;
152
+ (0, output_utils_1.printInfo)('Pre-resolved plugins', { count: Object.keys(resolvedPlugins).length });
153
+ }
143
154
  }
144
155
  // --show-resolved: print resolved config and exit (no CDK deploy)
145
156
  if (options.showResolved) {
@@ -260,4 +271,4 @@ function deploy(program) {
260
271
  }
261
272
  });
262
273
  }
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"]}
274
+ //# 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 { resolvePluginsForProps } from '../utils/plugin-resolver';\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          // Pre-resolve plugins from the platform API so the synthesized\n          // template ships with real CodeBuild image URIs baked in. Without\n          // this, CDK falls back to standard:7.0 (the deploy-time custom\n          // resource attribute is unresolvable at synth time). Skipped in\n          // --local-spec mode where there's no platform client.\n          const resolvedPlugins = await resolvePluginsForProps(platformClient, propsWithIds);\n          if (Object.keys(resolvedPlugins).length > 0) {\n            propsWithIds.resolvedPlugins = resolvedPlugins;\n            printInfo('Pre-resolved plugins', { count: Object.keys(resolvedPlugins).length });\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"]}
@@ -1 +1 @@
1
- {"version":3,"file":"store-token.d.ts","sourceRoot":"","sources":["../../src/commands/store-token.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAwBpC;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,UAAU,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAmLjD"}
1
+ {"version":3,"file":"store-token.d.ts","sourceRoot":"","sources":["../../src/commands/store-token.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAwBpC;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,UAAU,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA+LjD"}
@@ -146,8 +146,19 @@ function storeToken(program) {
146
146
  }
147
147
  (0, output_utils_1.printSuccess)(`Token generated (expires in ${actualExpiresIn}s)`);
148
148
  const expiresAt = new Date(Date.now() + actualExpiresIn * 1000).toISOString();
149
+ // Schema: { username: orgId, password: JWT, ...metadata }
150
+ // - username/password fields satisfy CodeBuild's `secretsManagerCredentials`
151
+ // (HTTP Basic, sent to pipeline-image-registry's /token endpoint)
152
+ // - The same JWT is consumed by the plugin-lookup Lambda by reading
153
+ // the `password` field
154
+ // - One Secret per customer account replaces the previous two-secret model
155
+ // Decode JWT for orgId — it's a stable identifier for the username field;
156
+ // the actual auth uses the password (JWT) which the token service verifies.
157
+ const payload = (0, auth_guard_1.decodeTokenPayload)(accessToken);
158
+ const orgId = payload?.organizationId ?? 'unknown-org';
149
159
  const secretValue = JSON.stringify({
150
- accessToken,
160
+ username: orgId,
161
+ password: accessToken,
151
162
  ...(refreshToken && { refreshToken }),
152
163
  platformUrl: client.getBaseUrl(),
153
164
  expiresIn: actualExpiresIn,
@@ -230,4 +241,4 @@ function storeToken(program) {
230
241
  }
231
242
  });
232
243
  }
233
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store-token.js","sourceRoot":"","sources":["../../src/commands/store-token.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 axios from 'axios';\nimport { Command } from 'commander';\nimport { validateNumber } from '../config/cli.constants';\nimport { auditLog } from '../utils/audit-log';\nimport { decodeTokenPayload } from '../utils/auth-guard';\nimport { upsertSecret, getSecretArn } from '../utils/aws-secrets';\nimport { createAuthenticatedClientAsync, printCommandHeader, printSslWarning } from '../utils/command-utils';\nimport { getConfigWithOptions } from '../utils/config-loader';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printInfo, printKeyValue, printSection, printSuccess } from '../utils/output-utils';\n\n/**\n * Build the secret name from the JWT token's organizationId.\n * Pattern: {SECRETS_PATH_PREFIX}/{orgId}/platform\n * @throws Error if organizationId is not present in the token\n */\nfunction resolveSecretName(token: string): string {\n  const payload = decodeTokenPayload(token);\n  const orgId = payload?.organizationId;\n  if (!orgId) {\n    throw new Error('Token does not contain organizationId — cannot derive secret name. Use --secret-name to specify explicitly.');\n  }\n  return `${CoreConstants.SECRETS_PATH_PREFIX}/${orgId}/platform`;\n}\n\n/**\n * Registers the `store-token` command with the CLI program.\n *\n * Generates a long-lived JWT token via the platform API and stores it\n * in AWS Secrets Manager for use by the synth/deploy --store-tokens flag.\n *\n * Requires PLATFORM_TOKEN to be set, or use --email/--password to login inline.\n *\n * @example\n * ```bash\n * pipeline-manager store-token --region us-east-1\n * pipeline-manager store-token --days 90 --region us-east-1\n * pipeline-manager store-token --days 7 --secret-name my-custom-secret --no-verify-ssl\n * pipeline-manager store-token --dry-run\n * pipeline-manager store-token -e admin -p '***' --region us-east-1\n * ```\n */\nexport function storeToken(program: Command): void {\n  program\n    .command('store-token')\n    .description('Generate JWT token and store in AWS Secrets Manager for CDK deployments')\n    .option('-e, --email <email>', 'Login email (skips PLATFORM_TOKEN requirement)')\n    .option('-p, --password <password>', 'Login password (used with --email)')\n    .option('--days <days>', 'Token lifetime in days', '30')\n    .option('--dry-run', 'Show what would be stored without writing to Secrets Manager', false)\n    .option('--secret-name <name>', 'Secrets Manager secret name (default: derived from token org)')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--json', 'Output result as JSON', false)\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .action(async (options) => {\n      const executionId = printCommandHeader('Store Token');\n\n      try {\n        printSslWarning(options.verifySsl);\n\n        const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n\n        const days = validateNumber(options.days, 'days', 1, 365);\n        const expiresInSeconds = days * 24 * 60 * 60;\n\n        // Step 0: If --email/--password provided and no PLATFORM_TOKEN, login first\n        if (options.email && options.password && !process.env.PLATFORM_TOKEN) {\n          printSection('Login');\n          printInfo('Authenticating with email/password...');\n\n          const config = getConfigWithOptions(options);\n          const loginUrl = `${config.api.baseUrl}/api/auth/login`;\n\n          const loginResponse = await axios.post(loginUrl, {\n            email: options.email,\n            password: options.password,\n          }, {\n            httpsAgent: config.api.rejectUnauthorized === false\n              ? new (await import('https')).Agent({ rejectUnauthorized: false })\n              : undefined,\n          });\n\n          const loginData = loginResponse.data?.data ?? loginResponse.data;\n          const loginToken = loginData?.accessToken;\n\n          if (!loginToken || typeof loginToken !== 'string') {\n            throw new Error('Login failed — no access token in response');\n          }\n\n          process.env.PLATFORM_TOKEN = loginToken;\n          printSuccess('Login successful');\n        }\n\n        // Step 1: Authenticate and generate long-lived token\n        printSection('Generate Token');\n\n        const client = await createAuthenticatedClientAsync(options);\n\n        // Resolve secret name from token's organizationId (unless --secret-name was explicitly set)\n        const secretName = options.secretName || resolveSecretName(process.env.PLATFORM_TOKEN!);\n\n        auditLog('store-token', { executionId, secretName, days: options.days, dryRun: options.dryRun });\n\n        printInfo('Parameters', {\n          secretName,\n          region,\n          days,\n          expiresIn: `${expiresInSeconds}s`,\n          dryRun: options.dryRun,\n        });\n\n        printInfo('Requesting token', { expiresIn: `${expiresInSeconds}s (${days} days)` });\n\n        const tokenResponse = await client.post<Record<string, unknown>>(\n          '/api/user/generate-token',\n          { expiresIn: expiresInSeconds },\n        );\n\n        const tokenData = (tokenResponse as Record<string, unknown>)?.data ?? tokenResponse;\n        const accessToken = (tokenData as Record<string, unknown>)?.accessToken as string | undefined;\n        const refreshToken = (tokenData as Record<string, unknown>)?.refreshToken as string | undefined;\n        const actualExpiresIn = ((tokenData as Record<string, unknown>)?.expiresIn as number) ?? expiresInSeconds;\n\n        if (!accessToken) {\n          throw new Error('Token generation failed — no access token in response');\n        }\n\n        printSuccess(`Token generated (expires in ${actualExpiresIn}s)`);\n\n        const expiresAt = new Date(Date.now() + actualExpiresIn * 1000).toISOString();\n\n        const secretValue = JSON.stringify({\n          accessToken,\n          ...(refreshToken && { refreshToken }),\n          platformUrl: client.getBaseUrl(),\n          expiresIn: actualExpiresIn,\n          expiresAt,\n          createdAt: new Date().toISOString(),\n        });\n\n        // Dry-run: show what would be stored\n        if (options.dryRun) {\n          if (options.json) {\n            console.log(JSON.stringify({\n              success: true,\n              dryRun: true,\n              secretName,\n              region,\n              expiresInDays: days,\n              expiresAt,\n              tokenLength: accessToken.length,\n              hasRefreshToken: !!refreshToken,\n            }, null, 2));\n          } else {\n            console.log('');\n            printSection('Dry Run — No Changes Made');\n            printKeyValue({\n              'Secret Name': secretName,\n              'Region': region,\n              'Expires In': `${days} days`,\n              'Renew By': expiresAt,\n              'Token Length': `${accessToken.length} chars`,\n              'Has Refresh Token': refreshToken ? 'Yes' : 'No',\n            });\n            printSuccess('Dry run complete — no secret was created or updated');\n          }\n          return;\n        }\n\n        // Step 2: Store token in Secrets Manager\n        printSection('Store Token');\n\n        const description = `Platform JWT token (renew by ${expiresAt})`;\n        await upsertSecret(secretName, secretValue, description, { region, profile: options.profile });\n\n        const arn = await getSecretArn(secretName, { region, profile: options.profile });\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            success: true,\n            secretName,\n            secretArn: arn,\n            region,\n            expiresInDays: days,\n            expiresAt,\n          }, null, 2));\n        } else {\n          console.log('');\n          printSection('Token Stored');\n\n          printKeyValue({\n            'Secret Name': secretName,\n            'Secret ARN': arn,\n            'Region': region,\n            'Expires In': `${days} days`,\n            'Renew By': expiresAt,\n            'Status': '✓ Stored',\n          });\n\n          console.log('');\n          printSuccess('Token stored. To use with synth/deploy:');\n          printInfo(`  export PLATFORM_SECRET_NAME=${secretName}`);\n          printInfo('  pipeline-manager synth --id <pipeline-id> --store-tokens');\n          console.log('');\n          printInfo(`Renew before ${expiresAt} with: pipeline-manager store-token --days ${days}`);\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'store-token',\n            executionId,\n            secretName: options.secretName || '(derived from token)',\n          },\n        });\n      }\n    });\n}\n"]}
244
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store-token.js","sourceRoot":"","sources":["../../src/commands/store-token.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 axios from 'axios';\nimport { Command } from 'commander';\nimport { validateNumber } from '../config/cli.constants';\nimport { auditLog } from '../utils/audit-log';\nimport { decodeTokenPayload } from '../utils/auth-guard';\nimport { upsertSecret, getSecretArn } from '../utils/aws-secrets';\nimport { createAuthenticatedClientAsync, printCommandHeader, printSslWarning } from '../utils/command-utils';\nimport { getConfigWithOptions } from '../utils/config-loader';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printInfo, printKeyValue, printSection, printSuccess } from '../utils/output-utils';\n\n/**\n * Build the secret name from the JWT token's organizationId.\n * Pattern: {SECRETS_PATH_PREFIX}/{orgId}/platform\n * @throws Error if organizationId is not present in the token\n */\nfunction resolveSecretName(token: string): string {\n  const payload = decodeTokenPayload(token);\n  const orgId = payload?.organizationId;\n  if (!orgId) {\n    throw new Error('Token does not contain organizationId — cannot derive secret name. Use --secret-name to specify explicitly.');\n  }\n  return `${CoreConstants.SECRETS_PATH_PREFIX}/${orgId}/platform`;\n}\n\n/**\n * Registers the `store-token` command with the CLI program.\n *\n * Generates a long-lived JWT token via the platform API and stores it\n * in AWS Secrets Manager for use by the synth/deploy --store-tokens flag.\n *\n * Requires PLATFORM_TOKEN to be set, or use --email/--password to login inline.\n *\n * @example\n * ```bash\n * pipeline-manager store-token --region us-east-1\n * pipeline-manager store-token --days 90 --region us-east-1\n * pipeline-manager store-token --days 7 --secret-name my-custom-secret --no-verify-ssl\n * pipeline-manager store-token --dry-run\n * pipeline-manager store-token -e admin -p '***' --region us-east-1\n * ```\n */\nexport function storeToken(program: Command): void {\n  program\n    .command('store-token')\n    .description('Generate JWT token and store in AWS Secrets Manager for CDK deployments')\n    .option('-e, --email <email>', 'Login email (skips PLATFORM_TOKEN requirement)')\n    .option('-p, --password <password>', 'Login password (used with --email)')\n    .option('--days <days>', 'Token lifetime in days', '30')\n    .option('--dry-run', 'Show what would be stored without writing to Secrets Manager', false)\n    .option('--secret-name <name>', 'Secrets Manager secret name (default: derived from token org)')\n    .option('--region <region>', 'AWS region (defaults to AWS_REGION env)')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .option('--json', 'Output result as JSON', false)\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .action(async (options) => {\n      const executionId = printCommandHeader('Store Token');\n\n      try {\n        printSslWarning(options.verifySsl);\n\n        const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION || 'us-east-1';\n\n        const days = validateNumber(options.days, 'days', 1, 365);\n        const expiresInSeconds = days * 24 * 60 * 60;\n\n        // Step 0: If --email/--password provided and no PLATFORM_TOKEN, login first\n        if (options.email && options.password && !process.env.PLATFORM_TOKEN) {\n          printSection('Login');\n          printInfo('Authenticating with email/password...');\n\n          const config = getConfigWithOptions(options);\n          const loginUrl = `${config.api.baseUrl}/api/auth/login`;\n\n          const loginResponse = await axios.post(loginUrl, {\n            email: options.email,\n            password: options.password,\n          }, {\n            httpsAgent: config.api.rejectUnauthorized === false\n              ? new (await import('https')).Agent({ rejectUnauthorized: false })\n              : undefined,\n          });\n\n          const loginData = loginResponse.data?.data ?? loginResponse.data;\n          const loginToken = loginData?.accessToken;\n\n          if (!loginToken || typeof loginToken !== 'string') {\n            throw new Error('Login failed — no access token in response');\n          }\n\n          process.env.PLATFORM_TOKEN = loginToken;\n          printSuccess('Login successful');\n        }\n\n        // Step 1: Authenticate and generate long-lived token\n        printSection('Generate Token');\n\n        const client = await createAuthenticatedClientAsync(options);\n\n        // Resolve secret name from token's organizationId (unless --secret-name was explicitly set)\n        const secretName = options.secretName || resolveSecretName(process.env.PLATFORM_TOKEN!);\n\n        auditLog('store-token', { executionId, secretName, days: options.days, dryRun: options.dryRun });\n\n        printInfo('Parameters', {\n          secretName,\n          region,\n          days,\n          expiresIn: `${expiresInSeconds}s`,\n          dryRun: options.dryRun,\n        });\n\n        printInfo('Requesting token', { expiresIn: `${expiresInSeconds}s (${days} days)` });\n\n        const tokenResponse = await client.post<Record<string, unknown>>(\n          '/api/user/generate-token',\n          { expiresIn: expiresInSeconds },\n        );\n\n        const tokenData = (tokenResponse as Record<string, unknown>)?.data ?? tokenResponse;\n        const accessToken = (tokenData as Record<string, unknown>)?.accessToken as string | undefined;\n        const refreshToken = (tokenData as Record<string, unknown>)?.refreshToken as string | undefined;\n        const actualExpiresIn = ((tokenData as Record<string, unknown>)?.expiresIn as number) ?? expiresInSeconds;\n\n        if (!accessToken) {\n          throw new Error('Token generation failed — no access token in response');\n        }\n\n        printSuccess(`Token generated (expires in ${actualExpiresIn}s)`);\n\n        const expiresAt = new Date(Date.now() + actualExpiresIn * 1000).toISOString();\n\n        // Schema: { username: orgId, password: JWT, ...metadata }\n        // - username/password fields satisfy CodeBuild's `secretsManagerCredentials`\n        //   (HTTP Basic, sent to pipeline-image-registry's /token endpoint)\n        // - The same JWT is consumed by the plugin-lookup Lambda by reading\n        //   the `password` field\n        // - One Secret per customer account replaces the previous two-secret model\n        // Decode JWT for orgId — it's a stable identifier for the username field;\n        // the actual auth uses the password (JWT) which the token service verifies.\n        const payload = decodeTokenPayload(accessToken);\n        const orgId = payload?.organizationId ?? 'unknown-org';\n\n        const secretValue = JSON.stringify({\n          username: orgId,\n          password: accessToken,\n          ...(refreshToken && { refreshToken }),\n          platformUrl: client.getBaseUrl(),\n          expiresIn: actualExpiresIn,\n          expiresAt,\n          createdAt: new Date().toISOString(),\n        });\n\n        // Dry-run: show what would be stored\n        if (options.dryRun) {\n          if (options.json) {\n            console.log(JSON.stringify({\n              success: true,\n              dryRun: true,\n              secretName,\n              region,\n              expiresInDays: days,\n              expiresAt,\n              tokenLength: accessToken.length,\n              hasRefreshToken: !!refreshToken,\n            }, null, 2));\n          } else {\n            console.log('');\n            printSection('Dry Run — No Changes Made');\n            printKeyValue({\n              'Secret Name': secretName,\n              'Region': region,\n              'Expires In': `${days} days`,\n              'Renew By': expiresAt,\n              'Token Length': `${accessToken.length} chars`,\n              'Has Refresh Token': refreshToken ? 'Yes' : 'No',\n            });\n            printSuccess('Dry run complete — no secret was created or updated');\n          }\n          return;\n        }\n\n        // Step 2: Store token in Secrets Manager\n        printSection('Store Token');\n\n        const description = `Platform JWT token (renew by ${expiresAt})`;\n        await upsertSecret(secretName, secretValue, description, { region, profile: options.profile });\n\n        const arn = await getSecretArn(secretName, { region, profile: options.profile });\n\n        if (options.json) {\n          console.log(JSON.stringify({\n            success: true,\n            secretName,\n            secretArn: arn,\n            region,\n            expiresInDays: days,\n            expiresAt,\n          }, null, 2));\n        } else {\n          console.log('');\n          printSection('Token Stored');\n\n          printKeyValue({\n            'Secret Name': secretName,\n            'Secret ARN': arn,\n            'Region': region,\n            'Expires In': `${days} days`,\n            'Renew By': expiresAt,\n            'Status': '✓ Stored',\n          });\n\n          console.log('');\n          printSuccess('Token stored. To use with synth/deploy:');\n          printInfo(`  export PLATFORM_SECRET_NAME=${secretName}`);\n          printInfo('  pipeline-manager synth --id <pipeline-id> --store-tokens');\n          console.log('');\n          printInfo(`Renew before ${expiresAt} with: pipeline-manager store-token --days ${days}`);\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'store-token',\n            executionId,\n            secretName: options.secretName || '(derived from token)',\n          },\n        });\n      }\n    });\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"synth.d.ts","sourceRoot":"","sources":["../../src/commands/synth.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAyCpC;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAgB,KAAK,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAsG5C"}
1
+ {"version":3,"file":"synth.d.ts","sourceRoot":"","sources":["../../src/commands/synth.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAoDpC;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAgB,KAAK,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAsG5C"}
@@ -9,6 +9,7 @@ const cdk_utils_1 = require("../utils/cdk-utils");
9
9
  const command_utils_1 = require("../utils/command-utils");
10
10
  const error_handler_1 = require("../utils/error-handler");
11
11
  const output_utils_1 = require("../utils/output-utils");
12
+ const plugin_resolver_1 = require("../utils/plugin-resolver");
12
13
  /**
13
14
  * Fetch pipeline config and set PIPELINE_PROPS env var for the boilerplate app.
14
15
  * Uses createAuthenticatedClientAsync which supports all three auth methods.
@@ -28,6 +29,15 @@ async function fetchPipelineConfig(pipelineId, options) {
28
29
  };
29
30
  if (pipeline.orgId)
30
31
  propsWithIds.orgId = pipeline.orgId;
32
+ // Pre-resolve plugins from the platform API so the synthesized template
33
+ // ships with real CodeBuild image URIs baked in. Without this, CDK falls
34
+ // back to standard:7.0 because the deploy-time custom resource attribute
35
+ // is unresolvable at synth time.
36
+ const resolvedPlugins = await (0, plugin_resolver_1.resolvePluginsForProps)(client, propsWithIds);
37
+ if (Object.keys(resolvedPlugins).length > 0) {
38
+ propsWithIds.resolvedPlugins = resolvedPlugins;
39
+ (0, output_utils_1.printInfo)('Pre-resolved plugins', { count: Object.keys(resolvedPlugins).length });
40
+ }
31
41
  process.env.PIPELINE_PROPS = Buffer.from(JSON.stringify(propsWithIds)).toString('base64');
32
42
  (0, output_utils_1.printSuccess)('Pipeline configuration loaded');
33
43
  }
@@ -177,4 +187,4 @@ async function showResolvedPipeline(pipelineId, options) {
177
187
  }
178
188
  console.log(JSON.stringify(decoded, null, 2));
179
189
  }
180
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synth.js","sourceRoot":"","sources":["../../src/commands/synth.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 { assertShellSafe } from '../config/cli.constants';\nimport type { Pipeline } from '../types/pipeline';\nimport { auditLog } from '../utils/audit-log';\nimport { ensureCdkAvailable, executeCdkShellCommand, resolveBoilerplatePath } from '../utils/cdk-utils';\nimport { createAuthenticatedClientAsync, printCommandHeader, printSslWarning } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { extractSingleResponse, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\n\n/**\n * Fetch pipeline config and set PIPELINE_PROPS env var for the boilerplate app.\n * Uses createAuthenticatedClientAsync which supports all three auth methods.\n */\nasync function fetchPipelineConfig(\n  pipelineId: string,\n  options: { storeTokens?: boolean; verifySsl?: boolean; region?: string; profile?: string },\n): Promise<void> {\n  const client = await createAuthenticatedClientAsync(options);\n  const config = client.getConfig();\n\n  const response = await client.get<Record<string, unknown>>(\n    `${config.api.pipelineUrl}/${pipelineId}`,\n  );\n\n  const pipeline = extractSingleResponse<Pipeline>(response, 'pipeline', 'props');\n\n  if (!pipeline?.props) {\n    printError('Pipeline has no props', { id: pipelineId });\n    throw new Error(`Failed to retrieve pipeline props for ID: ${pipelineId}`);\n  }\n\n  const propsWithIds: Record<string, unknown> = {\n    ...pipeline.props as Record<string, unknown>,\n    pipelineId: pipeline.id || pipelineId,\n  };\n  if (pipeline.orgId) propsWithIds.orgId = pipeline.orgId;\n\n  process.env.PIPELINE_PROPS = Buffer.from(JSON.stringify(propsWithIds)).toString('base64');\n  printSuccess('Pipeline configuration loaded');\n}\n\n/**\n * Registers the `synth` command with the CLI program.\n *\n * Runs CDK synthesis using the boilerplate app. Pipeline config is resolved from:\n * 1. --id flag or PIPELINE_ID env var → fetches config from platform API\n * 2. PIPELINE_PROPS env var (pre-encoded, from deploy command)\n *\n * Authentication methods (in priority order):\n * - PLATFORM_TOKEN env var\n * - --store-tokens → fetch from AWS Secrets Manager\n *\n * @example\n * ```bash\n * pipeline-manager synth --id <pipeline-id> --no-verify-ssl\n * pipeline-manager synth --id <pipeline-id> --store-tokens\n * pipeline-manager synth --quiet --no-notices          # CodePipeline (uses env vars)\n * ```\n */\nexport function synth(program: Command): void {\n  program\n    .command('synth')\n    .description('Run CDK synthesis using pipeline configuration')\n    .option('-i, --id <id>', 'Pipeline ID (or set PIPELINE_ID env var)')\n    .option('--store-tokens', 'Authenticate using token from AWS Secrets Manager (requires PLATFORM_SECRET_NAME env var)', false)\n    .option('--output <dir>', 'CDK output directory', 'cdk.out')\n    .option('--profile <profile>', 'AWS profile')\n    .option('--region <region>', 'AWS region (for --store-tokens)')\n    .option('--quiet', 'Suppress CDK output', false)\n    .option('--no-notices', 'Suppress CDK notices')\n    .option('--verbose', 'Show verbose CDK output', false)\n    .option('--json', 'Output result as JSON', false)\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 running CDK', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('CDK Synthesis');\n\n      try {\n        const pipelineId = options.id || process.env.PIPELINE_ID;\n\n        // --show-resolved: preview templates and exit\n        if (options.showResolved) {\n          await showResolvedPipeline(pipelineId, options);\n          return;\n        }\n\n        auditLog('synth', { executionId, pipelineId, output: options.output, profile: options.profile });\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        // Fetch pipeline config if ID is available and PIPELINE_PROPS not already set\n        if (pipelineId && !process.env.PIPELINE_PROPS) {\n          printInfo('Fetching pipeline configuration', { id: pipelineId });\n          await fetchPipelineConfig(pipelineId, options);\n        } else if (!pipelineId && !process.env.PIPELINE_PROPS) {\n          printWarning('No pipeline ID or PIPELINE_PROPS set');\n          throw new Error('Pipeline ID is required. Use --id <id> or set PIPELINE_ID env var.');\n        }\n\n        ensureCdkAvailable();\n        printSuccess('AWS CDK is available');\n\n        // Build cdk synth command\n        if (options.output) assertShellSafe(options.output, 'output');\n        if (options.profile) assertShellSafe(options.profile, 'profile');\n\n        const boilerplatePath = resolveBoilerplatePath(__dirname);\n        const parts = [\n          'cdk synth',\n          `--app=\"node ${boilerplatePath}\"`,\n          `--output=${options.output}`,\n        ];\n\n        if (options.profile) parts.push(`--profile=${options.profile}`);\n        if (options.quiet) parts.push('--quiet');\n        if (options.notices === false) parts.push('--no-notices');\n        if (options.verbose) parts.push('--verbose');\n\n        const command = parts.join(' ');\n\n        printInfo('Executing', { command: (command.split('--app')[0] ?? '').trim() + ' ...' });\n        console.log('');\n\n        const result = executeCdkShellCommand(command, {\n          showOutput: !options.quiet,\n        });\n\n        console.log('');\n        printSection('Synthesis Complete');\n\n        if (result.success) {\n          if (options.json) {\n            console.log(JSON.stringify({\n              success: true,\n              executionId,\n              duration: result.duration,\n              output: options.output,\n            }, null, 2));\n          } else {\n            printKeyValue({\n              'Execution ID': executionId,\n              'Duration': `${result.duration}ms`,\n              'Output': options.output,\n              'Status': '✓ Success',\n            });\n          }\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: { command: 'synth', executionId, pipelineId: options.id },\n        });\n      }\n    });\n}\n\n/**\n * Fetch the pipeline by ID, apply pass-1 self-referencing template\n * resolution, and print it to stdout. Used by `synth --show-resolved`\n * and `deploy --show-resolved` to preview substitution before running CDK.\n */\nasync function showResolvedPipeline(\n  pipelineId: string | undefined,\n  options: { profile?: string; region?: string; storeTokens?: boolean; verifySsl?: boolean },\n): Promise<void> {\n  if (!pipelineId) {\n    throw new Error('--show-resolved requires --id or PIPELINE_ID env var');\n  }\n  await fetchPipelineConfig(pipelineId, options);\n  // fetchPipelineConfig writes PIPELINE_PROPS base64-encoded on success\n  const encoded = process.env.PIPELINE_PROPS;\n  if (!encoded) throw new Error('Failed to fetch pipeline configuration');\n  const decoded = JSON.parse(Buffer.from(encoded, 'base64').toString('utf-8'));\n\n  // Lazy import to avoid pulling pipeline-core into every CLI invocation\n  // eslint-disable-next-line @typescript-eslint/no-require-imports\n  const { resolveSelfReferencing } = require('@pipeline-builder/pipeline-core');\n  const scope = { metadata: decoded.metadata ?? {}, vars: decoded.vars ?? {} };\n  const isTemplatable = (f: string) =>\n    f === 'projectName' || f.startsWith('metadata.') || f.startsWith('vars.');\n  const fieldToScope = (f: string) => isTemplatable(f) ? f : null;\n  const result = resolveSelfReferencing(decoded, scope, isTemplatable, fieldToScope, 'pipeline');\n\n  if (result.errors.length) {\n    console.error('Resolution errors:');\n    for (const e of result.errors) {\n      console.error(`  [${e.field ?? '?'}] ${e.message}`);\n    }\n    process.exit(1);\n  }\n  console.log(JSON.stringify(decoded, null, 2));\n}\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synth.js","sourceRoot":"","sources":["../../src/commands/synth.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 { assertShellSafe } from '../config/cli.constants';\nimport type { Pipeline } from '../types/pipeline';\nimport { auditLog } from '../utils/audit-log';\nimport { ensureCdkAvailable, executeCdkShellCommand, resolveBoilerplatePath } from '../utils/cdk-utils';\nimport { createAuthenticatedClientAsync, printCommandHeader, printSslWarning } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { extractSingleResponse, printError, printInfo, printKeyValue, printSection, printSuccess, printWarning } from '../utils/output-utils';\nimport { resolvePluginsForProps } from '../utils/plugin-resolver';\n\n/**\n * Fetch pipeline config and set PIPELINE_PROPS env var for the boilerplate app.\n * Uses createAuthenticatedClientAsync which supports all three auth methods.\n */\nasync function fetchPipelineConfig(\n  pipelineId: string,\n  options: { storeTokens?: boolean; verifySsl?: boolean; region?: string; profile?: string },\n): Promise<void> {\n  const client = await createAuthenticatedClientAsync(options);\n  const config = client.getConfig();\n\n  const response = await client.get<Record<string, unknown>>(\n    `${config.api.pipelineUrl}/${pipelineId}`,\n  );\n\n  const pipeline = extractSingleResponse<Pipeline>(response, 'pipeline', 'props');\n\n  if (!pipeline?.props) {\n    printError('Pipeline has no props', { id: pipelineId });\n    throw new Error(`Failed to retrieve pipeline props for ID: ${pipelineId}`);\n  }\n\n  const propsWithIds: Record<string, unknown> = {\n    ...pipeline.props as Record<string, unknown>,\n    pipelineId: pipeline.id || pipelineId,\n  };\n  if (pipeline.orgId) propsWithIds.orgId = pipeline.orgId;\n\n  // Pre-resolve plugins from the platform API so the synthesized template\n  // ships with real CodeBuild image URIs baked in. Without this, CDK falls\n  // back to standard:7.0 because the deploy-time custom resource attribute\n  // is unresolvable at synth time.\n  const resolvedPlugins = await resolvePluginsForProps(client, propsWithIds);\n  if (Object.keys(resolvedPlugins).length > 0) {\n    propsWithIds.resolvedPlugins = resolvedPlugins;\n    printInfo('Pre-resolved plugins', { count: Object.keys(resolvedPlugins).length });\n  }\n\n  process.env.PIPELINE_PROPS = Buffer.from(JSON.stringify(propsWithIds)).toString('base64');\n  printSuccess('Pipeline configuration loaded');\n}\n\n/**\n * Registers the `synth` command with the CLI program.\n *\n * Runs CDK synthesis using the boilerplate app. Pipeline config is resolved from:\n * 1. --id flag or PIPELINE_ID env var → fetches config from platform API\n * 2. PIPELINE_PROPS env var (pre-encoded, from deploy command)\n *\n * Authentication methods (in priority order):\n * - PLATFORM_TOKEN env var\n * - --store-tokens → fetch from AWS Secrets Manager\n *\n * @example\n * ```bash\n * pipeline-manager synth --id <pipeline-id> --no-verify-ssl\n * pipeline-manager synth --id <pipeline-id> --store-tokens\n * pipeline-manager synth --quiet --no-notices          # CodePipeline (uses env vars)\n * ```\n */\nexport function synth(program: Command): void {\n  program\n    .command('synth')\n    .description('Run CDK synthesis using pipeline configuration')\n    .option('-i, --id <id>', 'Pipeline ID (or set PIPELINE_ID env var)')\n    .option('--store-tokens', 'Authenticate using token from AWS Secrets Manager (requires PLATFORM_SECRET_NAME env var)', false)\n    .option('--output <dir>', 'CDK output directory', 'cdk.out')\n    .option('--profile <profile>', 'AWS profile')\n    .option('--region <region>', 'AWS region (for --store-tokens)')\n    .option('--quiet', 'Suppress CDK output', false)\n    .option('--no-notices', 'Suppress CDK notices')\n    .option('--verbose', 'Show verbose CDK output', false)\n    .option('--json', 'Output result as JSON', false)\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 running CDK', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('CDK Synthesis');\n\n      try {\n        const pipelineId = options.id || process.env.PIPELINE_ID;\n\n        // --show-resolved: preview templates and exit\n        if (options.showResolved) {\n          await showResolvedPipeline(pipelineId, options);\n          return;\n        }\n\n        auditLog('synth', { executionId, pipelineId, output: options.output, profile: options.profile });\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        // Fetch pipeline config if ID is available and PIPELINE_PROPS not already set\n        if (pipelineId && !process.env.PIPELINE_PROPS) {\n          printInfo('Fetching pipeline configuration', { id: pipelineId });\n          await fetchPipelineConfig(pipelineId, options);\n        } else if (!pipelineId && !process.env.PIPELINE_PROPS) {\n          printWarning('No pipeline ID or PIPELINE_PROPS set');\n          throw new Error('Pipeline ID is required. Use --id <id> or set PIPELINE_ID env var.');\n        }\n\n        ensureCdkAvailable();\n        printSuccess('AWS CDK is available');\n\n        // Build cdk synth command\n        if (options.output) assertShellSafe(options.output, 'output');\n        if (options.profile) assertShellSafe(options.profile, 'profile');\n\n        const boilerplatePath = resolveBoilerplatePath(__dirname);\n        const parts = [\n          'cdk synth',\n          `--app=\"node ${boilerplatePath}\"`,\n          `--output=${options.output}`,\n        ];\n\n        if (options.profile) parts.push(`--profile=${options.profile}`);\n        if (options.quiet) parts.push('--quiet');\n        if (options.notices === false) parts.push('--no-notices');\n        if (options.verbose) parts.push('--verbose');\n\n        const command = parts.join(' ');\n\n        printInfo('Executing', { command: (command.split('--app')[0] ?? '').trim() + ' ...' });\n        console.log('');\n\n        const result = executeCdkShellCommand(command, {\n          showOutput: !options.quiet,\n        });\n\n        console.log('');\n        printSection('Synthesis Complete');\n\n        if (result.success) {\n          if (options.json) {\n            console.log(JSON.stringify({\n              success: true,\n              executionId,\n              duration: result.duration,\n              output: options.output,\n            }, null, 2));\n          } else {\n            printKeyValue({\n              'Execution ID': executionId,\n              'Duration': `${result.duration}ms`,\n              'Output': options.output,\n              'Status': '✓ Success',\n            });\n          }\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: { command: 'synth', executionId, pipelineId: options.id },\n        });\n      }\n    });\n}\n\n/**\n * Fetch the pipeline by ID, apply pass-1 self-referencing template\n * resolution, and print it to stdout. Used by `synth --show-resolved`\n * and `deploy --show-resolved` to preview substitution before running CDK.\n */\nasync function showResolvedPipeline(\n  pipelineId: string | undefined,\n  options: { profile?: string; region?: string; storeTokens?: boolean; verifySsl?: boolean },\n): Promise<void> {\n  if (!pipelineId) {\n    throw new Error('--show-resolved requires --id or PIPELINE_ID env var');\n  }\n  await fetchPipelineConfig(pipelineId, options);\n  // fetchPipelineConfig writes PIPELINE_PROPS base64-encoded on success\n  const encoded = process.env.PIPELINE_PROPS;\n  if (!encoded) throw new Error('Failed to fetch pipeline configuration');\n  const decoded = JSON.parse(Buffer.from(encoded, 'base64').toString('utf-8'));\n\n  // Lazy import to avoid pulling pipeline-core into every CLI invocation\n  // eslint-disable-next-line @typescript-eslint/no-require-imports\n  const { resolveSelfReferencing } = require('@pipeline-builder/pipeline-core');\n  const scope = { metadata: decoded.metadata ?? {}, vars: decoded.vars ?? {} };\n  const isTemplatable = (f: string) =>\n    f === 'projectName' || f.startsWith('metadata.') || f.startsWith('vars.');\n  const fieldToScope = (f: string) => isTemplatable(f) ? f : null;\n  const result = resolveSelfReferencing(decoded, scope, isTemplatable, fieldToScope, 'pipeline');\n\n  if (result.errors.length) {\n    console.error('Resolution errors:');\n    for (const e of result.errors) {\n      console.error(`  [${e.field ?? '?'}] ${e.message}`);\n    }\n    process.exit(1);\n  }\n  console.log(JSON.stringify(decoded, null, 2));\n}\n"]}
@@ -0,0 +1,17 @@
1
+ import { ApiClient } from './api-client';
2
+ /**
3
+ * Pre-resolve plugins by calling the same `POST /api/plugins/lookup` endpoint
4
+ * the deploy-time custom resource Lambda uses. Returning the full Plugin
5
+ * record at synth time is what allows the resulting CFN template to ship
6
+ * with real CodeBuild image URIs (`<host>/<imageTag>:latest`) rather than
7
+ * the `aws/codebuild/standard:7.0` fallback that the synth-time token path
8
+ * forces.
9
+ *
10
+ * Failures are non-fatal: a missing plugin or unreachable API logs a warning
11
+ * and falls through to the deploy-time custom resource path so partial
12
+ * platform outages don't block synth/deploy.
13
+ *
14
+ * Keyed by `alias || name` to match `PluginLookup.plugin()`.
15
+ */
16
+ export declare function resolvePluginsForProps(client: ApiClient, props: Record<string, unknown>): Promise<Record<string, unknown>>;
17
+ //# sourceMappingURL=plugin-resolver.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin-resolver.d.ts","sourceRoot":"","sources":["../../src/utils/plugin-resolver.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAoDzC;;;;;;;;;;;;;GAaG;AACH,wBAAsB,sBAAsB,CAC1C,MAAM,EAAE,SAAS,EACjB,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAC7B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CA8BlC"}
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.resolvePluginsForProps = resolvePluginsForProps;
6
+ const output_utils_1 = require("./output-utils");
7
+ /**
8
+ * Cache key matching `PluginLookup.normalize()`: explicit alias when set,
9
+ * otherwise `${name}-alias`. Must stay in sync with the consumers in
10
+ * pipeline-core/plugin-lookup.ts and pipeline-builder.ts that read this map.
11
+ */
12
+ function cacheKey(ref) {
13
+ return ref.alias || `${ref.name}-alias`;
14
+ }
15
+ /**
16
+ * Walk the pipeline props tree to collect every plugin reference. Plugins
17
+ * appear at `synth.plugin` and at `stages[].steps[].plugin`. Entries are
18
+ * de-duplicated by `cacheKey()` — the same key `PluginLookup.plugin()` uses,
19
+ * so the resolved-plugins map lookups match downstream.
20
+ */
21
+ function collectPluginRefs(props) {
22
+ const refs = [];
23
+ const seen = new Set();
24
+ const push = (raw) => {
25
+ if (!raw || typeof raw !== 'object')
26
+ return;
27
+ const r = raw;
28
+ if (!r.name)
29
+ return;
30
+ const key = cacheKey(r);
31
+ if (seen.has(key))
32
+ return;
33
+ seen.add(key);
34
+ refs.push(r);
35
+ };
36
+ const synth = props.synth;
37
+ push(synth?.plugin);
38
+ const stages = props.stages;
39
+ for (const stage of stages ?? []) {
40
+ for (const step of stage.steps ?? []) {
41
+ push(step.plugin);
42
+ }
43
+ }
44
+ return refs;
45
+ }
46
+ /**
47
+ * Pre-resolve plugins by calling the same `POST /api/plugins/lookup` endpoint
48
+ * the deploy-time custom resource Lambda uses. Returning the full Plugin
49
+ * record at synth time is what allows the resulting CFN template to ship
50
+ * with real CodeBuild image URIs (`<host>/<imageTag>:latest`) rather than
51
+ * the `aws/codebuild/standard:7.0` fallback that the synth-time token path
52
+ * forces.
53
+ *
54
+ * Failures are non-fatal: a missing plugin or unreachable API logs a warning
55
+ * and falls through to the deploy-time custom resource path so partial
56
+ * platform outages don't block synth/deploy.
57
+ *
58
+ * Keyed by `alias || name` to match `PluginLookup.plugin()`.
59
+ */
60
+ async function resolvePluginsForProps(client, props) {
61
+ const refs = collectPluginRefs(props);
62
+ if (refs.length === 0)
63
+ return {};
64
+ const defaultFilter = (name) => ({ name, isActive: true, isDefault: true });
65
+ const resolved = {};
66
+ await Promise.all(refs.map(async (ref) => {
67
+ const key = cacheKey(ref);
68
+ const filter = ref.filter ?? defaultFilter(ref.name);
69
+ try {
70
+ const res = await client.post('/api/plugins/lookup', { filter });
71
+ // Lambda's handler calls `api.post<Plugin>(...)` and consumes the body
72
+ // directly; the platform may wrap it as `{ data: Plugin }` or
73
+ // `{ plugin: Plugin }` depending on the response middleware.
74
+ const plugin = res.plugin
75
+ ?? res.data
76
+ ?? res;
77
+ if (plugin && typeof plugin === 'object' && plugin.name) {
78
+ resolved[key] = plugin;
79
+ }
80
+ else {
81
+ (0, output_utils_1.printWarning)(`Plugin "${ref.name}" lookup returned no record — falling back to deploy-time resolution`);
82
+ }
83
+ }
84
+ catch (err) {
85
+ const msg = err instanceof Error ? err.message : String(err);
86
+ (0, output_utils_1.printWarning)(`Plugin "${ref.name}" pre-resolution failed (${msg}) — falling back to deploy-time resolution`);
87
+ }
88
+ }));
89
+ return resolved;
90
+ }
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-resolver.js","sourceRoot":"","sources":["../../src/utils/plugin-resolver.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { ApiClient } from './api-client';\nimport { printWarning } from './output-utils';\n\n/** Minimal plugin reference shape — matches PluginOptions from pipeline-core. */\ninterface PluginRef {\n  name: string;\n  alias?: string;\n  filter?: Record<string, unknown>;\n}\n\n/**\n * Cache key matching `PluginLookup.normalize()`: explicit alias when set,\n * otherwise `${name}-alias`. Must stay in sync with the consumers in\n * pipeline-core/plugin-lookup.ts and pipeline-builder.ts that read this map.\n */\nfunction cacheKey(ref: PluginRef): string {\n  return ref.alias || `${ref.name}-alias`;\n}\n\n/**\n * Walk the pipeline props tree to collect every plugin reference. Plugins\n * appear at `synth.plugin` and at `stages[].steps[].plugin`. Entries are\n * de-duplicated by `cacheKey()` — the same key `PluginLookup.plugin()` uses,\n * so the resolved-plugins map lookups match downstream.\n */\nfunction collectPluginRefs(props: Record<string, unknown>): PluginRef[] {\n  const refs: PluginRef[] = [];\n  const seen = new Set<string>();\n\n  const push = (raw: unknown): void => {\n    if (!raw || typeof raw !== 'object') return;\n    const r = raw as PluginRef;\n    if (!r.name) return;\n    const key = cacheKey(r);\n    if (seen.has(key)) return;\n    seen.add(key);\n    refs.push(r);\n  };\n\n  const synth = props.synth as { plugin?: unknown } | undefined;\n  push(synth?.plugin);\n\n  const stages = props.stages as Array<{ steps?: Array<{ plugin?: unknown }> }> | undefined;\n  for (const stage of stages ?? []) {\n    for (const step of stage.steps ?? []) {\n      push(step.plugin);\n    }\n  }\n\n  return refs;\n}\n\n/**\n * Pre-resolve plugins by calling the same `POST /api/plugins/lookup` endpoint\n * the deploy-time custom resource Lambda uses. Returning the full Plugin\n * record at synth time is what allows the resulting CFN template to ship\n * with real CodeBuild image URIs (`<host>/<imageTag>:latest`) rather than\n * the `aws/codebuild/standard:7.0` fallback that the synth-time token path\n * forces.\n *\n * Failures are non-fatal: a missing plugin or unreachable API logs a warning\n * and falls through to the deploy-time custom resource path so partial\n * platform outages don't block synth/deploy.\n *\n * Keyed by `alias || name` to match `PluginLookup.plugin()`.\n */\nexport async function resolvePluginsForProps(\n  client: ApiClient,\n  props: Record<string, unknown>,\n): Promise<Record<string, unknown>> {\n  const refs = collectPluginRefs(props);\n  if (refs.length === 0) return {};\n\n  const defaultFilter = (name: string) => ({ name, isActive: true, isDefault: true });\n  const resolved: Record<string, unknown> = {};\n\n  await Promise.all(refs.map(async (ref) => {\n    const key = cacheKey(ref);\n    const filter = ref.filter ?? defaultFilter(ref.name);\n    try {\n      const res = await client.post<unknown>('/api/plugins/lookup', { filter });\n      // Lambda's handler calls `api.post<Plugin>(...)` and consumes the body\n      // directly; the platform may wrap it as `{ data: Plugin }` or\n      // `{ plugin: Plugin }` depending on the response middleware.\n      const plugin = (res as { plugin?: unknown }).plugin\n        ?? (res as { data?: unknown }).data\n        ?? res;\n      if (plugin && typeof plugin === 'object' && (plugin as { name?: string }).name) {\n        resolved[key] = plugin;\n      } else {\n        printWarning(`Plugin \"${ref.name}\" lookup returned no record — falling back to deploy-time resolution`);\n      }\n    } catch (err) {\n      const msg = err instanceof Error ? err.message : String(err);\n      printWarning(`Plugin \"${ref.name}\" pre-resolution failed (${msg}) — falling back to deploy-time resolution`);\n    }\n  }));\n\n  return resolved;\n}\n"]}
@@ -41,8 +41,6 @@ export interface PipelineForRegistry {
41
41
  project: string;
42
42
  organization: string;
43
43
  }
44
- /** Where pending registration intents are persisted between CLI invocations. */
45
- export declare const PENDING_INTENTS_DIR: string;
46
44
  /**
47
45
  * Build a RegistryPayload from the platform's pipeline metadata. Reaches out
48
46
  * to STS to discover the AWS account, hashes it, and constructs the ARN.
@@ -1 +1 @@
1
- {"version":3,"file":"registry.d.ts","sourceRoot":"","sources":["../../src/utils/registry.ts"],"names":[],"mappings":"AASA;;;;;;;;;;;;;;;;;;;;;;GAsBG;AAEH,2DAA2D;AAC3D,MAAM,WAAW,eAAe;IAC9B,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,6DAA6D;AAC7D,MAAM,WAAW,mBAAmB;IAClC,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,gFAAgF;AAChF,eAAO,MAAM,mBAAmB,QAAgE,CAAC;AAEjG;;;;;;GAMG;AACH,wBAAsB,oBAAoB,CACxC,QAAQ,EAAE,mBAAmB,EAC7B,cAAc,CAAC,EAAE,MAAM,GACtB,OAAO,CAAC,eAAe,CAAC,CA8B1B;AAED;;;;;;;GAOG;AACH,wBAAsB,kBAAkB,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,MAAM,CAAC,CAKlF;AAED,wDAAwD;AACxD,wBAAsB,kBAAkB,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAQ1E;AAED,oFAAoF;AACpF,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,eAAe,EAAE,CAAC,CAqBrE"}
1
+ {"version":3,"file":"registry.d.ts","sourceRoot":"","sources":["../../src/utils/registry.ts"],"names":[],"mappings":"AASA;;;;;;;;;;;;;;;;;;;;;;GAsBG;AAEH,2DAA2D;AAC3D,MAAM,WAAW,eAAe;IAC9B,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,6DAA6D;AAC7D,MAAM,WAAW,mBAAmB;IAClC,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,MAAM,CAAC;CACtB;AAKD;;;;;;GAMG;AACH,wBAAsB,oBAAoB,CACxC,QAAQ,EAAE,mBAAmB,EAC7B,cAAc,CAAC,EAAE,MAAM,GACtB,OAAO,CAAC,eAAe,CAAC,CA8B1B;AAED;;;;;;;GAOG;AACH,wBAAsB,kBAAkB,CAAC,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,MAAM,CAAC,CAKlF;AAED,wDAAwD;AACxD,wBAAsB,kBAAkB,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAQ1E;AAED,oFAAoF;AACpF,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,eAAe,EAAE,CAAC,CAqBrE"}
@@ -2,7 +2,6 @@
2
2
  // Copyright 2026 Pipeline Builder Contributors
3
3
  // SPDX-License-Identifier: Apache-2.0
4
4
  Object.defineProperty(exports, "__esModule", { value: true });
5
- exports.PENDING_INTENTS_DIR = void 0;
6
5
  exports.buildRegistryPayload = buildRegistryPayload;
7
6
  exports.writePendingIntent = writePendingIntent;
8
7
  exports.clearPendingIntent = clearPendingIntent;
@@ -13,7 +12,7 @@ const os_1 = require("os");
13
12
  const path_1 = require("path");
14
13
  const client_sts_1 = require("@aws-sdk/client-sts");
15
14
  /** Where pending registration intents are persisted between CLI invocations. */
16
- exports.PENDING_INTENTS_DIR = (0, path_1.join)((0, os_1.homedir)(), '.pipeline-manager', 'pending-registrations');
15
+ const PENDING_INTENTS_DIR = (0, path_1.join)((0, os_1.homedir)(), '.pipeline-manager', 'pending-registrations');
17
16
  /**
18
17
  * Build a RegistryPayload from the platform's pipeline metadata. Reaches out
19
18
  * to STS to discover the AWS account, hashes it, and constructs the ARN.
@@ -58,14 +57,14 @@ async function buildRegistryPayload(pipeline, regionOverride) {
58
57
  * safe to leave around indefinitely.
59
58
  */
60
59
  async function writePendingIntent(payload) {
61
- await fs_1.promises.mkdir(exports.PENDING_INTENTS_DIR, { recursive: true });
62
- const path = (0, path_1.join)(exports.PENDING_INTENTS_DIR, `${payload.pipelineId}.json`);
60
+ await fs_1.promises.mkdir(PENDING_INTENTS_DIR, { recursive: true });
61
+ const path = (0, path_1.join)(PENDING_INTENTS_DIR, `${payload.pipelineId}.json`);
63
62
  await fs_1.promises.writeFile(path, JSON.stringify(payload, null, 2), 'utf8');
64
63
  return path;
65
64
  }
66
65
  /** Remove a pending intent after a successful drain. */
67
66
  async function clearPendingIntent(pipelineId) {
68
- const path = (0, path_1.join)(exports.PENDING_INTENTS_DIR, `${pipelineId}.json`);
67
+ const path = (0, path_1.join)(PENDING_INTENTS_DIR, `${pipelineId}.json`);
69
68
  try {
70
69
  await fs_1.promises.unlink(path);
71
70
  }
@@ -79,7 +78,7 @@ async function clearPendingIntent(pipelineId) {
79
78
  async function readPendingIntents() {
80
79
  let entries;
81
80
  try {
82
- entries = await fs_1.promises.readdir(exports.PENDING_INTENTS_DIR);
81
+ entries = await fs_1.promises.readdir(PENDING_INTENTS_DIR);
83
82
  }
84
83
  catch (err) {
85
84
  if (err.code === 'ENOENT')
@@ -91,7 +90,7 @@ async function readPendingIntents() {
91
90
  if (!file.endsWith('.json'))
92
91
  continue;
93
92
  try {
94
- const text = await fs_1.promises.readFile((0, path_1.join)(exports.PENDING_INTENTS_DIR, file), 'utf8');
93
+ const text = await fs_1.promises.readFile((0, path_1.join)(PENDING_INTENTS_DIR, file), 'utf8');
95
94
  intents.push(JSON.parse(text));
96
95
  }
97
96
  catch {
@@ -101,4 +100,4 @@ async function readPendingIntents() {
101
100
  }
102
101
  return intents;
103
102
  }
104
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/utils/registry.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 { promises as fs } from 'fs';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\n\n/**\n * Helpers for registering a deployed pipeline ARN with the platform.\n *\n * The registry table maps deployed CodePipeline ARNs back to pipeline records\n * and orgs. It's how the dashboard's \"Deployed pipelines\" panel and the event\n * reporting Lambda resolve incoming events to a pipeline definition.\n *\n * Two failure modes are handled here:\n *\n *   1. Platform unreachable at deploy time. The CDK stack lands in AWS but the\n *      registry POST fails. Without retry the deploy command would have to\n *      re-run cdk-deploy (slow, sometimes blocked by stack state) just to\n *      record the ARN. Instead, we write a pending intent to a local file;\n *      `pipeline-manager register` drains them.\n *\n *   2. User explicitly invokes `pipeline-manager register --id <pipelineId>`\n *      after a successful deploy that didn't register. Same path: rebuilds\n *      the ARN from STS, POSTs to the platform.\n *\n * Pending intents store ONLY the registration payload — never tokens or URLs.\n * Auth is supplied by whichever command drains the intent (deploy or register),\n * so a stale intent file is never an authentication risk.\n */\n\n/** Shape of the body POSTed to /api/pipelines/registry. */\nexport interface RegistryPayload {\n  pipelineId: string;\n  orgId: string;\n  pipelineArn: string;\n  pipelineName: string;\n  accountId: string;\n  region: string;\n  project: string;\n  organization: string;\n  stackName: string;\n}\n\n/** Pipeline fields needed to construct a RegistryPayload. */\nexport interface PipelineForRegistry {\n  id: string;\n  orgId: string;\n  pipelineName?: string;\n  project: string;\n  organization: string;\n}\n\n/** Where pending registration intents are persisted between CLI invocations. */\nexport const PENDING_INTENTS_DIR = join(homedir(), '.pipeline-manager', 'pending-registrations');\n\n/**\n * Build a RegistryPayload from the platform's pipeline metadata. Reaches out\n * to STS to discover the AWS account, hashes it, and constructs the ARN.\n *\n * The hash is the same one applied server-side as defense in depth — raw AWS\n * account numbers must never reach the platform DB.\n */\nexport async function buildRegistryPayload(\n  pipeline: PipelineForRegistry,\n  regionOverride?: string,\n): Promise<RegistryPayload> {\n  const region = regionOverride\n    || process.env.AWS_REGION\n    || process.env.CDK_DEFAULT_REGION\n    || 'us-east-1';\n\n  const stsClient = new STSClient({ region });\n  const identity = await stsClient.send(new GetCallerIdentityCommand({}));\n  const account = identity.Account ?? '';\n  if (!account) {\n    throw new Error('Could not determine AWS account from STS — check AWS credentials');\n  }\n\n  const hashedAccount = createHash('sha256').update(account).digest('hex').slice(0, 12);\n  const pipelineName = pipeline.pipelineName\n    || `${pipeline.organization}-${pipeline.project}-pipeline`.toLowerCase();\n  const pipelineArn = `arn:aws:codepipeline:${region}:${hashedAccount}:${pipelineName}`;\n  const stackName = `${pipeline.project}-${pipeline.organization}`.toLowerCase();\n\n  return {\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,\n  };\n}\n\n/**\n * Persist a registration intent for later retry.\n *\n * Intent files are keyed by pipelineId so re-running deploy on the same\n * pipeline overwrites rather than accumulates. We deliberately store the\n * payload plain (not the full pipeline doc, not auth) so a stale file is\n * safe to leave around indefinitely.\n */\nexport async function writePendingIntent(payload: RegistryPayload): Promise<string> {\n  await fs.mkdir(PENDING_INTENTS_DIR, { recursive: true });\n  const path = join(PENDING_INTENTS_DIR, `${payload.pipelineId}.json`);\n  await fs.writeFile(path, JSON.stringify(payload, null, 2), 'utf8');\n  return path;\n}\n\n/** Remove a pending intent after a successful drain. */\nexport async function clearPendingIntent(pipelineId: string): Promise<void> {\n  const path = join(PENDING_INTENTS_DIR, `${pipelineId}.json`);\n  try {\n    await fs.unlink(path);\n  } catch (err) {\n    // ENOENT is fine — already cleared. Anything else is a real problem.\n    if ((err as NodeJS.ErrnoException).code !== 'ENOENT') throw err;\n  }\n}\n\n/** Read all pending intents currently on disk. Empty array if dir doesn't exist. */\nexport async function readPendingIntents(): Promise<RegistryPayload[]> {\n  let entries: string[];\n  try {\n    entries = await fs.readdir(PENDING_INTENTS_DIR);\n  } catch (err) {\n    if ((err as NodeJS.ErrnoException).code === 'ENOENT') return [];\n    throw err;\n  }\n\n  const intents: RegistryPayload[] = [];\n  for (const file of entries) {\n    if (!file.endsWith('.json')) continue;\n    try {\n      const text = await fs.readFile(join(PENDING_INTENTS_DIR, file), 'utf8');\n      intents.push(JSON.parse(text) as RegistryPayload);\n    } catch {\n      // Skip unreadable / malformed files — they'll need manual cleanup but\n      // shouldn't block valid intents from draining.\n    }\n  }\n  return intents;\n}\n"]}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/utils/registry.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;AAgEtC,oDAiCC;AAUD,gDAKC;AAGD,gDAQC;AAGD,gDAqBC;AAjJD,mCAAoC;AACpC,2BAAoC;AACpC,2BAA6B;AAC7B,+BAA4B;AAC5B,oDAA0E;AAgD1E,gFAAgF;AAChF,MAAM,mBAAmB,GAAG,IAAA,WAAI,EAAC,IAAA,YAAO,GAAE,EAAE,mBAAmB,EAAE,uBAAuB,CAAC,CAAC;AAE1F;;;;;;GAMG;AACI,KAAK,UAAU,oBAAoB,CACxC,QAA6B,EAC7B,cAAuB;IAEvB,MAAM,MAAM,GAAG,cAAc;WACxB,OAAO,CAAC,GAAG,CAAC,UAAU;WACtB,OAAO,CAAC,GAAG,CAAC,kBAAkB;WAC9B,WAAW,CAAC;IAEjB,MAAM,SAAS,GAAG,IAAI,sBAAS,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IAC5C,MAAM,QAAQ,GAAG,MAAM,SAAS,CAAC,IAAI,CAAC,IAAI,qCAAwB,CAAC,EAAE,CAAC,CAAC,CAAC;IACxE,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAO,IAAI,EAAE,CAAC;IACvC,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,IAAI,KAAK,CAAC,kEAAkE,CAAC,CAAC;IACtF,CAAC;IAED,MAAM,aAAa,GAAG,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACtF,MAAM,YAAY,GAAG,QAAQ,CAAC,YAAY;WACrC,GAAG,QAAQ,CAAC,YAAY,IAAI,QAAQ,CAAC,OAAO,WAAW,CAAC,WAAW,EAAE,CAAC;IAC3E,MAAM,WAAW,GAAG,wBAAwB,MAAM,IAAI,aAAa,IAAI,YAAY,EAAE,CAAC;IACtF,MAAM,SAAS,GAAG,GAAG,QAAQ,CAAC,OAAO,IAAI,QAAQ,CAAC,YAAY,EAAE,CAAC,WAAW,EAAE,CAAC;IAE/E,OAAO;QACL,UAAU,EAAE,QAAQ,CAAC,EAAE;QACvB,KAAK,EAAE,QAAQ,CAAC,KAAK;QACrB,WAAW;QACX,YAAY;QACZ,SAAS,EAAE,aAAa;QACxB,MAAM;QACN,OAAO,EAAE,QAAQ,CAAC,OAAO;QACzB,YAAY,EAAE,QAAQ,CAAC,YAAY;QACnC,SAAS;KACV,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACI,KAAK,UAAU,kBAAkB,CAAC,OAAwB;IAC/D,MAAM,aAAE,CAAC,KAAK,CAAC,mBAAmB,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,IAAA,WAAI,EAAC,mBAAmB,EAAE,GAAG,OAAO,CAAC,UAAU,OAAO,CAAC,CAAC;IACrE,MAAM,aAAE,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;IACnE,OAAO,IAAI,CAAC;AACd,CAAC;AAED,wDAAwD;AACjD,KAAK,UAAU,kBAAkB,CAAC,UAAkB;IACzD,MAAM,IAAI,GAAG,IAAA,WAAI,EAAC,mBAAmB,EAAE,GAAG,UAAU,OAAO,CAAC,CAAC;IAC7D,IAAI,CAAC;QACH,MAAM,aAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,qEAAqE;QACrE,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ;YAAE,MAAM,GAAG,CAAC;IAClE,CAAC;AACH,CAAC;AAED,oFAAoF;AAC7E,KAAK,UAAU,kBAAkB;IACtC,IAAI,OAAiB,CAAC;IACtB,IAAI,CAAC;QACH,OAAO,GAAG,MAAM,aAAE,CAAC,OAAO,CAAC,mBAAmB,CAAC,CAAC;IAClD,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ;YAAE,OAAO,EAAE,CAAC;QAChE,MAAM,GAAG,CAAC;IACZ,CAAC;IAED,MAAM,OAAO,GAAsB,EAAE,CAAC;IACtC,KAAK,MAAM,IAAI,IAAI,OAAO,EAAE,CAAC;QAC3B,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,SAAS;QACtC,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,MAAM,aAAE,CAAC,QAAQ,CAAC,IAAA,WAAI,EAAC,mBAAmB,EAAE,IAAI,CAAC,EAAE,MAAM,CAAC,CAAC;YACxE,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAoB,CAAC,CAAC;QACpD,CAAC;QAAC,MAAM,CAAC;YACP,sEAAsE;YACtE,+CAA+C;QACjD,CAAC;IACH,CAAC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createHash } from 'crypto';\nimport { promises as fs } from 'fs';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\n\n/**\n * Helpers for registering a deployed pipeline ARN with the platform.\n *\n * The registry table maps deployed CodePipeline ARNs back to pipeline records\n * and orgs. It's how the dashboard's \"Deployed pipelines\" panel and the event\n * reporting Lambda resolve incoming events to a pipeline definition.\n *\n * Two failure modes are handled here:\n *\n *   1. Platform unreachable at deploy time. The CDK stack lands in AWS but the\n *      registry POST fails. Without retry the deploy command would have to\n *      re-run cdk-deploy (slow, sometimes blocked by stack state) just to\n *      record the ARN. Instead, we write a pending intent to a local file;\n *      `pipeline-manager register` drains them.\n *\n *   2. User explicitly invokes `pipeline-manager register --id <pipelineId>`\n *      after a successful deploy that didn't register. Same path: rebuilds\n *      the ARN from STS, POSTs to the platform.\n *\n * Pending intents store ONLY the registration payload — never tokens or URLs.\n * Auth is supplied by whichever command drains the intent (deploy or register),\n * so a stale intent file is never an authentication risk.\n */\n\n/** Shape of the body POSTed to /api/pipelines/registry. */\nexport interface RegistryPayload {\n  pipelineId: string;\n  orgId: string;\n  pipelineArn: string;\n  pipelineName: string;\n  accountId: string;\n  region: string;\n  project: string;\n  organization: string;\n  stackName: string;\n}\n\n/** Pipeline fields needed to construct a RegistryPayload. */\nexport interface PipelineForRegistry {\n  id: string;\n  orgId: string;\n  pipelineName?: string;\n  project: string;\n  organization: string;\n}\n\n/** Where pending registration intents are persisted between CLI invocations. */\nconst PENDING_INTENTS_DIR = join(homedir(), '.pipeline-manager', 'pending-registrations');\n\n/**\n * Build a RegistryPayload from the platform's pipeline metadata. Reaches out\n * to STS to discover the AWS account, hashes it, and constructs the ARN.\n *\n * The hash is the same one applied server-side as defense in depth — raw AWS\n * account numbers must never reach the platform DB.\n */\nexport async function buildRegistryPayload(\n  pipeline: PipelineForRegistry,\n  regionOverride?: string,\n): Promise<RegistryPayload> {\n  const region = regionOverride\n    || process.env.AWS_REGION\n    || process.env.CDK_DEFAULT_REGION\n    || 'us-east-1';\n\n  const stsClient = new STSClient({ region });\n  const identity = await stsClient.send(new GetCallerIdentityCommand({}));\n  const account = identity.Account ?? '';\n  if (!account) {\n    throw new Error('Could not determine AWS account from STS — check AWS credentials');\n  }\n\n  const hashedAccount = createHash('sha256').update(account).digest('hex').slice(0, 12);\n  const pipelineName = pipeline.pipelineName\n    || `${pipeline.organization}-${pipeline.project}-pipeline`.toLowerCase();\n  const pipelineArn = `arn:aws:codepipeline:${region}:${hashedAccount}:${pipelineName}`;\n  const stackName = `${pipeline.project}-${pipeline.organization}`.toLowerCase();\n\n  return {\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,\n  };\n}\n\n/**\n * Persist a registration intent for later retry.\n *\n * Intent files are keyed by pipelineId so re-running deploy on the same\n * pipeline overwrites rather than accumulates. We deliberately store the\n * payload plain (not the full pipeline doc, not auth) so a stale file is\n * safe to leave around indefinitely.\n */\nexport async function writePendingIntent(payload: RegistryPayload): Promise<string> {\n  await fs.mkdir(PENDING_INTENTS_DIR, { recursive: true });\n  const path = join(PENDING_INTENTS_DIR, `${payload.pipelineId}.json`);\n  await fs.writeFile(path, JSON.stringify(payload, null, 2), 'utf8');\n  return path;\n}\n\n/** Remove a pending intent after a successful drain. */\nexport async function clearPendingIntent(pipelineId: string): Promise<void> {\n  const path = join(PENDING_INTENTS_DIR, `${pipelineId}.json`);\n  try {\n    await fs.unlink(path);\n  } catch (err) {\n    // ENOENT is fine — already cleared. Anything else is a real problem.\n    if ((err as NodeJS.ErrnoException).code !== 'ENOENT') throw err;\n  }\n}\n\n/** Read all pending intents currently on disk. Empty array if dir doesn't exist. */\nexport async function readPendingIntents(): Promise<RegistryPayload[]> {\n  let entries: string[];\n  try {\n    entries = await fs.readdir(PENDING_INTENTS_DIR);\n  } catch (err) {\n    if ((err as NodeJS.ErrnoException).code === 'ENOENT') return [];\n    throw err;\n  }\n\n  const intents: RegistryPayload[] = [];\n  for (const file of entries) {\n    if (!file.endsWith('.json')) continue;\n    try {\n      const text = await fs.readFile(join(PENDING_INTENTS_DIR, file), 'utf8');\n      intents.push(JSON.parse(text) as RegistryPayload);\n    } catch {\n      // Skip unreadable / malformed files — they'll need manual cleanup but\n      // shouldn't block valid intents from draining.\n    }\n  }\n  return intents;\n}\n"]}
package/package.json CHANGED
@@ -30,7 +30,6 @@
30
30
  "@aws-sdk/client-lambda": "3.821.0",
31
31
  "@aws-sdk/client-secrets-manager": "3.821.0",
32
32
  "@aws-sdk/client-sts": "3.821.0",
33
- "@pipeline-builder/pipeline-core": "3.3.33",
34
33
  "aws-cdk-lib": "2.251.0",
35
34
  "axios": "1.13.5",
36
35
  "commander": "14.0.3",
@@ -40,7 +39,8 @@
40
39
  "picocolors": "1.1.1",
41
40
  "progress": "2.0.3",
42
41
  "typescript": "5.9.3",
43
- "yaml": "2.8.2"
42
+ "yaml": "2.8.2",
43
+ "@pipeline-builder/pipeline-core": "3.4.0"
44
44
  },
45
45
  "keywords": [
46
46
  "aws",
@@ -83,7 +83,7 @@
83
83
  "access": "public",
84
84
  "registry": "https://registry.npmjs.org/"
85
85
  },
86
- "version": "3.3.32",
86
+ "version": "3.4.0",
87
87
  "bugs": {
88
88
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
89
89
  },