@pipeline-builder/pipeline-core 3.3.33 → 3.3.35

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.
@@ -118,6 +118,17 @@ export interface RegistryConfig {
118
118
  readonly http: boolean;
119
119
  /** Skip TLS certificate verification for self-signed certs (env: `DOCKER_REGISTRY_INSECURE`). Defaults to true. */
120
120
  readonly insecure: boolean;
121
+ /**
122
+ * AWS Secrets Manager secret name containing CodeBuild image-pull
123
+ * credentials in the standard `{ "username": "...", "password": "..." }`
124
+ * JSON shape. Defaults to `pipeline-builder/system/registry`.
125
+ *
126
+ * Override via `IMAGE_REGISTRY_CREDS_SECRET` if you store the creds
127
+ * under a different name. Cannot share with `pipeline-builder/system/platform`
128
+ * — that secret stores a JWT, which has a different schema; CodeBuild's
129
+ * docker-pull credentials API requires `username`/`password` keys.
130
+ */
131
+ readonly credentialsSecret: string;
121
132
  }
122
133
  export interface RedisConfig {
123
134
  readonly host: string;
@@ -2,4 +2,4 @@
2
2
  // Copyright 2026 Pipeline Builder Contributors
3
3
  // SPDX-License-Identifier: Apache-2.0
4
4
  Object.defineProperty(exports, "__esModule", { value: true });
5
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config-types.js","sourceRoot":"","sources":["../../src/config/config-types.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport type { QuotaTier } from '@pipeline-builder/api-core';\nimport type { Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport type { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport type { Architecture, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport type { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { Algorithm } from 'jsonwebtoken';\n\n/**\n * Type-safe configuration interface\n */\nexport interface AppConfig {\n  readonly server: ServerConfig;\n  readonly auth: AuthConfig;\n  readonly database: DatabaseConfig;\n  readonly registry: RegistryConfig;\n  readonly redis: RedisConfig;\n  readonly pluginBuild: PluginBuildConfig;\n  readonly dockerConfig: BuildConfig;\n  readonly observability: ObservabilityConfig;\n  readonly compliance: ComplianceConfig;\n  readonly aws: AWSConfig;\n  readonly rateLimit: RateLimitConfig;\n  readonly billing: BillingConfig;\n}\n\n/** Express server configuration. */\nexport interface ServerConfig {\n  /** HTTP listen port (env: `PORT`). */\n  readonly port: number;\n  readonly cors: {\n    /** Whether to include credentials in CORS responses (env: `CORS_CREDENTIALS`). */\n    readonly credentials: boolean;\n    /** Allowed origin(s) — single string, array, or `'*'` (env: `CORS_ORIGIN`). */\n    readonly origin: string | string[];\n  };\n  /** Number of reverse proxy hops to trust (env: `TRUST_PROXY`). */\n  readonly trustProxy: number;\n  /** Frontend base URL, used as CORS fallback (env: `PLATFORM_BASE_URL`). */\n  readonly platformUrl: string;\n  readonly httpClient: {\n    /** Default HTTP request timeout in ms (env: `HTTP_CLIENT_TIMEOUT`). */\n    readonly timeout: number;\n    /** Maximum retry attempts for failed requests (env: `HTTP_CLIENT_MAX_RETRIES`). */\n    readonly maxRetries: number;\n    /** Base delay between retries in ms (env: `HTTP_CLIENT_RETRY_DELAY_MS`). */\n    readonly retryDelayMs: number;\n  };\n  readonly sse: {\n    /** Max SSE clients per request (env: `SSE_MAX_CLIENTS_PER_REQUEST`). */\n    readonly maxClientsPerRequest: number;\n    /** SSE client timeout in ms (env: `SSE_CLIENT_TIMEOUT_MS`). */\n    readonly clientTimeoutMs: number;\n    /** SSE cleanup interval in ms (env: `SSE_CLEANUP_INTERVAL_MS`). */\n    readonly cleanupIntervalMs: number;\n  };\n  readonly services: {\n    readonly pluginHost: string;\n    readonly pluginPort: number;\n    readonly pipelineHost: string;\n    readonly pipelinePort: number;\n    readonly messageHost: string;\n    readonly messagePort: number;\n    readonly complianceHost: string;\n    readonly compliancePort: number;\n    readonly billingHost: string;\n    readonly billingPort: number;\n    readonly billingTimeout: number;\n  };\n}\n\n/** JWT and refresh token authentication configuration. */\nexport interface AuthConfig {\n  readonly jwt: {\n    /** Signing secret for access tokens (env: `JWT_SECRET`). */\n    readonly secret: string;\n    /** Token lifetime in seconds (env: `JWT_EXPIRES_IN`). */\n    readonly expiresIn: number;\n    /** Signing algorithm, e.g. `'HS256'` (env: `JWT_ALGORITHM`). */\n    readonly algorithm: Algorithm;\n    /** bcrypt salt rounds for password hashing (env: `JWT_SALT_ROUNDS`). */\n    readonly saltRounds: number;\n  };\n  readonly refreshToken: {\n    /** Signing secret for refresh tokens (env: `REFRESH_TOKEN_SECRET`). */\n    readonly secret: string;\n    /** Token lifetime in seconds (env: `REFRESH_TOKEN_EXPIRES_IN`). */\n    readonly expiresIn: number;\n  };\n}\n\n/** PostgreSQL and Drizzle ORM database configuration. */\nexport interface DatabaseConfig {\n  readonly postgres: {\n    /** PostgreSQL host (env: `DB_HOST`). */\n    readonly host: string;\n    /** PostgreSQL port (env: `DB_PORT`). */\n    readonly port: number;\n    /** Database name (env: `DATABASE`). */\n    readonly database: string;\n    /** Database user (env: `DB_USER`). */\n    readonly user: string;\n    /** Database password (env: `DB_PASSWORD`). */\n    readonly password: string;\n  };\n  readonly drizzle: {\n    /** Maximum connection pool size (env: `DRIZZLE_MAX_POOL_SIZE`). */\n    readonly maxPoolSize: number;\n    /** Idle connection timeout in ms (env: `DRIZZLE_IDLE_TIMEOUT_MILLIS`). */\n    readonly idleTimeoutMillis: number;\n    /** New connection timeout in ms (env: `DRIZZLE_CONNECTION_TIMEOUT_MILLIS`). */\n    readonly connectionTimeoutMillis: number;\n  };\n}\n\nexport interface RegistryConfig {\n  readonly host: string;\n  readonly port: number;\n  readonly user: string;\n  readonly token: string;\n  /** Docker network for build/push (empty string = default). */\n  readonly network: string;\n  /** Use plain HTTP instead of HTTPS (env: `DOCKER_REGISTRY_HTTP`). Defaults to true. */\n  readonly http: boolean;\n  /** Skip TLS certificate verification for self-signed certs (env: `DOCKER_REGISTRY_INSECURE`). Defaults to true. */\n  readonly insecure: boolean;\n}\n\nexport interface RedisConfig {\n  readonly host: string;\n  readonly port: number;\n}\n\nexport interface PluginBuildConfig {\n  readonly concurrency: number;\n  readonly maxAttempts: number;\n  readonly backoffDelayMs: number;\n  readonly workerTimeoutMs: number;\n  readonly tempDirMaxAgeMs: number;\n  readonly dlqMaxAttempts: number;\n  readonly dlqBackoffBaseMs: number;\n  readonly dlqMaxSize: number;\n}\n\nexport interface BuildConfig {\n  /** Build strategy: 'podman' (default), 'docker', or 'kaniko'. */\n  readonly strategy: 'docker' | 'kaniko' | 'podman';\n  /** Root directory for build temp files. */\n  readonly tempRoot: string;\n  /** Build timeout in milliseconds. */\n  readonly timeoutMs: number;\n  /** Push timeout in milliseconds. */\n  readonly pushTimeoutMs: number;\n  /** Path to Kaniko executor binary (only used when strategy=kaniko). */\n  readonly kanikoExecutor: string;\n  /** Kaniko layer cache directory (only used when strategy=kaniko). */\n  readonly kanikoCacheDir: string;\n}\n\nexport interface ObservabilityConfig {\n  readonly logLevel: string;\n  readonly logFormat: string;\n  readonly serviceName: string;\n  readonly tracing: {\n    readonly enabled: boolean;\n    readonly endpoint: string;\n  };\n}\n\nexport interface ComplianceConfig {\n  readonly scanSchedulerIntervalMs: number;\n  /** When false, the scheduler skips scans/schedules for the system org. Default: false. */\n  readonly systemOrgScansEnabled: boolean;\n}\n\nexport interface AWSConfig {\n  readonly lambda: {\n    readonly runtime: Runtime;\n    readonly timeout: Duration;\n    readonly memorySize: number;\n    readonly architecture: Architecture;\n    readonly reservedConcurrentExecutions?: number;\n  };\n  readonly logging: {\n    readonly groupName: string;\n    readonly retention: RetentionDays;\n    readonly removalPolicy: RemovalPolicy;\n  };\n  readonly codeBuild: {\n    readonly computeType: ComputeType;\n  };\n  /** When true, resolve synth plugin via custom resource Lambda at deploy time.\n   *  When false (default), use fallback synth commands (pipeline-manager synth). */\n  readonly resolvedSynthPlugin: boolean;\n}\n\n/** Express rate limiting configuration. */\nexport interface RateLimitConfig {\n  /** Maximum requests per window (env: `LIMITER_MAX`). */\n  readonly max: number;\n  /** Rate limit window in milliseconds (env: `LIMITER_WINDOWMS`). */\n  readonly windowMs: number;\n  /** Include legacy `X-RateLimit-*` headers. */\n  readonly legacyHeaders: boolean;\n  /** Include standard `RateLimit-*` headers (RFC 6585). */\n  readonly standardHeaders: boolean;\n}\n\n/** Price configuration for a single billing plan (in cents). */\nexport interface BillingPlanPrices {\n  readonly monthly: number;\n  readonly annual: number;\n}\n\n/** Full billing plan definition used for seeding and runtime configuration. */\nexport interface BillingPlanConfig {\n  readonly id: string;\n  readonly name: string;\n  readonly description: string;\n  readonly tier: QuotaTier;\n  readonly prices: BillingPlanPrices;\n  readonly features: readonly string[];\n  readonly isActive: boolean;\n  readonly isDefault: boolean;\n  readonly sortOrder: number;\n}\n\n/** Billing plans configuration. */\nexport interface BillingConfig {\n  readonly plans: readonly BillingPlanConfig[];\n}\n"]}
5
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config-types.js","sourceRoot":"","sources":["../../src/config/config-types.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport type { QuotaTier } from '@pipeline-builder/api-core';\nimport type { Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport type { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport type { Architecture, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport type { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { Algorithm } from 'jsonwebtoken';\n\n/**\n * Type-safe configuration interface\n */\nexport interface AppConfig {\n  readonly server: ServerConfig;\n  readonly auth: AuthConfig;\n  readonly database: DatabaseConfig;\n  readonly registry: RegistryConfig;\n  readonly redis: RedisConfig;\n  readonly pluginBuild: PluginBuildConfig;\n  readonly dockerConfig: BuildConfig;\n  readonly observability: ObservabilityConfig;\n  readonly compliance: ComplianceConfig;\n  readonly aws: AWSConfig;\n  readonly rateLimit: RateLimitConfig;\n  readonly billing: BillingConfig;\n}\n\n/** Express server configuration. */\nexport interface ServerConfig {\n  /** HTTP listen port (env: `PORT`). */\n  readonly port: number;\n  readonly cors: {\n    /** Whether to include credentials in CORS responses (env: `CORS_CREDENTIALS`). */\n    readonly credentials: boolean;\n    /** Allowed origin(s) — single string, array, or `'*'` (env: `CORS_ORIGIN`). */\n    readonly origin: string | string[];\n  };\n  /** Number of reverse proxy hops to trust (env: `TRUST_PROXY`). */\n  readonly trustProxy: number;\n  /** Frontend base URL, used as CORS fallback (env: `PLATFORM_BASE_URL`). */\n  readonly platformUrl: string;\n  readonly httpClient: {\n    /** Default HTTP request timeout in ms (env: `HTTP_CLIENT_TIMEOUT`). */\n    readonly timeout: number;\n    /** Maximum retry attempts for failed requests (env: `HTTP_CLIENT_MAX_RETRIES`). */\n    readonly maxRetries: number;\n    /** Base delay between retries in ms (env: `HTTP_CLIENT_RETRY_DELAY_MS`). */\n    readonly retryDelayMs: number;\n  };\n  readonly sse: {\n    /** Max SSE clients per request (env: `SSE_MAX_CLIENTS_PER_REQUEST`). */\n    readonly maxClientsPerRequest: number;\n    /** SSE client timeout in ms (env: `SSE_CLIENT_TIMEOUT_MS`). */\n    readonly clientTimeoutMs: number;\n    /** SSE cleanup interval in ms (env: `SSE_CLEANUP_INTERVAL_MS`). */\n    readonly cleanupIntervalMs: number;\n  };\n  readonly services: {\n    readonly pluginHost: string;\n    readonly pluginPort: number;\n    readonly pipelineHost: string;\n    readonly pipelinePort: number;\n    readonly messageHost: string;\n    readonly messagePort: number;\n    readonly complianceHost: string;\n    readonly compliancePort: number;\n    readonly billingHost: string;\n    readonly billingPort: number;\n    readonly billingTimeout: number;\n  };\n}\n\n/** JWT and refresh token authentication configuration. */\nexport interface AuthConfig {\n  readonly jwt: {\n    /** Signing secret for access tokens (env: `JWT_SECRET`). */\n    readonly secret: string;\n    /** Token lifetime in seconds (env: `JWT_EXPIRES_IN`). */\n    readonly expiresIn: number;\n    /** Signing algorithm, e.g. `'HS256'` (env: `JWT_ALGORITHM`). */\n    readonly algorithm: Algorithm;\n    /** bcrypt salt rounds for password hashing (env: `JWT_SALT_ROUNDS`). */\n    readonly saltRounds: number;\n  };\n  readonly refreshToken: {\n    /** Signing secret for refresh tokens (env: `REFRESH_TOKEN_SECRET`). */\n    readonly secret: string;\n    /** Token lifetime in seconds (env: `REFRESH_TOKEN_EXPIRES_IN`). */\n    readonly expiresIn: number;\n  };\n}\n\n/** PostgreSQL and Drizzle ORM database configuration. */\nexport interface DatabaseConfig {\n  readonly postgres: {\n    /** PostgreSQL host (env: `DB_HOST`). */\n    readonly host: string;\n    /** PostgreSQL port (env: `DB_PORT`). */\n    readonly port: number;\n    /** Database name (env: `DATABASE`). */\n    readonly database: string;\n    /** Database user (env: `DB_USER`). */\n    readonly user: string;\n    /** Database password (env: `DB_PASSWORD`). */\n    readonly password: string;\n  };\n  readonly drizzle: {\n    /** Maximum connection pool size (env: `DRIZZLE_MAX_POOL_SIZE`). */\n    readonly maxPoolSize: number;\n    /** Idle connection timeout in ms (env: `DRIZZLE_IDLE_TIMEOUT_MILLIS`). */\n    readonly idleTimeoutMillis: number;\n    /** New connection timeout in ms (env: `DRIZZLE_CONNECTION_TIMEOUT_MILLIS`). */\n    readonly connectionTimeoutMillis: number;\n  };\n}\n\nexport interface RegistryConfig {\n  readonly host: string;\n  readonly port: number;\n  readonly user: string;\n  readonly token: string;\n  /** Docker network for build/push (empty string = default). */\n  readonly network: string;\n  /** Use plain HTTP instead of HTTPS (env: `DOCKER_REGISTRY_HTTP`). Defaults to true. */\n  readonly http: boolean;\n  /** Skip TLS certificate verification for self-signed certs (env: `DOCKER_REGISTRY_INSECURE`). Defaults to true. */\n  readonly insecure: boolean;\n  /**\n   * AWS Secrets Manager secret name containing CodeBuild image-pull\n   * credentials in the standard `{ \"username\": \"...\", \"password\": \"...\" }`\n   * JSON shape. Defaults to `pipeline-builder/system/registry`.\n   *\n   * Override via `IMAGE_REGISTRY_CREDS_SECRET` if you store the creds\n   * under a different name. Cannot share with `pipeline-builder/system/platform`\n   * — that secret stores a JWT, which has a different schema; CodeBuild's\n   * docker-pull credentials API requires `username`/`password` keys.\n   */\n  readonly credentialsSecret: string;\n}\n\nexport interface RedisConfig {\n  readonly host: string;\n  readonly port: number;\n}\n\nexport interface PluginBuildConfig {\n  readonly concurrency: number;\n  readonly maxAttempts: number;\n  readonly backoffDelayMs: number;\n  readonly workerTimeoutMs: number;\n  readonly tempDirMaxAgeMs: number;\n  readonly dlqMaxAttempts: number;\n  readonly dlqBackoffBaseMs: number;\n  readonly dlqMaxSize: number;\n}\n\nexport interface BuildConfig {\n  /** Build strategy: 'podman' (default), 'docker', or 'kaniko'. */\n  readonly strategy: 'docker' | 'kaniko' | 'podman';\n  /** Root directory for build temp files. */\n  readonly tempRoot: string;\n  /** Build timeout in milliseconds. */\n  readonly timeoutMs: number;\n  /** Push timeout in milliseconds. */\n  readonly pushTimeoutMs: number;\n  /** Path to Kaniko executor binary (only used when strategy=kaniko). */\n  readonly kanikoExecutor: string;\n  /** Kaniko layer cache directory (only used when strategy=kaniko). */\n  readonly kanikoCacheDir: string;\n}\n\nexport interface ObservabilityConfig {\n  readonly logLevel: string;\n  readonly logFormat: string;\n  readonly serviceName: string;\n  readonly tracing: {\n    readonly enabled: boolean;\n    readonly endpoint: string;\n  };\n}\n\nexport interface ComplianceConfig {\n  readonly scanSchedulerIntervalMs: number;\n  /** When false, the scheduler skips scans/schedules for the system org. Default: false. */\n  readonly systemOrgScansEnabled: boolean;\n}\n\nexport interface AWSConfig {\n  readonly lambda: {\n    readonly runtime: Runtime;\n    readonly timeout: Duration;\n    readonly memorySize: number;\n    readonly architecture: Architecture;\n    readonly reservedConcurrentExecutions?: number;\n  };\n  readonly logging: {\n    readonly groupName: string;\n    readonly retention: RetentionDays;\n    readonly removalPolicy: RemovalPolicy;\n  };\n  readonly codeBuild: {\n    readonly computeType: ComputeType;\n  };\n  /** When true, resolve synth plugin via custom resource Lambda at deploy time.\n   *  When false (default), use fallback synth commands (pipeline-manager synth). */\n  readonly resolvedSynthPlugin: boolean;\n}\n\n/** Express rate limiting configuration. */\nexport interface RateLimitConfig {\n  /** Maximum requests per window (env: `LIMITER_MAX`). */\n  readonly max: number;\n  /** Rate limit window in milliseconds (env: `LIMITER_WINDOWMS`). */\n  readonly windowMs: number;\n  /** Include legacy `X-RateLimit-*` headers. */\n  readonly legacyHeaders: boolean;\n  /** Include standard `RateLimit-*` headers (RFC 6585). */\n  readonly standardHeaders: boolean;\n}\n\n/** Price configuration for a single billing plan (in cents). */\nexport interface BillingPlanPrices {\n  readonly monthly: number;\n  readonly annual: number;\n}\n\n/** Full billing plan definition used for seeding and runtime configuration. */\nexport interface BillingPlanConfig {\n  readonly id: string;\n  readonly name: string;\n  readonly description: string;\n  readonly tier: QuotaTier;\n  readonly prices: BillingPlanPrices;\n  readonly features: readonly string[];\n  readonly isActive: boolean;\n  readonly isDefault: boolean;\n  readonly sortOrder: number;\n}\n\n/** Billing plans configuration. */\nexport interface BillingConfig {\n  readonly plans: readonly BillingPlanConfig[];\n}\n"]}
@@ -50,6 +50,8 @@ function loadRegistryConfig() {
50
50
  network: process.env.DOCKER_NETWORK || '',
51
51
  http: process.env.DOCKER_REGISTRY_HTTP !== 'false',
52
52
  insecure: process.env.DOCKER_REGISTRY_INSECURE !== 'false',
53
+ credentialsSecret: process.env.IMAGE_REGISTRY_CREDS_SECRET
54
+ || 'pipeline-builder/system/registry',
53
55
  };
54
56
  }
55
57
  function loadRedisConfig() {
@@ -198,4 +200,4 @@ function parseRetention(days) {
198
200
  const parsed = parseInt(days, 10);
199
201
  return VALID_RETENTION_DAYS.has(parsed) ? parsed : aws_logs_1.RetentionDays.ONE_DAY;
200
202
  }
201
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infrastructure-config.js","sourceRoot":"","sources":["../../src/config/infrastructure-config.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;;;;AAgCtC,gDAUC;AAED,0CAKC;AAQD,sDAWC;AAED,gDAeC;AAED,0DAUC;AAED,oDAKC;AAaD,4CAWC;AAiBD,sCA4BC;AA3KD,gDAAwB;AACxB,6CAAsD;AACtD,uDAA+D;AAC/D,mDAAqD;AAErD,+DAA0D;AAE1D,SAAS,mBAAmB,CAAC,MAAc,EAAE,UAAkB;IAC7D,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAClC,IAAI,KAAK;QAAE,OAAO,KAAK,CAAC;IACxB,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,EAAE,CAAC;QAC1C,MAAM,IAAI,KAAK,CAAC,GAAG,MAAM,4BAA4B,CAAC,CAAC;IACzD,CAAC;IACD,OAAO,UAAU,CAAC;AACpB,CAAC;AAED;;;;;;;;;;;;;GAaG;AACH,SAAgB,kBAAkB;IAChC,OAAO;QACL,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,UAAU;QACnD,IAAI,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,MAAM,EAAE,EAAE,CAAC;QAC7D,IAAI,EAAE,mBAAmB,CAAC,qBAAqB,EAAE,OAAO,CAAC;QACzD,KAAK,EAAE,mBAAmB,CAAC,sBAAsB,EAAE,UAAU,CAAC;QAC9D,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,EAAE;QACzC,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,KAAK,OAAO;QAClD,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,wBAAwB,KAAK,OAAO;KAC3D,CAAC;AACJ,CAAC;AAED,SAAgB,eAAe;IAC7B,OAAO;QACL,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,WAAW;QAC3C,IAAI,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,MAAM,EAAE,EAAE,CAAC;KACrD,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACH,SAAgB,qBAAqB;IACnC,OAAO;QACL,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,wBAAwB,IAAI,GAAG,EAAE,EAAE,CAAC;QACtE,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,yBAAyB,IAAI,GAAG,EAAE,EAAE,CAAC;QACvE,cAAc,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,6BAA6B,IAAI,MAAM,EAAE,EAAE,CAAC;QACjF,eAAe,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,8BAA8B,IAAI,OAAO,EAAE,EAAE,CAAC;QACpF,eAAe,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,UAAU,EAAE,EAAE,CAAC;QAC5E,cAAc,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,GAAG,EAAE,EAAE,CAAC;QACxE,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,0BAA0B,IAAI,QAAQ,EAAE,EAAE,CAAC;QAClF,UAAU,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,IAAI,EAAE,EAAE,CAAC;KAClE,CAAC;AACJ,CAAC;AAED,SAAgB,kBAAkB;IAChC,OAAO;QACL,QAAQ,EAAE;YACR,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,UAAU;YACvC,IAAI,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,MAAM,EAAE,EAAE,CAAC;YACjD,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ,IAAI,kBAAkB;YACpD,IAAI,EAAE,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,UAAU;YACvC,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,WAAW,IAAI,EAAE;SACxC;QACD,OAAO,EAAE;YACP,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,IAAI,IAAI,EAAE,EAAE,CAAC;YACpE,iBAAiB,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,2BAA2B,IAAI,OAAO,EAAE,EAAE,CAAC;YACnF,uBAAuB,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,iCAAiC,IAAI,OAAO,EAAE,EAAE,CAAC;SAChG;KACF,CAAC;AACJ,CAAC;AAED,SAAgB,uBAAuB;IACrC,OAAO;QACL,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,MAAM;QACzC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,MAAM;QAC3C,WAAW,EAAE,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,KAAK;QAC9C,OAAO,EAAE;YACP,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,KAAK,MAAM;YACpD,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,2BAA2B,IAAI,iCAAiC;SACvF;KACF,CAAC;AACJ,CAAC;AAED,SAAgB,oBAAoB;IAClC,OAAO;QACL,uBAAuB,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,0BAA0B,IAAI,OAAO,EAAE,EAAE,CAAC;QACxF,qBAAqB,EAAE,OAAO,CAAC,GAAG,CAAC,wBAAwB,KAAK,MAAM;KACvE,CAAC;AACJ,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,gBAAgB;IAC9B,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC;IAChE,MAAM,WAAW,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,qBAAqB,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IAC5E,OAAO;QACL,QAAQ,EAAE,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,WAAsC,CAAC,CAAC,CAAC,QAAQ;QAC9F,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,cAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,KAAK,CAAC;QAC/E,SAAS,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,QAAQ,EAAE,EAAE,CAAC;QACxE,aAAa,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,QAAQ,EAAE,EAAE,CAAC;QAC3E,cAAc,EAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,kBAAkB;QACtE,cAAc,EAAE,OAAO,CAAC,GAAG,CAAC,gBAAgB,IAAI,eAAe;KAChE,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;GAcG;AACH,SAAgB,aAAa;IAC3B,OAAO;QACL,MAAM,EAAE;YACN,OAAO,EAAE,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,YAAY,CAAC;YACjE,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,KAAK,EAAE,EAAE,CAAC,CAAC;YAC5E,UAAU,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,kBAAkB,IAAI,KAAK,EAAE,EAAE,CAAC;YACjE,YAAY,EAAE,OAAO,CAAC,GAAG,CAAC,mBAAmB,KAAK,QAAQ;gBACxD,CAAC,CAAC,yBAAY,CAAC,MAAM;gBACrB,CAAC,CAAC,yBAAY,CAAC,MAAM;YACvB,4BAA4B,EAAE,OAAO,CAAC,GAAG,CAAC,2BAA2B;gBACnE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,2BAA2B,EAAE,EAAE,CAAC;gBACvD,CAAC,CAAC,SAAS;SACd;QAED,OAAO,EAAE;YACP,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,wBAAwB;YACjE,SAAS,EAAE,cAAc,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,GAAG,CAAC;YAC3D,aAAa,EAAE,OAAO,CAAC,GAAG,CAAC,kBAAkB,KAAK,QAAQ;gBACxD,CAAC,CAAC,2BAAa,CAAC,MAAM;gBACtB,CAAC,CAAC,2BAAa,CAAC,OAAO;SAC1B;QAED,SAAS,EAAE;YACT,WAAW,EAAE,IAAA,iCAAc,EAAC,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,OAAO,CAAC;SAC3E;QAED,mBAAmB,EAAE,OAAO,CAAC,GAAG,CAAC,qBAAqB,KAAK,MAAM;KAClE,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACH,SAAS,YAAY,CAAC,OAAe;IACnC,MAAM,UAAU,GAA4B;QAC1C,YAAY,EAAE,oBAAO,CAAC,WAAW;KAClC,CAAC;IACF,OAAO,UAAU,CAAC,OAAO,CAAC,IAAI,oBAAO,CAAC,WAAW,CAAC;AACpD,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,wBAAa,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAe,EAAE,CAAC,OAAO,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC;AAErH,SAAS,cAAc,CAAC,IAAY;IAClC,MAAM,MAAM,GAAG,QAAQ,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IAClC,OAAO,oBAAoB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAuB,CAAC,CAAC,CAAC,wBAAa,CAAC,OAAO,CAAC;AAC5F,CAAC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport path from 'path';\nimport { Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Architecture, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { AWSConfig, BuildConfig, ComplianceConfig, DatabaseConfig, ObservabilityConfig, PluginBuildConfig, RedisConfig, RegistryConfig } from './config-types';\nimport { getComputeType } from '../core/pipeline-helpers';\n\nfunction requireInProduction(envVar: string, devDefault: string): string {\n  const value = process.env[envVar];\n  if (value) return value;\n  if (process.env.NODE_ENV === 'production') {\n    throw new Error(`${envVar} is required in production`);\n  }\n  return devDefault;\n}\n\n/**\n * Load Docker registry configuration from environment variables.\n *\n * Environment variables:\n * - `IMAGE_REGISTRY_HOST` — Registry hostname (default: `'registry'`)\n * - `IMAGE_REGISTRY_PORT` — Registry port (default: `5000`)\n * - `IMAGE_REGISTRY_USER` — Registry username (default: `'admin'`)\n * - `IMAGE_REGISTRY_TOKEN` — Registry auth token (default: `'password'`)\n * - `DOCKER_NETWORK` — Docker network for build/push (default: `''`)\n * - `DOCKER_REGISTRY_HTTP` — Use plain HTTP (default: `true`). Set `false` for HTTPS.\n * - `DOCKER_REGISTRY_INSECURE` — Skip TLS verification (default: `true`). Set `false` for production.\n *\n * @returns Registry configuration\n */\nexport function loadRegistryConfig(): RegistryConfig {\n  return {\n    host: process.env.IMAGE_REGISTRY_HOST || 'registry',\n    port: parseInt(process.env.IMAGE_REGISTRY_PORT || '5000', 10),\n    user: requireInProduction('IMAGE_REGISTRY_USER', 'admin'),\n    token: requireInProduction('IMAGE_REGISTRY_TOKEN', 'password'),\n    network: process.env.DOCKER_NETWORK || '',\n    http: process.env.DOCKER_REGISTRY_HTTP !== 'false',\n    insecure: process.env.DOCKER_REGISTRY_INSECURE !== 'false',\n  };\n}\n\nexport function loadRedisConfig(): RedisConfig {\n  return {\n    host: process.env.REDIS_HOST || 'localhost',\n    port: parseInt(process.env.REDIS_PORT || '6379', 10),\n  };\n}\n\n/**\n * Load plugin build queue configuration.\n *\n * Environment variables:\n * - `PLUGIN_BUILD_CONCURRENCY` — Max concurrent plugin builds (default: `1`)\n */\nexport function loadPluginBuildConfig(): PluginBuildConfig {\n  return {\n    concurrency: parseInt(process.env.PLUGIN_BUILD_CONCURRENCY || '1', 10),\n    maxAttempts: parseInt(process.env.PLUGIN_BUILD_MAX_ATTEMPTS || '2', 10),\n    backoffDelayMs: parseInt(process.env.PLUGIN_BUILD_BACKOFF_DELAY_MS || '5000', 10),\n    workerTimeoutMs: parseInt(process.env.PLUGIN_BUILD_WORKER_TIMEOUT_MS || '10000', 10),\n    tempDirMaxAgeMs: parseInt(process.env.TEMP_DIR_MAX_AGE_MS || '14400000', 10),\n    dlqMaxAttempts: parseInt(process.env.PLUGIN_DLQ_MAX_ATTEMPTS || '3', 10),\n    dlqBackoffBaseMs: parseInt(process.env.PLUGIN_DLQ_BACKOFF_BASE_MS || '300000', 10),\n    dlqMaxSize: parseInt(process.env.PLUGIN_DLQ_MAX_SIZE || '20', 10),\n  };\n}\n\nexport function loadDatabaseConfig(): DatabaseConfig {\n  return {\n    postgres: {\n      host: process.env.DB_HOST || 'postgres',\n      port: parseInt(process.env.DB_PORT || '5432', 10),\n      database: process.env.DATABASE || 'pipeline_builder',\n      user: process.env.DB_USER || 'postgres',\n      password: process.env.DB_PASSWORD || '',\n    },\n    drizzle: {\n      maxPoolSize: parseInt(process.env.DRIZZLE_MAX_POOL_SIZE || '20', 10),\n      idleTimeoutMillis: parseInt(process.env.DRIZZLE_IDLE_TIMEOUT_MILLIS || '30000', 10),\n      connectionTimeoutMillis: parseInt(process.env.DRIZZLE_CONNECTION_TIMEOUT_MILLIS || '10000', 10),\n    },\n  };\n}\n\nexport function loadObservabilityConfig(): ObservabilityConfig {\n  return {\n    logLevel: process.env.LOG_LEVEL || 'info',\n    logFormat: process.env.LOG_FORMAT || 'json',\n    serviceName: process.env.SERVICE_NAME || 'api',\n    tracing: {\n      enabled: process.env.OTEL_TRACING_ENABLED === 'true',\n      endpoint: process.env.OTEL_EXPORTER_OTLP_ENDPOINT || 'http://localhost:4318/v1/traces',\n    },\n  };\n}\n\nexport function loadComplianceConfig(): ComplianceConfig {\n  return {\n    scanSchedulerIntervalMs: parseInt(process.env.SCAN_SCHEDULER_INTERVAL_MS || '60000', 10),\n    systemOrgScansEnabled: process.env.SYSTEM_ORG_SCANS_ENABLED === 'true',\n  };\n}\n\n/**\n * Load Docker/Podman/Kaniko build configuration.\n *\n * Environment variables:\n * - `DOCKER_BUILD_STRATEGY` — Build strategy: `podman`, `docker`, or `kaniko` (default: `podman`)\n * - `DOCKER_BUILD_TEMP_ROOT` — Temp directory for build contexts (default: `<cwd>/tmp`)\n * - `DOCKER_BUILD_TIMEOUT_MS` — Build timeout in milliseconds (default: `900000` / 15 min)\n * - `DOCKER_PUSH_TIMEOUT_MS` — Push timeout in milliseconds (default: `300000` / 5 min)\n * - `KANIKO_EXECUTOR_PATH` — Path to Kaniko executor binary (default: `/kaniko/executor`)\n * - `KANIKO_CACHE_DIR` — Kaniko layer cache directory (default: `/kaniko/cache`)\n */\nexport function loadDockerConfig(): BuildConfig {\n  const validStrategies = new Set(['docker', 'kaniko', 'podman']);\n  const strategyEnv = (process.env.DOCKER_BUILD_STRATEGY || '').toLowerCase();\n  return {\n    strategy: validStrategies.has(strategyEnv) ? strategyEnv as BuildConfig['strategy'] : 'docker',\n    tempRoot: process.env.DOCKER_BUILD_TEMP_ROOT || path.join(process.cwd(), 'tmp'),\n    timeoutMs: parseInt(process.env.DOCKER_BUILD_TIMEOUT_MS || '900000', 10),\n    pushTimeoutMs: parseInt(process.env.DOCKER_PUSH_TIMEOUT_MS || '300000', 10),\n    kanikoExecutor: process.env.KANIKO_EXECUTOR_PATH || '/kaniko/executor',\n    kanikoCacheDir: process.env.KANIKO_CACHE_DIR || '/kaniko/cache',\n  };\n}\n\n/**\n * Load AWS infrastructure configuration from environment variables.\n *\n * Environment variables:\n * - `LAMBDA_RUNTIME` — Lambda runtime (default: `'nodejs24.x'`; supports nodejs22.x, nodejs24.x)\n * - `LAMBDA_TIMEOUT` — Lambda timeout in seconds (default: `900`)\n * - `LAMBDA_MEMORY_SIZE` — Lambda memory in MB (default: `128`)\n * - `LAMBDA_ARCHITECTURE` — `'x86_64'` or ARM (default: ARM_64)\n * - `LOG_GROUP_NAME` — CloudWatch log group (default: `'/pipeline-builder/logs'`)\n * - `LOG_RETENTION` — Log retention in days (default: `7`)\n * - `LOG_REMOVAL_POLICY` — `'RETAIN'` or destroy (default: DESTROY)\n * - `CODEBUILD_COMPUTE_TYPE` — CodeBuild compute type (default: `'SMALL'`)\n *\n * @returns AWS infrastructure configuration\n */\nexport function loadAWSConfig(): AWSConfig {\n  return {\n    lambda: {\n      runtime: parseRuntime(process.env.LAMBDA_RUNTIME || 'nodejs24.x'),\n      timeout: Duration.seconds(parseInt(process.env.LAMBDA_TIMEOUT || '900', 10)),\n      memorySize: parseInt(process.env.LAMBDA_MEMORY_SIZE || '512', 10),\n      architecture: process.env.LAMBDA_ARCHITECTURE === 'x86_64'\n        ? Architecture.X86_64\n        : Architecture.ARM_64,\n      reservedConcurrentExecutions: process.env.LAMBDA_RESERVED_CONCURRENCY\n        ? parseInt(process.env.LAMBDA_RESERVED_CONCURRENCY, 10)\n        : undefined,\n    },\n\n    logging: {\n      groupName: process.env.LOG_GROUP_NAME || '/pipeline-builder/logs',\n      retention: parseRetention(process.env.LOG_RETENTION || '7'),\n      removalPolicy: process.env.LOG_REMOVAL_POLICY === 'RETAIN'\n        ? RemovalPolicy.RETAIN\n        : RemovalPolicy.DESTROY,\n    },\n\n    codeBuild: {\n      computeType: getComputeType(process.env.CODEBUILD_COMPUTE_TYPE || 'SMALL'),\n    },\n\n    resolvedSynthPlugin: process.env.RESOLVED_SYNTH_PLUGIN === 'true',\n  };\n}\n\n/**\n * Parse Lambda runtime string into a CDK Runtime enum value.\n *\n * @param runtime - Runtime string (e.g. `'nodejs24.x'`)\n * @returns CDK Runtime enum; falls back to NODEJS_24_X for unknown values\n */\nfunction parseRuntime(runtime: string): Runtime {\n  const runtimeMap: Record<string, Runtime> = {\n    'nodejs24.x': Runtime.NODEJS_24_X,\n  };\n  return runtimeMap[runtime] || Runtime.NODEJS_24_X;\n}\n\n/**\n * Parse log retention days string into a CDK RetentionDays enum value.\n * RetentionDays enum values are the numeric day counts themselves,\n * so we parse the string and check if it's a valid enum member.\n *\n * @param days - Retention period in days as a string (e.g. `'30'`)\n * @returns CDK RetentionDays enum; falls back to ONE_DAY for unknown values\n */\nconst VALID_RETENTION_DAYS = new Set(Object.values(RetentionDays).filter((v): v is number => typeof v === 'number'));\n\nfunction parseRetention(days: string): RetentionDays {\n  const parsed = parseInt(days, 10);\n  return VALID_RETENTION_DAYS.has(parsed) ? parsed as RetentionDays : RetentionDays.ONE_DAY;\n}\n"]}
203
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infrastructure-config.js","sourceRoot":"","sources":["../../src/config/infrastructure-config.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport path from 'path';\nimport { Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { Architecture, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport type { AWSConfig, BuildConfig, ComplianceConfig, DatabaseConfig, ObservabilityConfig, PluginBuildConfig, RedisConfig, RegistryConfig } from './config-types';\nimport { getComputeType } from '../core/pipeline-helpers';\n\nfunction requireInProduction(envVar: string, devDefault: string): string {\n  const value = process.env[envVar];\n  if (value) return value;\n  if (process.env.NODE_ENV === 'production') {\n    throw new Error(`${envVar} is required in production`);\n  }\n  return devDefault;\n}\n\n/**\n * Load Docker registry configuration from environment variables.\n *\n * Environment variables:\n * - `IMAGE_REGISTRY_HOST` — Registry hostname (default: `'registry'`)\n * - `IMAGE_REGISTRY_PORT` — Registry port (default: `5000`)\n * - `IMAGE_REGISTRY_USER` — Registry username (default: `'admin'`)\n * - `IMAGE_REGISTRY_TOKEN` — Registry auth token (default: `'password'`)\n * - `DOCKER_NETWORK` — Docker network for build/push (default: `''`)\n * - `DOCKER_REGISTRY_HTTP` — Use plain HTTP (default: `true`). Set `false` for HTTPS.\n * - `DOCKER_REGISTRY_INSECURE` — Skip TLS verification (default: `true`). Set `false` for production.\n *\n * @returns Registry configuration\n */\nexport function loadRegistryConfig(): RegistryConfig {\n  return {\n    host: process.env.IMAGE_REGISTRY_HOST || 'registry',\n    port: parseInt(process.env.IMAGE_REGISTRY_PORT || '5000', 10),\n    user: requireInProduction('IMAGE_REGISTRY_USER', 'admin'),\n    token: requireInProduction('IMAGE_REGISTRY_TOKEN', 'password'),\n    network: process.env.DOCKER_NETWORK || '',\n    http: process.env.DOCKER_REGISTRY_HTTP !== 'false',\n    insecure: process.env.DOCKER_REGISTRY_INSECURE !== 'false',\n    credentialsSecret: process.env.IMAGE_REGISTRY_CREDS_SECRET\n      || 'pipeline-builder/system/registry',\n  };\n}\n\nexport function loadRedisConfig(): RedisConfig {\n  return {\n    host: process.env.REDIS_HOST || 'localhost',\n    port: parseInt(process.env.REDIS_PORT || '6379', 10),\n  };\n}\n\n/**\n * Load plugin build queue configuration.\n *\n * Environment variables:\n * - `PLUGIN_BUILD_CONCURRENCY` — Max concurrent plugin builds (default: `1`)\n */\nexport function loadPluginBuildConfig(): PluginBuildConfig {\n  return {\n    concurrency: parseInt(process.env.PLUGIN_BUILD_CONCURRENCY || '1', 10),\n    maxAttempts: parseInt(process.env.PLUGIN_BUILD_MAX_ATTEMPTS || '2', 10),\n    backoffDelayMs: parseInt(process.env.PLUGIN_BUILD_BACKOFF_DELAY_MS || '5000', 10),\n    workerTimeoutMs: parseInt(process.env.PLUGIN_BUILD_WORKER_TIMEOUT_MS || '10000', 10),\n    tempDirMaxAgeMs: parseInt(process.env.TEMP_DIR_MAX_AGE_MS || '14400000', 10),\n    dlqMaxAttempts: parseInt(process.env.PLUGIN_DLQ_MAX_ATTEMPTS || '3', 10),\n    dlqBackoffBaseMs: parseInt(process.env.PLUGIN_DLQ_BACKOFF_BASE_MS || '300000', 10),\n    dlqMaxSize: parseInt(process.env.PLUGIN_DLQ_MAX_SIZE || '20', 10),\n  };\n}\n\nexport function loadDatabaseConfig(): DatabaseConfig {\n  return {\n    postgres: {\n      host: process.env.DB_HOST || 'postgres',\n      port: parseInt(process.env.DB_PORT || '5432', 10),\n      database: process.env.DATABASE || 'pipeline_builder',\n      user: process.env.DB_USER || 'postgres',\n      password: process.env.DB_PASSWORD || '',\n    },\n    drizzle: {\n      maxPoolSize: parseInt(process.env.DRIZZLE_MAX_POOL_SIZE || '20', 10),\n      idleTimeoutMillis: parseInt(process.env.DRIZZLE_IDLE_TIMEOUT_MILLIS || '30000', 10),\n      connectionTimeoutMillis: parseInt(process.env.DRIZZLE_CONNECTION_TIMEOUT_MILLIS || '10000', 10),\n    },\n  };\n}\n\nexport function loadObservabilityConfig(): ObservabilityConfig {\n  return {\n    logLevel: process.env.LOG_LEVEL || 'info',\n    logFormat: process.env.LOG_FORMAT || 'json',\n    serviceName: process.env.SERVICE_NAME || 'api',\n    tracing: {\n      enabled: process.env.OTEL_TRACING_ENABLED === 'true',\n      endpoint: process.env.OTEL_EXPORTER_OTLP_ENDPOINT || 'http://localhost:4318/v1/traces',\n    },\n  };\n}\n\nexport function loadComplianceConfig(): ComplianceConfig {\n  return {\n    scanSchedulerIntervalMs: parseInt(process.env.SCAN_SCHEDULER_INTERVAL_MS || '60000', 10),\n    systemOrgScansEnabled: process.env.SYSTEM_ORG_SCANS_ENABLED === 'true',\n  };\n}\n\n/**\n * Load Docker/Podman/Kaniko build configuration.\n *\n * Environment variables:\n * - `DOCKER_BUILD_STRATEGY` — Build strategy: `podman`, `docker`, or `kaniko` (default: `podman`)\n * - `DOCKER_BUILD_TEMP_ROOT` — Temp directory for build contexts (default: `<cwd>/tmp`)\n * - `DOCKER_BUILD_TIMEOUT_MS` — Build timeout in milliseconds (default: `900000` / 15 min)\n * - `DOCKER_PUSH_TIMEOUT_MS` — Push timeout in milliseconds (default: `300000` / 5 min)\n * - `KANIKO_EXECUTOR_PATH` — Path to Kaniko executor binary (default: `/kaniko/executor`)\n * - `KANIKO_CACHE_DIR` — Kaniko layer cache directory (default: `/kaniko/cache`)\n */\nexport function loadDockerConfig(): BuildConfig {\n  const validStrategies = new Set(['docker', 'kaniko', 'podman']);\n  const strategyEnv = (process.env.DOCKER_BUILD_STRATEGY || '').toLowerCase();\n  return {\n    strategy: validStrategies.has(strategyEnv) ? strategyEnv as BuildConfig['strategy'] : 'docker',\n    tempRoot: process.env.DOCKER_BUILD_TEMP_ROOT || path.join(process.cwd(), 'tmp'),\n    timeoutMs: parseInt(process.env.DOCKER_BUILD_TIMEOUT_MS || '900000', 10),\n    pushTimeoutMs: parseInt(process.env.DOCKER_PUSH_TIMEOUT_MS || '300000', 10),\n    kanikoExecutor: process.env.KANIKO_EXECUTOR_PATH || '/kaniko/executor',\n    kanikoCacheDir: process.env.KANIKO_CACHE_DIR || '/kaniko/cache',\n  };\n}\n\n/**\n * Load AWS infrastructure configuration from environment variables.\n *\n * Environment variables:\n * - `LAMBDA_RUNTIME` — Lambda runtime (default: `'nodejs24.x'`; supports nodejs22.x, nodejs24.x)\n * - `LAMBDA_TIMEOUT` — Lambda timeout in seconds (default: `900`)\n * - `LAMBDA_MEMORY_SIZE` — Lambda memory in MB (default: `128`)\n * - `LAMBDA_ARCHITECTURE` — `'x86_64'` or ARM (default: ARM_64)\n * - `LOG_GROUP_NAME` — CloudWatch log group (default: `'/pipeline-builder/logs'`)\n * - `LOG_RETENTION` — Log retention in days (default: `7`)\n * - `LOG_REMOVAL_POLICY` — `'RETAIN'` or destroy (default: DESTROY)\n * - `CODEBUILD_COMPUTE_TYPE` — CodeBuild compute type (default: `'SMALL'`)\n *\n * @returns AWS infrastructure configuration\n */\nexport function loadAWSConfig(): AWSConfig {\n  return {\n    lambda: {\n      runtime: parseRuntime(process.env.LAMBDA_RUNTIME || 'nodejs24.x'),\n      timeout: Duration.seconds(parseInt(process.env.LAMBDA_TIMEOUT || '900', 10)),\n      memorySize: parseInt(process.env.LAMBDA_MEMORY_SIZE || '512', 10),\n      architecture: process.env.LAMBDA_ARCHITECTURE === 'x86_64'\n        ? Architecture.X86_64\n        : Architecture.ARM_64,\n      reservedConcurrentExecutions: process.env.LAMBDA_RESERVED_CONCURRENCY\n        ? parseInt(process.env.LAMBDA_RESERVED_CONCURRENCY, 10)\n        : undefined,\n    },\n\n    logging: {\n      groupName: process.env.LOG_GROUP_NAME || '/pipeline-builder/logs',\n      retention: parseRetention(process.env.LOG_RETENTION || '7'),\n      removalPolicy: process.env.LOG_REMOVAL_POLICY === 'RETAIN'\n        ? RemovalPolicy.RETAIN\n        : RemovalPolicy.DESTROY,\n    },\n\n    codeBuild: {\n      computeType: getComputeType(process.env.CODEBUILD_COMPUTE_TYPE || 'SMALL'),\n    },\n\n    resolvedSynthPlugin: process.env.RESOLVED_SYNTH_PLUGIN === 'true',\n  };\n}\n\n/**\n * Parse Lambda runtime string into a CDK Runtime enum value.\n *\n * @param runtime - Runtime string (e.g. `'nodejs24.x'`)\n * @returns CDK Runtime enum; falls back to NODEJS_24_X for unknown values\n */\nfunction parseRuntime(runtime: string): Runtime {\n  const runtimeMap: Record<string, Runtime> = {\n    'nodejs24.x': Runtime.NODEJS_24_X,\n  };\n  return runtimeMap[runtime] || Runtime.NODEJS_24_X;\n}\n\n/**\n * Parse log retention days string into a CDK RetentionDays enum value.\n * RetentionDays enum values are the numeric day counts themselves,\n * so we parse the string and check if it's a valid enum member.\n *\n * @param days - Retention period in days as a string (e.g. `'30'`)\n * @returns CDK RetentionDays enum; falls back to ONE_DAY for unknown values\n */\nconst VALID_RETENTION_DAYS = new Set(Object.values(RetentionDays).filter((v): v is number => typeof v === 'number'));\n\nfunction parseRetention(days: string): RetentionDays {\n  const parsed = parseInt(days, 10);\n  return VALID_RETENTION_DAYS.has(parsed) ? parsed as RetentionDays : RetentionDays.ONE_DAY;\n}\n"]}
@@ -1,6 +1,8 @@
1
+ import type { Plugin } from '@pipeline-builder/pipeline-data';
1
2
  import { SecretValue } from 'aws-cdk-lib';
2
- import { ComputeType as CDKComputeType } from 'aws-cdk-lib/aws-codebuild';
3
+ import { ComputeType as CDKComputeType, IBuildImage } from 'aws-cdk-lib/aws-codebuild';
3
4
  import { CodeBuildStep, ManualApprovalStep, ShellStep } from 'aws-cdk-lib/pipelines';
5
+ import type { Construct } from 'constructs';
4
6
  import { MetaDataType } from './pipeline-types';
5
7
  import type { CodeBuildStepOptions } from '../pipeline/step-types';
6
8
  /**
@@ -33,6 +35,27 @@ export declare function extractMetadataEnv(metadata: MetaDataType): Record<strin
33
35
  * This means metadata keys like `aws:cdk:pipelines:codebuildstep:commands`
34
36
  * will override the plugin-derived commands when explicitly set.
35
37
  */
38
+ /**
39
+ * Resolve the CodeBuild image to use for a plugin.
40
+ *
41
+ * Strategy:
42
+ * 1. If the plugin sets `aws:cdk:codebuild:buildenvironment:buildImage`
43
+ * explicitly in metadata, the metadata-builder passthrough handles it
44
+ * (via `metadataForBuildEnvironment`). This function returns
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>/<imageTag>:latest`. CodeBuild
49
+ * pulls from our private registry — credentials come from the
50
+ * `pipeline-builder/system/registry` Secrets Manager secret.
51
+ * 3. If neither (e.g., a `metadata_only` plugin or one with no image),
52
+ * return `undefined` so CodeBuild uses its default (`standard:7.0`).
53
+ *
54
+ * `scope` is required because `Secret.fromSecretNameV2()` needs a parent
55
+ * Construct to anchor the imported secret to. Pass the stack/construct
56
+ * that owns the CodeBuild step.
57
+ */
58
+ export declare function resolvePluginImage(scope: Construct | undefined, plugin: Plugin): IBuildImage | undefined;
36
59
  export declare function createCodeBuildStep(options: CodeBuildStepOptions): ShellStep | CodeBuildStep | ManualApprovalStep;
37
60
  /**
38
61
  * Convert string or ComputeType enum to CDK ComputeType
@@ -4,6 +4,7 @@
4
4
  Object.defineProperty(exports, "__esModule", { value: true });
5
5
  exports.merge = merge;
6
6
  exports.extractMetadataEnv = extractMetadataEnv;
7
+ exports.resolvePluginImage = resolvePluginImage;
7
8
  exports.createCodeBuildStep = createCodeBuildStep;
8
9
  exports.getComputeType = getComputeType;
9
10
  exports.replaceNonAlphanumeric = replaceNonAlphanumeric;
@@ -11,6 +12,7 @@ exports.unwrapSecret = unwrapSecret;
11
12
  const api_core_1 = require("@pipeline-builder/api-core");
12
13
  const aws_cdk_lib_1 = require("aws-cdk-lib");
13
14
  const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild");
15
+ const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
14
16
  const pipelines_1 = require("aws-cdk-lib/pipelines");
15
17
  const metadata_builder_1 = require("./metadata-builder");
16
18
  const network_1 = require("./network");
@@ -146,6 +148,90 @@ function toSecretEnvVars(secrets, orgId) {
146
148
  * This means metadata keys like `aws:cdk:pipelines:codebuildstep:commands`
147
149
  * will override the plugin-derived commands when explicitly set.
148
150
  */
151
+ /**
152
+ * Resolve the CodeBuild image to use for a plugin.
153
+ *
154
+ * Strategy:
155
+ * 1. If the plugin sets `aws:cdk:codebuild:buildenvironment:buildImage`
156
+ * explicitly in metadata, the metadata-builder passthrough handles it
157
+ * (via `metadataForBuildEnvironment`). This function returns
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>/<imageTag>:latest`. CodeBuild
162
+ * pulls from our private registry — credentials come from the
163
+ * `pipeline-builder/system/registry` Secrets Manager secret.
164
+ * 3. If neither (e.g., a `metadata_only` plugin or one with no image),
165
+ * return `undefined` so CodeBuild uses its default (`standard:7.0`).
166
+ *
167
+ * `scope` is required because `Secret.fromSecretNameV2()` needs a parent
168
+ * Construct to anchor the imported secret to. Pass the stack/construct
169
+ * that owns the CodeBuild step.
170
+ */
171
+ function resolvePluginImage(scope, plugin) {
172
+ const imageTag = plugin.imageTag;
173
+ // `metadata_only` plugins legitimately have no image — their work runs
174
+ // in the default CodeBuild image. Quiet skip.
175
+ if (plugin.buildType === 'metadata_only')
176
+ return undefined;
177
+ // No image tag means there's nothing to pull. This SHOULDN'T happen for
178
+ // build_image / prebuilt plugins (the platform writes imageTag at upload
179
+ // time); if it does, the plugin record is malformed.
180
+ if (!imageTag || imageTag === '') {
181
+ log.warn(`Plugin "${plugin.name}" has buildType=${plugin.buildType} but no imageTag — ` +
182
+ `CodeBuild will run on aws/codebuild/standard:7.0 and won't have the plugin's baked tools. ` +
183
+ `Verify the plugin was uploaded correctly via load-plugins.sh.`);
184
+ return undefined;
185
+ }
186
+ let registry;
187
+ try {
188
+ registry = app_config_1.Config.get('registry');
189
+ }
190
+ catch {
191
+ // Config namespace not loaded (e.g., unit tests without full config).
192
+ log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but registry config not loaded — ` +
193
+ `CodeBuild will fall back to aws/codebuild/standard:7.0. ` +
194
+ `Set IMAGE_REGISTRY_HOST + IMAGE_REGISTRY_PORT in pipeline-manager's environment.`);
195
+ return undefined;
196
+ }
197
+ if (!registry?.host) {
198
+ log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but IMAGE_REGISTRY_HOST is empty — ` +
199
+ `CodeBuild will fall back to aws/codebuild/standard:7.0. ` +
200
+ `Set IMAGE_REGISTRY_HOST in pipeline-manager's environment to use the plugin image.`);
201
+ return undefined;
202
+ }
203
+ if (!scope) {
204
+ log.warn(`Plugin "${plugin.name}" image resolution skipped: no construct scope provided`);
205
+ return undefined;
206
+ }
207
+ // Compose the image URI: `<host>:<port>/<imageTag>:latest`.
208
+ // The trailing `:latest` is required — Docker assumes `latest` only when
209
+ // pulling without a tag, and we want explicit pull semantics.
210
+ const portPart = registry.port && registry.port !== 80 && registry.port !== 443
211
+ ? `:${registry.port}`
212
+ : '';
213
+ const imageUri = `${registry.host}${portPart}/${imageTag}:latest`;
214
+ // Credentials live in a Secrets Manager secret as
215
+ // `{ "username": "...", "password": "..." }` JSON. The secret name is
216
+ // configurable via `IMAGE_REGISTRY_CREDS_SECRET` env var (default:
217
+ // `pipeline-builder/system/registry`). The CodeBuild role needs
218
+ // `secretsmanager:GetSecretValue` for that ARN — CDK adds this
219
+ // automatically when the secret reference flows through
220
+ // `secretsManagerCredentials`.
221
+ //
222
+ // Use a deterministic construct ID so multiple steps that share the same
223
+ // registry secret reuse the same imported construct (avoids "construct
224
+ // already exists" errors when many steps resolve the same image).
225
+ const stack = aws_cdk_lib_1.Stack.of(scope);
226
+ const secretName = registry.credentialsSecret || 'pipeline-builder/system/registry';
227
+ // CDK construct IDs allow only [A-Za-z0-9_-]; sanitize the secret name.
228
+ const secretConstructId = `RegistryCreds_${secretName.replace(/[^A-Za-z0-9_-]/g, '_')}`;
229
+ const credentialsSecret = stack.node.tryFindChild(secretConstructId)
230
+ ?? aws_secretsmanager_1.Secret.fromSecretNameV2(stack, secretConstructId, secretName);
231
+ return aws_codebuild_1.LinuxBuildImage.fromDockerRegistry(imageUri, {
232
+ secretsManagerCredentials: credentialsSecret,
233
+ });
234
+ }
149
235
  function createCodeBuildStep(options) {
150
236
  const { id, input, metadata, network, scope, preInstallCommands, postInstallCommands, preCommands, postCommands, env: customEnv, additionalInputs, timeout, failureBehavior, artifactManager, stageName, stageAlias, pluginAlias, orgId, pipelineScope, } = options;
151
237
  // Resolve {{ ... }} templates in plugin spec fields against the pipeline scope.
@@ -187,13 +273,36 @@ function createCodeBuildStep(options) {
187
273
  postCommands,
188
274
  }, failureBehavior);
189
275
  const programmatic = { input, installCommands, commands };
190
- // Return ShellStep if plugin type is SHELL_STEP
276
+ // Resolve the plugin's runtime image. CodeBuild defaults to
277
+ // `aws/codebuild/standard:7.0` when no `buildImage` is set — that's the
278
+ // AWS-managed Ubuntu image and DOES NOT have any plugin-baked tools
279
+ // (pipeline-manager, snyk, terraform, etc.). Wiring the plugin's image
280
+ // here means CodeBuild pulls from our private registry and the tools
281
+ // installed in the plugin's Dockerfile are actually available.
282
+ const pluginBuildImage = resolvePluginImage(scope, plugin);
283
+ // ShellStep branch.
284
+ // ShellStep itself doesn't accept `buildEnvironment`/`buildImage`. CDK
285
+ // pipelines wraps ShellSteps in a default CodeBuild action that runs on
286
+ // `aws/codebuild/standard:7.0` — which won't have the plugin's tools.
287
+ //
288
+ // So when a SHELL_STEP plugin DOES have an image (because its author
289
+ // baked tools into a Dockerfile), we PROMOTE it to a CodeBuildStep with
290
+ // the resolved image. The plugin author's intent ("use my baked tools")
291
+ // is preserved without forcing them to change pluginType.
292
+ //
293
+ // When the plugin has no image (or registry isn't configured), we keep
294
+ // the original ShellStep — it's lighter weight and the default
295
+ // CodeBuild image is fine.
191
296
  if (plugin.pluginType === pipeline_types_1.PluginType.SHELL_STEP) {
192
- return new pipelines_1.ShellStep(id, {
193
- ...programmatic,
194
- env: { ...env, ...actionEnv },
195
- ...(0, metadata_builder_1.metadataForShellStep)(merged),
196
- });
297
+ if (!pluginBuildImage) {
298
+ return new pipelines_1.ShellStep(id, {
299
+ ...programmatic,
300
+ env: { ...env, ...actionEnv },
301
+ ...(0, metadata_builder_1.metadataForShellStep)(merged),
302
+ });
303
+ }
304
+ log.debug(`[CreateCodeBuildStep] SHELL_STEP plugin "${plugin.name}" has imageTag — promoting to CodeBuildStep so the plugin image is actually used`);
305
+ // Fall through to the CodeBuildStep path below.
197
306
  }
198
307
  const computeType = getComputeType(plugin.computeType ?? options.defaultComputeType ?? 'SMALL');
199
308
  const networkProps = network
@@ -215,6 +324,7 @@ function createCodeBuildStep(options) {
215
324
  primaryOutputDirectory: plugin.primaryOutputDirectory ?? undefined,
216
325
  buildEnvironment: {
217
326
  computeType,
327
+ ...(pluginBuildImage && { buildImage: pluginBuildImage }),
218
328
  environmentVariables: {
219
329
  ...toCodeBuildEnvVars(env),
220
330
  ...secretEnvVars,
@@ -275,4 +385,4 @@ function replaceNonAlphanumeric(input, replaceValue = '_') {
275
385
  function unwrapSecret(value) {
276
386
  return typeof value === 'string' ? value : value.unsafeUnwrap();
277
387
  }
278
- //# 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 } from 'aws-cdk-lib';\nimport { BuildEnvironmentVariableType, ComputeType as CDKComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { CodeBuildStep, ManualApprovalStep, ShellStep } from 'aws-cdk-lib/pipelines';\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 { 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 */\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  // Return ShellStep if plugin type is SHELL_STEP\n  if (plugin.pluginType === PluginType.SHELL_STEP) {\n    return new ShellStep(id, {\n      ...programmatic,\n      env: { ...env, ...actionEnv },\n      ...metadataForShellStep(merged),\n    });\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      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"]}
388
+ //# 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>/<imageTag>:latest`. CodeBuild\n *      pulls from our private registry — credentials come from the\n *      `pipeline-builder/system/registry` Secrets Manager secret.\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` is required because `Secret.fromSecretNameV2()` needs a parent\n * Construct to anchor the imported secret to. Pass the stack/construct\n * that owns the CodeBuild step.\n */\nexport function resolvePluginImage(scope: Construct | undefined, plugin: Plugin): 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  // Compose the image URI: `<host>:<port>/<imageTag>:latest`.\n  // The trailing `:latest` is required — Docker assumes `latest` only when\n  // pulling without a tag, and we want explicit pull semantics.\n  const portPart = registry.port && registry.port !== 80 && registry.port !== 443\n    ? `:${registry.port}`\n    : '';\n  const imageUri = `${registry.host}${portPart}/${imageTag}:latest`;\n\n  // Credentials live in a Secrets Manager secret as\n  // `{ \"username\": \"...\", \"password\": \"...\" }` JSON. The secret name is\n  // configurable via `IMAGE_REGISTRY_CREDS_SECRET` env var (default:\n  // `pipeline-builder/system/registry`). The CodeBuild role needs\n  // `secretsmanager:GetSecretValue` for that ARN — CDK adds this\n  // automatically when the secret reference flows through\n  // `secretsManagerCredentials`.\n  //\n  // Use a deterministic construct ID so multiple steps that share the same\n  // registry secret reuse the same imported construct (avoids \"construct\n  // already exists\" errors when many steps resolve the same image).\n  const stack = Stack.of(scope);\n  const secretName = registry.credentialsSecret || 'pipeline-builder/system/registry';\n  // CDK construct IDs allow only [A-Za-z0-9_-]; sanitize the secret name.\n  const secretConstructId = `RegistryCreds_${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);\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"]}
package/package.json CHANGED
@@ -25,8 +25,8 @@
25
25
  "typescript": "5.9.3"
26
26
  },
27
27
  "dependencies": {
28
- "@pipeline-builder/api-core": "3.3.31",
29
- "@pipeline-builder/pipeline-data": "3.3.30",
28
+ "@pipeline-builder/api-core": "3.3.33",
29
+ "@pipeline-builder/pipeline-data": "3.3.32",
30
30
  "aws-cdk-lib": "2.251.0",
31
31
  "axios": "1.13.5",
32
32
  "constructs": "10.5.1",
@@ -75,7 +75,7 @@
75
75
  "access": "public",
76
76
  "registry": "https://registry.npmjs.org/"
77
77
  },
78
- "version": "3.3.33",
78
+ "version": "3.3.35",
79
79
  "bugs": {
80
80
  "url": "https://github.com/mwashburn160/pipeline-builder/issues"
81
81
  },