@pipeline-builder/pipeline-core 3.4.9 → 3.4.11

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.
@@ -43,15 +43,15 @@ export declare function extractMetadataEnv(metadata: MetaDataType): Record<strin
43
43
  * explicitly in metadata, the metadata-builder passthrough handles it
44
44
  * (via `metadataForBuildEnvironment`). This function returns
45
45
  * `undefined` and the metadata wins.
46
- * 2. Otherwise, if the plugin has an `imageTag` AND the registry config
47
- * is populated, build a `LinuxBuildImage.fromDockerRegistry()` image
48
- * pointing at `<registry-host>:<port>/system/<imageTag>:latest`. CodeBuild
49
- * authenticates by sending the per-org platform Secret as Basic auth to
50
- * `pipeline-image-registry`'s `/token` endpoint, which the registry's
51
- * bearer challenge points at; the JWT in `password` resolves to a
52
- * registry token scoped to the org.
53
- * 3. If neither (e.g., a `metadata_only` plugin or one with no image),
54
- * return `undefined` so CodeBuild uses its default (`standard:7.0`).
46
+ * 2. Otherwise, if the plugin has a `name`+`version` AND the registry
47
+ * config is populated, build a `LinuxBuildImage.fromDockerRegistry()`
48
+ * image pointing at `<registry-host>:<port>/<ns>/<name>:<version>`
49
+ * where `<ns>` is `system` or `org-<orgId>`. CodeBuild authenticates by
50
+ * sending the per-org platform Secret as Basic auth to
51
+ * `pipeline-image-registry`'s `/token` endpoint; the JWT in `password`
52
+ * resolves to a registry token scoped to the org.
53
+ * 3. If `metadata_only`, return `undefined` so CodeBuild uses its default
54
+ * (`standard:7.0`).
55
55
  *
56
56
  * `scope` and `orgId` are required: the per-org platform Secret is named
57
57
  * `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`
@@ -156,33 +156,30 @@ function toSecretEnvVars(secrets, orgId) {
156
156
  * explicitly in metadata, the metadata-builder passthrough handles it
157
157
  * (via `metadataForBuildEnvironment`). This function returns
158
158
  * `undefined` and the metadata wins.
159
- * 2. Otherwise, if the plugin has an `imageTag` AND the registry config
160
- * is populated, build a `LinuxBuildImage.fromDockerRegistry()` image
161
- * pointing at `<registry-host>:<port>/system/<imageTag>:latest`. CodeBuild
162
- * authenticates by sending the per-org platform Secret as Basic auth to
163
- * `pipeline-image-registry`'s `/token` endpoint, which the registry's
164
- * bearer challenge points at; the JWT in `password` resolves to a
165
- * registry token scoped to the org.
166
- * 3. If neither (e.g., a `metadata_only` plugin or one with no image),
167
- * return `undefined` so CodeBuild uses its default (`standard:7.0`).
159
+ * 2. Otherwise, if the plugin has a `name`+`version` AND the registry
160
+ * config is populated, build a `LinuxBuildImage.fromDockerRegistry()`
161
+ * image pointing at `<registry-host>:<port>/<ns>/<name>:<version>`
162
+ * where `<ns>` is `system` or `org-<orgId>`. CodeBuild authenticates by
163
+ * sending the per-org platform Secret as Basic auth to
164
+ * `pipeline-image-registry`'s `/token` endpoint; the JWT in `password`
165
+ * resolves to a registry token scoped to the org.
166
+ * 3. If `metadata_only`, return `undefined` so CodeBuild uses its default
167
+ * (`standard:7.0`).
168
168
  *
169
169
  * `scope` and `orgId` are required: the per-org platform Secret is named
170
170
  * `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`
171
171
  * needs a Construct to anchor the imported secret to.
172
172
  */
173
173
  function resolvePluginImage(scope, plugin, orgId) {
174
- const imageTag = plugin.imageTag;
175
174
  // `metadata_only` plugins legitimately have no image — their work runs
176
175
  // in the default CodeBuild image. Quiet skip.
177
176
  if (plugin.buildType === 'metadata_only')
178
177
  return undefined;
179
- // No image tag means there's nothing to pull. This SHOULDN'T happen for
180
- // build_image / prebuilt plugins (the platform writes imageTag at upload
181
- // time); if it does, the plugin record is malformed.
182
- if (!imageTag || imageTag === '') {
183
- log.warn(`Plugin "${plugin.name}" has buildType=${plugin.buildType} but no imageTag ` +
184
- 'CodeBuild will run on aws/codebuild/standard:7.0 and won\'t have the plugin\'s baked tools. ' +
185
- 'Verify the plugin was uploaded correctly via load-plugins.sh.');
178
+ // build_image / prebuilt plugins must have name + version (DB constraint
179
+ // makes both NOT NULL). Defensive check in case a malformed row slips in.
180
+ if (!plugin.name || !plugin.version) {
181
+ log.warn(`Plugin "${plugin.name}" has buildType=${plugin.buildType} but missing name/version — ` +
182
+ 'CodeBuild will run on aws/codebuild/standard:7.0 and won\'t have the plugin\'s baked tools.');
186
183
  return undefined;
187
184
  }
188
185
  let registry;
@@ -191,13 +188,13 @@ function resolvePluginImage(scope, plugin, orgId) {
191
188
  }
192
189
  catch {
193
190
  // Config namespace not loaded (e.g., unit tests without full config).
194
- log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but registry config not loaded — ` +
191
+ log.warn(`Plugin "${plugin.name}:${plugin.version}" needs the registry config but it's not loaded — ` +
195
192
  'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +
196
193
  'Set IMAGE_REGISTRY_HOST + IMAGE_REGISTRY_PORT in pipeline-manager\'s environment.');
197
194
  return undefined;
198
195
  }
199
196
  if (!registry?.host) {
200
- log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but IMAGE_REGISTRY_HOST is empty — ` +
197
+ log.warn(`Plugin "${plugin.name}:${plugin.version}" needs IMAGE_REGISTRY_HOST but it's empty — ` +
201
198
  'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +
202
199
  'Set IMAGE_REGISTRY_HOST in pipeline-manager\'s environment to use the plugin image.');
203
200
  return undefined;
@@ -213,10 +210,10 @@ function resolvePluginImage(scope, plugin, orgId) {
213
210
  return undefined;
214
211
  }
215
212
  // Compose the image URI. Namespace by ownership:
216
- // - Plugins owned by the system org → `system/<imageTag>:latest`.
213
+ // - Plugins owned by the system org → `system/<name>:<version>`.
217
214
  // pipeline-image-registry's token service grants pull on `system/*`
218
215
  // to any authenticated org user (read-only catalog of shared plugins).
219
- // - Plugins owned by a tenant org → `org-<orgId>/<imageTag>:latest`.
216
+ // - Plugins owned by a tenant org → `org-<orgId>/<name>:<version>`.
220
217
  // The token service grants pull,push only to members of that org.
221
218
  // The plugin's own `orgId` (not the caller's) decides the namespace —
222
219
  // tenant pipelines pulling shared system plugins still get the `system/`
@@ -228,7 +225,7 @@ function resolvePluginImage(scope, plugin, orgId) {
228
225
  const namespace = plugin.orgId === SYSTEM_ORG_ID
229
226
  ? 'system'
230
227
  : `org-${plugin.orgId}`;
231
- const imageUri = `${registry.host}${portPart}/${namespace}/${imageTag}:latest`;
228
+ const imageUri = `${registry.host}${portPart}/${namespace}/${plugin.name}:${plugin.version}`;
232
229
  // CodeBuild reads `pipeline-builder/<orgId>/platform` and sends its
233
230
  // `username`/`password` fields as HTTP Basic to the registry. The
234
231
  // registry challenges with a Bearer realm pointing at
@@ -317,7 +314,7 @@ function createCodeBuildStep(options) {
317
314
  ...(0, metadata_builder_1.metadataForShellStep)(merged),
318
315
  });
319
316
  }
320
- log.debug(`[CreateCodeBuildStep] SHELL_STEP plugin "${plugin.name}" has imageTag — promoting to CodeBuildStep so the plugin image is actually used`);
317
+ log.debug(`[CreateCodeBuildStep] SHELL_STEP plugin "${plugin.name}" has a registry image — promoting to CodeBuildStep so the plugin image is actually used`);
321
318
  // Fall through to the CodeBuildStep path below.
322
319
  }
323
320
  const computeType = getComputeType(plugin.computeType ?? options.defaultComputeType ?? 'SMALL');
@@ -401,4 +398,4 @@ function replaceNonAlphanumeric(input, replaceValue = '_') {
401
398
  function unwrapSecret(value) {
402
399
  return typeof value === 'string' ? value : value.unsafeUnwrap();
403
400
  }
404
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-helpers.js","sourceRoot":"","sources":["../../src/core/pipeline-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createLogger } from '@pipeline-builder/api-core';\nimport type { Plugin } from '@pipeline-builder/pipeline-data';\nimport { Duration, SecretValue, Stack } from 'aws-cdk-lib';\nimport { BuildEnvironmentVariableType, ComputeType as CDKComputeType, LinuxBuildImage, IBuildImage } from 'aws-cdk-lib/aws-codebuild';\nimport { Secret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { CodeBuildStep, ManualApprovalStep, ShellStep } from 'aws-cdk-lib/pipelines';\nimport type { Construct } from 'constructs';\nimport type { ArtifactKey } from './artifact-manager';\nimport { metadataForShellStep, metadataForCodeBuildStep, metadataForBuildEnvironment } from './metadata-builder';\nimport { resolveNetwork } from './network';\nimport { PluginType, ComputeType, MetaDataType, CDK_METADATA_PREFIX } from './pipeline-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport type { CodeBuildStepOptions, StepCustomization } from '../pipeline/step-types';\nimport { resolvePluginTemplates } from '../template/plugin-resolver';\n\nconst log = createLogger('Helper');\n\n/**\n * Merge multiple metadata objects into one. Later sources override earlier ones.\n */\nexport function merge(...sources: Array<Partial<MetaDataType>>): MetaDataType {\n  return Object.assign({}, ...sources) as MetaDataType;\n}\n\n/**\n * Extract non-namespaced metadata keys as environment variable strings.\n * Keys starting with 'aws:cdk:' are reserved for CDK construct props\n * (processed by metadata extraction functions) and are excluded here.\n *\n * All values are converted to strings for CodeBuild compatibility.\n */\nexport function extractMetadataEnv(metadata: MetaDataType): Record<string, string> {\n  const env: Record<string, string> = {};\n  for (const [key, value] of Object.entries(metadata)) {\n    if (!key.startsWith(CDK_METADATA_PREFIX)) {\n      env[key] = String(value);\n    }\n  }\n  return env;\n}\n\n/**\n * Build environment variables from plugin config, merged metadata, and custom env.\n *\n * Merge order (last wins):\n *   1. plugin.env — plugin default env vars (lowest priority)\n *   2. non-namespaced metadata keys — e.g. PYTHON_VERSION, WORKDIR\n *   3. customEnv — per-step custom env vars (highest priority)\n */\nconst BOOTSTRAP_CMD = 'export WORKDIR=${WORKDIR:-./}; cd ${WORKDIR}';\n\nfunction buildEnv(plugin: Plugin, metadata: MetaDataType, customEnv?: Record<string, string>): Record<string, string> {\n  return {\n    ...(plugin.env ?? {}),\n    ...extractMetadataEnv(metadata),\n    ...(customEnv ?? {}),\n  };\n}\n\n/**\n * Wrap build commands based on failure behavior.\n * - 'fail' (default): No wrapping — commands fail the pipeline naturally.\n * - 'warn': Run commands with `set +e`, capture failures, log warnings, continue.\n * - 'ignore': Append `|| true` to each command — failures are silently swallowed.\n *\n * Only applied to build commands, not install commands (install failures should always stop the build).\n */\nfunction wrapCommandsForFailureBehavior(commands: string[], behavior?: 'fail' | 'warn' | 'ignore'): string[] {\n  if (!behavior || behavior === 'fail') return commands;\n\n  if (behavior === 'ignore') {\n    return commands.map(cmd => `${cmd} || true`);\n  }\n\n  // 'warn': run all commands, capture failures, but don't stop\n  return [\n    'set +e',\n    '_STEP_EXIT=0',\n    ...commands.map(cmd => `${cmd} || { echo \"WARNING: Command failed with exit code $?\"; _STEP_EXIT=1; }`),\n    'set -e',\n    'if [ \"$_STEP_EXIT\" -ne 0 ]; then echo \"WARNING: One or more commands in this step failed\"; fi',\n  ];\n}\n\n/**\n * Build bootstrap-prefixed install and build commands from plugin config.\n * Each command list is prepended with a WORKDIR bootstrap that defaults to './'.\n * When custom commands are provided, they are injected before/after the plugin's commands.\n * Build commands are optionally wrapped by failureBehavior logic.\n */\nfunction buildCommands(plugin: Plugin, custom?: StepCustomization, failureBehavior?: 'fail' | 'warn' | 'ignore'): { installCommands: string[]; commands: string[] } {\n  const userCommands = [\n    ...(custom?.preCommands ?? []),\n    ...(plugin.commands?.length ? plugin.commands : []),\n    ...(custom?.postCommands ?? []),\n  ];\n\n  return {\n    installCommands: [\n      BOOTSTRAP_CMD,\n      ...(custom?.preInstallCommands ?? []),\n      ...(plugin.installCommands ?? []),\n      ...(custom?.postInstallCommands ?? []),\n    ],\n    commands: [BOOTSTRAP_CMD, ...wrapCommandsForFailureBehavior(userCommands, failureBehavior)],\n  };\n}\n\n/**\n * Convert a plain env record to CodeBuild's environmentVariables format (PLAINTEXT).\n */\nfunction toCodeBuildEnvVars(env: Record<string, string>): Record<string, { value: string }> {\n  return Object.fromEntries(\n    Object.entries(env).map(([name, value]) => [name, { value }]),\n  );\n}\n\n/**\n * Build SECRETS_MANAGER-type environment variables from plugin secret declarations.\n * Uses naming convention: pipeline-builder/{orgId}/{secretName}\n * Each org manages these secrets in their own AWS Secrets Manager.\n */\nconst VALID_SECRET_NAME = /^[a-zA-Z0-9/_+=.@-]+$/;\n\nfunction toSecretEnvVars(\n  secrets: Array<{ name: string; required: boolean }>,\n  orgId: string,\n): Record<string, { value: string; type: BuildEnvironmentVariableType }> {\n  return Object.fromEntries(\n    secrets.map(({ name }) => {\n      const secretPath = CoreConstants.secretPath(orgId, name);\n      if (!VALID_SECRET_NAME.test(secretPath)) {\n        throw new Error(`Secret path \"${secretPath}\" contains invalid characters for AWS Secrets Manager`);\n      }\n      return [\n        name,\n        {\n          value: secretPath,\n          type: BuildEnvironmentVariableType.SECRETS_MANAGER,\n        },\n      ];\n    }),\n  );\n}\n\n/**\n * Create a CodeBuild step or Shell step based on plugin configuration.\n *\n * Metadata merge order (last wins):\n *   1. Step-level metadata (from options.metadata)\n *   2. Plugin metadata (from plugin.metadata in database)\n *\n * Environment merge order (last wins):\n *   1. Plugin env vars (from plugin.env)\n *   2. Custom env vars (from options.env)\n *   3. WORKDIR from merged metadata\n *\n * CDK prop spread order (last wins):\n *   programmatic defaults (input, commands, env, network) → metadata overrides\n *\n * This means metadata keys like `aws:cdk:pipelines:codebuildstep:commands`\n * will override the plugin-derived commands when explicitly set.\n */\n/**\n * Resolve the CodeBuild image to use for a plugin.\n *\n * Strategy:\n *   1. If the plugin sets `aws:cdk:codebuild:buildenvironment:buildImage`\n *      explicitly in metadata, the metadata-builder passthrough handles it\n *      (via `metadataForBuildEnvironment`). This function returns\n *      `undefined` and the metadata wins.\n *   2. Otherwise, if the plugin has an `imageTag` AND the registry config\n *      is populated, build a `LinuxBuildImage.fromDockerRegistry()` image\n *      pointing at `<registry-host>:<port>/system/<imageTag>:latest`. CodeBuild\n *      authenticates by sending the per-org platform Secret as Basic auth to\n *      `pipeline-image-registry`'s `/token` endpoint, which the registry's\n *      bearer challenge points at; the JWT in `password` resolves to a\n *      registry token scoped to the org.\n *   3. If neither (e.g., a `metadata_only` plugin or one with no image),\n *      return `undefined` so CodeBuild uses its default (`standard:7.0`).\n *\n * `scope` and `orgId` are required: the per-org platform Secret is named\n * `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`\n * needs a Construct to anchor the imported secret to.\n */\nexport function resolvePluginImage(scope: Construct | undefined, plugin: Plugin, orgId?: string): IBuildImage | undefined {\n  const imageTag = plugin.imageTag;\n\n  // `metadata_only` plugins legitimately have no image — their work runs\n  // in the default CodeBuild image. Quiet skip.\n  if (plugin.buildType === 'metadata_only') return undefined;\n\n  // No image tag means there's nothing to pull. This SHOULDN'T happen for\n  // build_image / prebuilt plugins (the platform writes imageTag at upload\n  // time); if it does, the plugin record is malformed.\n  if (!imageTag || imageTag === '') {\n    log.warn(\n      `Plugin \"${plugin.name}\" has buildType=${plugin.buildType} but no imageTag — ` +\n      'CodeBuild will run on aws/codebuild/standard:7.0 and won\\'t have the plugin\\'s baked tools. ' +\n      'Verify the plugin was uploaded correctly via load-plugins.sh.',\n    );\n    return undefined;\n  }\n\n  let registry;\n  try {\n    registry = Config.get('registry');\n  } catch {\n    // Config namespace not loaded (e.g., unit tests without full config).\n    log.warn(\n      `Plugin \"${plugin.name}\" has imageTag=\"${imageTag}\" but registry config not loaded — ` +\n      'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +\n      'Set IMAGE_REGISTRY_HOST + IMAGE_REGISTRY_PORT in pipeline-manager\\'s environment.',\n    );\n    return undefined;\n  }\n\n  if (!registry?.host) {\n    log.warn(\n      `Plugin \"${plugin.name}\" has imageTag=\"${imageTag}\" but IMAGE_REGISTRY_HOST is empty — ` +\n      'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +\n      'Set IMAGE_REGISTRY_HOST in pipeline-manager\\'s environment to use the plugin image.',\n    );\n    return undefined;\n  }\n  if (!scope) {\n    log.warn(`Plugin \"${plugin.name}\" image resolution skipped: no construct scope provided`);\n    return undefined;\n  }\n\n  // Per-org auth requires orgId — the Secret is named per org.\n  if (!orgId) {\n    log.warn(\n      `Plugin \"${plugin.name}\" image resolution skipped: orgId is required to ` +\n      'resolve the per-org platform Secret used for CodeBuild Basic auth.',\n    );\n    return undefined;\n  }\n\n  // Compose the image URI. Namespace by ownership:\n  //   - Plugins owned by the system org → `system/<imageTag>:latest`.\n  //     pipeline-image-registry's token service grants pull on `system/*`\n  //     to any authenticated org user (read-only catalog of shared plugins).\n  //   - Plugins owned by a tenant org → `org-<orgId>/<imageTag>:latest`.\n  //     The token service grants pull,push only to members of that org.\n  // The plugin's own `orgId` (not the caller's) decides the namespace —\n  // tenant pipelines pulling shared system plugins still get the `system/`\n  // path because that's where the image actually lives.\n  const portPart = registry.port && registry.port !== 80 && registry.port !== 443\n    ? `:${registry.port}`\n    : '';\n  const SYSTEM_ORG_ID = 'system';\n  const namespace = plugin.orgId === SYSTEM_ORG_ID\n    ? 'system'\n    : `org-${plugin.orgId}`;\n  const imageUri = `${registry.host}${portPart}/${namespace}/${imageTag}:latest`;\n\n  // CodeBuild reads `pipeline-builder/<orgId>/platform` and sends its\n  // `username`/`password` fields as HTTP Basic to the registry. The\n  // registry challenges with a Bearer realm pointing at\n  // pipeline-image-registry's `/token` endpoint; the Docker client forwards\n  // those creds, the password is verified as a platform JWT, and a registry\n  // token scoped to the org is issued.\n  //\n  // Same Secret is read by the plugin-lookup Lambda (via the `password`\n  // field) — one Secret serves both flows.\n  const stack = Stack.of(scope);\n  const secretName = CoreConstants.secretPath(orgId, 'platform');\n  // CDK construct IDs allow only [A-Za-z0-9_-]; sanitize the secret name.\n  const secretConstructId = `PlatformCreds_${secretName.replace(/[^A-Za-z0-9_-]/g, '_')}`;\n  const credentialsSecret = (stack.node.tryFindChild(secretConstructId) as Secret | undefined)\n    ?? Secret.fromSecretNameV2(stack, secretConstructId, secretName);\n\n  return LinuxBuildImage.fromDockerRegistry(imageUri, {\n    secretsManagerCredentials: credentialsSecret,\n  });\n}\n\nexport function createCodeBuildStep(options: CodeBuildStepOptions): ShellStep | CodeBuildStep | ManualApprovalStep {\n  const {\n    id, input, metadata, network, scope,\n    preInstallCommands, postInstallCommands, preCommands, postCommands,\n    env: customEnv, additionalInputs, timeout, failureBehavior,\n    artifactManager, stageName, stageAlias, pluginAlias, orgId, pipelineScope,\n  } = options;\n\n  // Resolve {{ ... }} templates in plugin spec fields against the pipeline scope.\n  // Every call goes through the resolver — plugins without template tokens are\n  // a no-op fast path inside resolvePluginTemplates().\n  const plugin = resolvePluginTemplates(options.plugin, pipelineScope);\n\n  const merged = merge(metadata ?? {}, plugin.metadata ?? {});\n\n  // ManualApprovalStep: no commands, env, compute, or network — just id + optional comment\n  if (plugin.pluginType === PluginType.MANUAL_APPROVAL_STEP) {\n    return new ManualApprovalStep(id, {\n      comment: typeof merged.APPROVAL_COMMENT === 'string' ? merged.APPROVAL_COMMENT : undefined,\n    });\n  }\n\n  log.debug('[CreateCodeBuildStep] Building step with merged metadata');\n\n  // Warn about required secrets without orgId (can't resolve)\n  const requiredSecrets = plugin.secrets?.filter(s => s.required) ?? [];\n  if (requiredSecrets.length > 0 && !orgId) {\n    log.warn(\n      `Plugin \"${plugin.name}\" declares ${requiredSecrets.length} required secret(s) but no orgId is available. ` +\n      `Secrets will not be injected: ${requiredSecrets.map(s => s.name).join(', ')}`,\n    );\n  }\n\n  // Resolve plugin secrets as SECRETS_MANAGER env vars\n  const secretEnvVars = (plugin.secrets?.length && orgId)\n    ? toSecretEnvVars(plugin.secrets, orgId)\n    : {};\n\n  const env = buildEnv(plugin, merged, customEnv);\n\n  // CodePipeline resolved variables — must be in CodeBuildStep.env (action-level),\n  // not buildEnvironment.environmentVariables (project-level)\n  const actionEnv: Record<string, string> = {\n    PIPELINE_EXECUTION_ID: '#{codepipeline.PipelineExecutionId}',\n  };\n\n  const outputDir = plugin.primaryOutputDirectory;\n  const ensureOutputDir = (outputDir && !outputDir.includes('*'))\n    ? [`mkdir -p \"${outputDir}\" && touch \"${outputDir}/.gitkeep\"`]\n    : [];\n\n  const { installCommands, commands } = buildCommands(plugin, {\n    preInstallCommands: [...ensureOutputDir, ...(preInstallCommands ?? [])],\n    postInstallCommands,\n    preCommands,\n    postCommands,\n  }, failureBehavior);\n\n  const programmatic = { input, installCommands, commands };\n\n  // Resolve the plugin's runtime image. CodeBuild defaults to\n  // `aws/codebuild/standard:7.0` when no `buildImage` is set — that's the\n  // AWS-managed Ubuntu image and DOES NOT have any plugin-baked tools\n  // (pipeline-manager, snyk, terraform, etc.). Wiring the plugin's image\n  // here means CodeBuild pulls from our private registry and the tools\n  // installed in the plugin's Dockerfile are actually available.\n  const pluginBuildImage = resolvePluginImage(scope, plugin, orgId);\n\n  // ShellStep branch.\n  // ShellStep itself doesn't accept `buildEnvironment`/`buildImage`. CDK\n  // pipelines wraps ShellSteps in a default CodeBuild action that runs on\n  // `aws/codebuild/standard:7.0` — which won't have the plugin's tools.\n  //\n  // So when a SHELL_STEP plugin DOES have an image (because its author\n  // baked tools into a Dockerfile), we PROMOTE it to a CodeBuildStep with\n  // the resolved image. The plugin author's intent (\"use my baked tools\")\n  // is preserved without forcing them to change pluginType.\n  //\n  // When the plugin has no image (or registry isn't configured), we keep\n  // the original ShellStep — it's lighter weight and the default\n  // CodeBuild image is fine.\n  if (plugin.pluginType === PluginType.SHELL_STEP) {\n    if (!pluginBuildImage) {\n      return new ShellStep(id, {\n        ...programmatic,\n        env: { ...env, ...actionEnv },\n        ...metadataForShellStep(merged),\n      });\n    }\n    log.debug(`[CreateCodeBuildStep] SHELL_STEP plugin \"${plugin.name}\" has imageTag — promoting to CodeBuildStep so the plugin image is actually used`);\n    // Fall through to the CodeBuildStep path below.\n  }\n\n  const computeType = getComputeType(\n    plugin.computeType ?? options.defaultComputeType ?? 'SMALL',\n  );\n\n  const networkProps = network\n    ? resolveNetwork(scope, options.uniqueId, network)\n    : {};\n\n  // Metadata spread last so it can override programmatic defaults.\n  // NOTE: Caching is supported via two metadata paths:\n  //   1. MetadataKeys.CACHE ('aws:cdk:pipelines:codebuildstep:cache') — passed directly\n  //      as the CodeBuildStep `cache` prop (expects a codebuild.Cache object).\n  //   2. MetadataKeys.PARTIAL_BUILD_SPEC ('aws:cdk:pipelines:codebuildstep:partialbuildspec')\n  //      — passed as `partialBuildSpec`, which can include a `cache:` section for\n  //      S3 or local caching (e.g., BuildSpec.fromObject({ cache: { paths: [...] } })).\n  const step = new CodeBuildStep(id, {\n    ...programmatic,\n    ...networkProps,\n    ...(additionalInputs && { additionalInputs }),\n    ...(timeout && { timeout: Duration.minutes(timeout) }),\n    env: actionEnv,\n    primaryOutputDirectory: plugin.primaryOutputDirectory ?? undefined,\n    buildEnvironment: {\n      computeType,\n      ...(pluginBuildImage && { buildImage: pluginBuildImage }),\n      environmentVariables: {\n        ...toCodeBuildEnvVars(env),\n        ...secretEnvVars,\n      },\n      ...metadataForBuildEnvironment(merged),\n    },\n    ...metadataForCodeBuildStep(merged),\n  });\n\n  // Register with artifact manager if primaryOutputDirectory is set\n  if (plugin.primaryOutputDirectory && artifactManager && stageName) {\n    const artifactKey: ArtifactKey = {\n      stageName,\n      stageAlias: stageAlias ?? `${stageName}-alias`,\n      pluginName: plugin.name,\n      pluginAlias: pluginAlias ?? `${plugin.name}-alias`,\n      outputDirectory: plugin.primaryOutputDirectory,\n    };\n    artifactManager.add(artifactKey, step);\n  }\n\n  return step;\n}\n\n/**\n * Convert string or ComputeType enum to CDK ComputeType\n */\nexport function getComputeType(input: string | CDKComputeType = 'SMALL'): CDKComputeType {\n  // If already a CDK ComputeType, return as-is\n  if (typeof input !== 'string') {\n    return input;\n  }\n\n  const normalized = input.toUpperCase() as ComputeType;\n\n  const mapping: Record<ComputeType, CDKComputeType> = {\n    [ComputeType.SMALL]: CDKComputeType.SMALL,\n    [ComputeType.MEDIUM]: CDKComputeType.MEDIUM,\n    [ComputeType.LARGE]: CDKComputeType.LARGE,\n    [ComputeType.X2_LARGE]: CDKComputeType.X2_LARGE,\n  };\n\n  const result = mapping[normalized];\n  if (!result) {\n    log.warn(`Unknown compute type \"${input}\", falling back to SMALL`);\n    return CDKComputeType.SMALL;\n  }\n  return result;\n}\n\n/**\n * Replaces all characters that are not letters or numbers with the specified value\n * @param input - The string to process\n * @param replaceValue - The character(s) to replace non-alphanumeric characters with (default: '_')\n * @returns The string with non-alphanumeric characters replaced\n */\nexport function replaceNonAlphanumeric(input: string, replaceValue: string = '_'): string {\n  return input.replace(/[^a-zA-Z0-9]/g, replaceValue);\n}\n\n/**\n * Unwrap a SecretValue | string into a plain string.\n * When a SecretValue is provided (e.g. from Secrets Manager), calls unsafeUnwrap()\n * to extract the underlying value.\n */\nexport function unwrapSecret(value: SecretValue | string): string {\n  return typeof value === 'string' ? value : value.unsafeUnwrap();\n}\n"]}
401
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-helpers.js","sourceRoot":"","sources":["../../src/core/pipeline-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { createLogger } from '@pipeline-builder/api-core';\nimport type { Plugin } from '@pipeline-builder/pipeline-data';\nimport { Duration, SecretValue, Stack } from 'aws-cdk-lib';\nimport { BuildEnvironmentVariableType, ComputeType as CDKComputeType, LinuxBuildImage, IBuildImage } from 'aws-cdk-lib/aws-codebuild';\nimport { Secret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { CodeBuildStep, ManualApprovalStep, ShellStep } from 'aws-cdk-lib/pipelines';\nimport type { Construct } from 'constructs';\nimport type { ArtifactKey } from './artifact-manager';\nimport { metadataForShellStep, metadataForCodeBuildStep, metadataForBuildEnvironment } from './metadata-builder';\nimport { resolveNetwork } from './network';\nimport { PluginType, ComputeType, MetaDataType, CDK_METADATA_PREFIX } from './pipeline-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport type { CodeBuildStepOptions, StepCustomization } from '../pipeline/step-types';\nimport { resolvePluginTemplates } from '../template/plugin-resolver';\n\nconst log = createLogger('Helper');\n\n/**\n * Merge multiple metadata objects into one. Later sources override earlier ones.\n */\nexport function merge(...sources: Array<Partial<MetaDataType>>): MetaDataType {\n  return Object.assign({}, ...sources) as MetaDataType;\n}\n\n/**\n * Extract non-namespaced metadata keys as environment variable strings.\n * Keys starting with 'aws:cdk:' are reserved for CDK construct props\n * (processed by metadata extraction functions) and are excluded here.\n *\n * All values are converted to strings for CodeBuild compatibility.\n */\nexport function extractMetadataEnv(metadata: MetaDataType): Record<string, string> {\n  const env: Record<string, string> = {};\n  for (const [key, value] of Object.entries(metadata)) {\n    if (!key.startsWith(CDK_METADATA_PREFIX)) {\n      env[key] = String(value);\n    }\n  }\n  return env;\n}\n\n/**\n * Build environment variables from plugin config, merged metadata, and custom env.\n *\n * Merge order (last wins):\n *   1. plugin.env — plugin default env vars (lowest priority)\n *   2. non-namespaced metadata keys — e.g. PYTHON_VERSION, WORKDIR\n *   3. customEnv — per-step custom env vars (highest priority)\n */\nconst BOOTSTRAP_CMD = 'export WORKDIR=${WORKDIR:-./}; cd ${WORKDIR}';\n\nfunction buildEnv(plugin: Plugin, metadata: MetaDataType, customEnv?: Record<string, string>): Record<string, string> {\n  return {\n    ...(plugin.env ?? {}),\n    ...extractMetadataEnv(metadata),\n    ...(customEnv ?? {}),\n  };\n}\n\n/**\n * Wrap build commands based on failure behavior.\n * - 'fail' (default): No wrapping — commands fail the pipeline naturally.\n * - 'warn': Run commands with `set +e`, capture failures, log warnings, continue.\n * - 'ignore': Append `|| true` to each command — failures are silently swallowed.\n *\n * Only applied to build commands, not install commands (install failures should always stop the build).\n */\nfunction wrapCommandsForFailureBehavior(commands: string[], behavior?: 'fail' | 'warn' | 'ignore'): string[] {\n  if (!behavior || behavior === 'fail') return commands;\n\n  if (behavior === 'ignore') {\n    return commands.map(cmd => `${cmd} || true`);\n  }\n\n  // 'warn': run all commands, capture failures, but don't stop\n  return [\n    'set +e',\n    '_STEP_EXIT=0',\n    ...commands.map(cmd => `${cmd} || { echo \"WARNING: Command failed with exit code $?\"; _STEP_EXIT=1; }`),\n    'set -e',\n    'if [ \"$_STEP_EXIT\" -ne 0 ]; then echo \"WARNING: One or more commands in this step failed\"; fi',\n  ];\n}\n\n/**\n * Build bootstrap-prefixed install and build commands from plugin config.\n * Each command list is prepended with a WORKDIR bootstrap that defaults to './'.\n * When custom commands are provided, they are injected before/after the plugin's commands.\n * Build commands are optionally wrapped by failureBehavior logic.\n */\nfunction buildCommands(plugin: Plugin, custom?: StepCustomization, failureBehavior?: 'fail' | 'warn' | 'ignore'): { installCommands: string[]; commands: string[] } {\n  const userCommands = [\n    ...(custom?.preCommands ?? []),\n    ...(plugin.commands?.length ? plugin.commands : []),\n    ...(custom?.postCommands ?? []),\n  ];\n\n  return {\n    installCommands: [\n      BOOTSTRAP_CMD,\n      ...(custom?.preInstallCommands ?? []),\n      ...(plugin.installCommands ?? []),\n      ...(custom?.postInstallCommands ?? []),\n    ],\n    commands: [BOOTSTRAP_CMD, ...wrapCommandsForFailureBehavior(userCommands, failureBehavior)],\n  };\n}\n\n/**\n * Convert a plain env record to CodeBuild's environmentVariables format (PLAINTEXT).\n */\nfunction toCodeBuildEnvVars(env: Record<string, string>): Record<string, { value: string }> {\n  return Object.fromEntries(\n    Object.entries(env).map(([name, value]) => [name, { value }]),\n  );\n}\n\n/**\n * Build SECRETS_MANAGER-type environment variables from plugin secret declarations.\n * Uses naming convention: pipeline-builder/{orgId}/{secretName}\n * Each org manages these secrets in their own AWS Secrets Manager.\n */\nconst VALID_SECRET_NAME = /^[a-zA-Z0-9/_+=.@-]+$/;\n\nfunction toSecretEnvVars(\n  secrets: Array<{ name: string; required: boolean }>,\n  orgId: string,\n): Record<string, { value: string; type: BuildEnvironmentVariableType }> {\n  return Object.fromEntries(\n    secrets.map(({ name }) => {\n      const secretPath = CoreConstants.secretPath(orgId, name);\n      if (!VALID_SECRET_NAME.test(secretPath)) {\n        throw new Error(`Secret path \"${secretPath}\" contains invalid characters for AWS Secrets Manager`);\n      }\n      return [\n        name,\n        {\n          value: secretPath,\n          type: BuildEnvironmentVariableType.SECRETS_MANAGER,\n        },\n      ];\n    }),\n  );\n}\n\n/**\n * Create a CodeBuild step or Shell step based on plugin configuration.\n *\n * Metadata merge order (last wins):\n *   1. Step-level metadata (from options.metadata)\n *   2. Plugin metadata (from plugin.metadata in database)\n *\n * Environment merge order (last wins):\n *   1. Plugin env vars (from plugin.env)\n *   2. Custom env vars (from options.env)\n *   3. WORKDIR from merged metadata\n *\n * CDK prop spread order (last wins):\n *   programmatic defaults (input, commands, env, network) → metadata overrides\n *\n * This means metadata keys like `aws:cdk:pipelines:codebuildstep:commands`\n * will override the plugin-derived commands when explicitly set.\n */\n/**\n * Resolve the CodeBuild image to use for a plugin.\n *\n * Strategy:\n *   1. If the plugin sets `aws:cdk:codebuild:buildenvironment:buildImage`\n *      explicitly in metadata, the metadata-builder passthrough handles it\n *      (via `metadataForBuildEnvironment`). This function returns\n *      `undefined` and the metadata wins.\n *   2. Otherwise, if the plugin has a `name`+`version` AND the registry\n *      config is populated, build a `LinuxBuildImage.fromDockerRegistry()`\n *      image pointing at `<registry-host>:<port>/<ns>/<name>:<version>`\n *      where `<ns>` is `system` or `org-<orgId>`. CodeBuild authenticates by\n *      sending the per-org platform Secret as Basic auth to\n *      `pipeline-image-registry`'s `/token` endpoint; the JWT in `password`\n *      resolves to a registry token scoped to the org.\n *   3. If `metadata_only`, return `undefined` so CodeBuild uses its default\n *      (`standard:7.0`).\n *\n * `scope` and `orgId` are required: the per-org platform Secret is named\n * `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`\n * needs a Construct to anchor the imported secret to.\n */\nexport function resolvePluginImage(scope: Construct | undefined, plugin: Plugin, orgId?: string): IBuildImage | undefined {\n  // `metadata_only` plugins legitimately have no image — their work runs\n  // in the default CodeBuild image. Quiet skip.\n  if (plugin.buildType === 'metadata_only') return undefined;\n\n  // build_image / prebuilt plugins must have name + version (DB constraint\n  // makes both NOT NULL). Defensive check in case a malformed row slips in.\n  if (!plugin.name || !plugin.version) {\n    log.warn(\n      `Plugin \"${plugin.name}\" has buildType=${plugin.buildType} but missing name/version — ` +\n      'CodeBuild will run on aws/codebuild/standard:7.0 and won\\'t have the plugin\\'s baked tools.',\n    );\n    return undefined;\n  }\n\n  let registry;\n  try {\n    registry = Config.get('registry');\n  } catch {\n    // Config namespace not loaded (e.g., unit tests without full config).\n    log.warn(\n      `Plugin \"${plugin.name}:${plugin.version}\" needs the registry config but it's not loaded — ` +\n      'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +\n      'Set IMAGE_REGISTRY_HOST + IMAGE_REGISTRY_PORT in pipeline-manager\\'s environment.',\n    );\n    return undefined;\n  }\n\n  if (!registry?.host) {\n    log.warn(\n      `Plugin \"${plugin.name}:${plugin.version}\" needs IMAGE_REGISTRY_HOST but it's empty — ` +\n      'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +\n      'Set IMAGE_REGISTRY_HOST in pipeline-manager\\'s environment to use the plugin image.',\n    );\n    return undefined;\n  }\n  if (!scope) {\n    log.warn(`Plugin \"${plugin.name}\" image resolution skipped: no construct scope provided`);\n    return undefined;\n  }\n\n  // Per-org auth requires orgId — the Secret is named per org.\n  if (!orgId) {\n    log.warn(\n      `Plugin \"${plugin.name}\" image resolution skipped: orgId is required to ` +\n      'resolve the per-org platform Secret used for CodeBuild Basic auth.',\n    );\n    return undefined;\n  }\n\n  // Compose the image URI. Namespace by ownership:\n  //   - Plugins owned by the system org → `system/<name>:<version>`.\n  //     pipeline-image-registry's token service grants pull on `system/*`\n  //     to any authenticated org user (read-only catalog of shared plugins).\n  //   - Plugins owned by a tenant org → `org-<orgId>/<name>:<version>`.\n  //     The token service grants pull,push only to members of that org.\n  // The plugin's own `orgId` (not the caller's) decides the namespace —\n  // tenant pipelines pulling shared system plugins still get the `system/`\n  // path because that's where the image actually lives.\n  const portPart = registry.port && registry.port !== 80 && registry.port !== 443\n    ? `:${registry.port}`\n    : '';\n  const SYSTEM_ORG_ID = 'system';\n  const namespace = plugin.orgId === SYSTEM_ORG_ID\n    ? 'system'\n    : `org-${plugin.orgId}`;\n  const imageUri = `${registry.host}${portPart}/${namespace}/${plugin.name}:${plugin.version}`;\n\n  // CodeBuild reads `pipeline-builder/<orgId>/platform` and sends its\n  // `username`/`password` fields as HTTP Basic to the registry. The\n  // registry challenges with a Bearer realm pointing at\n  // pipeline-image-registry's `/token` endpoint; the Docker client forwards\n  // those creds, the password is verified as a platform JWT, and a registry\n  // token scoped to the org is issued.\n  //\n  // Same Secret is read by the plugin-lookup Lambda (via the `password`\n  // field) — one Secret serves both flows.\n  const stack = Stack.of(scope);\n  const secretName = CoreConstants.secretPath(orgId, 'platform');\n  // CDK construct IDs allow only [A-Za-z0-9_-]; sanitize the secret name.\n  const secretConstructId = `PlatformCreds_${secretName.replace(/[^A-Za-z0-9_-]/g, '_')}`;\n  const credentialsSecret = (stack.node.tryFindChild(secretConstructId) as Secret | undefined)\n    ?? Secret.fromSecretNameV2(stack, secretConstructId, secretName);\n\n  return LinuxBuildImage.fromDockerRegistry(imageUri, {\n    secretsManagerCredentials: credentialsSecret,\n  });\n}\n\nexport function createCodeBuildStep(options: CodeBuildStepOptions): ShellStep | CodeBuildStep | ManualApprovalStep {\n  const {\n    id, input, metadata, network, scope,\n    preInstallCommands, postInstallCommands, preCommands, postCommands,\n    env: customEnv, additionalInputs, timeout, failureBehavior,\n    artifactManager, stageName, stageAlias, pluginAlias, orgId, pipelineScope,\n  } = options;\n\n  // Resolve {{ ... }} templates in plugin spec fields against the pipeline scope.\n  // Every call goes through the resolver — plugins without template tokens are\n  // a no-op fast path inside resolvePluginTemplates().\n  const plugin = resolvePluginTemplates(options.plugin, pipelineScope);\n\n  const merged = merge(metadata ?? {}, plugin.metadata ?? {});\n\n  // ManualApprovalStep: no commands, env, compute, or network — just id + optional comment\n  if (plugin.pluginType === PluginType.MANUAL_APPROVAL_STEP) {\n    return new ManualApprovalStep(id, {\n      comment: typeof merged.APPROVAL_COMMENT === 'string' ? merged.APPROVAL_COMMENT : undefined,\n    });\n  }\n\n  log.debug('[CreateCodeBuildStep] Building step with merged metadata');\n\n  // Warn about required secrets without orgId (can't resolve)\n  const requiredSecrets = plugin.secrets?.filter(s => s.required) ?? [];\n  if (requiredSecrets.length > 0 && !orgId) {\n    log.warn(\n      `Plugin \"${plugin.name}\" declares ${requiredSecrets.length} required secret(s) but no orgId is available. ` +\n      `Secrets will not be injected: ${requiredSecrets.map(s => s.name).join(', ')}`,\n    );\n  }\n\n  // Resolve plugin secrets as SECRETS_MANAGER env vars\n  const secretEnvVars = (plugin.secrets?.length && orgId)\n    ? toSecretEnvVars(plugin.secrets, orgId)\n    : {};\n\n  const env = buildEnv(plugin, merged, customEnv);\n\n  // CodePipeline resolved variables — must be in CodeBuildStep.env (action-level),\n  // not buildEnvironment.environmentVariables (project-level)\n  const actionEnv: Record<string, string> = {\n    PIPELINE_EXECUTION_ID: '#{codepipeline.PipelineExecutionId}',\n  };\n\n  const outputDir = plugin.primaryOutputDirectory;\n  const ensureOutputDir = (outputDir && !outputDir.includes('*'))\n    ? [`mkdir -p \"${outputDir}\" && touch \"${outputDir}/.gitkeep\"`]\n    : [];\n\n  const { installCommands, commands } = buildCommands(plugin, {\n    preInstallCommands: [...ensureOutputDir, ...(preInstallCommands ?? [])],\n    postInstallCommands,\n    preCommands,\n    postCommands,\n  }, failureBehavior);\n\n  const programmatic = { input, installCommands, commands };\n\n  // Resolve the plugin's runtime image. CodeBuild defaults to\n  // `aws/codebuild/standard:7.0` when no `buildImage` is set — that's the\n  // AWS-managed Ubuntu image and DOES NOT have any plugin-baked tools\n  // (pipeline-manager, snyk, terraform, etc.). Wiring the plugin's image\n  // here means CodeBuild pulls from our private registry and the tools\n  // installed in the plugin's Dockerfile are actually available.\n  const pluginBuildImage = resolvePluginImage(scope, plugin, orgId);\n\n  // ShellStep branch.\n  // ShellStep itself doesn't accept `buildEnvironment`/`buildImage`. CDK\n  // pipelines wraps ShellSteps in a default CodeBuild action that runs on\n  // `aws/codebuild/standard:7.0` — which won't have the plugin's tools.\n  //\n  // So when a SHELL_STEP plugin DOES have an image (because its author\n  // baked tools into a Dockerfile), we PROMOTE it to a CodeBuildStep with\n  // the resolved image. The plugin author's intent (\"use my baked tools\")\n  // is preserved without forcing them to change pluginType.\n  //\n  // When the plugin has no image (or registry isn't configured), we keep\n  // the original ShellStep — it's lighter weight and the default\n  // CodeBuild image is fine.\n  if (plugin.pluginType === PluginType.SHELL_STEP) {\n    if (!pluginBuildImage) {\n      return new ShellStep(id, {\n        ...programmatic,\n        env: { ...env, ...actionEnv },\n        ...metadataForShellStep(merged),\n      });\n    }\n    log.debug(`[CreateCodeBuildStep] SHELL_STEP plugin \"${plugin.name}\" has a registry image — promoting to CodeBuildStep so the plugin image is actually used`);\n    // Fall through to the CodeBuildStep path below.\n  }\n\n  const computeType = getComputeType(\n    plugin.computeType ?? options.defaultComputeType ?? 'SMALL',\n  );\n\n  const networkProps = network\n    ? resolveNetwork(scope, options.uniqueId, network)\n    : {};\n\n  // Metadata spread last so it can override programmatic defaults.\n  // NOTE: Caching is supported via two metadata paths:\n  //   1. MetadataKeys.CACHE ('aws:cdk:pipelines:codebuildstep:cache') — passed directly\n  //      as the CodeBuildStep `cache` prop (expects a codebuild.Cache object).\n  //   2. MetadataKeys.PARTIAL_BUILD_SPEC ('aws:cdk:pipelines:codebuildstep:partialbuildspec')\n  //      — passed as `partialBuildSpec`, which can include a `cache:` section for\n  //      S3 or local caching (e.g., BuildSpec.fromObject({ cache: { paths: [...] } })).\n  const step = new CodeBuildStep(id, {\n    ...programmatic,\n    ...networkProps,\n    ...(additionalInputs && { additionalInputs }),\n    ...(timeout && { timeout: Duration.minutes(timeout) }),\n    env: actionEnv,\n    primaryOutputDirectory: plugin.primaryOutputDirectory ?? undefined,\n    buildEnvironment: {\n      computeType,\n      ...(pluginBuildImage && { buildImage: pluginBuildImage }),\n      environmentVariables: {\n        ...toCodeBuildEnvVars(env),\n        ...secretEnvVars,\n      },\n      ...metadataForBuildEnvironment(merged),\n    },\n    ...metadataForCodeBuildStep(merged),\n  });\n\n  // Register with artifact manager if primaryOutputDirectory is set\n  if (plugin.primaryOutputDirectory && artifactManager && stageName) {\n    const artifactKey: ArtifactKey = {\n      stageName,\n      stageAlias: stageAlias ?? `${stageName}-alias`,\n      pluginName: plugin.name,\n      pluginAlias: pluginAlias ?? `${plugin.name}-alias`,\n      outputDirectory: plugin.primaryOutputDirectory,\n    };\n    artifactManager.add(artifactKey, step);\n  }\n\n  return step;\n}\n\n/**\n * Convert string or ComputeType enum to CDK ComputeType\n */\nexport function getComputeType(input: string | CDKComputeType = 'SMALL'): CDKComputeType {\n  // If already a CDK ComputeType, return as-is\n  if (typeof input !== 'string') {\n    return input;\n  }\n\n  const normalized = input.toUpperCase() as ComputeType;\n\n  const mapping: Record<ComputeType, CDKComputeType> = {\n    [ComputeType.SMALL]: CDKComputeType.SMALL,\n    [ComputeType.MEDIUM]: CDKComputeType.MEDIUM,\n    [ComputeType.LARGE]: CDKComputeType.LARGE,\n    [ComputeType.X2_LARGE]: CDKComputeType.X2_LARGE,\n  };\n\n  const result = mapping[normalized];\n  if (!result) {\n    log.warn(`Unknown compute type \"${input}\", falling back to SMALL`);\n    return CDKComputeType.SMALL;\n  }\n  return result;\n}\n\n/**\n * Replaces all characters that are not letters or numbers with the specified value\n * @param input - The string to process\n * @param replaceValue - The character(s) to replace non-alphanumeric characters with (default: '_')\n * @returns The string with non-alphanumeric characters replaced\n */\nexport function replaceNonAlphanumeric(input: string, replaceValue: string = '_'): string {\n  return input.replace(/[^a-zA-Z0-9]/g, replaceValue);\n}\n\n/**\n * Unwrap a SecretValue | string into a plain string.\n * When a SecretValue is provided (e.g. from Secrets Manager), calls unsafeUnwrap()\n * to extract the underlying value.\n */\nexport function unwrapSecret(value: SecretValue | string): string {\n  return typeof value === 'string' ? value : value.unsafeUnwrap();\n}\n"]}
@@ -285,7 +285,6 @@ const handler = async (event) => {
285
285
  secrets: plugin.secrets,
286
286
  failureBehavior: plugin.failureBehavior,
287
287
  timeout: plugin.timeout,
288
- imageTag: plugin.imageTag,
289
288
  };
290
289
  const encoded = Buffer.from(JSON.stringify(slim), 'utf-8').toString('base64');
291
290
  lambdaLog.debug('ENCODE', 'Encoded plugin data', { length: encoded.length });
@@ -315,4 +314,4 @@ const handler = async (event) => {
315
314
  }
316
315
  };
317
316
  exports.handler = handler;
318
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-lookup-handler.js","sourceRoot":"","sources":["../../src/handlers/plugin-lookup-handler.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { GetSecretValueCommand, SecretsManagerClient } from '@aws-sdk/client-secrets-manager';\nimport { PluginFilter, Plugin } from '@pipeline-builder/pipeline-data';\nimport { CloudFormationCustomResourceEvent, CloudFormationCustomResourceResponse } from 'aws-lambda';\nimport axios, { AxiosInstance, AxiosError } from 'axios';\nimport { CoreConstants } from '../config/app-config';\n\n/**\n * Structured logger for Lambda (outputs JSON to CloudWatch).\n * Debug messages only emitted when LOG_LEVEL=debug.\n */\nfunction logEntry(level: string, tag: string, message: string, data?: unknown) {\n  const line = JSON.stringify({ level, tag, message, data, ts: new Date().toISOString() });\n  switch (level) {\n    case 'ERROR': console.error(line); break;\n    case 'DEBUG': if (process.env.LOG_LEVEL === 'debug') console.debug(line); break;\n    default: console.log(line);\n  }\n}\n\nconst lambdaLog = {\n  info: (tag: string, message: string, data?: unknown) => logEntry('INFO', tag, message, data),\n  error: (tag: string, message: string, data?: unknown) => logEntry('ERROR', tag, message, data),\n  debug: (tag: string, message: string, data?: unknown) => logEntry('DEBUG', tag, message, data),\n};\n\nconst RETRYABLE_STATUSES = new Set([429, 502, 503, 504]);\nconst RETRYABLE_CODES = new Set(['ECONNRESET', 'ECONNREFUSED', 'ETIMEDOUT']);\n\nfunction sleep(ms: number): Promise<void> {\n  return new Promise(resolve => setTimeout(resolve, ms));\n}\n\n/** Platform secret name — injected as PLATFORM_SECRET_NAME env var by PluginLookup construct. */\nconst PLATFORM_SECRET_NAME = process.env.PLATFORM_SECRET_NAME;\nif (!PLATFORM_SECRET_NAME) {\n  throw new Error('PLATFORM_SECRET_NAME environment variable is required');\n}\n\n/** Cached token to avoid repeated Secrets Manager calls within a single invocation. */\nlet cachedToken: string | null = null;\n\n/** @internal Reset cached token (for testing only). */\nexport function _resetCredentialsCache(): void { cachedToken = null; }\n\n/**\n * Fetch JWT token from AWS Secrets Manager.\n * Caches the result for the lifetime of the Lambda execution context.\n *\n * The secret name is set via PLATFORM_SECRET_NAME env var (e.g. `{prefix}/{orgId}/platform`)\n * Create it with: `pipeline-manager store-token`\n */\nasync function getToken(): Promise<string> {\n  if (cachedToken) return cachedToken;\n\n  lambdaLog.info('AUTH', `Fetching token from Secrets Manager: ${PLATFORM_SECRET_NAME}`);\n\n  const client = new SecretsManagerClient({});\n  const response = await client.send(new GetSecretValueCommand({ SecretId: PLATFORM_SECRET_NAME }));\n\n  if (!response.SecretString) {\n    throw new Error(`Secret \"${PLATFORM_SECRET_NAME}\" is empty`);\n  }\n\n  const parsed = JSON.parse(response.SecretString) as Record<string, string>;\n  // Schema is `{ username, password, ... }` — the `password` field is the\n  // platform JWT. Same secret is read by CodeBuild's `secretsManagerCredentials`\n  // for registry pulls (Basic auth: username:password = orgId:JWT). The\n  // pipeline-image-registry token endpoint validates the password as a JWT\n  // and issues a registry token scoped to the JWT's org.\n  if (!parsed.password) {\n    throw new Error(`Secret \"${PLATFORM_SECRET_NAME}\" missing password — run \"pipeline-manager store-token\" to generate`);\n  }\n\n  cachedToken = parsed.password;\n  lambdaLog.info('AUTH', 'Token retrieved from Secrets Manager');\n  return cachedToken;\n}\n\n/**\n * Creates a pre-configured Axios instance for API requests.\n *\n * @param baseURL - Base URL of the target API\n * @param token - JWT token for authorization\n * @returns Configured Axios instance\n */\nfunction create(baseURL: string, token: string): AxiosInstance {\n  return axios.create({\n    baseURL,\n    timeout: CoreConstants.HANDLER_TIMEOUT_MS,\n    headers: {\n      'Content-Type': 'application/json',\n      'Authorization': `Bearer ${token}`,\n    },\n  });\n}\n\n/**\n * Fetches plugin configuration from the external API with retry logic.\n * Retries on transient failures (429, 502, 503, 504, network errors)\n * with exponential backoff.\n *\n * @param api - Configured Axios instance\n * @param pluginFilter - Filter criteria for the plugin lookup\n * @returns The plugin data returned by the API\n * @throws Error on persistent failure, timeout or invalid response\n */\nasync function fetch(api: AxiosInstance, pluginFilter: PluginFilter): Promise<Plugin> {\n  lambdaLog.debug('FETCH', 'Starting plugin fetch', { filter: pluginFilter });\n\n  let lastError: Error | undefined;\n\n  for (let attempt = 0; attempt <= CoreConstants.HANDLER_MAX_RETRIES; attempt++) {\n    if (attempt > 0) {\n      const delay = CoreConstants.HANDLER_RETRY_DELAY_MS * Math.pow(2, attempt - 1);\n      lambdaLog.info('RETRY', `Attempt ${attempt + 1}/${CoreConstants.HANDLER_MAX_RETRIES + 1} after ${delay}ms`);\n      await sleep(delay);\n    }\n\n    try {\n      const { data, status } = await api.post<Plugin>('/api/plugins/lookup', {\n        filter: pluginFilter,\n      });\n\n      if (!data) {\n        throw new Error('Empty response data from API');\n      }\n\n      lambdaLog.info('FETCH', 'Plugin fetched successfully', {\n        status,\n        plugin: data.name,\n        version: data.version,\n        id: data.id,\n      });\n\n      return data;\n    } catch (error) {\n      if (error instanceof AxiosError) {\n        if (error.code === 'ECONNABORTED') {\n          lambdaLog.error('FETCH', `Plugin lookup timed out after ${CoreConstants.HANDLER_TIMEOUT_MS}ms`);\n          throw new Error(`Plugin lookup timed out after ${CoreConstants.HANDLER_TIMEOUT_MS}ms`);\n        }\n\n        const retryable = error.response\n          ? RETRYABLE_STATUSES.has(error.response.status)\n          : RETRYABLE_CODES.has(error.code ?? '');\n\n        const msg = error.response\n          ? `API error ${error.response.status}: ${error.response.statusText}`\n          : error.code || error.message;\n\n        if (retryable && attempt < CoreConstants.HANDLER_MAX_RETRIES) {\n          lambdaLog.info('RETRY', `Retryable error: ${msg}`, { attempt: attempt + 1 });\n          lastError = new Error(`Failed to fetch plugin: ${msg}`);\n          continue;\n        }\n\n        lambdaLog.error('FETCH', msg, { responseData: error.response?.data });\n        throw new Error(`Failed to fetch plugin: ${msg}`);\n      }\n\n      throw error instanceof Error ? error : new Error('Unknown error during plugin fetch');\n    }\n  }\n\n  throw lastError ?? new Error('Failed to fetch plugin after retries');\n}\n\n/**\n * Validates the plugin filter object\n *\n * @param pluginFilter - Filter to validate\n * @returns true if valid\n * @throws Error if invalid\n */\nfunction validatePluginFilter(pluginFilter: unknown): pluginFilter is PluginFilter {\n  if (!pluginFilter || typeof pluginFilter !== 'object') {\n    throw new Error('Missing or invalid pluginFilter');\n  }\n\n  const filter = pluginFilter as Record<string, unknown>;\n  if (!filter.name && !filter.id && !filter.version && !filter.orgId) {\n    throw new Error('PluginFilter must have at least one criterion (name, id, version, or orgId)');\n  }\n\n  return true;\n}\n\n/**\n * Lambda handler for CloudFormation Custom Resource that performs plugin lookup.\n *\n * Authenticates using JWT token from AWS Secrets Manager (PLATFORM_SECRET_NAME env var).\n * Create the secret with: `pipeline-manager store-token`\n *\n * Request Types:\n * - Create/Update: fetches and returns plugin configuration from API\n * - Delete: no-op (always succeeds)\n *\n * Response:\n * - Success: Returns base64-encoded plugin JSON in Data.ResultValue\n * - Failure: Returns error message in Reason\n *\n * @param event - CloudFormation custom resource event\n * @returns CloudFormation response\n *\n * @example\n * Custom Resource Properties:\n * ```json\n * {\n *   \"baseURL\": \"https://api.example.com\",\n *   \"pluginFilter\": {\n *     \"name\": \"nodejs-build\",\n *     \"version\": \"1.0.0\",\n *     \"isActive\": true\n *   }\n * }\n * ```\n */\nexport const handler = async (\n  event: CloudFormationCustomResourceEvent,\n): Promise<CloudFormationCustomResourceResponse> => {\n  lambdaLog.info('START', `${event.RequestType} request received`, {\n    logicalResourceId: event.LogicalResourceId,\n    requestId: event.RequestId,\n    stackId: event.StackId,\n  });\n\n  const baseResponse: Partial<CloudFormationCustomResourceResponse> = {\n    StackId: event.StackId,\n    RequestId: event.RequestId,\n    LogicalResourceId: event.LogicalResourceId,\n    PhysicalResourceId: event.LogicalResourceId,\n  };\n\n  try {\n    // Handle Delete - always succeed (no-op)\n    if (event.RequestType === 'Delete') {\n      lambdaLog.info('DELETE', 'No-op - returning SUCCESS');\n      return {\n        ...baseResponse,\n        Status: 'SUCCESS',\n        Reason: 'Delete completed (no-op)',\n      } as CloudFormationCustomResourceResponse;\n    }\n\n    // Extract and validate properties\n    const pluginFilter = event.ResourceProperties.pluginFilter;\n    const baseURL = event.ResourceProperties.baseURL || CoreConstants.HANDLER_DEFAULT_BASE_URL;\n\n    if (!baseURL.startsWith('https://') && !baseURL.startsWith('http://')) {\n      throw new Error(`Invalid baseURL: \"${baseURL}\" — must start with http:// or https://`);\n    }\n\n    lambdaLog.info('CONFIG', 'Configuration loaded', { baseURL, pluginFilter });\n\n    validatePluginFilter(pluginFilter);\n\n    // Get token from Secrets Manager and create API client\n    const token = await getToken();\n    const api = create(baseURL, token);\n\n    // Fetch plugin\n    lambdaLog.info('FETCH', 'Initiating plugin lookup...');\n    const plugin = await fetch(api, pluginFilter);\n    lambdaLog.info('FETCH', 'Plugin retrieved successfully', {\n      name: plugin.name,\n      version: plugin.version,\n      id: plugin.id,\n    });\n\n    // Strip large fields to stay within CloudFormation's 4096-byte Data limit.\n    // CDK constructs only need the fields used by createCodeBuildStep().\n    const slim = {\n      id: plugin.id,\n      name: plugin.name,\n      version: plugin.version,\n      pluginType: plugin.pluginType,\n      computeType: plugin.computeType,\n      commands: plugin.commands,\n      installCommands: plugin.installCommands,\n      env: plugin.env,\n      metadata: plugin.metadata,\n      primaryOutputDirectory: plugin.primaryOutputDirectory,\n      secrets: plugin.secrets,\n      failureBehavior: plugin.failureBehavior,\n      timeout: plugin.timeout,\n      imageTag: plugin.imageTag,\n    };\n\n    const encoded = Buffer.from(JSON.stringify(slim), 'utf-8').toString('base64');\n    lambdaLog.debug('ENCODE', 'Encoded plugin data', { length: encoded.length });\n\n    return {\n      ...baseResponse,\n      Status: 'SUCCESS',\n      Reason: `Plugin '${plugin.name}' (v${plugin.version}) retrieved successfully`,\n      Data: {\n        ResultValue: encoded,\n      },\n    } as CloudFormationCustomResourceResponse;\n  } catch (error) {\n    const reason = error instanceof Error ? error.message : 'Unexpected error occurred';\n    lambdaLog.error('ERROR', 'Handler failed', {\n      reason,\n      stack: error instanceof Error ? error.stack : undefined,\n    });\n\n    return {\n      ...baseResponse,\n      Status: 'FAILED',\n      Reason: reason,\n    } as CloudFormationCustomResourceResponse;\n  } finally {\n    lambdaLog.info('END', 'Custom resource execution completed');\n  }\n};\n"]}
317
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-lookup-handler.js","sourceRoot":"","sources":["../../src/handlers/plugin-lookup-handler.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { GetSecretValueCommand, SecretsManagerClient } from '@aws-sdk/client-secrets-manager';\nimport { PluginFilter, Plugin } from '@pipeline-builder/pipeline-data';\nimport { CloudFormationCustomResourceEvent, CloudFormationCustomResourceResponse } from 'aws-lambda';\nimport axios, { AxiosInstance, AxiosError } from 'axios';\nimport { CoreConstants } from '../config/app-config';\n\n/**\n * Structured logger for Lambda (outputs JSON to CloudWatch).\n * Debug messages only emitted when LOG_LEVEL=debug.\n */\nfunction logEntry(level: string, tag: string, message: string, data?: unknown) {\n  const line = JSON.stringify({ level, tag, message, data, ts: new Date().toISOString() });\n  switch (level) {\n    case 'ERROR': console.error(line); break;\n    case 'DEBUG': if (process.env.LOG_LEVEL === 'debug') console.debug(line); break;\n    default: console.log(line);\n  }\n}\n\nconst lambdaLog = {\n  info: (tag: string, message: string, data?: unknown) => logEntry('INFO', tag, message, data),\n  error: (tag: string, message: string, data?: unknown) => logEntry('ERROR', tag, message, data),\n  debug: (tag: string, message: string, data?: unknown) => logEntry('DEBUG', tag, message, data),\n};\n\nconst RETRYABLE_STATUSES = new Set([429, 502, 503, 504]);\nconst RETRYABLE_CODES = new Set(['ECONNRESET', 'ECONNREFUSED', 'ETIMEDOUT']);\n\nfunction sleep(ms: number): Promise<void> {\n  return new Promise(resolve => setTimeout(resolve, ms));\n}\n\n/** Platform secret name — injected as PLATFORM_SECRET_NAME env var by PluginLookup construct. */\nconst PLATFORM_SECRET_NAME = process.env.PLATFORM_SECRET_NAME;\nif (!PLATFORM_SECRET_NAME) {\n  throw new Error('PLATFORM_SECRET_NAME environment variable is required');\n}\n\n/** Cached token to avoid repeated Secrets Manager calls within a single invocation. */\nlet cachedToken: string | null = null;\n\n/** @internal Reset cached token (for testing only). */\nexport function _resetCredentialsCache(): void { cachedToken = null; }\n\n/**\n * Fetch JWT token from AWS Secrets Manager.\n * Caches the result for the lifetime of the Lambda execution context.\n *\n * The secret name is set via PLATFORM_SECRET_NAME env var (e.g. `{prefix}/{orgId}/platform`)\n * Create it with: `pipeline-manager store-token`\n */\nasync function getToken(): Promise<string> {\n  if (cachedToken) return cachedToken;\n\n  lambdaLog.info('AUTH', `Fetching token from Secrets Manager: ${PLATFORM_SECRET_NAME}`);\n\n  const client = new SecretsManagerClient({});\n  const response = await client.send(new GetSecretValueCommand({ SecretId: PLATFORM_SECRET_NAME }));\n\n  if (!response.SecretString) {\n    throw new Error(`Secret \"${PLATFORM_SECRET_NAME}\" is empty`);\n  }\n\n  const parsed = JSON.parse(response.SecretString) as Record<string, string>;\n  // Schema is `{ username, password, ... }` — the `password` field is the\n  // platform JWT. Same secret is read by CodeBuild's `secretsManagerCredentials`\n  // for registry pulls (Basic auth: username:password = orgId:JWT). The\n  // pipeline-image-registry token endpoint validates the password as a JWT\n  // and issues a registry token scoped to the JWT's org.\n  if (!parsed.password) {\n    throw new Error(`Secret \"${PLATFORM_SECRET_NAME}\" missing password — run \"pipeline-manager store-token\" to generate`);\n  }\n\n  cachedToken = parsed.password;\n  lambdaLog.info('AUTH', 'Token retrieved from Secrets Manager');\n  return cachedToken;\n}\n\n/**\n * Creates a pre-configured Axios instance for API requests.\n *\n * @param baseURL - Base URL of the target API\n * @param token - JWT token for authorization\n * @returns Configured Axios instance\n */\nfunction create(baseURL: string, token: string): AxiosInstance {\n  return axios.create({\n    baseURL,\n    timeout: CoreConstants.HANDLER_TIMEOUT_MS,\n    headers: {\n      'Content-Type': 'application/json',\n      'Authorization': `Bearer ${token}`,\n    },\n  });\n}\n\n/**\n * Fetches plugin configuration from the external API with retry logic.\n * Retries on transient failures (429, 502, 503, 504, network errors)\n * with exponential backoff.\n *\n * @param api - Configured Axios instance\n * @param pluginFilter - Filter criteria for the plugin lookup\n * @returns The plugin data returned by the API\n * @throws Error on persistent failure, timeout or invalid response\n */\nasync function fetch(api: AxiosInstance, pluginFilter: PluginFilter): Promise<Plugin> {\n  lambdaLog.debug('FETCH', 'Starting plugin fetch', { filter: pluginFilter });\n\n  let lastError: Error | undefined;\n\n  for (let attempt = 0; attempt <= CoreConstants.HANDLER_MAX_RETRIES; attempt++) {\n    if (attempt > 0) {\n      const delay = CoreConstants.HANDLER_RETRY_DELAY_MS * Math.pow(2, attempt - 1);\n      lambdaLog.info('RETRY', `Attempt ${attempt + 1}/${CoreConstants.HANDLER_MAX_RETRIES + 1} after ${delay}ms`);\n      await sleep(delay);\n    }\n\n    try {\n      const { data, status } = await api.post<Plugin>('/api/plugins/lookup', {\n        filter: pluginFilter,\n      });\n\n      if (!data) {\n        throw new Error('Empty response data from API');\n      }\n\n      lambdaLog.info('FETCH', 'Plugin fetched successfully', {\n        status,\n        plugin: data.name,\n        version: data.version,\n        id: data.id,\n      });\n\n      return data;\n    } catch (error) {\n      if (error instanceof AxiosError) {\n        if (error.code === 'ECONNABORTED') {\n          lambdaLog.error('FETCH', `Plugin lookup timed out after ${CoreConstants.HANDLER_TIMEOUT_MS}ms`);\n          throw new Error(`Plugin lookup timed out after ${CoreConstants.HANDLER_TIMEOUT_MS}ms`);\n        }\n\n        const retryable = error.response\n          ? RETRYABLE_STATUSES.has(error.response.status)\n          : RETRYABLE_CODES.has(error.code ?? '');\n\n        const msg = error.response\n          ? `API error ${error.response.status}: ${error.response.statusText}`\n          : error.code || error.message;\n\n        if (retryable && attempt < CoreConstants.HANDLER_MAX_RETRIES) {\n          lambdaLog.info('RETRY', `Retryable error: ${msg}`, { attempt: attempt + 1 });\n          lastError = new Error(`Failed to fetch plugin: ${msg}`);\n          continue;\n        }\n\n        lambdaLog.error('FETCH', msg, { responseData: error.response?.data });\n        throw new Error(`Failed to fetch plugin: ${msg}`);\n      }\n\n      throw error instanceof Error ? error : new Error('Unknown error during plugin fetch');\n    }\n  }\n\n  throw lastError ?? new Error('Failed to fetch plugin after retries');\n}\n\n/**\n * Validates the plugin filter object\n *\n * @param pluginFilter - Filter to validate\n * @returns true if valid\n * @throws Error if invalid\n */\nfunction validatePluginFilter(pluginFilter: unknown): pluginFilter is PluginFilter {\n  if (!pluginFilter || typeof pluginFilter !== 'object') {\n    throw new Error('Missing or invalid pluginFilter');\n  }\n\n  const filter = pluginFilter as Record<string, unknown>;\n  if (!filter.name && !filter.id && !filter.version && !filter.orgId) {\n    throw new Error('PluginFilter must have at least one criterion (name, id, version, or orgId)');\n  }\n\n  return true;\n}\n\n/**\n * Lambda handler for CloudFormation Custom Resource that performs plugin lookup.\n *\n * Authenticates using JWT token from AWS Secrets Manager (PLATFORM_SECRET_NAME env var).\n * Create the secret with: `pipeline-manager store-token`\n *\n * Request Types:\n * - Create/Update: fetches and returns plugin configuration from API\n * - Delete: no-op (always succeeds)\n *\n * Response:\n * - Success: Returns base64-encoded plugin JSON in Data.ResultValue\n * - Failure: Returns error message in Reason\n *\n * @param event - CloudFormation custom resource event\n * @returns CloudFormation response\n *\n * @example\n * Custom Resource Properties:\n * ```json\n * {\n *   \"baseURL\": \"https://api.example.com\",\n *   \"pluginFilter\": {\n *     \"name\": \"nodejs-build\",\n *     \"version\": \"1.0.0\",\n *     \"isActive\": true\n *   }\n * }\n * ```\n */\nexport const handler = async (\n  event: CloudFormationCustomResourceEvent,\n): Promise<CloudFormationCustomResourceResponse> => {\n  lambdaLog.info('START', `${event.RequestType} request received`, {\n    logicalResourceId: event.LogicalResourceId,\n    requestId: event.RequestId,\n    stackId: event.StackId,\n  });\n\n  const baseResponse: Partial<CloudFormationCustomResourceResponse> = {\n    StackId: event.StackId,\n    RequestId: event.RequestId,\n    LogicalResourceId: event.LogicalResourceId,\n    PhysicalResourceId: event.LogicalResourceId,\n  };\n\n  try {\n    // Handle Delete - always succeed (no-op)\n    if (event.RequestType === 'Delete') {\n      lambdaLog.info('DELETE', 'No-op - returning SUCCESS');\n      return {\n        ...baseResponse,\n        Status: 'SUCCESS',\n        Reason: 'Delete completed (no-op)',\n      } as CloudFormationCustomResourceResponse;\n    }\n\n    // Extract and validate properties\n    const pluginFilter = event.ResourceProperties.pluginFilter;\n    const baseURL = event.ResourceProperties.baseURL || CoreConstants.HANDLER_DEFAULT_BASE_URL;\n\n    if (!baseURL.startsWith('https://') && !baseURL.startsWith('http://')) {\n      throw new Error(`Invalid baseURL: \"${baseURL}\" — must start with http:// or https://`);\n    }\n\n    lambdaLog.info('CONFIG', 'Configuration loaded', { baseURL, pluginFilter });\n\n    validatePluginFilter(pluginFilter);\n\n    // Get token from Secrets Manager and create API client\n    const token = await getToken();\n    const api = create(baseURL, token);\n\n    // Fetch plugin\n    lambdaLog.info('FETCH', 'Initiating plugin lookup...');\n    const plugin = await fetch(api, pluginFilter);\n    lambdaLog.info('FETCH', 'Plugin retrieved successfully', {\n      name: plugin.name,\n      version: plugin.version,\n      id: plugin.id,\n    });\n\n    // Strip large fields to stay within CloudFormation's 4096-byte Data limit.\n    // CDK constructs only need the fields used by createCodeBuildStep().\n    const slim = {\n      id: plugin.id,\n      name: plugin.name,\n      version: plugin.version,\n      pluginType: plugin.pluginType,\n      computeType: plugin.computeType,\n      commands: plugin.commands,\n      installCommands: plugin.installCommands,\n      env: plugin.env,\n      metadata: plugin.metadata,\n      primaryOutputDirectory: plugin.primaryOutputDirectory,\n      secrets: plugin.secrets,\n      failureBehavior: plugin.failureBehavior,\n      timeout: plugin.timeout,\n    };\n\n    const encoded = Buffer.from(JSON.stringify(slim), 'utf-8').toString('base64');\n    lambdaLog.debug('ENCODE', 'Encoded plugin data', { length: encoded.length });\n\n    return {\n      ...baseResponse,\n      Status: 'SUCCESS',\n      Reason: `Plugin '${plugin.name}' (v${plugin.version}) retrieved successfully`,\n      Data: {\n        ResultValue: encoded,\n      },\n    } as CloudFormationCustomResourceResponse;\n  } catch (error) {\n    const reason = error instanceof Error ? error.message : 'Unexpected error occurred';\n    lambdaLog.error('ERROR', 'Handler failed', {\n      reason,\n      stack: error instanceof Error ? error.stack : undefined,\n    });\n\n    return {\n      ...baseResponse,\n      Status: 'FAILED',\n      Reason: reason,\n    } as CloudFormationCustomResourceResponse;\n  } finally {\n    lambdaLog.info('END', 'Custom resource execution completed');\n  }\n};\n"]}
@@ -29,7 +29,7 @@ export interface PluginLookupProps {
29
29
  * Keyed by `alias || name` (matches the construct's normalize() output).
30
30
  * When a lookup hits this map, `plugin()` returns the resolved Plugin
31
31
  * directly — no custom resource is created. This is what makes the
32
- * imageTag, commands, env, etc. available at synth time so the resulting
32
+ * name, version, commands, env, etc. available at synth time so the resulting
33
33
  * CFN template ships with the real values baked in.
34
34
  */
35
35
  readonly resolvedPlugins?: Record<string, Plugin>;
@@ -92,7 +92,7 @@ class PluginLookup extends constructs_1.Construct {
92
92
  const props = this.normalize(plugin);
93
93
  // Pre-resolved by `pipeline-manager synth` from the platform API.
94
94
  // Skip the custom resource entirely — we already know the plugin's
95
- // imageTag, commands, env, etc. at synth time, so the resulting CFN
95
+ // name, version, commands, env, etc. at synth time, so the resulting CFN
96
96
  // template can ship with the real CodeBuild image baked in.
97
97
  const cacheKey = props.alias || props.name;
98
98
  const preResolved = this._resolvedPlugins?.[cacheKey];
@@ -232,7 +232,6 @@ class PluginLookup extends constructs_1.Construct {
232
232
  buildArgs: {},
233
233
  installCommands: [],
234
234
  commands: [],
235
- imageTag: '',
236
235
  dockerfile: null,
237
236
  buildType: 'metadata_only',
238
237
  accessModifier: 'public',
@@ -268,4 +267,4 @@ class PluginLookup extends constructs_1.Construct {
268
267
  }
269
268
  }
270
269
  exports.PluginLookup = PluginLookup;
271
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-lookup.js","sourceRoot":"","sources":["../../src/pipeline/plugin-lookup.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { join } from 'path';\nimport { createLogger } from '@pipeline-builder/api-core';\nimport { PluginFilter, Plugin } from '@pipeline-builder/pipeline-data';\nimport { CustomResource, Token, Duration } from 'aws-cdk-lib';\nimport { PolicyStatement, Effect } from 'aws-cdk-lib/aws-iam';\nimport { Runtime, Architecture } from 'aws-cdk-lib/aws-lambda';\nimport { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport type { PluginOptions } from './step-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport { UniqueId } from '../core/id-generator';\n\nconst log = createLogger('Lookup');\n\ninterface InputProps {\n  readonly baseURL: string;\n  readonly pluginFilter: PluginFilter;\n}\n\n/**\n * Configuration for PluginLookup construct\n */\nexport interface PluginLookupProps {\n  readonly organization: string;\n  readonly project: string;\n  readonly platformUrl: string;\n  readonly uniqueId: UniqueId;\n  /** Organization ID for resolving per-org secrets from Secrets Manager */\n  readonly orgId?: string;\n  readonly runtime?: Runtime;\n  /** Lambda timeout (default: 30s) */\n  readonly timeout?: Duration;\n  /** Lambda memory in MB (default: 512) */\n  readonly memorySize?: number;\n  /** Log retention (default: ONE_WEEK) */\n  readonly logRetention?: RetentionDays;\n  /** Reserved concurrent executions for the lookup Lambda (default: 30) */\n  readonly reservedConcurrentExecutions?: number;\n  /**\n   * Plugins pre-resolved by `pipeline-manager synth` from the platform API.\n   * Keyed by `alias || name` (matches the construct's normalize() output).\n   * When a lookup hits this map, `plugin()` returns the resolved Plugin\n   * directly — no custom resource is created. This is what makes the\n   * imageTag, commands, env, etc. available at synth time so the resulting\n   * CFN template ships with the real values baked in.\n   */\n  readonly resolvedPlugins?: Record<string, Plugin>;\n}\n\n/**\n * CDK Construct responsible for looking up plugin configurations from an external platform\n * using AWS CloudFormation Custom Resources backed by a Lambda function.\n *\n * This construct creates:\n * - A Lambda function (plugin-lookup-handler) that fetches plugin configs\n * - A CloudWatch Log Group for the Lambda\n * - A Custom Resource Provider that invokes the Lambda\n * - An IAM policy granting the Lambda access to the credentials secret\n *\n * ## Prerequisites\n *\n * Before deploying, store a JWT token in Secrets Manager:\n * ```sh\n * pipeline-manager store-token --days 30 --region <region>\n * ```\n *\n * The Lambda resolves the secret by name at runtime:\n * `{SECRETS_PATH_PREFIX}/{orgId}/platform`\n *\n * @see handlers/plugin-lookup-handler.ts for the Lambda implementation\n */\nexport class PluginLookup extends Construct {\n  private readonly _uniqueId: UniqueId;\n  private readonly _provider: Provider;\n  private readonly _platformUrl: string;\n  private readonly _runtime: Runtime;\n  private readonly _timeout: Duration;\n  private readonly _memorySize: number;\n  private readonly _reservedConcurrentExecutions?: number;\n  private readonly _orgId?: string;\n  private readonly _resolvedPlugins?: Record<string, Plugin>;\n\n  constructor(scope: Construct, id: string, props: PluginLookupProps) {\n    super(scope, id);\n\n    if (!props.organization || !props.project) {\n      throw new Error('Both organization and project are required.');\n    }\n\n    this._uniqueId = props.uniqueId;\n    this._platformUrl = props.platformUrl;\n    this._orgId = props.orgId;\n    this._runtime = props.runtime ?? Runtime.NODEJS_24_X;\n    this._timeout = props.timeout ?? Duration.seconds(30);\n    this._memorySize = props.memorySize ?? Config.get('aws').lambda.memorySize;\n    this._reservedConcurrentExecutions = props.reservedConcurrentExecutions;\n    this._resolvedPlugins = props.resolvedPlugins;\n\n    const onEventHandler = this.createLambdaFunction();\n\n    // Log-group strategy:\n    //   Previous code created an `AWS::Logs::LogGroup` resource with an\n    //   EXPLICIT name (`/aws/lambda/plugin-lookup-N`). That breaks every\n    //   re-deploy after a rolled-back stack: AWS auto-creates the group on\n    //   first Lambda invocation, the rollback leaves it as an orphan, and\n    //   the next CDK run fails with \"Resource ... already exists\" because\n    //   the explicit name collides with the orphan.\n    //\n    //   Fix: use `LogGroup.fromLogGroupName()` to adopt-or-pass-through.\n    //   If the group exists, CDK references it without trying to recreate.\n    //   If it doesn't, AWS Lambda auto-creates it on first invocation.\n    //   Retention is then set via a separate retention-policy custom\n    //   resource which is idempotent (safe to apply to existing groups).\n    const logGroupName = `/aws/lambda/${this._uniqueId.generate('plugin:lookup').replace(/:/g, '-')}`;\n    const logGroup = LogGroup.fromLogGroupName(\n      this,\n      this._uniqueId.generate('log:group'),\n      logGroupName,\n    );\n\n    this._provider = new Provider(this, this._uniqueId.generate('resource:provider'), {\n      onEventHandler,\n      logGroup,\n    });\n\n    log.debug(`PluginLookup initialized for ${props.organization}/${props.project}`);\n  }\n\n  /**\n   * Looks up and resolves plugin configuration using either a simple name or full PluginOptions object\n   * During synthesis, if the value is unresolved (token), returns fallback plugin\n   * During deployment, attempts to parse the actual value returned by the custom resource\n   * @param plugin - Plugin name (string) or complete PluginOptions configuration\n   * @returns Resolved Plugin object or fallback default configuration\n   */\n  public plugin(plugin: string | PluginOptions): Plugin {\n    const props = this.normalize(plugin);\n\n    // Pre-resolved by `pipeline-manager synth` from the platform API.\n    // Skip the custom resource entirely — we already know the plugin's\n    // imageTag, commands, env, etc. at synth time, so the resulting CFN\n    // template can ship with the real CodeBuild image baked in.\n    const cacheKey = props.alias || props.name;\n    const preResolved = this._resolvedPlugins?.[cacheKey];\n    if (preResolved) {\n      log.debug(`Plugin \"${props.name}\" pre-resolved (alias=${cacheKey}) — skipping custom resource`);\n      return preResolved;\n    }\n\n    const custom = this.createCustomResource(props);\n    const encoded = custom.getAttString('ResultValue');\n\n    if (Token.isUnresolved(encoded)) {\n      log.debug(`Plugin \"${props.name}\" value is unresolved (token) during synthesis — using fallback. The actual plugin will be resolved at deployment time.`);\n      return this.fallback();\n    }\n\n    try {\n      const decoded = Buffer.from(encoded, 'base64').toString('utf-8');\n      const data = JSON.parse(decoded);\n\n      if (!data || typeof data !== 'object' || !data.name || !Array.isArray(data.commands)) {\n        throw new Error('Invalid plugin response: missing required fields (name, commands)');\n      }\n\n      return data as Plugin;\n    } catch (error) {\n      const errorMsg = error instanceof Error ? error.message : 'Unknown error';\n      throw new Error(`Failed to parse plugin \"${props.name}\" data: ${errorMsg}`);\n    }\n  }\n\n  /**\n   * Creates the Lambda function that serves as the event handler for the custom resource provider.\n   *\n   * JWT token is stored in a pre-existing Secrets Manager secret at\n   * `{SECRETS_PATH_PREFIX}/{orgId}/platform`. The Lambda resolves the\n   * secret by name at runtime using `CoreConstants.SECRETS_PATH_PREFIX`.\n   *\n   * Create the secret before deploying with:\n   *   pipeline-manager store-token --days 30 --region <region>\n   */\n  private createLambdaFunction(): NodejsFunction {\n    if (!this._orgId) {\n      throw new Error('orgId is required for PluginLookup — needed to resolve the per-org platform secret');\n    }\n    const secretName = CoreConstants.secretPath(this._orgId, 'platform');\n\n    const fn = new NodejsFunction(this, this._uniqueId.generate('onevent:handler'), {\n      runtime: this._runtime,\n      timeout: this._timeout,\n      memorySize: this._memorySize,\n      architecture: Architecture.ARM_64,\n      entry: join(__dirname, '/../handlers/plugin-lookup-handler.js'),\n      depsLockFilePath: join(__dirname, '/../handlers/pnpm-lock.yaml'),\n      reservedConcurrentExecutions: this._reservedConcurrentExecutions,\n      environment: {\n        PLATFORM_SECRET_NAME: secretName,\n        // Allow self-signed certs when platform uses HTTPS without a CA-signed certificate\n        ...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {\n          NODE_TLS_REJECT_UNAUTHORIZED: '0',\n        }),\n      },\n      bundling: {\n        minify: true,\n        sourceMap: false,\n        target: 'es2022',\n        externalModules: ['@aws-sdk/*'],\n      },\n    });\n\n    // Grant the Lambda permission to read the per-org platform secret.\n    // The wildcard suffix handles the 6-char random ID that Secrets Manager appends.\n    fn.addToRolePolicy(new PolicyStatement({\n      effect: Effect.ALLOW,\n      actions: ['secretsmanager:GetSecretValue'],\n      resources: [`arn:aws:secretsmanager:*:*:secret:${secretName}-*`],\n    }));\n\n    return fn;\n  }\n\n  /**\n   * Build the default plugin filter.\n   * Access control (orgId scoping, public/private visibility) is handled by\n   * the platform's access control query builder based on the JWT's organizationId.\n   */\n  private defaultFilter(name: string): PluginFilter {\n    return {\n      name,\n      isActive: true,\n      isDefault: true,\n    };\n  }\n\n  private normalize(plugin: string | PluginOptions): PluginOptions {\n    if (typeof plugin === 'string') {\n      return {\n        name: plugin,\n        filter: this.defaultFilter(plugin),\n        alias: `${plugin}-alias`,\n      };\n    }\n\n    return {\n      name: plugin.name,\n      alias: plugin.alias ?? `${plugin.name}-alias`,\n      filter: plugin.filter ?? this.defaultFilter(plugin.name),\n      metadata: plugin.metadata,\n    };\n  }\n\n  /**\n   * Creates a CustomResource instance that triggers plugin lookup during deployment\n   */\n  private createCustomResource(props: PluginOptions): CustomResource {\n    const resourceId = this._uniqueId.generate(props.alias || props.name);\n\n    return new CustomResource(this, resourceId, {\n      serviceToken: this._provider.serviceToken,\n      resourceType: 'Custom::PluginLookup',\n      properties: {\n        baseURL: this._platformUrl,\n        pluginFilter: props.filter,\n      } as InputProps,\n    });\n  }\n\n  /** Base plugin shape with no-op defaults for fields CDK doesn't use. */\n  private static basePlugin(): Plugin {\n    const now = new Date();\n    return {\n      id: '00000000-0000-0000-0000-000000000000',\n      orgId: 'system',\n      createdBy: 'system',\n      createdAt: now,\n      updatedBy: 'system',\n      updatedAt: now,\n      name: 'fallback',\n      description: null,\n      keywords: [],\n      category: 'unknown',\n      version: '1.0.0',\n      metadata: {},\n      pluginType: 'CodeBuildStep',\n      computeType: 'SMALL',\n      timeout: null,\n      failureBehavior: 'fail',\n      secrets: [],\n      primaryOutputDirectory: 'cdk.out',\n      env: {},\n      buildArgs: {},\n      installCommands: [],\n      commands: [],\n      imageTag: '',\n      dockerfile: null,\n      buildType: 'metadata_only',\n      accessModifier: 'public',\n      isDefault: false,\n      isActive: true,\n      deletedAt: null,\n      deletedBy: null,\n    };\n  }\n\n  /** Fallback for unresolved plugin lookup tokens during synthesis. */\n  private fallback(): Plugin {\n    return {\n      ...PluginLookup.basePlugin(),\n      commands: ['echo \"FALLBACK: Plugin lookup unresolved — will be resolved at deployment time\"'],\n    };\n  }\n\n  /**\n   * Synth plugin with pipeline-manager commands. Cold-start fallback for the\n   * synth step when pre-resolution by `pipeline-manager synth/deploy` didn't\n   * populate `resolvedPlugins` for the synth plugin. Runs on `standard:7.0`\n   * via the default CodeBuild image and self-bootstraps the real synth via\n   * `pipeline-manager synth --id ${PIPELINE_ID}`.\n   */\n  public fallbackSynth(): Plugin {\n    return {\n      ...PluginLookup.basePlugin(),\n      name: 'cdk-synth',\n      primaryOutputDirectory: 'cdk.out',\n      commands: [\n        'pipeline-manager synth --id ${PIPELINE_ID} --store-tokens --quiet --no-notices --no-verify-ssl',\n      ],\n    };\n  }\n}\n"]}
270
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-lookup.js","sourceRoot":"","sources":["../../src/pipeline/plugin-lookup.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { join } from 'path';\nimport { createLogger } from '@pipeline-builder/api-core';\nimport { PluginFilter, Plugin } from '@pipeline-builder/pipeline-data';\nimport { CustomResource, Token, Duration } from 'aws-cdk-lib';\nimport { PolicyStatement, Effect } from 'aws-cdk-lib/aws-iam';\nimport { Runtime, Architecture } from 'aws-cdk-lib/aws-lambda';\nimport { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';\nimport { LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport type { PluginOptions } from './step-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport { UniqueId } from '../core/id-generator';\n\nconst log = createLogger('Lookup');\n\ninterface InputProps {\n  readonly baseURL: string;\n  readonly pluginFilter: PluginFilter;\n}\n\n/**\n * Configuration for PluginLookup construct\n */\nexport interface PluginLookupProps {\n  readonly organization: string;\n  readonly project: string;\n  readonly platformUrl: string;\n  readonly uniqueId: UniqueId;\n  /** Organization ID for resolving per-org secrets from Secrets Manager */\n  readonly orgId?: string;\n  readonly runtime?: Runtime;\n  /** Lambda timeout (default: 30s) */\n  readonly timeout?: Duration;\n  /** Lambda memory in MB (default: 512) */\n  readonly memorySize?: number;\n  /** Log retention (default: ONE_WEEK) */\n  readonly logRetention?: RetentionDays;\n  /** Reserved concurrent executions for the lookup Lambda (default: 30) */\n  readonly reservedConcurrentExecutions?: number;\n  /**\n   * Plugins pre-resolved by `pipeline-manager synth` from the platform API.\n   * Keyed by `alias || name` (matches the construct's normalize() output).\n   * When a lookup hits this map, `plugin()` returns the resolved Plugin\n   * directly — no custom resource is created. This is what makes the\n   * name, version, commands, env, etc. available at synth time so the resulting\n   * CFN template ships with the real values baked in.\n   */\n  readonly resolvedPlugins?: Record<string, Plugin>;\n}\n\n/**\n * CDK Construct responsible for looking up plugin configurations from an external platform\n * using AWS CloudFormation Custom Resources backed by a Lambda function.\n *\n * This construct creates:\n * - A Lambda function (plugin-lookup-handler) that fetches plugin configs\n * - A CloudWatch Log Group for the Lambda\n * - A Custom Resource Provider that invokes the Lambda\n * - An IAM policy granting the Lambda access to the credentials secret\n *\n * ## Prerequisites\n *\n * Before deploying, store a JWT token in Secrets Manager:\n * ```sh\n * pipeline-manager store-token --days 30 --region <region>\n * ```\n *\n * The Lambda resolves the secret by name at runtime:\n * `{SECRETS_PATH_PREFIX}/{orgId}/platform`\n *\n * @see handlers/plugin-lookup-handler.ts for the Lambda implementation\n */\nexport class PluginLookup extends Construct {\n  private readonly _uniqueId: UniqueId;\n  private readonly _provider: Provider;\n  private readonly _platformUrl: string;\n  private readonly _runtime: Runtime;\n  private readonly _timeout: Duration;\n  private readonly _memorySize: number;\n  private readonly _reservedConcurrentExecutions?: number;\n  private readonly _orgId?: string;\n  private readonly _resolvedPlugins?: Record<string, Plugin>;\n\n  constructor(scope: Construct, id: string, props: PluginLookupProps) {\n    super(scope, id);\n\n    if (!props.organization || !props.project) {\n      throw new Error('Both organization and project are required.');\n    }\n\n    this._uniqueId = props.uniqueId;\n    this._platformUrl = props.platformUrl;\n    this._orgId = props.orgId;\n    this._runtime = props.runtime ?? Runtime.NODEJS_24_X;\n    this._timeout = props.timeout ?? Duration.seconds(30);\n    this._memorySize = props.memorySize ?? Config.get('aws').lambda.memorySize;\n    this._reservedConcurrentExecutions = props.reservedConcurrentExecutions;\n    this._resolvedPlugins = props.resolvedPlugins;\n\n    const onEventHandler = this.createLambdaFunction();\n\n    // Log-group strategy:\n    //   Previous code created an `AWS::Logs::LogGroup` resource with an\n    //   EXPLICIT name (`/aws/lambda/plugin-lookup-N`). That breaks every\n    //   re-deploy after a rolled-back stack: AWS auto-creates the group on\n    //   first Lambda invocation, the rollback leaves it as an orphan, and\n    //   the next CDK run fails with \"Resource ... already exists\" because\n    //   the explicit name collides with the orphan.\n    //\n    //   Fix: use `LogGroup.fromLogGroupName()` to adopt-or-pass-through.\n    //   If the group exists, CDK references it without trying to recreate.\n    //   If it doesn't, AWS Lambda auto-creates it on first invocation.\n    //   Retention is then set via a separate retention-policy custom\n    //   resource which is idempotent (safe to apply to existing groups).\n    const logGroupName = `/aws/lambda/${this._uniqueId.generate('plugin:lookup').replace(/:/g, '-')}`;\n    const logGroup = LogGroup.fromLogGroupName(\n      this,\n      this._uniqueId.generate('log:group'),\n      logGroupName,\n    );\n\n    this._provider = new Provider(this, this._uniqueId.generate('resource:provider'), {\n      onEventHandler,\n      logGroup,\n    });\n\n    log.debug(`PluginLookup initialized for ${props.organization}/${props.project}`);\n  }\n\n  /**\n   * Looks up and resolves plugin configuration using either a simple name or full PluginOptions object\n   * During synthesis, if the value is unresolved (token), returns fallback plugin\n   * During deployment, attempts to parse the actual value returned by the custom resource\n   * @param plugin - Plugin name (string) or complete PluginOptions configuration\n   * @returns Resolved Plugin object or fallback default configuration\n   */\n  public plugin(plugin: string | PluginOptions): Plugin {\n    const props = this.normalize(plugin);\n\n    // Pre-resolved by `pipeline-manager synth` from the platform API.\n    // Skip the custom resource entirely — we already know the plugin's\n    // name, version, commands, env, etc. at synth time, so the resulting CFN\n    // template can ship with the real CodeBuild image baked in.\n    const cacheKey = props.alias || props.name;\n    const preResolved = this._resolvedPlugins?.[cacheKey];\n    if (preResolved) {\n      log.debug(`Plugin \"${props.name}\" pre-resolved (alias=${cacheKey}) — skipping custom resource`);\n      return preResolved;\n    }\n\n    const custom = this.createCustomResource(props);\n    const encoded = custom.getAttString('ResultValue');\n\n    if (Token.isUnresolved(encoded)) {\n      log.debug(`Plugin \"${props.name}\" value is unresolved (token) during synthesis — using fallback. The actual plugin will be resolved at deployment time.`);\n      return this.fallback();\n    }\n\n    try {\n      const decoded = Buffer.from(encoded, 'base64').toString('utf-8');\n      const data = JSON.parse(decoded);\n\n      if (!data || typeof data !== 'object' || !data.name || !Array.isArray(data.commands)) {\n        throw new Error('Invalid plugin response: missing required fields (name, commands)');\n      }\n\n      return data as Plugin;\n    } catch (error) {\n      const errorMsg = error instanceof Error ? error.message : 'Unknown error';\n      throw new Error(`Failed to parse plugin \"${props.name}\" data: ${errorMsg}`);\n    }\n  }\n\n  /**\n   * Creates the Lambda function that serves as the event handler for the custom resource provider.\n   *\n   * JWT token is stored in a pre-existing Secrets Manager secret at\n   * `{SECRETS_PATH_PREFIX}/{orgId}/platform`. The Lambda resolves the\n   * secret by name at runtime using `CoreConstants.SECRETS_PATH_PREFIX`.\n   *\n   * Create the secret before deploying with:\n   *   pipeline-manager store-token --days 30 --region <region>\n   */\n  private createLambdaFunction(): NodejsFunction {\n    if (!this._orgId) {\n      throw new Error('orgId is required for PluginLookup — needed to resolve the per-org platform secret');\n    }\n    const secretName = CoreConstants.secretPath(this._orgId, 'platform');\n\n    const fn = new NodejsFunction(this, this._uniqueId.generate('onevent:handler'), {\n      runtime: this._runtime,\n      timeout: this._timeout,\n      memorySize: this._memorySize,\n      architecture: Architecture.ARM_64,\n      entry: join(__dirname, '/../handlers/plugin-lookup-handler.js'),\n      depsLockFilePath: join(__dirname, '/../handlers/pnpm-lock.yaml'),\n      reservedConcurrentExecutions: this._reservedConcurrentExecutions,\n      environment: {\n        PLATFORM_SECRET_NAME: secretName,\n        // Allow self-signed certs when platform uses HTTPS without a CA-signed certificate\n        ...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {\n          NODE_TLS_REJECT_UNAUTHORIZED: '0',\n        }),\n      },\n      bundling: {\n        minify: true,\n        sourceMap: false,\n        target: 'es2022',\n        externalModules: ['@aws-sdk/*'],\n      },\n    });\n\n    // Grant the Lambda permission to read the per-org platform secret.\n    // The wildcard suffix handles the 6-char random ID that Secrets Manager appends.\n    fn.addToRolePolicy(new PolicyStatement({\n      effect: Effect.ALLOW,\n      actions: ['secretsmanager:GetSecretValue'],\n      resources: [`arn:aws:secretsmanager:*:*:secret:${secretName}-*`],\n    }));\n\n    return fn;\n  }\n\n  /**\n   * Build the default plugin filter.\n   * Access control (orgId scoping, public/private visibility) is handled by\n   * the platform's access control query builder based on the JWT's organizationId.\n   */\n  private defaultFilter(name: string): PluginFilter {\n    return {\n      name,\n      isActive: true,\n      isDefault: true,\n    };\n  }\n\n  private normalize(plugin: string | PluginOptions): PluginOptions {\n    if (typeof plugin === 'string') {\n      return {\n        name: plugin,\n        filter: this.defaultFilter(plugin),\n        alias: `${plugin}-alias`,\n      };\n    }\n\n    return {\n      name: plugin.name,\n      alias: plugin.alias ?? `${plugin.name}-alias`,\n      filter: plugin.filter ?? this.defaultFilter(plugin.name),\n      metadata: plugin.metadata,\n    };\n  }\n\n  /**\n   * Creates a CustomResource instance that triggers plugin lookup during deployment\n   */\n  private createCustomResource(props: PluginOptions): CustomResource {\n    const resourceId = this._uniqueId.generate(props.alias || props.name);\n\n    return new CustomResource(this, resourceId, {\n      serviceToken: this._provider.serviceToken,\n      resourceType: 'Custom::PluginLookup',\n      properties: {\n        baseURL: this._platformUrl,\n        pluginFilter: props.filter,\n      } as InputProps,\n    });\n  }\n\n  /** Base plugin shape with no-op defaults for fields CDK doesn't use. */\n  private static basePlugin(): Plugin {\n    const now = new Date();\n    return {\n      id: '00000000-0000-0000-0000-000000000000',\n      orgId: 'system',\n      createdBy: 'system',\n      createdAt: now,\n      updatedBy: 'system',\n      updatedAt: now,\n      name: 'fallback',\n      description: null,\n      keywords: [],\n      category: 'unknown',\n      version: '1.0.0',\n      metadata: {},\n      pluginType: 'CodeBuildStep',\n      computeType: 'SMALL',\n      timeout: null,\n      failureBehavior: 'fail',\n      secrets: [],\n      primaryOutputDirectory: 'cdk.out',\n      env: {},\n      buildArgs: {},\n      installCommands: [],\n      commands: [],\n      dockerfile: null,\n      buildType: 'metadata_only',\n      accessModifier: 'public',\n      isDefault: false,\n      isActive: true,\n      deletedAt: null,\n      deletedBy: null,\n    };\n  }\n\n  /** Fallback for unresolved plugin lookup tokens during synthesis. */\n  private fallback(): Plugin {\n    return {\n      ...PluginLookup.basePlugin(),\n      commands: ['echo \"FALLBACK: Plugin lookup unresolved — will be resolved at deployment time\"'],\n    };\n  }\n\n  /**\n   * Synth plugin with pipeline-manager commands. Cold-start fallback for the\n   * synth step when pre-resolution by `pipeline-manager synth/deploy` didn't\n   * populate `resolvedPlugins` for the synth plugin. Runs on `standard:7.0`\n   * via the default CodeBuild image and self-bootstraps the real synth via\n   * `pipeline-manager synth --id ${PIPELINE_ID}`.\n   */\n  public fallbackSynth(): Plugin {\n    return {\n      ...PluginLookup.basePlugin(),\n      name: 'cdk-synth',\n      primaryOutputDirectory: 'cdk.out',\n      commands: [\n        'pipeline-manager synth --id ${PIPELINE_ID} --store-tokens --quiet --no-notices --no-verify-ssl',\n      ],\n    };\n  }\n}\n"]}
@@ -36,7 +36,7 @@ function resolvePluginTemplates(plugin, pipelineScope) {
36
36
  const clone = structuredClone(plugin);
37
37
  const scope = {
38
38
  ...pipelineScope,
39
- plugin: { name: plugin.name, version: plugin.version, imageTag: plugin.imageTag },
39
+ plugin: { name: plugin.name, version: plugin.version },
40
40
  env: plugin.env ?? {},
41
41
  };
42
42
  const { errors } = (0, index_1.resolveTemplates)(clone, scope, isPluginTemplatableField, 'plugin');
@@ -49,4 +49,4 @@ function resolvePluginTemplates(plugin, pipelineScope) {
49
49
  }
50
50
  return clone;
51
51
  }
52
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2luLXJlc29sdmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3RlbXBsYXRlL3BsdWdpbi1yZXNvbHZlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7QUFxQnRDLDREQUlDO0FBT0Qsd0RBdUJDO0FBcERELG1DQUEyQztBQUUzQzs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxrQkFBa0IsR0FBRztJQUN6QixhQUFhO0lBQ2IsVUFBVSxFQUFFLFdBQVc7SUFDdkIsaUJBQWlCLEVBQUUsV0FBVztJQUM5QixLQUFLLEVBQUUseUJBQXlCO0lBQ2hDLFdBQVcsRUFBRSx5QkFBeUI7Q0FDOUIsQ0FBQztBQUVYLFNBQWdCLHdCQUF3QixDQUFDLEtBQWE7SUFDcEQsb0ZBQW9GO0lBQ3BGLDZEQUE2RDtJQUM3RCxPQUFPLGtCQUFrQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEtBQUssS0FBSyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUM3RyxDQUFDO0FBRUQ7Ozs7R0FJRztBQUNILFNBQWdCLHNCQUFzQixDQUNwQyxNQUFjLEVBQ2QsYUFBc0M7SUFFdEMsa0VBQWtFO0lBQ2xFLDJFQUEyRTtJQUMzRSxNQUFNLEtBQUssR0FBRyxlQUFlLENBQUMsTUFBTSxDQUFxQyxDQUFDO0lBRTFFLE1BQU0sS0FBSyxHQUFHO1FBQ1osR0FBRyxhQUFhO1FBQ2hCLE1BQU0sRUFBRSxFQUFFLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxFQUFFO1FBQ2pGLEdBQUcsRUFBRSxNQUFNLENBQUMsR0FBRyxJQUFJLEVBQUU7S0FDdEIsQ0FBQztJQUVGLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFBLHdCQUFnQixFQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsd0JBQXdCLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDdEYsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1FBQ3RCLDJFQUEyRTtRQUMzRSwwRUFBMEU7UUFDMUUsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBRSxDQUFDO1FBQ3JCLE1BQU0sR0FBRyxHQUFHLHlDQUF5QyxNQUFNLENBQUMsSUFBSSxlQUFlLENBQUMsQ0FBQyxLQUFLLE1BQU0sQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3hHLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDdkIsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFDO0FBQ2YsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgdHlwZSB7IFBsdWdpbiB9IGZyb20gJ0BwaXBlbGluZS1idWlsZGVyL3BpcGVsaW5lLWRhdGEnO1xuaW1wb3J0IHsgcmVzb2x2ZVRlbXBsYXRlcyB9IGZyb20gJy4vaW5kZXgnO1xuXG4vKipcbiAqIEZpZWxkcyBpbnNpZGUgYSBQbHVnaW4gcmVjb3JkIHRoYXQgYWNjZXB0IGB7eyAuLi4gfX1gIHRlbXBsYXRlcy5cbiAqXG4gKiBQdXJlLXN0cmluZyBsZWF2ZXMgb25seS4gYG5hbWVgLCBgdmVyc2lvbmAsIGBwbHVnaW5UeXBlYCwgYGNvbXB1dGVUeXBlYCxcbiAqIGB0aW1lb3V0YCwgYHNlY3JldHNgLCBgZmFpbHVyZUJlaGF2aW9yYCwgYHJlcXVpcmVkTWV0YWRhdGFgLCBgcmVxdWlyZWRWYXJzYFxuICogc3RheSBsaXRlcmFsLiBgbWV0YWRhdGEuKmAgaXMgZXhjbHVkZWQgYmVjYXVzZSBDREstbWV0YWRhdGEgdmFsdWVzIGFyZVxuICogc3RydWN0dXJhbCwgbm90IHVzZXItaW50ZXJwb2xhdGVkLlxuICovXG5jb25zdCBURU1QTEFUQUJMRV9GSUVMRFMgPSBbXG4gICdkZXNjcmlwdGlvbicsXG4gICdjb21tYW5kcycsIC8vIHN0cmluZ1tdXG4gICdpbnN0YWxsQ29tbWFuZHMnLCAvLyBzdHJpbmdbXVxuICAnZW52JywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuICAnYnVpbGRBcmdzJywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuXSBhcyBjb25zdDtcblxuZXhwb3J0IGZ1bmN0aW9uIGlzUGx1Z2luVGVtcGxhdGFibGVGaWVsZChmaWVsZDogc3RyaW5nKTogYm9vbGVhbiB7XG4gIC8vIGBjb21tYW5kc2AgLyBgaW5zdGFsbENvbW1hbmRzYCBhcmUgYXJyYXlzIG9mIHN0cmluZ3Mg4oaSIGVudHJpZXMgbGlrZSAnY29tbWFuZHNbMF0nXG4gIC8vIGBlbnZgIC8gYGJ1aWxkQXJnc2AgYXJlIG9iamVjdHMg4oaSIGVudHJpZXMgbGlrZSAnZW52LlNUQUdFJ1xuICByZXR1cm4gVEVNUExBVEFCTEVfRklFTERTLnNvbWUoZiA9PiBmaWVsZCA9PT0gZiB8fCBmaWVsZC5zdGFydHNXaXRoKGAke2Z9W2ApIHx8IGZpZWxkLnN0YXJ0c1dpdGgoYCR7Zn0uYCkpO1xufVxuXG4vKipcbiAqIFJldHVybiBhIHNoYWxsb3cgY2xvbmUgb2YgYHBsdWdpbmAgd2l0aCBhbGwgYHt7IC4uLiB9fWAgdGVtcGxhdGVzXG4gKiByZXNvbHZlZCBhZ2FpbnN0IHRoZSBnaXZlbiBwaXBlbGluZSBzY29wZS4gQ2FsbGVyIG11c3QgcHJlLXBvcHVsYXRlXG4gKiBgcGlwZWxpbmVTY29wZWAgd2l0aCBgeyBwaXBlbGluZSwgcGx1Z2luLCBlbnYgfWAga2V5cy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVQbHVnaW5UZW1wbGF0ZXMoXG4gIHBsdWdpbjogUGx1Z2luLFxuICBwaXBlbGluZVNjb3BlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbik6IFBsdWdpbiB7XG4gIC8vIERlZXAgY2xvbmUgc28gbXV0YXRpb25zIGRvbid0IGxlYWsgYmFjayB0byB0aGUgY2FsbGVyJ3MgUGx1Z2luLlxuICAvLyBTdHJ1Y3R1cmFsIGZpZWxkcyB3ZSBjYXJlIGFib3V0IGFyZSBwbGFpbiBKU09OOyBzdHJ1Y3R1cmVkQ2xvbmUgaXMgc2FmZS5cbiAgY29uc3QgY2xvbmUgPSBzdHJ1Y3R1cmVkQ2xvbmUocGx1Z2luKSBhcyBQbHVnaW4gJiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcblxuICBjb25zdCBzY29wZSA9IHtcbiAgICAuLi5waXBlbGluZVNjb3BlLFxuICAgIHBsdWdpbjogeyBuYW1lOiBwbHVnaW4ubmFtZSwgdmVyc2lvbjogcGx1Z2luLnZlcnNpb24sIGltYWdlVGFnOiBwbHVnaW4uaW1hZ2VUYWcgfSxcbiAgICBlbnY6IHBsdWdpbi5lbnYgPz8ge30sXG4gIH07XG5cbiAgY29uc3QgeyBlcnJvcnMgfSA9IHJlc29sdmVUZW1wbGF0ZXMoY2xvbmUsIHNjb3BlLCBpc1BsdWdpblRlbXBsYXRhYmxlRmllbGQsICdwbHVnaW4nKTtcbiAgaWYgKGVycm9ycy5sZW5ndGggPiAwKSB7XG4gICAgLy8gRmlyc3QgZXJyb3Igd2lucyDigJQgcmVzb2x2ZXIgZXJyb3JzIHNob3VsZCBuZXZlciBiZSBiYXRjaGVkIGF0IHN5bnRoIHRpbWVcbiAgICAvLyBiZWNhdXNlIGEgYnJva2VuIHRlbXBsYXRlIGlzIGEgcHJvZ3JhbW1lciBlcnJvciwgbm90IGEgdmFsaWRhdGlvbiBzdGVwLlxuICAgIGNvbnN0IGUgPSBlcnJvcnNbMF0hO1xuICAgIGNvbnN0IG1zZyA9IGBUZW1wbGF0ZSByZXNvbHV0aW9uIGZhaWxlZCBpbiBwbHVnaW4gXCIke3BsdWdpbi5uYW1lfVwiIGF0IGZpZWxkICcke2UuZmllbGR9JzogJHtlLm1lc3NhZ2V9YDtcbiAgICB0aHJvdyBuZXcgRXJyb3IobXNnKTtcbiAgfVxuICByZXR1cm4gY2xvbmU7XG59XG4iXX0=
52
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2luLXJlc29sdmVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3RlbXBsYXRlL3BsdWdpbi1yZXNvbHZlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7QUFxQnRDLDREQUlDO0FBT0Qsd0RBdUJDO0FBcERELG1DQUEyQztBQUUzQzs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxrQkFBa0IsR0FBRztJQUN6QixhQUFhO0lBQ2IsVUFBVSxFQUFFLFdBQVc7SUFDdkIsaUJBQWlCLEVBQUUsV0FBVztJQUM5QixLQUFLLEVBQUUseUJBQXlCO0lBQ2hDLFdBQVcsRUFBRSx5QkFBeUI7Q0FDOUIsQ0FBQztBQUVYLFNBQWdCLHdCQUF3QixDQUFDLEtBQWE7SUFDcEQsb0ZBQW9GO0lBQ3BGLDZEQUE2RDtJQUM3RCxPQUFPLGtCQUFrQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEtBQUssS0FBSyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUM3RyxDQUFDO0FBRUQ7Ozs7R0FJRztBQUNILFNBQWdCLHNCQUFzQixDQUNwQyxNQUFjLEVBQ2QsYUFBc0M7SUFFdEMsa0VBQWtFO0lBQ2xFLDJFQUEyRTtJQUMzRSxNQUFNLEtBQUssR0FBRyxlQUFlLENBQUMsTUFBTSxDQUFxQyxDQUFDO0lBRTFFLE1BQU0sS0FBSyxHQUFHO1FBQ1osR0FBRyxhQUFhO1FBQ2hCLE1BQU0sRUFBRSxFQUFFLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxFQUFFO1FBQ3RELEdBQUcsRUFBRSxNQUFNLENBQUMsR0FBRyxJQUFJLEVBQUU7S0FDdEIsQ0FBQztJQUVGLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxJQUFBLHdCQUFnQixFQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsd0JBQXdCLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDdEYsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1FBQ3RCLDJFQUEyRTtRQUMzRSwwRUFBMEU7UUFDMUUsTUFBTSxDQUFDLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBRSxDQUFDO1FBQ3JCLE1BQU0sR0FBRyxHQUFHLHlDQUF5QyxNQUFNLENBQUMsSUFBSSxlQUFlLENBQUMsQ0FBQyxLQUFLLE1BQU0sQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ3hHLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDdkIsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFDO0FBQ2YsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgdHlwZSB7IFBsdWdpbiB9IGZyb20gJ0BwaXBlbGluZS1idWlsZGVyL3BpcGVsaW5lLWRhdGEnO1xuaW1wb3J0IHsgcmVzb2x2ZVRlbXBsYXRlcyB9IGZyb20gJy4vaW5kZXgnO1xuXG4vKipcbiAqIEZpZWxkcyBpbnNpZGUgYSBQbHVnaW4gcmVjb3JkIHRoYXQgYWNjZXB0IGB7eyAuLi4gfX1gIHRlbXBsYXRlcy5cbiAqXG4gKiBQdXJlLXN0cmluZyBsZWF2ZXMgb25seS4gYG5hbWVgLCBgdmVyc2lvbmAsIGBwbHVnaW5UeXBlYCwgYGNvbXB1dGVUeXBlYCxcbiAqIGB0aW1lb3V0YCwgYHNlY3JldHNgLCBgZmFpbHVyZUJlaGF2aW9yYCwgYHJlcXVpcmVkTWV0YWRhdGFgLCBgcmVxdWlyZWRWYXJzYFxuICogc3RheSBsaXRlcmFsLiBgbWV0YWRhdGEuKmAgaXMgZXhjbHVkZWQgYmVjYXVzZSBDREstbWV0YWRhdGEgdmFsdWVzIGFyZVxuICogc3RydWN0dXJhbCwgbm90IHVzZXItaW50ZXJwb2xhdGVkLlxuICovXG5jb25zdCBURU1QTEFUQUJMRV9GSUVMRFMgPSBbXG4gICdkZXNjcmlwdGlvbicsXG4gICdjb21tYW5kcycsIC8vIHN0cmluZ1tdXG4gICdpbnN0YWxsQ29tbWFuZHMnLCAvLyBzdHJpbmdbXVxuICAnZW52JywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuICAnYnVpbGRBcmdzJywgLy8gUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuXSBhcyBjb25zdDtcblxuZXhwb3J0IGZ1bmN0aW9uIGlzUGx1Z2luVGVtcGxhdGFibGVGaWVsZChmaWVsZDogc3RyaW5nKTogYm9vbGVhbiB7XG4gIC8vIGBjb21tYW5kc2AgLyBgaW5zdGFsbENvbW1hbmRzYCBhcmUgYXJyYXlzIG9mIHN0cmluZ3Mg4oaSIGVudHJpZXMgbGlrZSAnY29tbWFuZHNbMF0nXG4gIC8vIGBlbnZgIC8gYGJ1aWxkQXJnc2AgYXJlIG9iamVjdHMg4oaSIGVudHJpZXMgbGlrZSAnZW52LlNUQUdFJ1xuICByZXR1cm4gVEVNUExBVEFCTEVfRklFTERTLnNvbWUoZiA9PiBmaWVsZCA9PT0gZiB8fCBmaWVsZC5zdGFydHNXaXRoKGAke2Z9W2ApIHx8IGZpZWxkLnN0YXJ0c1dpdGgoYCR7Zn0uYCkpO1xufVxuXG4vKipcbiAqIFJldHVybiBhIHNoYWxsb3cgY2xvbmUgb2YgYHBsdWdpbmAgd2l0aCBhbGwgYHt7IC4uLiB9fWAgdGVtcGxhdGVzXG4gKiByZXNvbHZlZCBhZ2FpbnN0IHRoZSBnaXZlbiBwaXBlbGluZSBzY29wZS4gQ2FsbGVyIG11c3QgcHJlLXBvcHVsYXRlXG4gKiBgcGlwZWxpbmVTY29wZWAgd2l0aCBgeyBwaXBlbGluZSwgcGx1Z2luLCBlbnYgfWAga2V5cy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVQbHVnaW5UZW1wbGF0ZXMoXG4gIHBsdWdpbjogUGx1Z2luLFxuICBwaXBlbGluZVNjb3BlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbik6IFBsdWdpbiB7XG4gIC8vIERlZXAgY2xvbmUgc28gbXV0YXRpb25zIGRvbid0IGxlYWsgYmFjayB0byB0aGUgY2FsbGVyJ3MgUGx1Z2luLlxuICAvLyBTdHJ1Y3R1cmFsIGZpZWxkcyB3ZSBjYXJlIGFib3V0IGFyZSBwbGFpbiBKU09OOyBzdHJ1Y3R1cmVkQ2xvbmUgaXMgc2FmZS5cbiAgY29uc3QgY2xvbmUgPSBzdHJ1Y3R1cmVkQ2xvbmUocGx1Z2luKSBhcyBQbHVnaW4gJiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcblxuICBjb25zdCBzY29wZSA9IHtcbiAgICAuLi5waXBlbGluZVNjb3BlLFxuICAgIHBsdWdpbjogeyBuYW1lOiBwbHVnaW4ubmFtZSwgdmVyc2lvbjogcGx1Z2luLnZlcnNpb24gfSxcbiAgICBlbnY6IHBsdWdpbi5lbnYgPz8ge30sXG4gIH07XG5cbiAgY29uc3QgeyBlcnJvcnMgfSA9IHJlc29sdmVUZW1wbGF0ZXMoY2xvbmUsIHNjb3BlLCBpc1BsdWdpblRlbXBsYXRhYmxlRmllbGQsICdwbHVnaW4nKTtcbiAgaWYgKGVycm9ycy5sZW5ndGggPiAwKSB7XG4gICAgLy8gRmlyc3QgZXJyb3Igd2lucyDigJQgcmVzb2x2ZXIgZXJyb3JzIHNob3VsZCBuZXZlciBiZSBiYXRjaGVkIGF0IHN5bnRoIHRpbWVcbiAgICAvLyBiZWNhdXNlIGEgYnJva2VuIHRlbXBsYXRlIGlzIGEgcHJvZ3JhbW1lciBlcnJvciwgbm90IGEgdmFsaWRhdGlvbiBzdGVwLlxuICAgIGNvbnN0IGUgPSBlcnJvcnNbMF0hO1xuICAgIGNvbnN0IG1zZyA9IGBUZW1wbGF0ZSByZXNvbHV0aW9uIGZhaWxlZCBpbiBwbHVnaW4gXCIke3BsdWdpbi5uYW1lfVwiIGF0IGZpZWxkICcke2UuZmllbGR9JzogJHtlLm1lc3NhZ2V9YDtcbiAgICB0aHJvdyBuZXcgRXJyb3IobXNnKTtcbiAgfVxuICByZXR1cm4gY2xvbmU7XG59XG4iXX0=
package/package.json CHANGED
@@ -25,13 +25,13 @@
25
25
  "typescript": "5.9.3"
26
26
  },
27
27
  "dependencies": {
28
- "@pipeline-builder/api-core": "3.4.8",
29
- "@pipeline-builder/pipeline-data": "3.4.8",
28
+ "@pipeline-builder/api-core": "3.4.10",
30
29
  "aws-cdk-lib": "2.251.0",
31
30
  "axios": "1.13.5",
32
31
  "constructs": "10.5.1",
33
32
  "jsonwebtoken": "9.0.3",
34
- "uuid": "13.0.0"
33
+ "uuid": "13.0.0",
34
+ "@pipeline-builder/pipeline-data": "3.4.11"
35
35
  },
36
36
  "keywords": [
37
37
  "aws",
@@ -75,7 +75,7 @@
75
75
  "access": "public",
76
76
  "registry": "https://registry.npmjs.org/"
77
77
  },
78
- "version": "3.4.9",
78
+ "version": "3.4.11",
79
79
  "bugs": {
80
80
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
81
81
  },