@pipeline-builder/pipeline-core 3.3.35 → 3.4.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/config/config-types.d.ts +0 -16
- package/lib/config/config-types.js +1 -1
- package/lib/config/infrastructure-config.d.ts +0 -2
- package/lib/config/infrastructure-config.js +1 -17
- package/lib/core/pipeline-helpers.d.ts +9 -7
- package/lib/core/pipeline-helpers.js +47 -31
- package/lib/handlers/plugin-lookup-handler.js +9 -4
- package/lib/pipeline/pipeline-builder.d.ts +10 -0
- package/lib/pipeline/pipeline-builder.js +15 -6
- package/lib/pipeline/plugin-lookup.d.ts +15 -3
- package/lib/pipeline/plugin-lookup.js +18 -4
- package/package.json +3 -3
|
@@ -110,25 +110,12 @@ export interface DatabaseConfig {
|
|
|
110
110
|
export interface RegistryConfig {
|
|
111
111
|
readonly host: string;
|
|
112
112
|
readonly port: number;
|
|
113
|
-
readonly user: string;
|
|
114
|
-
readonly token: string;
|
|
115
113
|
/** Docker network for build/push (empty string = default). */
|
|
116
114
|
readonly network: string;
|
|
117
115
|
/** Use plain HTTP instead of HTTPS (env: `DOCKER_REGISTRY_HTTP`). Defaults to true. */
|
|
118
116
|
readonly http: boolean;
|
|
119
117
|
/** Skip TLS certificate verification for self-signed certs (env: `DOCKER_REGISTRY_INSECURE`). Defaults to true. */
|
|
120
118
|
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;
|
|
132
119
|
}
|
|
133
120
|
export interface RedisConfig {
|
|
134
121
|
readonly host: string;
|
|
@@ -188,9 +175,6 @@ export interface AWSConfig {
|
|
|
188
175
|
readonly codeBuild: {
|
|
189
176
|
readonly computeType: ComputeType;
|
|
190
177
|
};
|
|
191
|
-
/** When true, resolve synth plugin via custom resource Lambda at deploy time.
|
|
192
|
-
* When false (default), use fallback synth commands (pipeline-manager synth). */
|
|
193
|
-
readonly resolvedSynthPlugin: boolean;
|
|
194
178
|
}
|
|
195
179
|
/** Express rate limiting configuration. */
|
|
196
180
|
export interface RateLimitConfig {
|
|
@@ -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   * 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"]}
|
|
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  /** 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}\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,8 +5,6 @@ import type { AWSConfig, BuildConfig, ComplianceConfig, DatabaseConfig, Observab
|
|
|
5
5
|
* Environment variables:
|
|
6
6
|
* - `IMAGE_REGISTRY_HOST` — Registry hostname (default: `'registry'`)
|
|
7
7
|
* - `IMAGE_REGISTRY_PORT` — Registry port (default: `5000`)
|
|
8
|
-
* - `IMAGE_REGISTRY_USER` — Registry username (default: `'admin'`)
|
|
9
|
-
* - `IMAGE_REGISTRY_TOKEN` — Registry auth token (default: `'password'`)
|
|
10
8
|
* - `DOCKER_NETWORK` — Docker network for build/push (default: `''`)
|
|
11
9
|
* - `DOCKER_REGISTRY_HTTP` — Use plain HTTP (default: `true`). Set `false` for HTTPS.
|
|
12
10
|
* - `DOCKER_REGISTRY_INSECURE` — Skip TLS verification (default: `true`). Set `false` for production.
|
|
@@ -18,23 +18,12 @@ const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
|
18
18
|
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
19
19
|
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
20
20
|
const pipeline_helpers_1 = require("../core/pipeline-helpers");
|
|
21
|
-
function requireInProduction(envVar, devDefault) {
|
|
22
|
-
const value = process.env[envVar];
|
|
23
|
-
if (value)
|
|
24
|
-
return value;
|
|
25
|
-
if (process.env.NODE_ENV === 'production') {
|
|
26
|
-
throw new Error(`${envVar} is required in production`);
|
|
27
|
-
}
|
|
28
|
-
return devDefault;
|
|
29
|
-
}
|
|
30
21
|
/**
|
|
31
22
|
* Load Docker registry configuration from environment variables.
|
|
32
23
|
*
|
|
33
24
|
* Environment variables:
|
|
34
25
|
* - `IMAGE_REGISTRY_HOST` — Registry hostname (default: `'registry'`)
|
|
35
26
|
* - `IMAGE_REGISTRY_PORT` — Registry port (default: `5000`)
|
|
36
|
-
* - `IMAGE_REGISTRY_USER` — Registry username (default: `'admin'`)
|
|
37
|
-
* - `IMAGE_REGISTRY_TOKEN` — Registry auth token (default: `'password'`)
|
|
38
27
|
* - `DOCKER_NETWORK` — Docker network for build/push (default: `''`)
|
|
39
28
|
* - `DOCKER_REGISTRY_HTTP` — Use plain HTTP (default: `true`). Set `false` for HTTPS.
|
|
40
29
|
* - `DOCKER_REGISTRY_INSECURE` — Skip TLS verification (default: `true`). Set `false` for production.
|
|
@@ -45,13 +34,9 @@ function loadRegistryConfig() {
|
|
|
45
34
|
return {
|
|
46
35
|
host: process.env.IMAGE_REGISTRY_HOST || 'registry',
|
|
47
36
|
port: parseInt(process.env.IMAGE_REGISTRY_PORT || '5000', 10),
|
|
48
|
-
user: requireInProduction('IMAGE_REGISTRY_USER', 'admin'),
|
|
49
|
-
token: requireInProduction('IMAGE_REGISTRY_TOKEN', 'password'),
|
|
50
37
|
network: process.env.DOCKER_NETWORK || '',
|
|
51
38
|
http: process.env.DOCKER_REGISTRY_HTTP !== 'false',
|
|
52
39
|
insecure: process.env.DOCKER_REGISTRY_INSECURE !== 'false',
|
|
53
|
-
credentialsSecret: process.env.IMAGE_REGISTRY_CREDS_SECRET
|
|
54
|
-
|| 'pipeline-builder/system/registry',
|
|
55
40
|
};
|
|
56
41
|
}
|
|
57
42
|
function loadRedisConfig() {
|
|
@@ -172,7 +157,6 @@ function loadAWSConfig() {
|
|
|
172
157
|
codeBuild: {
|
|
173
158
|
computeType: (0, pipeline_helpers_1.getComputeType)(process.env.CODEBUILD_COMPUTE_TYPE || 'SMALL'),
|
|
174
159
|
},
|
|
175
|
-
resolvedSynthPlugin: process.env.RESOLVED_SYNTH_PLUGIN === 'true',
|
|
176
160
|
};
|
|
177
161
|
}
|
|
178
162
|
/**
|
|
@@ -200,4 +184,4 @@ function parseRetention(days) {
|
|
|
200
184
|
const parsed = parseInt(days, 10);
|
|
201
185
|
return VALID_RETENTION_DAYS.has(parsed) ? parsed : aws_logs_1.RetentionDays.ONE_DAY;
|
|
202
186
|
}
|
|
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"]}
|
|
187
|
+
//# 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\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 * - `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    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}\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"]}
|
|
@@ -45,17 +45,19 @@ export declare function extractMetadataEnv(metadata: MetaDataType): Record<strin
|
|
|
45
45
|
* `undefined` and the metadata wins.
|
|
46
46
|
* 2. Otherwise, if the plugin has an `imageTag` AND the registry config
|
|
47
47
|
* is populated, build a `LinuxBuildImage.fromDockerRegistry()` image
|
|
48
|
-
* pointing at `<registry-host>:<port
|
|
49
|
-
*
|
|
50
|
-
* `pipeline-
|
|
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.
|
|
51
53
|
* 3. If neither (e.g., a `metadata_only` plugin or one with no image),
|
|
52
54
|
* return `undefined` so CodeBuild uses its default (`standard:7.0`).
|
|
53
55
|
*
|
|
54
|
-
* `scope`
|
|
55
|
-
*
|
|
56
|
-
*
|
|
56
|
+
* `scope` and `orgId` are required: the per-org platform Secret is named
|
|
57
|
+
* `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`
|
|
58
|
+
* needs a Construct to anchor the imported secret to.
|
|
57
59
|
*/
|
|
58
|
-
export declare function resolvePluginImage(scope: Construct | undefined, plugin: Plugin): IBuildImage | undefined;
|
|
60
|
+
export declare function resolvePluginImage(scope: Construct | undefined, plugin: Plugin, orgId?: string): IBuildImage | undefined;
|
|
59
61
|
export declare function createCodeBuildStep(options: CodeBuildStepOptions): ShellStep | CodeBuildStep | ManualApprovalStep;
|
|
60
62
|
/**
|
|
61
63
|
* Convert string or ComputeType enum to CDK ComputeType
|
|
@@ -158,17 +158,19 @@ function toSecretEnvVars(secrets, orgId) {
|
|
|
158
158
|
* `undefined` and the metadata wins.
|
|
159
159
|
* 2. Otherwise, if the plugin has an `imageTag` AND the registry config
|
|
160
160
|
* is populated, build a `LinuxBuildImage.fromDockerRegistry()` image
|
|
161
|
-
* pointing at `<registry-host>:<port
|
|
162
|
-
*
|
|
163
|
-
* `pipeline-
|
|
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.
|
|
164
166
|
* 3. If neither (e.g., a `metadata_only` plugin or one with no image),
|
|
165
167
|
* return `undefined` so CodeBuild uses its default (`standard:7.0`).
|
|
166
168
|
*
|
|
167
|
-
* `scope`
|
|
168
|
-
*
|
|
169
|
-
*
|
|
169
|
+
* `scope` and `orgId` are required: the per-org platform Secret is named
|
|
170
|
+
* `pipeline-builder/<orgId>/platform`, and `Secret.fromSecretNameV2()`
|
|
171
|
+
* needs a Construct to anchor the imported secret to.
|
|
170
172
|
*/
|
|
171
|
-
function resolvePluginImage(scope, plugin) {
|
|
173
|
+
function resolvePluginImage(scope, plugin, orgId) {
|
|
172
174
|
const imageTag = plugin.imageTag;
|
|
173
175
|
// `metadata_only` plugins legitimately have no image — their work runs
|
|
174
176
|
// in the default CodeBuild image. Quiet skip.
|
|
@@ -179,8 +181,8 @@ function resolvePluginImage(scope, plugin) {
|
|
|
179
181
|
// time); if it does, the plugin record is malformed.
|
|
180
182
|
if (!imageTag || imageTag === '') {
|
|
181
183
|
log.warn(`Plugin "${plugin.name}" has buildType=${plugin.buildType} but no imageTag — ` +
|
|
182
|
-
|
|
183
|
-
|
|
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.');
|
|
184
186
|
return undefined;
|
|
185
187
|
}
|
|
186
188
|
let registry;
|
|
@@ -190,42 +192,56 @@ function resolvePluginImage(scope, plugin) {
|
|
|
190
192
|
catch {
|
|
191
193
|
// Config namespace not loaded (e.g., unit tests without full config).
|
|
192
194
|
log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but registry config not loaded — ` +
|
|
193
|
-
|
|
194
|
-
|
|
195
|
+
'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +
|
|
196
|
+
'Set IMAGE_REGISTRY_HOST + IMAGE_REGISTRY_PORT in pipeline-manager\'s environment.');
|
|
195
197
|
return undefined;
|
|
196
198
|
}
|
|
197
199
|
if (!registry?.host) {
|
|
198
200
|
log.warn(`Plugin "${plugin.name}" has imageTag="${imageTag}" but IMAGE_REGISTRY_HOST is empty — ` +
|
|
199
|
-
|
|
200
|
-
|
|
201
|
+
'CodeBuild will fall back to aws/codebuild/standard:7.0. ' +
|
|
202
|
+
'Set IMAGE_REGISTRY_HOST in pipeline-manager\'s environment to use the plugin image.');
|
|
201
203
|
return undefined;
|
|
202
204
|
}
|
|
203
205
|
if (!scope) {
|
|
204
206
|
log.warn(`Plugin "${plugin.name}" image resolution skipped: no construct scope provided`);
|
|
205
207
|
return undefined;
|
|
206
208
|
}
|
|
207
|
-
//
|
|
208
|
-
|
|
209
|
-
|
|
209
|
+
// Per-org auth requires orgId — the Secret is named per org.
|
|
210
|
+
if (!orgId) {
|
|
211
|
+
log.warn(`Plugin "${plugin.name}" image resolution skipped: orgId is required to ` +
|
|
212
|
+
'resolve the per-org platform Secret used for CodeBuild Basic auth.');
|
|
213
|
+
return undefined;
|
|
214
|
+
}
|
|
215
|
+
// Compose the image URI. Namespace by ownership:
|
|
216
|
+
// - Plugins owned by the system org → `system/<imageTag>:latest`.
|
|
217
|
+
// pipeline-image-registry's token service grants pull on `system/*`
|
|
218
|
+
// to any authenticated org user (read-only catalog of shared plugins).
|
|
219
|
+
// - Plugins owned by a tenant org → `org-<orgId>/<imageTag>:latest`.
|
|
220
|
+
// The token service grants pull,push only to members of that org.
|
|
221
|
+
// The plugin's own `orgId` (not the caller's) decides the namespace —
|
|
222
|
+
// tenant pipelines pulling shared system plugins still get the `system/`
|
|
223
|
+
// path because that's where the image actually lives.
|
|
210
224
|
const portPart = registry.port && registry.port !== 80 && registry.port !== 443
|
|
211
225
|
? `:${registry.port}`
|
|
212
226
|
: '';
|
|
213
|
-
const
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
// `
|
|
219
|
-
//
|
|
220
|
-
//
|
|
227
|
+
const SYSTEM_ORG_ID = 'system';
|
|
228
|
+
const namespace = plugin.orgId === SYSTEM_ORG_ID
|
|
229
|
+
? 'system'
|
|
230
|
+
: `org-${plugin.orgId}`;
|
|
231
|
+
const imageUri = `${registry.host}${portPart}/${namespace}/${imageTag}:latest`;
|
|
232
|
+
// CodeBuild reads `pipeline-builder/<orgId>/platform` and sends its
|
|
233
|
+
// `username`/`password` fields as HTTP Basic to the registry. The
|
|
234
|
+
// registry challenges with a Bearer realm pointing at
|
|
235
|
+
// pipeline-image-registry's `/token` endpoint; the Docker client forwards
|
|
236
|
+
// those creds, the password is verified as a platform JWT, and a registry
|
|
237
|
+
// token scoped to the org is issued.
|
|
221
238
|
//
|
|
222
|
-
//
|
|
223
|
-
//
|
|
224
|
-
// already exists" errors when many steps resolve the same image).
|
|
239
|
+
// Same Secret is read by the plugin-lookup Lambda (via the `password`
|
|
240
|
+
// field) — one Secret serves both flows.
|
|
225
241
|
const stack = aws_cdk_lib_1.Stack.of(scope);
|
|
226
|
-
const secretName =
|
|
242
|
+
const secretName = app_config_1.CoreConstants.secretPath(orgId, 'platform');
|
|
227
243
|
// CDK construct IDs allow only [A-Za-z0-9_-]; sanitize the secret name.
|
|
228
|
-
const secretConstructId = `
|
|
244
|
+
const secretConstructId = `PlatformCreds_${secretName.replace(/[^A-Za-z0-9_-]/g, '_')}`;
|
|
229
245
|
const credentialsSecret = stack.node.tryFindChild(secretConstructId)
|
|
230
246
|
?? aws_secretsmanager_1.Secret.fromSecretNameV2(stack, secretConstructId, secretName);
|
|
231
247
|
return aws_codebuild_1.LinuxBuildImage.fromDockerRegistry(imageUri, {
|
|
@@ -279,7 +295,7 @@ function createCodeBuildStep(options) {
|
|
|
279
295
|
// (pipeline-manager, snyk, terraform, etc.). Wiring the plugin's image
|
|
280
296
|
// here means CodeBuild pulls from our private registry and the tools
|
|
281
297
|
// installed in the plugin's Dockerfile are actually available.
|
|
282
|
-
const pluginBuildImage = resolvePluginImage(scope, plugin);
|
|
298
|
+
const pluginBuildImage = resolvePluginImage(scope, plugin, orgId);
|
|
283
299
|
// ShellStep branch.
|
|
284
300
|
// ShellStep itself doesn't accept `buildEnvironment`/`buildImage`. CDK
|
|
285
301
|
// pipelines wraps ShellSteps in a default CodeBuild action that runs on
|
|
@@ -385,4 +401,4 @@ function replaceNonAlphanumeric(input, replaceValue = '_') {
|
|
|
385
401
|
function unwrapSecret(value) {
|
|
386
402
|
return typeof value === 'string' ? value : value.unsafeUnwrap();
|
|
387
403
|
}
|
|
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"]}
|
|
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"]}
|
|
@@ -93,10 +93,15 @@ async function getToken() {
|
|
|
93
93
|
throw new Error(`Secret "${PLATFORM_SECRET_NAME}" is empty`);
|
|
94
94
|
}
|
|
95
95
|
const parsed = JSON.parse(response.SecretString);
|
|
96
|
-
|
|
97
|
-
|
|
96
|
+
// Schema is `{ username, password, ... }` — the `password` field is the
|
|
97
|
+
// platform JWT. Same secret is read by CodeBuild's `secretsManagerCredentials`
|
|
98
|
+
// for registry pulls (Basic auth: username:password = orgId:JWT). The
|
|
99
|
+
// pipeline-image-registry token endpoint validates the password as a JWT
|
|
100
|
+
// and issues a registry token scoped to the JWT's org.
|
|
101
|
+
if (!parsed.password) {
|
|
102
|
+
throw new Error(`Secret "${PLATFORM_SECRET_NAME}" missing password — run "pipeline-manager store-token" to generate`);
|
|
98
103
|
}
|
|
99
|
-
cachedToken = parsed.
|
|
104
|
+
cachedToken = parsed.password;
|
|
100
105
|
lambdaLog.info('AUTH', 'Token retrieved from Secrets Manager');
|
|
101
106
|
return cachedToken;
|
|
102
107
|
}
|
|
@@ -310,4 +315,4 @@ const handler = async (event) => {
|
|
|
310
315
|
}
|
|
311
316
|
};
|
|
312
317
|
exports.handler = handler;
|
|
313
|
-
//# 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  if (!parsed.accessToken) {\n    throw new Error(`Secret \"${PLATFORM_SECRET_NAME}\" missing accessToken — run \"pipeline-manager store-token\" to generate`);\n  }\n\n  cachedToken = parsed.accessToken;\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"]}
|
|
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"]}
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import type { Plugin } from '@pipeline-builder/pipeline-data';
|
|
1
2
|
import { CodePipeline } from 'aws-cdk-lib/pipelines';
|
|
2
3
|
import { Construct } from 'constructs';
|
|
3
4
|
import { PipelineConfiguration } from './pipeline-configuration';
|
|
@@ -43,6 +44,15 @@ export interface BuilderProps {
|
|
|
43
44
|
readonly schedule?: string;
|
|
44
45
|
/** Custom tags applied to all pipeline resources. */
|
|
45
46
|
readonly tags?: Record<string, string>;
|
|
47
|
+
/**
|
|
48
|
+
* Plugins pre-resolved by `pipeline-manager synth` from the platform API,
|
|
49
|
+
* keyed by `alias || name`. When present, `PluginLookup.plugin()` returns
|
|
50
|
+
* the matching entry directly and skips the custom resource — so the
|
|
51
|
+
* synthesized CFN template ships with the real CodeBuild image baked in.
|
|
52
|
+
* Populated by the CLI before invoking the boilerplate app; CDK consumers
|
|
53
|
+
* who construct PipelineBuilder directly normally leave this unset.
|
|
54
|
+
*/
|
|
55
|
+
readonly resolvedPlugins?: Record<string, Plugin>;
|
|
46
56
|
}
|
|
47
57
|
/**
|
|
48
58
|
* CDK construct that creates and configures a CodePipeline for continuous deployment.
|
|
@@ -123,13 +123,24 @@ class PipelineBuilder extends constructs_1.Construct {
|
|
|
123
123
|
runtime: awsConfig.lambda.runtime,
|
|
124
124
|
timeout: awsConfig.lambda.timeout,
|
|
125
125
|
reservedConcurrentExecutions: awsConfig.lambda.reservedConcurrentExecutions,
|
|
126
|
+
resolvedPlugins: props.resolvedPlugins,
|
|
126
127
|
});
|
|
127
128
|
// Create source and build step
|
|
128
129
|
const sourceBuilder = new source_builder_1.SourceBuilder(this, this.config);
|
|
129
130
|
const source = sourceBuilder.create(uniqueId);
|
|
130
|
-
//
|
|
131
|
-
//
|
|
132
|
-
|
|
131
|
+
// Synth-plugin resolution:
|
|
132
|
+
// 1. Pre-resolved by `pipeline-manager synth/deploy` from the platform
|
|
133
|
+
// API → use it. Synth step runs on the real `cdk-synth` image with
|
|
134
|
+
// real commands baked into the template.
|
|
135
|
+
// 2. Otherwise → `fallbackSynth()`: hardcoded `pipeline-manager synth`
|
|
136
|
+
// on standard:7.0. Cold-start path used when the platform isn't
|
|
137
|
+
// reachable at synth time (CLI logs a warning per missed plugin).
|
|
138
|
+
//
|
|
139
|
+
// pluginLookup.plugin() reads `resolvedPlugins` internally and returns
|
|
140
|
+
// the cached entry when present — so calling it always wins over
|
|
141
|
+
// fallbackSynth() when pre-resolution succeeded.
|
|
142
|
+
const synthCacheKey = this.config.plugin.alias || `${this.config.plugin.name}-alias`;
|
|
143
|
+
const plugin = props.resolvedPlugins?.[synthCacheKey]
|
|
133
144
|
? pluginLookup.plugin(this.config.plugin)
|
|
134
145
|
: pluginLookup.fallbackSynth();
|
|
135
146
|
const defaultComputeType = awsConfig.codeBuild.computeType;
|
|
@@ -298,8 +309,6 @@ class PipelineBuilder extends constructs_1.Construct {
|
|
|
298
309
|
PLATFORM_BASE_URL: { value: platformUrl },
|
|
299
310
|
...(pipelineId && { PIPELINE_ID: { value: pipelineId } }),
|
|
300
311
|
...(platformSecretName && { PLATFORM_SECRET_NAME: { value: platformSecretName } }),
|
|
301
|
-
// Enable plugin resolution via custom resource Lambda inside CodePipeline
|
|
302
|
-
RESOLVED_SYNTH_PLUGIN: { value: 'true' },
|
|
303
312
|
// Propagate TLS verification setting so all CodeBuild steps can reach
|
|
304
313
|
// the platform API when using self-signed certificates
|
|
305
314
|
...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {
|
|
@@ -318,4 +327,4 @@ class PipelineBuilder extends constructs_1.Construct {
|
|
|
318
327
|
}
|
|
319
328
|
}
|
|
320
329
|
exports.PipelineBuilder = PipelineBuilder;
|
|
321
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-builder.js","sourceRoot":"","sources":["../../src/pipeline/pipeline-builder.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 { Duration, Tags } from 'aws-cdk-lib';\nimport * as cloudwatch from 'aws-cdk-lib/aws-cloudwatch';\nimport { PipelineNotificationEvents, PipelineType } from 'aws-cdk-lib/aws-codepipeline';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport * as kms from 'aws-cdk-lib/aws-kms';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport { CodePipeline, type CodeBuildOptions } from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\nimport { PipelineConfiguration } from './pipeline-configuration';\nimport { PluginLookup } from './plugin-lookup';\nimport { SourceBuilder } from './source-builder';\nimport { StageBuilder } from './stage-builder';\nimport type { StageOptions, SynthOptions } from './step-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport { ArtifactManager } from '../core/artifact-manager';\nimport { UniqueId } from '../core/id-generator';\nimport { metadataForCodePipeline } from '../core/metadata-builder';\nimport { resolveNetwork } from '../core/network';\nimport type { CodeBuildDefaults } from '../core/network-types';\nimport { createCodeBuildStep } from '../core/pipeline-helpers';\nimport { MetadataKeys, TriggerType } from '../core/pipeline-types';\nimport type { MetaDataType } from '../core/pipeline-types';\nimport { resolveRole } from '../core/role';\nimport type { RoleConfig } from '../core/role-types';\nimport { resolveSecurityGroup } from '../core/security-group';\n\nconst PIPELINE_EVENT_MAP: Record<string, PipelineNotificationEvents> = {\n  FAILED: PipelineNotificationEvents.PIPELINE_EXECUTION_FAILED,\n  SUCCEEDED: PipelineNotificationEvents.PIPELINE_EXECUTION_SUCCEEDED,\n  STARTED: PipelineNotificationEvents.PIPELINE_EXECUTION_STARTED,\n  CANCELED: PipelineNotificationEvents.PIPELINE_EXECUTION_CANCELED,\n  SUPERSEDED: PipelineNotificationEvents.PIPELINE_EXECUTION_SUPERSEDED,\n};\n\nfunction parseNotificationEvents(value: unknown): string[] {\n  if (Array.isArray(value)) return value;\n  if (typeof value === 'string') return value.split(',').map(s => s.trim());\n  return ['FAILED', 'SUCCEEDED'];\n}\n\n/**\n * Configuration properties for the PipelineBuilder construct\n */\nexport interface BuilderProps {\n  /** Project identifier (will be sanitized to lowercase alphanumeric with underscores) */\n  readonly project: string;\n\n  /** Organization identifier (will be sanitized to lowercase alphanumeric with underscores) */\n  readonly organization: string;\n\n  /** Tenant identifier for resolving per-org secrets from AWS Secrets Manager */\n  readonly orgId?: string;\n\n  /** Pipeline database record ID — injected as PIPELINE_ID env var for autonomous synth */\n  readonly pipelineId?: string;\n\n  /** Optional custom pipeline name. Defaults to: {organization}-{project}-pipeline */\n  readonly pipelineName?: string;\n\n  /** Global metadata inherited by all pipeline steps */\n  readonly global?: MetaDataType;\n\n  /**\n   * Pipeline-level CodeBuild defaults applied to all CodeBuild actions\n   * (synth, self-mutation, asset publishing) via `codeBuildDefaults`.\n   */\n  readonly defaults?: CodeBuildDefaults;\n\n  /**\n   * Optional IAM role for the CodePipeline.\n   * When provided, resolves to a CDK IRole and is passed to the CodePipeline construct.\n   * When omitted, CDK auto-creates a role with the correct codepipeline.amazonaws.com principal.\n   */\n  readonly role?: RoleConfig;\n\n  /** Synthesis configuration including source and plugin details */\n  readonly synth: SynthOptions;\n\n  /**\n   * Optional pipeline stages, each containing one or more CodeBuild steps.\n   * Stages are added as waves to the CodePipeline after the synth step.\n   */\n  readonly stages?: StageOptions[];\n\n  /** Optional cron/rate expression for scheduled pipeline execution. */\n  readonly schedule?: string;\n\n  /** Custom tags applied to all pipeline resources. */\n  readonly tags?: Record<string, string>;\n}\n\n/**\n * CDK construct that creates and configures a CodePipeline for continuous deployment.\n *\n * Features:\n * - Multi-source support (S3, GitHub, CodeStar)\n * - Plugin-based build steps\n * - Metadata-driven configuration\n * - Automatic tagging\n * - Automatic sanitization of project and organization names\n *\n * @example\n * ```typescript\n * new PipelineBuilder(this, 'MyPipeline', {\n *   project: 'my-app',\n *   organization: 'my-org',\n *   synth: {\n *     source: {\n *       type: 'github',\n *       options: { repo: 'owner/repo', branch: 'main' }\n *     },\n *     plugin: { name: 'synth' }\n *   }\n * });\n * ```\n */\nexport class PipelineBuilder extends Construct {\n  public readonly pipeline: CodePipeline;\n  public readonly config: PipelineConfiguration;\n\n  constructor(scope: Construct, id: string, props: BuilderProps) {\n    super(scope, id);\n\n    // Use PipelineConfiguration for all business logic (validation, sanitization, metadata merging)\n    this.config = new PipelineConfiguration(props);\n\n    const serverConfig = Config.get('server');\n    const awsConfig = Config.get('aws');\n    // Pass org+project so log group / IAM role names get a stable hash\n    // suffix per pipeline. Prevents `Resource already exists` collisions\n    // across stacks deployed to the same AWS account.\n    const uniqueId = new UniqueId({\n      organization: this.config.organization,\n      project: this.config.project,\n    });\n    const pluginLookup = new PluginLookup(\n      this,\n      uniqueId.generate('plugin:lookup'),\n      {\n        organization: this.config.organization,\n        project: this.config.project,\n        platformUrl: serverConfig.platformUrl,\n        uniqueId,\n        orgId: props.orgId,\n        runtime: awsConfig.lambda.runtime,\n        timeout: awsConfig.lambda.timeout,\n        reservedConcurrentExecutions: awsConfig.lambda.reservedConcurrentExecutions,\n      },\n    );\n\n    // Create source and build step\n    const sourceBuilder = new SourceBuilder(this, this.config);\n    const source = sourceBuilder.create(uniqueId);\n\n    // RESOLVED_SYNTH_PLUGIN=true (CodePipeline): resolve plugin via custom resource Lambda\n    // RESOLVED_SYNTH_PLUGIN=false (default/CLI): use fallback with pipeline-manager synth commands\n    const plugin = awsConfig.resolvedSynthPlugin\n      ? pluginLookup.plugin(this.config.plugin)\n      : pluginLookup.fallbackSynth();\n    const defaultComputeType = awsConfig.codeBuild.computeType;\n    const artifactManager = new ArtifactManager();\n    const synthAlias = this.config.plugin.alias ?? this.config.plugin.name;\n\n    // Scope exposed to plugin-spec templates as `pipeline.*`. Built once\n    // here so both the synth step and every stage step resolve against\n    // the same snapshot.\n    const pipelineScope: Record<string, unknown> = {\n      pipeline: {\n        projectName: this.config.project,\n        project: this.config.project,\n        orgId: this.config.organization,\n        organization: this.config.organization,\n        pipelineName: this.config.pipelineName,\n        metadata: this.config.metadata.merged,\n        vars: (props as { vars?: Record<string, unknown> }).vars ?? {},\n      },\n    };\n\n    const synth = createCodeBuildStep({\n      ...this.config.synthCustomization,\n      id: uniqueId.generate('cdk:synth'),\n      uniqueId,\n      plugin,\n      input: source,\n      metadata: this.config.metadata.merged,\n      network: this.config.network,\n      scope: this,\n      defaultComputeType,\n      artifactManager,\n      stageName: 'no-stage',\n      stageAlias: 'no-stage-alias',\n      pluginAlias: `${synthAlias}-alias`,\n      orgId: props.orgId,\n      pipelineScope,\n    });\n\n    // Resolve pipeline-level defaults into codeBuildDefaults\n    // Build the per-org platform secret name for CodeBuild env vars\n    const platformSecretName = props.orgId\n      ? CoreConstants.secretPath(props.orgId, 'platform')\n      : undefined;\n\n    const codeBuildDefaults = this.resolveDefaults(this.config.defaults, uniqueId, props.pipelineId, platformSecretName, serverConfig.platformUrl);\n\n    // Resolve IAM role if explicitly provided; otherwise let CDK auto-create\n    // the pipeline role with the correct codepipeline.amazonaws.com principal.\n    if (props.role?.type === 'codeBuildDefault') {\n      createLogger('PipelineBuilder').warn(\n        'codeBuildDefault role type uses codebuild.amazonaws.com trust principal — ' +\n        'this is not suitable as the pipeline-level role. Consider using roleArn/roleName ' +\n        'or omitting the role to let CDK auto-create one with codepipeline.amazonaws.com.',\n      );\n    }\n    const role = props.role\n      ? resolveRole(this, uniqueId, props.role)\n      : undefined;\n\n    // Create CodePipeline construct\n    this.pipeline = new CodePipeline(this, uniqueId.generate('pipelines:codepipeline'), {\n      ...(codeBuildDefaults && { codeBuildDefaults }),\n      ...(role && { role }),\n      pipelineType: PipelineType.V2,\n      pipelineName: this.config.pipelineName,\n      synth,\n      ...metadataForCodePipeline(this.config.metadata.merged),\n    });\n\n    if (props.stages) {\n      const stageBuilder = new StageBuilder({\n        scope: this,\n        pluginLookup,\n        uniqueId,\n        globalMetadata: this.config.metadata.merged,\n        defaultComputeType,\n        artifactManager,\n        orgId: props.orgId,\n        pipelineScope,\n      });\n      stageBuilder.addStages(this.pipeline, props.stages);\n    }\n\n    // ── Tags ──\n    // The first three are operations-essential and used by `pipeline-manager\n    // audit-stacks` to diff CFN stacks against the pipeline_registry table.\n    // `OrgId` is the canonical key for cost attribution (AWS Cost Explorer\n    // groups by tag key/value when activated in Billing settings).\n    Tags.of(this.pipeline).add('pipeline-builder', 'true');\n    Tags.of(this.pipeline).add('project', this.config.project);\n    Tags.of(this.pipeline).add('organization', this.config.organization);\n    if (props.orgId) {\n      Tags.of(this.pipeline).add('OrgId', props.orgId);\n    }\n    if (props.tags) {\n      for (const [key, value] of Object.entries(props.tags)) {\n        Tags.of(this.pipeline).add(key, value);\n      }\n    }\n\n    // Build the internal pipeline before accessing its properties\n    this.pipeline.buildPipeline();\n    const cdkPipeline = this.pipeline.pipeline;\n    const meta = this.config.metadata.merged;\n\n    // ── SNS Notifications ──\n    const notificationTopicArn = meta[MetadataKeys.NOTIFICATION_TOPIC_ARN];\n    if (typeof notificationTopicArn === 'string') {\n      const topic = sns.Topic.fromTopicArn(this, 'NotificationTopic', notificationTopicArn);\n      const notificationEvents = parseNotificationEvents(meta[MetadataKeys.NOTIFICATION_EVENTS])\n        .map(e => PIPELINE_EVENT_MAP[e.toUpperCase()])\n        .filter(Boolean);\n      if (notificationEvents.length > 0) {\n        cdkPipeline.notifyOn('PipelineNotification', topic, { events: notificationEvents });\n      }\n    }\n\n    // ── Scheduled Execution ──\n    if (props.synth.source.options?.trigger === TriggerType.SCHEDULE || props.schedule) {\n      const expr = props.schedule || (props.synth.source.options as { schedule?: string })?.schedule || 'rate(1 day)';\n      new events.Rule(this, 'ScheduleRule', {\n        schedule: events.Schedule.expression(expr),\n        targets: [new targets.CodePipeline(cdkPipeline)],\n      });\n    }\n\n    // ── Execution Event Tracking (forward pipeline state changes to SNS) ──\n    if (meta[MetadataKeys.ENABLE_EXECUTION_EVENTS] && typeof notificationTopicArn === 'string') {\n      new events.Rule(this, 'ExecutionEventRule', {\n        eventPattern: {\n          source: ['aws.codepipeline'],\n          detailType: ['CodePipeline Pipeline Execution State Change'],\n          resources: [cdkPipeline.pipelineArn],\n        },\n        targets: [new targets.SnsTopic(\n          sns.Topic.fromTopicArn(this, 'ExecutionEventTopic', notificationTopicArn),\n        )],\n      });\n    }\n\n    // ── Artifact Encryption (KMS key) ──\n    const kmsKeyArn = this.config.metadata.merged[MetadataKeys.KMS_KEY_ARN];\n    if (typeof kmsKeyArn === 'string') {\n      const key = kms.Key.fromKeyArn(this, 'ArtifactKey', kmsKeyArn);\n      Tags.of(key).add('pipeline', this.config.pipelineName);\n    }\n\n    // ── Pipeline Metrics & Alarms ──\n    const enableMetrics = this.config.metadata.merged[MetadataKeys.ENABLE_METRICS];\n    if (enableMetrics) {\n      new cloudwatch.Alarm(this, 'PipelineFailureAlarm', {\n        metric: new cloudwatch.Metric({\n          namespace: 'AWS/CodePipeline',\n          metricName: 'FailedPipelineExecutionCount',\n          dimensionsMap: { PipelineName: this.config.pipelineName },\n          statistic: 'Sum',\n          period: Duration.minutes(5),\n        }),\n        threshold: 1,\n        evaluationPeriods: 1,\n        alarmDescription: `Pipeline ${this.config.pipelineName} execution failed`,\n        comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n    }\n  }\n\n  /**\n   * Resolves CodeBuildDefaults into the shape expected by CDK's codeBuildDefaults.\n   * Combines network config, security groups, and pipeline-level environment variables\n   * (PIPELINE_ID, EXECUTION_ID, PLATFORM_BASE_URL) available to all CodeBuild actions.\n   */\n  private resolveDefaults(\n    defaults: CodeBuildDefaults | undefined,\n    id: UniqueId,\n    pipelineId: string | undefined,\n    platformSecretName: string | undefined,\n    platformUrl: string,\n  ): CodeBuildOptions | undefined {\n    const networkProps = defaults?.network\n      ? resolveNetwork(this, id, defaults.network)\n      : undefined;\n\n    const standaloneSecurityGroups = defaults?.securityGroups\n      ? resolveSecurityGroup(this, id, defaults.securityGroups)\n      : undefined;\n\n    // Pipeline-level env vars available to all CodeBuild actions\n    // Note: #{codepipeline.*} resolved variables must go through CodeBuildStep.env\n    // (action-level), not buildEnvironment.environmentVariables (project-level).\n    const pipelineEnvVars: Record<string, { value: string }> = {\n      PLATFORM_BASE_URL: { value: platformUrl },\n      ...(pipelineId && { PIPELINE_ID: { value: pipelineId } }),\n      ...(platformSecretName && { PLATFORM_SECRET_NAME: { value: platformSecretName } }),\n      // Enable plugin resolution via custom resource Lambda inside CodePipeline\n      RESOLVED_SYNTH_PLUGIN: { value: 'true' },\n      // Propagate TLS verification setting so all CodeBuild steps can reach\n      // the platform API when using self-signed certificates\n      ...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {\n        NODE_TLS_REJECT_UNAUTHORIZED: { value: '0' },\n      }),\n    };\n\n    const securityGroups = [\n      ...(networkProps?.securityGroups ?? []),\n      ...(standaloneSecurityGroups ?? []),\n    ];\n\n    return {\n      ...(networkProps && { vpc: networkProps.vpc, subnetSelection: networkProps.subnetSelection }),\n      ...(securityGroups.length > 0 && { securityGroups }),\n      buildEnvironment: { environmentVariables: pipelineEnvVars },\n    };\n  }\n}\n"]}
|
|
330
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-builder.js","sourceRoot":"","sources":["../../src/pipeline/pipeline-builder.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, Tags } from 'aws-cdk-lib';\nimport * as cloudwatch from 'aws-cdk-lib/aws-cloudwatch';\nimport { PipelineNotificationEvents, PipelineType } from 'aws-cdk-lib/aws-codepipeline';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport * as kms from 'aws-cdk-lib/aws-kms';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport { CodePipeline, type CodeBuildOptions } from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\nimport { PipelineConfiguration } from './pipeline-configuration';\nimport { PluginLookup } from './plugin-lookup';\nimport { SourceBuilder } from './source-builder';\nimport { StageBuilder } from './stage-builder';\nimport type { StageOptions, SynthOptions } from './step-types';\nimport { Config, CoreConstants } from '../config/app-config';\nimport { ArtifactManager } from '../core/artifact-manager';\nimport { UniqueId } from '../core/id-generator';\nimport { metadataForCodePipeline } from '../core/metadata-builder';\nimport { resolveNetwork } from '../core/network';\nimport type { CodeBuildDefaults } from '../core/network-types';\nimport { createCodeBuildStep } from '../core/pipeline-helpers';\nimport { MetadataKeys, TriggerType } from '../core/pipeline-types';\nimport type { MetaDataType } from '../core/pipeline-types';\nimport { resolveRole } from '../core/role';\nimport type { RoleConfig } from '../core/role-types';\nimport { resolveSecurityGroup } from '../core/security-group';\n\nconst PIPELINE_EVENT_MAP: Record<string, PipelineNotificationEvents> = {\n  FAILED: PipelineNotificationEvents.PIPELINE_EXECUTION_FAILED,\n  SUCCEEDED: PipelineNotificationEvents.PIPELINE_EXECUTION_SUCCEEDED,\n  STARTED: PipelineNotificationEvents.PIPELINE_EXECUTION_STARTED,\n  CANCELED: PipelineNotificationEvents.PIPELINE_EXECUTION_CANCELED,\n  SUPERSEDED: PipelineNotificationEvents.PIPELINE_EXECUTION_SUPERSEDED,\n};\n\nfunction parseNotificationEvents(value: unknown): string[] {\n  if (Array.isArray(value)) return value;\n  if (typeof value === 'string') return value.split(',').map(s => s.trim());\n  return ['FAILED', 'SUCCEEDED'];\n}\n\n/**\n * Configuration properties for the PipelineBuilder construct\n */\nexport interface BuilderProps {\n  /** Project identifier (will be sanitized to lowercase alphanumeric with underscores) */\n  readonly project: string;\n\n  /** Organization identifier (will be sanitized to lowercase alphanumeric with underscores) */\n  readonly organization: string;\n\n  /** Tenant identifier for resolving per-org secrets from AWS Secrets Manager */\n  readonly orgId?: string;\n\n  /** Pipeline database record ID — injected as PIPELINE_ID env var for autonomous synth */\n  readonly pipelineId?: string;\n\n  /** Optional custom pipeline name. Defaults to: {organization}-{project}-pipeline */\n  readonly pipelineName?: string;\n\n  /** Global metadata inherited by all pipeline steps */\n  readonly global?: MetaDataType;\n\n  /**\n   * Pipeline-level CodeBuild defaults applied to all CodeBuild actions\n   * (synth, self-mutation, asset publishing) via `codeBuildDefaults`.\n   */\n  readonly defaults?: CodeBuildDefaults;\n\n  /**\n   * Optional IAM role for the CodePipeline.\n   * When provided, resolves to a CDK IRole and is passed to the CodePipeline construct.\n   * When omitted, CDK auto-creates a role with the correct codepipeline.amazonaws.com principal.\n   */\n  readonly role?: RoleConfig;\n\n  /** Synthesis configuration including source and plugin details */\n  readonly synth: SynthOptions;\n\n  /**\n   * Optional pipeline stages, each containing one or more CodeBuild steps.\n   * Stages are added as waves to the CodePipeline after the synth step.\n   */\n  readonly stages?: StageOptions[];\n\n  /** Optional cron/rate expression for scheduled pipeline execution. */\n  readonly schedule?: string;\n\n  /** Custom tags applied to all pipeline resources. */\n  readonly tags?: Record<string, string>;\n\n  /**\n   * Plugins pre-resolved by `pipeline-manager synth` from the platform API,\n   * keyed by `alias || name`. When present, `PluginLookup.plugin()` returns\n   * the matching entry directly and skips the custom resource — so the\n   * synthesized CFN template ships with the real CodeBuild image baked in.\n   * Populated by the CLI before invoking the boilerplate app; CDK consumers\n   * who construct PipelineBuilder directly normally leave this unset.\n   */\n  readonly resolvedPlugins?: Record<string, Plugin>;\n}\n\n/**\n * CDK construct that creates and configures a CodePipeline for continuous deployment.\n *\n * Features:\n * - Multi-source support (S3, GitHub, CodeStar)\n * - Plugin-based build steps\n * - Metadata-driven configuration\n * - Automatic tagging\n * - Automatic sanitization of project and organization names\n *\n * @example\n * ```typescript\n * new PipelineBuilder(this, 'MyPipeline', {\n *   project: 'my-app',\n *   organization: 'my-org',\n *   synth: {\n *     source: {\n *       type: 'github',\n *       options: { repo: 'owner/repo', branch: 'main' }\n *     },\n *     plugin: { name: 'synth' }\n *   }\n * });\n * ```\n */\nexport class PipelineBuilder extends Construct {\n  public readonly pipeline: CodePipeline;\n  public readonly config: PipelineConfiguration;\n\n  constructor(scope: Construct, id: string, props: BuilderProps) {\n    super(scope, id);\n\n    // Use PipelineConfiguration for all business logic (validation, sanitization, metadata merging)\n    this.config = new PipelineConfiguration(props);\n\n    const serverConfig = Config.get('server');\n    const awsConfig = Config.get('aws');\n    // Pass org+project so log group / IAM role names get a stable hash\n    // suffix per pipeline. Prevents `Resource already exists` collisions\n    // across stacks deployed to the same AWS account.\n    const uniqueId = new UniqueId({\n      organization: this.config.organization,\n      project: this.config.project,\n    });\n    const pluginLookup = new PluginLookup(\n      this,\n      uniqueId.generate('plugin:lookup'),\n      {\n        organization: this.config.organization,\n        project: this.config.project,\n        platformUrl: serverConfig.platformUrl,\n        uniqueId,\n        orgId: props.orgId,\n        runtime: awsConfig.lambda.runtime,\n        timeout: awsConfig.lambda.timeout,\n        reservedConcurrentExecutions: awsConfig.lambda.reservedConcurrentExecutions,\n        resolvedPlugins: props.resolvedPlugins,\n      },\n    );\n\n    // Create source and build step\n    const sourceBuilder = new SourceBuilder(this, this.config);\n    const source = sourceBuilder.create(uniqueId);\n\n    // Synth-plugin resolution:\n    //   1. Pre-resolved by `pipeline-manager synth/deploy` from the platform\n    //      API → use it. Synth step runs on the real `cdk-synth` image with\n    //      real commands baked into the template.\n    //   2. Otherwise → `fallbackSynth()`: hardcoded `pipeline-manager synth`\n    //      on standard:7.0. Cold-start path used when the platform isn't\n    //      reachable at synth time (CLI logs a warning per missed plugin).\n    //\n    // pluginLookup.plugin() reads `resolvedPlugins` internally and returns\n    // the cached entry when present — so calling it always wins over\n    // fallbackSynth() when pre-resolution succeeded.\n    const synthCacheKey =\n      this.config.plugin.alias || `${this.config.plugin.name}-alias`;\n    const plugin = props.resolvedPlugins?.[synthCacheKey]\n      ? pluginLookup.plugin(this.config.plugin)\n      : pluginLookup.fallbackSynth();\n    const defaultComputeType = awsConfig.codeBuild.computeType;\n    const artifactManager = new ArtifactManager();\n    const synthAlias = this.config.plugin.alias ?? this.config.plugin.name;\n\n    // Scope exposed to plugin-spec templates as `pipeline.*`. Built once\n    // here so both the synth step and every stage step resolve against\n    // the same snapshot.\n    const pipelineScope: Record<string, unknown> = {\n      pipeline: {\n        projectName: this.config.project,\n        project: this.config.project,\n        orgId: this.config.organization,\n        organization: this.config.organization,\n        pipelineName: this.config.pipelineName,\n        metadata: this.config.metadata.merged,\n        vars: (props as { vars?: Record<string, unknown> }).vars ?? {},\n      },\n    };\n\n    const synth = createCodeBuildStep({\n      ...this.config.synthCustomization,\n      id: uniqueId.generate('cdk:synth'),\n      uniqueId,\n      plugin,\n      input: source,\n      metadata: this.config.metadata.merged,\n      network: this.config.network,\n      scope: this,\n      defaultComputeType,\n      artifactManager,\n      stageName: 'no-stage',\n      stageAlias: 'no-stage-alias',\n      pluginAlias: `${synthAlias}-alias`,\n      orgId: props.orgId,\n      pipelineScope,\n    });\n\n    // Resolve pipeline-level defaults into codeBuildDefaults\n    // Build the per-org platform secret name for CodeBuild env vars\n    const platformSecretName = props.orgId\n      ? CoreConstants.secretPath(props.orgId, 'platform')\n      : undefined;\n\n    const codeBuildDefaults = this.resolveDefaults(this.config.defaults, uniqueId, props.pipelineId, platformSecretName, serverConfig.platformUrl);\n\n    // Resolve IAM role if explicitly provided; otherwise let CDK auto-create\n    // the pipeline role with the correct codepipeline.amazonaws.com principal.\n    if (props.role?.type === 'codeBuildDefault') {\n      createLogger('PipelineBuilder').warn(\n        'codeBuildDefault role type uses codebuild.amazonaws.com trust principal — ' +\n        'this is not suitable as the pipeline-level role. Consider using roleArn/roleName ' +\n        'or omitting the role to let CDK auto-create one with codepipeline.amazonaws.com.',\n      );\n    }\n    const role = props.role\n      ? resolveRole(this, uniqueId, props.role)\n      : undefined;\n\n    // Create CodePipeline construct\n    this.pipeline = new CodePipeline(this, uniqueId.generate('pipelines:codepipeline'), {\n      ...(codeBuildDefaults && { codeBuildDefaults }),\n      ...(role && { role }),\n      pipelineType: PipelineType.V2,\n      pipelineName: this.config.pipelineName,\n      synth,\n      ...metadataForCodePipeline(this.config.metadata.merged),\n    });\n\n    if (props.stages) {\n      const stageBuilder = new StageBuilder({\n        scope: this,\n        pluginLookup,\n        uniqueId,\n        globalMetadata: this.config.metadata.merged,\n        defaultComputeType,\n        artifactManager,\n        orgId: props.orgId,\n        pipelineScope,\n      });\n      stageBuilder.addStages(this.pipeline, props.stages);\n    }\n\n    // ── Tags ──\n    // The first three are operations-essential and used by `pipeline-manager\n    // audit-stacks` to diff CFN stacks against the pipeline_registry table.\n    // `OrgId` is the canonical key for cost attribution (AWS Cost Explorer\n    // groups by tag key/value when activated in Billing settings).\n    Tags.of(this.pipeline).add('pipeline-builder', 'true');\n    Tags.of(this.pipeline).add('project', this.config.project);\n    Tags.of(this.pipeline).add('organization', this.config.organization);\n    if (props.orgId) {\n      Tags.of(this.pipeline).add('OrgId', props.orgId);\n    }\n    if (props.tags) {\n      for (const [key, value] of Object.entries(props.tags)) {\n        Tags.of(this.pipeline).add(key, value);\n      }\n    }\n\n    // Build the internal pipeline before accessing its properties\n    this.pipeline.buildPipeline();\n    const cdkPipeline = this.pipeline.pipeline;\n    const meta = this.config.metadata.merged;\n\n    // ── SNS Notifications ──\n    const notificationTopicArn = meta[MetadataKeys.NOTIFICATION_TOPIC_ARN];\n    if (typeof notificationTopicArn === 'string') {\n      const topic = sns.Topic.fromTopicArn(this, 'NotificationTopic', notificationTopicArn);\n      const notificationEvents = parseNotificationEvents(meta[MetadataKeys.NOTIFICATION_EVENTS])\n        .map(e => PIPELINE_EVENT_MAP[e.toUpperCase()])\n        .filter(Boolean);\n      if (notificationEvents.length > 0) {\n        cdkPipeline.notifyOn('PipelineNotification', topic, { events: notificationEvents });\n      }\n    }\n\n    // ── Scheduled Execution ──\n    if (props.synth.source.options?.trigger === TriggerType.SCHEDULE || props.schedule) {\n      const expr = props.schedule || (props.synth.source.options as { schedule?: string })?.schedule || 'rate(1 day)';\n      new events.Rule(this, 'ScheduleRule', {\n        schedule: events.Schedule.expression(expr),\n        targets: [new targets.CodePipeline(cdkPipeline)],\n      });\n    }\n\n    // ── Execution Event Tracking (forward pipeline state changes to SNS) ──\n    if (meta[MetadataKeys.ENABLE_EXECUTION_EVENTS] && typeof notificationTopicArn === 'string') {\n      new events.Rule(this, 'ExecutionEventRule', {\n        eventPattern: {\n          source: ['aws.codepipeline'],\n          detailType: ['CodePipeline Pipeline Execution State Change'],\n          resources: [cdkPipeline.pipelineArn],\n        },\n        targets: [new targets.SnsTopic(\n          sns.Topic.fromTopicArn(this, 'ExecutionEventTopic', notificationTopicArn),\n        )],\n      });\n    }\n\n    // ── Artifact Encryption (KMS key) ──\n    const kmsKeyArn = this.config.metadata.merged[MetadataKeys.KMS_KEY_ARN];\n    if (typeof kmsKeyArn === 'string') {\n      const key = kms.Key.fromKeyArn(this, 'ArtifactKey', kmsKeyArn);\n      Tags.of(key).add('pipeline', this.config.pipelineName);\n    }\n\n    // ── Pipeline Metrics & Alarms ──\n    const enableMetrics = this.config.metadata.merged[MetadataKeys.ENABLE_METRICS];\n    if (enableMetrics) {\n      new cloudwatch.Alarm(this, 'PipelineFailureAlarm', {\n        metric: new cloudwatch.Metric({\n          namespace: 'AWS/CodePipeline',\n          metricName: 'FailedPipelineExecutionCount',\n          dimensionsMap: { PipelineName: this.config.pipelineName },\n          statistic: 'Sum',\n          period: Duration.minutes(5),\n        }),\n        threshold: 1,\n        evaluationPeriods: 1,\n        alarmDescription: `Pipeline ${this.config.pipelineName} execution failed`,\n        comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n    }\n  }\n\n  /**\n   * Resolves CodeBuildDefaults into the shape expected by CDK's codeBuildDefaults.\n   * Combines network config, security groups, and pipeline-level environment variables\n   * (PIPELINE_ID, EXECUTION_ID, PLATFORM_BASE_URL) available to all CodeBuild actions.\n   */\n  private resolveDefaults(\n    defaults: CodeBuildDefaults | undefined,\n    id: UniqueId,\n    pipelineId: string | undefined,\n    platformSecretName: string | undefined,\n    platformUrl: string,\n  ): CodeBuildOptions | undefined {\n    const networkProps = defaults?.network\n      ? resolveNetwork(this, id, defaults.network)\n      : undefined;\n\n    const standaloneSecurityGroups = defaults?.securityGroups\n      ? resolveSecurityGroup(this, id, defaults.securityGroups)\n      : undefined;\n\n    // Pipeline-level env vars available to all CodeBuild actions\n    // Note: #{codepipeline.*} resolved variables must go through CodeBuildStep.env\n    // (action-level), not buildEnvironment.environmentVariables (project-level).\n    const pipelineEnvVars: Record<string, { value: string }> = {\n      PLATFORM_BASE_URL: { value: platformUrl },\n      ...(pipelineId && { PIPELINE_ID: { value: pipelineId } }),\n      ...(platformSecretName && { PLATFORM_SECRET_NAME: { value: platformSecretName } }),\n      // Propagate TLS verification setting so all CodeBuild steps can reach\n      // the platform API when using self-signed certificates\n      ...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {\n        NODE_TLS_REJECT_UNAUTHORIZED: { value: '0' },\n      }),\n    };\n\n    const securityGroups = [\n      ...(networkProps?.securityGroups ?? []),\n      ...(standaloneSecurityGroups ?? []),\n    ];\n\n    return {\n      ...(networkProps && { vpc: networkProps.vpc, subnetSelection: networkProps.subnetSelection }),\n      ...(securityGroups.length > 0 && { securityGroups }),\n      buildEnvironment: { environmentVariables: pipelineEnvVars },\n    };\n  }\n}\n"]}
|
|
@@ -24,6 +24,15 @@ export interface PluginLookupProps {
|
|
|
24
24
|
readonly logRetention?: RetentionDays;
|
|
25
25
|
/** Reserved concurrent executions for the lookup Lambda (default: 30) */
|
|
26
26
|
readonly reservedConcurrentExecutions?: number;
|
|
27
|
+
/**
|
|
28
|
+
* Plugins pre-resolved by `pipeline-manager synth` from the platform API.
|
|
29
|
+
* Keyed by `alias || name` (matches the construct's normalize() output).
|
|
30
|
+
* When a lookup hits this map, `plugin()` returns the resolved Plugin
|
|
31
|
+
* directly — no custom resource is created. This is what makes the
|
|
32
|
+
* imageTag, commands, env, etc. available at synth time so the resulting
|
|
33
|
+
* CFN template ships with the real values baked in.
|
|
34
|
+
*/
|
|
35
|
+
readonly resolvedPlugins?: Record<string, Plugin>;
|
|
27
36
|
}
|
|
28
37
|
/**
|
|
29
38
|
* CDK Construct responsible for looking up plugin configurations from an external platform
|
|
@@ -56,6 +65,7 @@ export declare class PluginLookup extends Construct {
|
|
|
56
65
|
private readonly _memorySize;
|
|
57
66
|
private readonly _reservedConcurrentExecutions?;
|
|
58
67
|
private readonly _orgId?;
|
|
68
|
+
private readonly _resolvedPlugins?;
|
|
59
69
|
constructor(scope: Construct, id: string, props: PluginLookupProps);
|
|
60
70
|
/**
|
|
61
71
|
* Looks up and resolves plugin configuration using either a simple name or full PluginOptions object
|
|
@@ -92,9 +102,11 @@ export declare class PluginLookup extends Construct {
|
|
|
92
102
|
/** Fallback for unresolved plugin lookup tokens during synthesis. */
|
|
93
103
|
private fallback;
|
|
94
104
|
/**
|
|
95
|
-
* Synth plugin with pipeline-manager commands.
|
|
96
|
-
*
|
|
97
|
-
*
|
|
105
|
+
* Synth plugin with pipeline-manager commands. Cold-start fallback for the
|
|
106
|
+
* synth step when pre-resolution by `pipeline-manager synth/deploy` didn't
|
|
107
|
+
* populate `resolvedPlugins` for the synth plugin. Runs on `standard:7.0`
|
|
108
|
+
* via the default CodeBuild image and self-bootstraps the real synth via
|
|
109
|
+
* `pipeline-manager synth --id ${PIPELINE_ID}`.
|
|
98
110
|
*/
|
|
99
111
|
fallbackSynth(): Plugin;
|
|
100
112
|
}
|
|
@@ -45,6 +45,7 @@ class PluginLookup extends constructs_1.Construct {
|
|
|
45
45
|
_memorySize;
|
|
46
46
|
_reservedConcurrentExecutions;
|
|
47
47
|
_orgId;
|
|
48
|
+
_resolvedPlugins;
|
|
48
49
|
constructor(scope, id, props) {
|
|
49
50
|
super(scope, id);
|
|
50
51
|
if (!props.organization || !props.project) {
|
|
@@ -57,6 +58,7 @@ class PluginLookup extends constructs_1.Construct {
|
|
|
57
58
|
this._timeout = props.timeout ?? aws_cdk_lib_1.Duration.seconds(30);
|
|
58
59
|
this._memorySize = props.memorySize ?? app_config_1.Config.get('aws').lambda.memorySize;
|
|
59
60
|
this._reservedConcurrentExecutions = props.reservedConcurrentExecutions;
|
|
61
|
+
this._resolvedPlugins = props.resolvedPlugins;
|
|
60
62
|
const onEventHandler = this.createLambdaFunction();
|
|
61
63
|
// Log-group strategy:
|
|
62
64
|
// Previous code created an `AWS::Logs::LogGroup` resource with an
|
|
@@ -88,6 +90,16 @@ class PluginLookup extends constructs_1.Construct {
|
|
|
88
90
|
*/
|
|
89
91
|
plugin(plugin) {
|
|
90
92
|
const props = this.normalize(plugin);
|
|
93
|
+
// Pre-resolved by `pipeline-manager synth` from the platform API.
|
|
94
|
+
// Skip the custom resource entirely — we already know the plugin's
|
|
95
|
+
// imageTag, commands, env, etc. at synth time, so the resulting CFN
|
|
96
|
+
// template can ship with the real CodeBuild image baked in.
|
|
97
|
+
const cacheKey = props.alias || props.name;
|
|
98
|
+
const preResolved = this._resolvedPlugins?.[cacheKey];
|
|
99
|
+
if (preResolved) {
|
|
100
|
+
log.debug(`Plugin "${props.name}" pre-resolved (alias=${cacheKey}) — skipping custom resource`);
|
|
101
|
+
return preResolved;
|
|
102
|
+
}
|
|
91
103
|
const custom = this.createCustomResource(props);
|
|
92
104
|
const encoded = custom.getAttString('ResultValue');
|
|
93
105
|
if (aws_cdk_lib_1.Token.isUnresolved(encoded)) {
|
|
@@ -238,9 +250,11 @@ class PluginLookup extends constructs_1.Construct {
|
|
|
238
250
|
};
|
|
239
251
|
}
|
|
240
252
|
/**
|
|
241
|
-
* Synth plugin with pipeline-manager commands.
|
|
242
|
-
*
|
|
243
|
-
*
|
|
253
|
+
* Synth plugin with pipeline-manager commands. Cold-start fallback for the
|
|
254
|
+
* synth step when pre-resolution by `pipeline-manager synth/deploy` didn't
|
|
255
|
+
* populate `resolvedPlugins` for the synth plugin. Runs on `standard:7.0`
|
|
256
|
+
* via the default CodeBuild image and self-bootstraps the real synth via
|
|
257
|
+
* `pipeline-manager synth --id ${PIPELINE_ID}`.
|
|
244
258
|
*/
|
|
245
259
|
fallbackSynth() {
|
|
246
260
|
return {
|
|
@@ -254,4 +268,4 @@ class PluginLookup extends constructs_1.Construct {
|
|
|
254
268
|
}
|
|
255
269
|
}
|
|
256
270
|
exports.PluginLookup = PluginLookup;
|
|
257
|
-
//# 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\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\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\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    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.\n   * Used when RESOLVED_SYNTH_PLUGIN is not set (default/CLI) — CDK needs real\n   * commands at synthesis time, but the custom resource resolves at deploy time.\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"]}
|
|
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"]}
|
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.
|
|
29
|
-
"@pipeline-builder/pipeline-data": "3.
|
|
28
|
+
"@pipeline-builder/api-core": "3.4.0",
|
|
29
|
+
"@pipeline-builder/pipeline-data": "3.4.0",
|
|
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.
|
|
78
|
+
"version": "3.4.1",
|
|
79
79
|
"bugs": {
|
|
80
80
|
"url": "https://github.com/mwashburn160/pipeline-builder/issues"
|
|
81
81
|
},
|