@pipeline-builder/pipeline-core 3.1.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.
Files changed (54) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +32 -0
  3. package/lib/config/app-config.d.ts +81 -0
  4. package/lib/config/app-config.js +151 -0
  5. package/lib/config/billing-config.d.ts +17 -0
  6. package/lib/config/billing-config.js +95 -0
  7. package/lib/config/config-types.d.ts +213 -0
  8. package/lib/config/config-types.js +5 -0
  9. package/lib/config/infrastructure-config.d.ts +55 -0
  10. package/lib/config/infrastructure-config.js +200 -0
  11. package/lib/config/server-config.d.ts +53 -0
  12. package/lib/config/server-config.js +180 -0
  13. package/lib/core/artifact-manager.d.ts +62 -0
  14. package/lib/core/artifact-manager.js +86 -0
  15. package/lib/core/id-generator.d.ts +26 -0
  16. package/lib/core/id-generator.js +44 -0
  17. package/lib/core/metadata-builder.d.ts +13 -0
  18. package/lib/core/metadata-builder.js +81 -0
  19. package/lib/core/network-types.d.ts +200 -0
  20. package/lib/core/network-types.js +5 -0
  21. package/lib/core/network.d.ts +20 -0
  22. package/lib/core/network.js +84 -0
  23. package/lib/core/pipeline-helpers.d.ts +53 -0
  24. package/lib/core/pipeline-helpers.js +273 -0
  25. package/lib/core/pipeline-types.d.ts +136 -0
  26. package/lib/core/pipeline-types.js +140 -0
  27. package/lib/core/role-types.d.ts +254 -0
  28. package/lib/core/role-types.js +5 -0
  29. package/lib/core/role.d.ts +14 -0
  30. package/lib/core/role.js +118 -0
  31. package/lib/core/security-group-types.d.ts +84 -0
  32. package/lib/core/security-group-types.js +5 -0
  33. package/lib/core/security-group.d.ts +14 -0
  34. package/lib/core/security-group.js +34 -0
  35. package/lib/handlers/plugin-lookup-handler.d.ts +32 -0
  36. package/lib/handlers/plugin-lookup-handler.js +313 -0
  37. package/lib/handlers/pnpm-lock.yaml +12 -0
  38. package/lib/index.d.ts +54 -0
  39. package/lib/index.js +112 -0
  40. package/lib/pipeline/pipeline-builder.d.ts +82 -0
  41. package/lib/pipeline/pipeline-builder.js +292 -0
  42. package/lib/pipeline/pipeline-configuration.d.ts +72 -0
  43. package/lib/pipeline/pipeline-configuration.js +196 -0
  44. package/lib/pipeline/plugin-lookup.d.ts +100 -0
  45. package/lib/pipeline/plugin-lookup.js +247 -0
  46. package/lib/pipeline/source-builder.d.ts +47 -0
  47. package/lib/pipeline/source-builder.js +111 -0
  48. package/lib/pipeline/source-types.d.ts +191 -0
  49. package/lib/pipeline/source-types.js +5 -0
  50. package/lib/pipeline/stage-builder.d.ts +71 -0
  51. package/lib/pipeline/stage-builder.js +118 -0
  52. package/lib/pipeline/step-types.d.ts +307 -0
  53. package/lib/pipeline/step-types.js +5 -0
  54. package/package.json +137 -0
@@ -0,0 +1,180 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.loadServerConfig = loadServerConfig;
6
+ exports.loadAuthConfig = loadAuthConfig;
7
+ exports.loadRateLimitConfig = loadRateLimitConfig;
8
+ exports.validateServerConfig = validateServerConfig;
9
+ exports.validateAuthConfig = validateAuthConfig;
10
+ const api_core_1 = require("@pipeline-builder/api-core");
11
+ const app_config_1 = require("./app-config");
12
+ const log = (0, api_core_1.createLogger)('ServerConfig');
13
+ /**
14
+ * Load server configuration from environment variables.
15
+ *
16
+ * Environment variables:
17
+ * - `PORT` — HTTP listen port (default: `3000`)
18
+ * - `CORS_ORIGIN` — Comma-separated allowed origins (default: `PLATFORM_BASE_URL`)
19
+ * - `CORS_CREDENTIALS` — Allow credentials; set to `'false'` to disable (default: `true`)
20
+ * - `TRUST_PROXY` — Express trust proxy hops (default: `1`)
21
+ * - `PLATFORM_BASE_URL` — Frontend URL used as CORS fallback (default: `'https://localhost:8443'`)
22
+ *
23
+ * @returns Server configuration with port, CORS, trust proxy, and platform URL
24
+ */
25
+ function loadServerConfig() {
26
+ return {
27
+ port: parseInt(process.env.PORT || '3000', 10),
28
+ cors: {
29
+ credentials: process.env.CORS_CREDENTIALS !== 'false',
30
+ origin: process.env.CORS_ORIGIN
31
+ ? process.env.CORS_ORIGIN.split(',').map(o => o.trim())
32
+ : [process.env.PLATFORM_BASE_URL || app_config_1.CoreConstants.DEFAULT_PLATFORM_URL],
33
+ },
34
+ trustProxy: parseInt(process.env.TRUST_PROXY || '1', 10),
35
+ platformUrl: process.env.PLATFORM_BASE_URL || app_config_1.CoreConstants.DEFAULT_PLATFORM_URL,
36
+ httpClient: {
37
+ timeout: parseInt(process.env.HTTP_CLIENT_TIMEOUT || '5000', 10),
38
+ maxRetries: parseInt(process.env.HTTP_CLIENT_MAX_RETRIES || '2', 10),
39
+ retryDelayMs: parseInt(process.env.HTTP_CLIENT_RETRY_DELAY_MS || '200', 10),
40
+ },
41
+ sse: {
42
+ maxClientsPerRequest: parseInt(process.env.SSE_MAX_CLIENTS_PER_REQUEST || '10', 10),
43
+ clientTimeoutMs: parseInt(process.env.SSE_CLIENT_TIMEOUT_MS || '1800000', 10),
44
+ cleanupIntervalMs: parseInt(process.env.SSE_CLEANUP_INTERVAL_MS || '300000', 10),
45
+ },
46
+ services: {
47
+ pluginHost: process.env.PLUGIN_SERVICE_HOST || 'plugin',
48
+ pluginPort: parseInt(process.env.PLUGIN_SERVICE_PORT || '3000', 10),
49
+ pipelineHost: process.env.PIPELINE_SERVICE_HOST || 'pipeline',
50
+ pipelinePort: parseInt(process.env.PIPELINE_SERVICE_PORT || '3000', 10),
51
+ messageHost: process.env.MESSAGE_SERVICE_HOST || 'message',
52
+ messagePort: parseInt(process.env.MESSAGE_SERVICE_PORT || '3000', 10),
53
+ complianceHost: process.env.COMPLIANCE_SERVICE_HOST || 'compliance',
54
+ compliancePort: parseInt(process.env.COMPLIANCE_SERVICE_PORT || '3000', 10),
55
+ billingHost: process.env.BILLING_SERVICE_HOST || 'billing',
56
+ billingPort: parseInt(process.env.BILLING_SERVICE_PORT || '3000', 10),
57
+ billingTimeout: parseInt(process.env.BILLING_SERVICE_TIMEOUT || '5000', 10),
58
+ },
59
+ };
60
+ }
61
+ /**
62
+ * Load authentication configuration from environment variables.
63
+ *
64
+ * Environment variables:
65
+ * - `JWT_SECRET` — **Required.** Secret key for signing JWTs
66
+ * - `REFRESH_TOKEN_SECRET` — **Required.** Secret key for signing refresh tokens
67
+ * - `JWT_EXPIRES_IN` — JWT lifetime in seconds (default: `7200` = 2 hours)
68
+ * - `JWT_ALGORITHM` — JWT signing algorithm (default: `'HS256'`)
69
+ * - `JWT_SALT_ROUNDS` — bcrypt salt rounds for password hashing (default: `12`)
70
+ * - `REFRESH_TOKEN_EXPIRES_IN` — Refresh token lifetime in seconds (default: `2592000` = 30 days)
71
+ *
72
+ * @returns Authentication configuration with safe defaults (empty strings when env vars are unset).
73
+ * Call {@link validateAuthConfig} at server startup to enforce required secrets.
74
+ */
75
+ function loadAuthConfig() {
76
+ return {
77
+ jwt: {
78
+ secret: process.env.JWT_SECRET ?? '',
79
+ expiresIn: parseInt(process.env.JWT_EXPIRES_IN || '7200', 10),
80
+ algorithm: (process.env.JWT_ALGORITHM || 'HS256'),
81
+ saltRounds: parseInt(process.env.JWT_SALT_ROUNDS || '12', 10),
82
+ },
83
+ refreshToken: {
84
+ secret: process.env.REFRESH_TOKEN_SECRET || '',
85
+ expiresIn: parseInt(process.env.REFRESH_TOKEN_EXPIRES_IN || '2592000', 10),
86
+ },
87
+ };
88
+ }
89
+ /**
90
+ * Load rate limiting configuration from environment variables.
91
+ *
92
+ * Environment variables:
93
+ * - `LIMITER_MAX` — Max requests per window (default: `100`)
94
+ * - `LIMITER_WINDOWMS` — Rate limit window in ms (default: `900000` = 15 minutes)
95
+ *
96
+ * @returns Rate limit configuration
97
+ */
98
+ function loadRateLimitConfig() {
99
+ return {
100
+ max: parseInt(process.env.LIMITER_MAX || '100', 10),
101
+ windowMs: parseInt(process.env.LIMITER_WINDOWMS || '900000', 10),
102
+ legacyHeaders: false,
103
+ standardHeaders: true,
104
+ };
105
+ }
106
+ /**
107
+ * Validate server configuration and log warnings for insecure settings.
108
+ *
109
+ * @param config - Server configuration to validate
110
+ */
111
+ function validateServerConfig(config) {
112
+ const warnings = [];
113
+ // Check CORS configuration
114
+ const origin = config.cors.origin;
115
+ const isWildcard = origin === '*' || (Array.isArray(origin) && origin.includes('*'));
116
+ if (isWildcard) {
117
+ warnings.push('CORS origin set to wildcard (*) - consider restricting to specific domains');
118
+ }
119
+ if (isWildcard && config.cors.credentials) {
120
+ // Browsers reject this combination, but it indicates a misconfiguration
121
+ throw new Error('SECURITY ERROR: CORS_ORIGIN=* with CORS_CREDENTIALS=true is an invalid and insecure configuration. ' +
122
+ 'Set CORS_ORIGIN to specific domains or disable CORS_CREDENTIALS.');
123
+ }
124
+ // Check platform URL uses HTTPS
125
+ if (config.platformUrl.startsWith('http://') &&
126
+ !config.platformUrl.includes('localhost')) {
127
+ warnings.push('Platform URL uses HTTP instead of HTTPS - insecure for production');
128
+ }
129
+ // Display warnings
130
+ if (warnings.length > 0) {
131
+ log.warn('Server configuration warnings:');
132
+ warnings.forEach(warning => log.warn(` - ${warning}`));
133
+ }
134
+ }
135
+ const INSECURE_PATTERNS = ['secret', 'password', 'changeme', 'default', '123456', 'admin'];
136
+ function isInsecureSecret(secret) {
137
+ const lower = secret.toLowerCase();
138
+ return INSECURE_PATTERNS.some(s => lower === s || (lower.length < 64 && lower.includes(s)));
139
+ }
140
+ /**
141
+ * Validate authentication configuration (JWT secrets, algorithms, expiration).
142
+ * Call this at server startup, not during CDK synthesis.
143
+ *
144
+ * @param config - Auth configuration to validate
145
+ * @throws {Error} If secrets are insecure, too short (<32 chars), or use disallowed algorithms
146
+ */
147
+ function validateAuthConfig(config) {
148
+ const errors = [];
149
+ const warnings = [];
150
+ // Validate both secrets with the same checks
151
+ for (const [label, secret] of [['JWT', config.jwt.secret], ['Refresh token', config.refreshToken.secret]]) {
152
+ if (secret.length < 32) {
153
+ errors.push(`${label} secret should be at least 32 characters long`);
154
+ }
155
+ if (isInsecureSecret(secret)) {
156
+ errors.push(`${label} secret appears to be insecure or default value`);
157
+ }
158
+ }
159
+ // Check JWT expiration times
160
+ if (config.jwt.expiresIn > 86400) {
161
+ errors.push('JWT expiration must not exceed 24 hours (86400 seconds)');
162
+ }
163
+ else if (config.jwt.expiresIn > 7200) {
164
+ warnings.push('JWT expiration time is greater than 2 hours - shorter expiration recommended');
165
+ }
166
+ // Check algorithm
167
+ if (!app_config_1.CoreConstants.ALLOWED_JWT_ALGORITHMS.includes(config.jwt.algorithm)) {
168
+ errors.push(`JWT algorithm ${config.jwt.algorithm} is not in the allowed list`);
169
+ }
170
+ // Display warnings
171
+ if (warnings.length > 0) {
172
+ log.warn('Auth configuration warnings:');
173
+ warnings.forEach(warning => log.warn(` - ${warning}`));
174
+ }
175
+ // Throw errors
176
+ if (errors.length > 0) {
177
+ throw new Error(`Auth configuration validation failed:\n${errors.map(e => ` - ${e}`).join('\n')}`);
178
+ }
179
+ }
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server-config.js","sourceRoot":"","sources":["../../src/config/server-config.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 { Algorithm } from 'jsonwebtoken';\nimport { CoreConstants } from './app-config';\nimport type { ServerConfig, AuthConfig, RateLimitConfig } from './config-types';\n\nconst log = createLogger('ServerConfig');\n\n/**\n * Load server configuration from environment variables.\n *\n * Environment variables:\n * - `PORT` — HTTP listen port (default: `3000`)\n * - `CORS_ORIGIN` — Comma-separated allowed origins (default: `PLATFORM_BASE_URL`)\n * - `CORS_CREDENTIALS` — Allow credentials; set to `'false'` to disable (default: `true`)\n * - `TRUST_PROXY` — Express trust proxy hops (default: `1`)\n * - `PLATFORM_BASE_URL` — Frontend URL used as CORS fallback (default: `'https://localhost:8443'`)\n *\n * @returns Server configuration with port, CORS, trust proxy, and platform URL\n */\nexport function loadServerConfig(): ServerConfig {\n  return {\n    port: parseInt(process.env.PORT || '3000', 10),\n    cors: {\n      credentials: process.env.CORS_CREDENTIALS !== 'false',\n      origin: process.env.CORS_ORIGIN\n        ? process.env.CORS_ORIGIN.split(',').map(o => o.trim())\n        : [process.env.PLATFORM_BASE_URL || CoreConstants.DEFAULT_PLATFORM_URL],\n    },\n    trustProxy: parseInt(process.env.TRUST_PROXY || '1', 10),\n    platformUrl: process.env.PLATFORM_BASE_URL || CoreConstants.DEFAULT_PLATFORM_URL,\n\n    httpClient: {\n      timeout: parseInt(process.env.HTTP_CLIENT_TIMEOUT || '5000', 10),\n      maxRetries: parseInt(process.env.HTTP_CLIENT_MAX_RETRIES || '2', 10),\n      retryDelayMs: parseInt(process.env.HTTP_CLIENT_RETRY_DELAY_MS || '200', 10),\n    },\n\n    sse: {\n      maxClientsPerRequest: parseInt(process.env.SSE_MAX_CLIENTS_PER_REQUEST || '10', 10),\n      clientTimeoutMs: parseInt(process.env.SSE_CLIENT_TIMEOUT_MS || '1800000', 10),\n      cleanupIntervalMs: parseInt(process.env.SSE_CLEANUP_INTERVAL_MS || '300000', 10),\n    },\n\n    services: {\n      pluginHost: process.env.PLUGIN_SERVICE_HOST || 'plugin',\n      pluginPort: parseInt(process.env.PLUGIN_SERVICE_PORT || '3000', 10),\n      pipelineHost: process.env.PIPELINE_SERVICE_HOST || 'pipeline',\n      pipelinePort: parseInt(process.env.PIPELINE_SERVICE_PORT || '3000', 10),\n      messageHost: process.env.MESSAGE_SERVICE_HOST || 'message',\n      messagePort: parseInt(process.env.MESSAGE_SERVICE_PORT || '3000', 10),\n      complianceHost: process.env.COMPLIANCE_SERVICE_HOST || 'compliance',\n      compliancePort: parseInt(process.env.COMPLIANCE_SERVICE_PORT || '3000', 10),\n      billingHost: process.env.BILLING_SERVICE_HOST || 'billing',\n      billingPort: parseInt(process.env.BILLING_SERVICE_PORT || '3000', 10),\n      billingTimeout: parseInt(process.env.BILLING_SERVICE_TIMEOUT || '5000', 10),\n    },\n  };\n}\n\n/**\n * Load authentication configuration from environment variables.\n *\n * Environment variables:\n * - `JWT_SECRET` — **Required.** Secret key for signing JWTs\n * - `REFRESH_TOKEN_SECRET` — **Required.** Secret key for signing refresh tokens\n * - `JWT_EXPIRES_IN` — JWT lifetime in seconds (default: `7200` = 2 hours)\n * - `JWT_ALGORITHM` — JWT signing algorithm (default: `'HS256'`)\n * - `JWT_SALT_ROUNDS` — bcrypt salt rounds for password hashing (default: `12`)\n * - `REFRESH_TOKEN_EXPIRES_IN` — Refresh token lifetime in seconds (default: `2592000` = 30 days)\n *\n * @returns Authentication configuration with safe defaults (empty strings when env vars are unset).\n * Call {@link validateAuthConfig} at server startup to enforce required secrets.\n */\nexport function loadAuthConfig(): AuthConfig {\n  return {\n    jwt: {\n      secret: process.env.JWT_SECRET ?? '',\n      expiresIn: parseInt(process.env.JWT_EXPIRES_IN || '7200', 10),\n      algorithm: (process.env.JWT_ALGORITHM || 'HS256') as Algorithm,\n      saltRounds: parseInt(process.env.JWT_SALT_ROUNDS || '12', 10),\n    },\n    refreshToken: {\n      secret: process.env.REFRESH_TOKEN_SECRET || '',\n      expiresIn: parseInt(process.env.REFRESH_TOKEN_EXPIRES_IN || '2592000', 10),\n    },\n  };\n}\n\n/**\n * Load rate limiting configuration from environment variables.\n *\n * Environment variables:\n * - `LIMITER_MAX` — Max requests per window (default: `100`)\n * - `LIMITER_WINDOWMS` — Rate limit window in ms (default: `900000` = 15 minutes)\n *\n * @returns Rate limit configuration\n */\nexport function loadRateLimitConfig(): RateLimitConfig {\n  return {\n    max: parseInt(process.env.LIMITER_MAX || '100', 10),\n    windowMs: parseInt(process.env.LIMITER_WINDOWMS || '900000', 10),\n    legacyHeaders: false,\n    standardHeaders: true,\n  };\n}\n\n/**\n * Validate server configuration and log warnings for insecure settings.\n *\n * @param config - Server configuration to validate\n */\nexport function validateServerConfig(config: ServerConfig): void {\n  const warnings: string[] = [];\n\n  // Check CORS configuration\n  const origin = config.cors.origin;\n  const isWildcard = origin === '*' || (Array.isArray(origin) && origin.includes('*'));\n  if (isWildcard) {\n    warnings.push('CORS origin set to wildcard (*) - consider restricting to specific domains');\n  }\n  if (isWildcard && config.cors.credentials) {\n    // Browsers reject this combination, but it indicates a misconfiguration\n    throw new Error(\n      'SECURITY ERROR: CORS_ORIGIN=* with CORS_CREDENTIALS=true is an invalid and insecure configuration. ' +\n      'Set CORS_ORIGIN to specific domains or disable CORS_CREDENTIALS.',\n    );\n  }\n\n  // Check platform URL uses HTTPS\n  if (config.platformUrl.startsWith('http://') &&\n    !config.platformUrl.includes('localhost')) {\n    warnings.push('Platform URL uses HTTP instead of HTTPS - insecure for production');\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    log.warn('Server configuration warnings:');\n    warnings.forEach(warning => log.warn(`  - ${warning}`));\n  }\n}\n\nconst INSECURE_PATTERNS = ['secret', 'password', 'changeme', 'default', '123456', 'admin'];\n\nfunction isInsecureSecret(secret: string): boolean {\n  const lower = secret.toLowerCase();\n  return INSECURE_PATTERNS.some(s => lower === s || (lower.length < 64 && lower.includes(s)));\n}\n\n/**\n * Validate authentication configuration (JWT secrets, algorithms, expiration).\n * Call this at server startup, not during CDK synthesis.\n *\n * @param config - Auth configuration to validate\n * @throws {Error} If secrets are insecure, too short (<32 chars), or use disallowed algorithms\n */\nexport function validateAuthConfig(config: AuthConfig): void {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n\n  // Validate both secrets with the same checks\n  for (const [label, secret] of [['JWT', config.jwt.secret], ['Refresh token', config.refreshToken.secret]] as const) {\n    if (secret.length < 32) {\n      errors.push(`${label} secret should be at least 32 characters long`);\n    }\n    if (isInsecureSecret(secret)) {\n      errors.push(`${label} secret appears to be insecure or default value`);\n    }\n  }\n\n  // Check JWT expiration times\n  if (config.jwt.expiresIn > 86400) {\n    errors.push('JWT expiration must not exceed 24 hours (86400 seconds)');\n  } else if (config.jwt.expiresIn > 7200) {\n    warnings.push('JWT expiration time is greater than 2 hours - shorter expiration recommended');\n  }\n\n  // Check algorithm\n  if (!CoreConstants.ALLOWED_JWT_ALGORITHMS.includes(config.jwt.algorithm)) {\n    errors.push(`JWT algorithm ${config.jwt.algorithm} is not in the allowed list`);\n  }\n\n  // Display warnings\n  if (warnings.length > 0) {\n    log.warn('Auth configuration warnings:');\n    warnings.forEach(warning => log.warn(`  - ${warning}`));\n  }\n\n  // Throw errors\n  if (errors.length > 0) {\n    throw new Error(\n      `Auth configuration validation failed:\\n${errors.map(e => `  - ${e}`).join('\\n')}`,\n    );\n  }\n}\n"]}
@@ -0,0 +1,62 @@
1
+ import { CodeBuildStep, FileSet, ShellStep } from 'aws-cdk-lib/pipelines';
2
+ export interface ArtifactKey {
3
+ readonly stageName: string;
4
+ readonly stageAlias: string;
5
+ readonly pluginName: string;
6
+ readonly pluginAlias: string;
7
+ readonly outputDirectory: string;
8
+ }
9
+ /**
10
+ * Manages build step artifacts with hierarchical key-based lookup.
11
+ * Keys follow the pattern: stageName:stageAlias:pluginName:pluginAlias:outputDirectory
12
+ *
13
+ * @example
14
+ * ```typescript
15
+ * // Synth step: "no-stage:no-stage-alias:cdk-synth:cdk-synth-alias:cdk.out"
16
+ * // Build step: "build:build-alias:nodejs-build:nodejs-build-alias:dist"
17
+ * ```
18
+ */
19
+ export declare class ArtifactManager {
20
+ private readonly artifacts;
21
+ /**
22
+ * Generate a key string from artifact parameters
23
+ */
24
+ private generateKey;
25
+ /**
26
+ * Register a build step artifact
27
+ * @param key - The hierarchical key identifying this artifact (includes output directory)
28
+ * @param step - The CodeBuildStep or ShellStep to store
29
+ */
30
+ add(key: ArtifactKey, step: CodeBuildStep | ShellStep): void;
31
+ /**
32
+ * Get a build step by its artifact key
33
+ * @param key - The hierarchical key identifying the artifact
34
+ * @returns The stored step, or undefined if not found
35
+ */
36
+ get(key: ArtifactKey): CodeBuildStep | ShellStep | undefined;
37
+ /**
38
+ * Get the primary output FileSet from a registered step.
39
+ * @param key - The artifact key identifying the step
40
+ * @returns The primary output FileSet
41
+ * @throws Error if the step is not found or has no primary output
42
+ */
43
+ getOutput(key: ArtifactKey): FileSet;
44
+ /**
45
+ * Register an additional output directory on a stored step and return its FileSet.
46
+ * Calls CDK's addOutputDirectory() to create a named output beyond the primary.
47
+ * @param key - The artifact key identifying the step
48
+ * @param directory - The additional output directory path to register
49
+ * @returns The FileSet for the additional output directory
50
+ * @throws Error if the step is not found
51
+ */
52
+ addOutput(key: ArtifactKey, directory: string): FileSet;
53
+ /**
54
+ * Get a step by key or throw with a helpful error listing available artifacts.
55
+ */
56
+ private require;
57
+ /**
58
+ * List all artifact keys
59
+ * @returns Array of all stored artifact key strings
60
+ */
61
+ list(): string[];
62
+ }
@@ -0,0 +1,86 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.ArtifactManager = void 0;
6
+ /**
7
+ * Manages build step artifacts with hierarchical key-based lookup.
8
+ * Keys follow the pattern: stageName:stageAlias:pluginName:pluginAlias:outputDirectory
9
+ *
10
+ * @example
11
+ * ```typescript
12
+ * // Synth step: "no-stage:no-stage-alias:cdk-synth:cdk-synth-alias:cdk.out"
13
+ * // Build step: "build:build-alias:nodejs-build:nodejs-build-alias:dist"
14
+ * ```
15
+ */
16
+ class ArtifactManager {
17
+ artifacts = new Map();
18
+ /**
19
+ * Generate a key string from artifact parameters
20
+ */
21
+ generateKey(key) {
22
+ return `${key.stageName}:${key.stageAlias}:${key.pluginName}:${key.pluginAlias}:${key.outputDirectory}`;
23
+ }
24
+ /**
25
+ * Register a build step artifact
26
+ * @param key - The hierarchical key identifying this artifact (includes output directory)
27
+ * @param step - The CodeBuildStep or ShellStep to store
28
+ */
29
+ add(key, step) {
30
+ this.artifacts.set(this.generateKey(key), step);
31
+ }
32
+ /**
33
+ * Get a build step by its artifact key
34
+ * @param key - The hierarchical key identifying the artifact
35
+ * @returns The stored step, or undefined if not found
36
+ */
37
+ get(key) {
38
+ return this.artifacts.get(this.generateKey(key));
39
+ }
40
+ /**
41
+ * Get the primary output FileSet from a registered step.
42
+ * @param key - The artifact key identifying the step
43
+ * @returns The primary output FileSet
44
+ * @throws Error if the step is not found or has no primary output
45
+ */
46
+ getOutput(key) {
47
+ const step = this.require(key);
48
+ const output = step.primaryOutput;
49
+ if (!output) {
50
+ throw new Error(`Step '${key.pluginName}' has no primary output`);
51
+ }
52
+ return output;
53
+ }
54
+ /**
55
+ * Register an additional output directory on a stored step and return its FileSet.
56
+ * Calls CDK's addOutputDirectory() to create a named output beyond the primary.
57
+ * @param key - The artifact key identifying the step
58
+ * @param directory - The additional output directory path to register
59
+ * @returns The FileSet for the additional output directory
60
+ * @throws Error if the step is not found
61
+ */
62
+ addOutput(key, directory) {
63
+ return this.require(key).addOutputDirectory(directory);
64
+ }
65
+ /**
66
+ * Get a step by key or throw with a helpful error listing available artifacts.
67
+ */
68
+ require(key) {
69
+ const step = this.get(key);
70
+ if (!step) {
71
+ const registered = this.list();
72
+ const available = registered.length > 0 ? ` Available: [${registered.join(', ')}]` : ' No artifacts registered.';
73
+ throw new Error(`No artifact registered for "${this.generateKey(key)}".${available}`);
74
+ }
75
+ return step;
76
+ }
77
+ /**
78
+ * List all artifact keys
79
+ * @returns Array of all stored artifact key strings
80
+ */
81
+ list() {
82
+ return Array.from(this.artifacts.keys());
83
+ }
84
+ }
85
+ exports.ArtifactManager = ArtifactManager;
86
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXJ0aWZhY3QtbWFuYWdlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb3JlL2FydGlmYWN0LW1hbmFnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7OztBQVl0Qzs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFhLGVBQWU7SUFDVCxTQUFTLEdBQTJDLElBQUksR0FBRyxFQUFFLENBQUM7SUFFL0U7O09BRUc7SUFDSyxXQUFXLENBQUMsR0FBZ0I7UUFDbEMsT0FBTyxHQUFHLEdBQUcsQ0FBQyxTQUFTLElBQUksR0FBRyxDQUFDLFVBQVUsSUFBSSxHQUFHLENBQUMsVUFBVSxJQUFJLEdBQUcsQ0FBQyxXQUFXLElBQUksR0FBRyxDQUFDLGVBQWUsRUFBRSxDQUFDO0lBQzFHLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsR0FBRyxDQUFDLEdBQWdCLEVBQUUsSUFBK0I7UUFDbkQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILEdBQUcsQ0FBQyxHQUFnQjtRQUNsQixPQUFPLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztJQUNuRCxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSCxTQUFTLENBQUMsR0FBZ0I7UUFDeEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUMvQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDO1FBQ2xDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNaLE1BQU0sSUFBSSxLQUFLLENBQUMsU0FBUyxHQUFHLENBQUMsVUFBVSx5QkFBeUIsQ0FBQyxDQUFDO1FBQ3BFLENBQUM7UUFDRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRUQ7Ozs7Ozs7T0FPRztJQUNILFNBQVMsQ0FBQyxHQUFnQixFQUFFLFNBQWlCO1FBQzNDLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQyxrQkFBa0IsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUN6RCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxPQUFPLENBQUMsR0FBZ0I7UUFDOUIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUMzQixJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDVixNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDL0IsTUFBTSxTQUFTLEdBQUcsVUFBVSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLGdCQUFnQixVQUFVLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLDJCQUEyQixDQUFDO1lBQ2pILE1BQU0sSUFBSSxLQUFLLENBQUMsK0JBQStCLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEtBQUssU0FBUyxFQUFFLENBQUMsQ0FBQztRQUN4RixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsSUFBSTtRQUNGLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7SUFDM0MsQ0FBQztDQUNGO0FBM0VELDBDQTJFQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgeyBDb2RlQnVpbGRTdGVwLCBGaWxlU2V0LCBTaGVsbFN0ZXAgfSBmcm9tICdhd3MtY2RrLWxpYi9waXBlbGluZXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEFydGlmYWN0S2V5IHtcbiAgcmVhZG9ubHkgc3RhZ2VOYW1lOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHN0YWdlQWxpYXM6IHN0cmluZztcbiAgcmVhZG9ubHkgcGx1Z2luTmFtZTogc3RyaW5nO1xuICByZWFkb25seSBwbHVnaW5BbGlhczogc3RyaW5nO1xuICByZWFkb25seSBvdXRwdXREaXJlY3Rvcnk6IHN0cmluZztcbn1cblxuLyoqXG4gKiBNYW5hZ2VzIGJ1aWxkIHN0ZXAgYXJ0aWZhY3RzIHdpdGggaGllcmFyY2hpY2FsIGtleS1iYXNlZCBsb29rdXAuXG4gKiBLZXlzIGZvbGxvdyB0aGUgcGF0dGVybjogc3RhZ2VOYW1lOnN0YWdlQWxpYXM6cGx1Z2luTmFtZTpwbHVnaW5BbGlhczpvdXRwdXREaXJlY3RvcnlcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gU3ludGggc3RlcDogXCJuby1zdGFnZTpuby1zdGFnZS1hbGlhczpjZGstc3ludGg6Y2RrLXN5bnRoLWFsaWFzOmNkay5vdXRcIlxuICogLy8gQnVpbGQgc3RlcDogXCJidWlsZDpidWlsZC1hbGlhczpub2RlanMtYnVpbGQ6bm9kZWpzLWJ1aWxkLWFsaWFzOmRpc3RcIlxuICogYGBgXG4gKi9cbmV4cG9ydCBjbGFzcyBBcnRpZmFjdE1hbmFnZXIge1xuICBwcml2YXRlIHJlYWRvbmx5IGFydGlmYWN0czogTWFwPHN0cmluZywgQ29kZUJ1aWxkU3RlcCB8IFNoZWxsU3RlcD4gPSBuZXcgTWFwKCk7XG5cbiAgLyoqXG4gICAqIEdlbmVyYXRlIGEga2V5IHN0cmluZyBmcm9tIGFydGlmYWN0IHBhcmFtZXRlcnNcbiAgICovXG4gIHByaXZhdGUgZ2VuZXJhdGVLZXkoa2V5OiBBcnRpZmFjdEtleSk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGAke2tleS5zdGFnZU5hbWV9OiR7a2V5LnN0YWdlQWxpYXN9OiR7a2V5LnBsdWdpbk5hbWV9OiR7a2V5LnBsdWdpbkFsaWFzfToke2tleS5vdXRwdXREaXJlY3Rvcnl9YDtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlciBhIGJ1aWxkIHN0ZXAgYXJ0aWZhY3RcbiAgICogQHBhcmFtIGtleSAtIFRoZSBoaWVyYXJjaGljYWwga2V5IGlkZW50aWZ5aW5nIHRoaXMgYXJ0aWZhY3QgKGluY2x1ZGVzIG91dHB1dCBkaXJlY3RvcnkpXG4gICAqIEBwYXJhbSBzdGVwIC0gVGhlIENvZGVCdWlsZFN0ZXAgb3IgU2hlbGxTdGVwIHRvIHN0b3JlXG4gICAqL1xuICBhZGQoa2V5OiBBcnRpZmFjdEtleSwgc3RlcDogQ29kZUJ1aWxkU3RlcCB8IFNoZWxsU3RlcCk6IHZvaWQge1xuICAgIHRoaXMuYXJ0aWZhY3RzLnNldCh0aGlzLmdlbmVyYXRlS2V5KGtleSksIHN0ZXApO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCBhIGJ1aWxkIHN0ZXAgYnkgaXRzIGFydGlmYWN0IGtleVxuICAgKiBAcGFyYW0ga2V5IC0gVGhlIGhpZXJhcmNoaWNhbCBrZXkgaWRlbnRpZnlpbmcgdGhlIGFydGlmYWN0XG4gICAqIEByZXR1cm5zIFRoZSBzdG9yZWQgc3RlcCwgb3IgdW5kZWZpbmVkIGlmIG5vdCBmb3VuZFxuICAgKi9cbiAgZ2V0KGtleTogQXJ0aWZhY3RLZXkpOiBDb2RlQnVpbGRTdGVwIHwgU2hlbGxTdGVwIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gdGhpcy5hcnRpZmFjdHMuZ2V0KHRoaXMuZ2VuZXJhdGVLZXkoa2V5KSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IHRoZSBwcmltYXJ5IG91dHB1dCBGaWxlU2V0IGZyb20gYSByZWdpc3RlcmVkIHN0ZXAuXG4gICAqIEBwYXJhbSBrZXkgLSBUaGUgYXJ0aWZhY3Qga2V5IGlkZW50aWZ5aW5nIHRoZSBzdGVwXG4gICAqIEByZXR1cm5zIFRoZSBwcmltYXJ5IG91dHB1dCBGaWxlU2V0XG4gICAqIEB0aHJvd3MgRXJyb3IgaWYgdGhlIHN0ZXAgaXMgbm90IGZvdW5kIG9yIGhhcyBubyBwcmltYXJ5IG91dHB1dFxuICAgKi9cbiAgZ2V0T3V0cHV0KGtleTogQXJ0aWZhY3RLZXkpOiBGaWxlU2V0IHtcbiAgICBjb25zdCBzdGVwID0gdGhpcy5yZXF1aXJlKGtleSk7XG4gICAgY29uc3Qgb3V0cHV0ID0gc3RlcC5wcmltYXJ5T3V0cHV0O1xuICAgIGlmICghb3V0cHV0KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYFN0ZXAgJyR7a2V5LnBsdWdpbk5hbWV9JyBoYXMgbm8gcHJpbWFyeSBvdXRwdXRgKTtcbiAgICB9XG4gICAgcmV0dXJuIG91dHB1dDtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZWdpc3RlciBhbiBhZGRpdGlvbmFsIG91dHB1dCBkaXJlY3Rvcnkgb24gYSBzdG9yZWQgc3RlcCBhbmQgcmV0dXJuIGl0cyBGaWxlU2V0LlxuICAgKiBDYWxscyBDREsncyBhZGRPdXRwdXREaXJlY3RvcnkoKSB0byBjcmVhdGUgYSBuYW1lZCBvdXRwdXQgYmV5b25kIHRoZSBwcmltYXJ5LlxuICAgKiBAcGFyYW0ga2V5IC0gVGhlIGFydGlmYWN0IGtleSBpZGVudGlmeWluZyB0aGUgc3RlcFxuICAgKiBAcGFyYW0gZGlyZWN0b3J5IC0gVGhlIGFkZGl0aW9uYWwgb3V0cHV0IGRpcmVjdG9yeSBwYXRoIHRvIHJlZ2lzdGVyXG4gICAqIEByZXR1cm5zIFRoZSBGaWxlU2V0IGZvciB0aGUgYWRkaXRpb25hbCBvdXRwdXQgZGlyZWN0b3J5XG4gICAqIEB0aHJvd3MgRXJyb3IgaWYgdGhlIHN0ZXAgaXMgbm90IGZvdW5kXG4gICAqL1xuICBhZGRPdXRwdXQoa2V5OiBBcnRpZmFjdEtleSwgZGlyZWN0b3J5OiBzdHJpbmcpOiBGaWxlU2V0IHtcbiAgICByZXR1cm4gdGhpcy5yZXF1aXJlKGtleSkuYWRkT3V0cHV0RGlyZWN0b3J5KGRpcmVjdG9yeSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGEgc3RlcCBieSBrZXkgb3IgdGhyb3cgd2l0aCBhIGhlbHBmdWwgZXJyb3IgbGlzdGluZyBhdmFpbGFibGUgYXJ0aWZhY3RzLlxuICAgKi9cbiAgcHJpdmF0ZSByZXF1aXJlKGtleTogQXJ0aWZhY3RLZXkpOiBDb2RlQnVpbGRTdGVwIHwgU2hlbGxTdGVwIHtcbiAgICBjb25zdCBzdGVwID0gdGhpcy5nZXQoa2V5KTtcbiAgICBpZiAoIXN0ZXApIHtcbiAgICAgIGNvbnN0IHJlZ2lzdGVyZWQgPSB0aGlzLmxpc3QoKTtcbiAgICAgIGNvbnN0IGF2YWlsYWJsZSA9IHJlZ2lzdGVyZWQubGVuZ3RoID4gMCA/IGAgQXZhaWxhYmxlOiBbJHtyZWdpc3RlcmVkLmpvaW4oJywgJyl9XWAgOiAnIE5vIGFydGlmYWN0cyByZWdpc3RlcmVkLic7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYE5vIGFydGlmYWN0IHJlZ2lzdGVyZWQgZm9yIFwiJHt0aGlzLmdlbmVyYXRlS2V5KGtleSl9XCIuJHthdmFpbGFibGV9YCk7XG4gICAgfVxuICAgIHJldHVybiBzdGVwO1xuICB9XG5cbiAgLyoqXG4gICAqIExpc3QgYWxsIGFydGlmYWN0IGtleXNcbiAgICogQHJldHVybnMgQXJyYXkgb2YgYWxsIHN0b3JlZCBhcnRpZmFjdCBrZXkgc3RyaW5nc1xuICAgKi9cbiAgbGlzdCgpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIEFycmF5LmZyb20odGhpcy5hcnRpZmFjdHMua2V5cygpKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,26 @@
1
+ /**
2
+ * Generates unique CDK construct IDs by appending auto-incrementing counters to labels.
3
+ * Labels that already end with a numeric counter (e.g., 'cdk:pipeline:1') are returned as-is.
4
+ *
5
+ * @example
6
+ * ```typescript
7
+ * const id = new UniqueId();
8
+ *
9
+ * id.generate('plugin:lookup'); // "plugin:lookup:1"
10
+ * id.generate('cdk:synth'); // "cdk:synth:1"
11
+ * id.generate('plugin:lookup'); // "plugin:lookup:2"
12
+ * id.generate('cdk:pipeline:1'); // "cdk:pipeline:1" (already has counter)
13
+ * ```
14
+ */
15
+ export declare class UniqueId {
16
+ private readonly _counters;
17
+ /**
18
+ * Returns a unique construct ID for the given label.
19
+ * If the label already ends with a numeric counter, it is returned as-is.
20
+ * Otherwise, an auto-incrementing counter is appended.
21
+ *
22
+ * @param label - Colon-separated namespace (e.g., 'plugin:lookup')
23
+ * @returns The label with counter appended, or unchanged if it already has one
24
+ */
25
+ generate(label: string): string;
26
+ }
@@ -0,0 +1,44 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.UniqueId = void 0;
6
+ /**
7
+ * Generates unique CDK construct IDs by appending auto-incrementing counters to labels.
8
+ * Labels that already end with a numeric counter (e.g., 'cdk:pipeline:1') are returned as-is.
9
+ *
10
+ * @example
11
+ * ```typescript
12
+ * const id = new UniqueId();
13
+ *
14
+ * id.generate('plugin:lookup'); // "plugin:lookup:1"
15
+ * id.generate('cdk:synth'); // "cdk:synth:1"
16
+ * id.generate('plugin:lookup'); // "plugin:lookup:2"
17
+ * id.generate('cdk:pipeline:1'); // "cdk:pipeline:1" (already has counter)
18
+ * ```
19
+ */
20
+ class UniqueId {
21
+ _counters = new Map();
22
+ /**
23
+ * Returns a unique construct ID for the given label.
24
+ * If the label already ends with a numeric counter, it is returned as-is.
25
+ * Otherwise, an auto-incrementing counter is appended.
26
+ *
27
+ * @param label - Colon-separated namespace (e.g., 'plugin:lookup')
28
+ * @returns The label with counter appended, or unchanged if it already has one
29
+ */
30
+ generate(label) {
31
+ if (!label || typeof label !== 'string') {
32
+ throw new Error('Label must be a non-empty string');
33
+ }
34
+ // If label already ends with a numeric counter, return as-is
35
+ if (/:\d+$/.test(label)) {
36
+ return label;
37
+ }
38
+ const count = (this._counters.get(label) ?? 0) + 1;
39
+ this._counters.set(label, count);
40
+ return `${label}:${count}`;
41
+ }
42
+ }
43
+ exports.UniqueId = UniqueId;
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWQtZ2VuZXJhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvcmUvaWQtZ2VuZXJhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSwrQ0FBK0M7QUFDL0Msc0NBQXNDOzs7QUFFdEM7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQWEsUUFBUTtJQUNGLFNBQVMsR0FBRyxJQUFJLEdBQUcsRUFBa0IsQ0FBQztJQUV2RDs7Ozs7OztPQU9HO0lBQ0gsUUFBUSxDQUFDLEtBQWE7UUFDcEIsSUFBSSxDQUFDLEtBQUssSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUN4QyxNQUFNLElBQUksS0FBSyxDQUFDLGtDQUFrQyxDQUFDLENBQUM7UUFDdEQsQ0FBQztRQUVELDZEQUE2RDtRQUM3RCxJQUFJLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUN4QixPQUFPLEtBQUssQ0FBQztRQUNmLENBQUM7UUFFRCxNQUFNLEtBQUssR0FBRyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNuRCxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFFakMsT0FBTyxHQUFHLEtBQUssSUFBSSxLQUFLLEVBQUUsQ0FBQztJQUM3QixDQUFDO0NBQ0Y7QUExQkQsNEJBMEJDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IDIwMjYgUGlwZWxpbmUgQnVpbGRlciBDb250cmlidXRvcnNcbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5cbi8qKlxuICogR2VuZXJhdGVzIHVuaXF1ZSBDREsgY29uc3RydWN0IElEcyBieSBhcHBlbmRpbmcgYXV0by1pbmNyZW1lbnRpbmcgY291bnRlcnMgdG8gbGFiZWxzLlxuICogTGFiZWxzIHRoYXQgYWxyZWFkeSBlbmQgd2l0aCBhIG51bWVyaWMgY291bnRlciAoZS5nLiwgJ2NkazpwaXBlbGluZToxJykgYXJlIHJldHVybmVkIGFzLWlzLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBpZCA9IG5ldyBVbmlxdWVJZCgpO1xuICpcbiAqIGlkLmdlbmVyYXRlKCdwbHVnaW46bG9va3VwJyk7ICAgLy8gXCJwbHVnaW46bG9va3VwOjFcIlxuICogaWQuZ2VuZXJhdGUoJ2NkazpzeW50aCcpOyAgICAgICAvLyBcImNkazpzeW50aDoxXCJcbiAqIGlkLmdlbmVyYXRlKCdwbHVnaW46bG9va3VwJyk7ICAgLy8gXCJwbHVnaW46bG9va3VwOjJcIlxuICogaWQuZ2VuZXJhdGUoJ2NkazpwaXBlbGluZToxJyk7IC8vIFwiY2RrOnBpcGVsaW5lOjFcIiAoYWxyZWFkeSBoYXMgY291bnRlcilcbiAqIGBgYFxuICovXG5leHBvcnQgY2xhc3MgVW5pcXVlSWQge1xuICBwcml2YXRlIHJlYWRvbmx5IF9jb3VudGVycyA9IG5ldyBNYXA8c3RyaW5nLCBudW1iZXI+KCk7XG5cbiAgLyoqXG4gICAqIFJldHVybnMgYSB1bmlxdWUgY29uc3RydWN0IElEIGZvciB0aGUgZ2l2ZW4gbGFiZWwuXG4gICAqIElmIHRoZSBsYWJlbCBhbHJlYWR5IGVuZHMgd2l0aCBhIG51bWVyaWMgY291bnRlciwgaXQgaXMgcmV0dXJuZWQgYXMtaXMuXG4gICAqIE90aGVyd2lzZSwgYW4gYXV0by1pbmNyZW1lbnRpbmcgY291bnRlciBpcyBhcHBlbmRlZC5cbiAgICpcbiAgICogQHBhcmFtIGxhYmVsIC0gQ29sb24tc2VwYXJhdGVkIG5hbWVzcGFjZSAoZS5nLiwgJ3BsdWdpbjpsb29rdXAnKVxuICAgKiBAcmV0dXJucyBUaGUgbGFiZWwgd2l0aCBjb3VudGVyIGFwcGVuZGVkLCBvciB1bmNoYW5nZWQgaWYgaXQgYWxyZWFkeSBoYXMgb25lXG4gICAqL1xuICBnZW5lcmF0ZShsYWJlbDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBpZiAoIWxhYmVsIHx8IHR5cGVvZiBsYWJlbCAhPT0gJ3N0cmluZycpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignTGFiZWwgbXVzdCBiZSBhIG5vbi1lbXB0eSBzdHJpbmcnKTtcbiAgICB9XG5cbiAgICAvLyBJZiBsYWJlbCBhbHJlYWR5IGVuZHMgd2l0aCBhIG51bWVyaWMgY291bnRlciwgcmV0dXJuIGFzLWlzXG4gICAgaWYgKC86XFxkKyQvLnRlc3QobGFiZWwpKSB7XG4gICAgICByZXR1cm4gbGFiZWw7XG4gICAgfVxuXG4gICAgY29uc3QgY291bnQgPSAodGhpcy5fY291bnRlcnMuZ2V0KGxhYmVsKSA/PyAwKSArIDE7XG4gICAgdGhpcy5fY291bnRlcnMuc2V0KGxhYmVsLCBjb3VudCk7XG5cbiAgICByZXR1cm4gYCR7bGFiZWx9OiR7Y291bnR9YDtcbiAgfVxufVxuIl19
@@ -0,0 +1,13 @@
1
+ import type { BuildEnvironment } from 'aws-cdk-lib/aws-codebuild';
2
+ import type { CodeBuildStepProps, CodePipelineProps, ShellStepProps } from 'aws-cdk-lib/pipelines';
3
+ import type { MetaDataType } from './pipeline-types';
4
+ /** Extract CDK construct config from metadata for a given namespace. */
5
+ export declare function buildConfigFromMetadata(metadata: MetaDataType, namespace: string): Record<string, unknown>;
6
+ /** Extract CodePipeline config from metadata. */
7
+ export declare function metadataForCodePipeline(metadata: MetaDataType): Partial<CodePipelineProps>;
8
+ /** Extract CodeBuildStep config from metadata. */
9
+ export declare function metadataForCodeBuildStep(metadata: MetaDataType): Partial<CodeBuildStepProps>;
10
+ /** Extract ShellStep config from metadata. */
11
+ export declare function metadataForShellStep(metadata: MetaDataType): Partial<ShellStepProps>;
12
+ /** Extract BuildEnvironment config from metadata. */
13
+ export declare function metadataForBuildEnvironment(metadata: MetaDataType): Partial<BuildEnvironment>;
@@ -0,0 +1,81 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.buildConfigFromMetadata = buildConfigFromMetadata;
6
+ exports.metadataForCodePipeline = metadataForCodePipeline;
7
+ exports.metadataForCodeBuildStep = metadataForCodeBuildStep;
8
+ exports.metadataForShellStep = metadataForShellStep;
9
+ exports.metadataForBuildEnvironment = metadataForBuildEnvironment;
10
+ const pipeline_types_1 = require("./pipeline-types");
11
+ /**
12
+ * Type-safe namespace constants for metadata configuration.
13
+ */
14
+ const NAMESPACE = {
15
+ SHELL_STEP: 'pipelines:shellstep',
16
+ CODE_BUILD_STEP: 'pipelines:codebuildstep',
17
+ BUILD_ENVIRONMENT: 'codebuild:buildenvironment',
18
+ CODE_PIPELINE: 'pipelines:codepipeline',
19
+ };
20
+ const NAMESPACE_KEY_MAP = {
21
+ [NAMESPACE.SHELL_STEP]: {
22
+ booleanKeys: [],
23
+ passthroughKeys: ['additionalInputs', 'commands', 'env', 'envFromCfnOutputs', 'input', 'installCommands', 'primaryOutputDirectory'],
24
+ },
25
+ [NAMESPACE.CODE_BUILD_STEP]: {
26
+ booleanKeys: [],
27
+ passthroughKeys: ['actionRole', 'additionalInputs', 'buildEnvironment', 'cache', 'commands', 'env', 'envFromCfnOutputs', 'fileSystemLocations', 'input', 'installCommands', 'logging', 'partialBuildSpec', 'primaryOutputDirectory', 'projectName', 'role', 'rolePolicyStatements', 'timeout'],
28
+ },
29
+ [NAMESPACE.BUILD_ENVIRONMENT]: {
30
+ booleanKeys: ['privileged'],
31
+ passthroughKeys: ['buildImage', 'certificate', 'computeType', 'dockerServer', 'environmentVariables', 'fleet'],
32
+ },
33
+ [NAMESPACE.CODE_PIPELINE]: {
34
+ booleanKeys: ['crossAccountKeys', 'dockerEnabledForSelfMutation', 'dockerEnabledForSynth', 'enableKeyRotation', 'publishAssetsInParallel', 'reuseCrossRegionSupportStacks', 'selfMutation', 'useChangeSets', 'usePipelineRoleForActions'],
35
+ passthroughKeys: ['artifactBucket', 'assetPublishingCodeBuildDefaults', 'cdkAssetsCliVersion', 'cliVersion', 'codeBuildDefaults', 'codePipeline', 'crossRegionReplicationBuckets', 'dockerCredentials', 'pipelineName', 'pipelineType', 'role', 'selfMutationCodeBuildDefaults', 'synth', 'synthCodeBuildDefaults'],
36
+ },
37
+ };
38
+ const EMPTY_KEY_CONFIG = { booleanKeys: [], passthroughKeys: [] };
39
+ function getCustomKey(prefix, key) {
40
+ return `${pipeline_types_1.CDK_METADATA_PREFIX}${prefix}:${key}`.toLowerCase();
41
+ }
42
+ function isTrue(value) {
43
+ if (typeof value === 'boolean')
44
+ return value;
45
+ if (typeof value === 'string')
46
+ return value.toLowerCase() === 'true';
47
+ return false;
48
+ }
49
+ /** Extract CDK construct config from metadata for a given namespace. */
50
+ function buildConfigFromMetadata(metadata, namespace) {
51
+ const { booleanKeys, passthroughKeys } = NAMESPACE_KEY_MAP[namespace] ?? EMPTY_KEY_CONFIG;
52
+ const result = {};
53
+ for (const key of booleanKeys) {
54
+ const raw = metadata[getCustomKey(namespace, key)];
55
+ if (raw !== undefined)
56
+ result[key] = isTrue(raw);
57
+ }
58
+ for (const key of passthroughKeys) {
59
+ const raw = metadata[getCustomKey(namespace, key)];
60
+ if (raw !== undefined)
61
+ result[key] = raw;
62
+ }
63
+ return result;
64
+ }
65
+ /** Extract CodePipeline config from metadata. */
66
+ function metadataForCodePipeline(metadata) {
67
+ return buildConfigFromMetadata(metadata, NAMESPACE.CODE_PIPELINE);
68
+ }
69
+ /** Extract CodeBuildStep config from metadata. */
70
+ function metadataForCodeBuildStep(metadata) {
71
+ return buildConfigFromMetadata(metadata, NAMESPACE.CODE_BUILD_STEP);
72
+ }
73
+ /** Extract ShellStep config from metadata. */
74
+ function metadataForShellStep(metadata) {
75
+ return buildConfigFromMetadata(metadata, NAMESPACE.SHELL_STEP);
76
+ }
77
+ /** Extract BuildEnvironment config from metadata. */
78
+ function metadataForBuildEnvironment(metadata) {
79
+ return buildConfigFromMetadata(metadata, NAMESPACE.BUILD_ENVIRONMENT);
80
+ }
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metadata-builder.js","sourceRoot":"","sources":["../../src/core/metadata-builder.ts"],"names":[],"mappings":";AAAA,+CAA+C;AAC/C,sCAAsC;;AAuDtC,0DAoBC;AAGD,0DAEC;AAGD,4DAEC;AAGD,oDAEC;AAGD,kEAEC;AA3FD,qDAAuD;AAGvD;;GAEG;AACH,MAAM,SAAS,GAAG;IAChB,UAAU,EAAE,qBAAqB;IACjC,eAAe,EAAE,yBAAyB;IAC1C,iBAAiB,EAAE,4BAA4B;IAC/C,aAAa,EAAE,wBAAwB;CAC/B,CAAC;AAQX,MAAM,iBAAiB,GAA0C;IAC/D,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE;QACtB,WAAW,EAAE,EAAE;QACf,eAAe,EAAE,CAAC,kBAAkB,EAAE,UAAU,EAAE,KAAK,EAAE,mBAAmB,EAAE,OAAO,EAAE,iBAAiB,EAAE,wBAAwB,CAAC;KACpI;IACD,CAAC,SAAS,CAAC,eAAe,CAAC,EAAE;QAC3B,WAAW,EAAE,EAAE;QACf,eAAe,EAAE,CAAC,YAAY,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,mBAAmB,EAAE,qBAAqB,EAAE,OAAO,EAAE,iBAAiB,EAAE,SAAS,EAAE,kBAAkB,EAAE,wBAAwB,EAAE,aAAa,EAAE,MAAM,EAAE,sBAAsB,EAAE,SAAS,CAAC;KAC/R;IACD,CAAC,SAAS,CAAC,iBAAiB,CAAC,EAAE;QAC7B,WAAW,EAAE,CAAC,YAAY,CAAC;QAC3B,eAAe,EAAE,CAAC,YAAY,EAAE,aAAa,EAAE,aAAa,EAAE,cAAc,EAAE,sBAAsB,EAAE,OAAO,CAAC;KAC/G;IACD,CAAC,SAAS,CAAC,aAAa,CAAC,EAAE;QACzB,WAAW,EAAE,CAAC,kBAAkB,EAAE,8BAA8B,EAAE,uBAAuB,EAAE,mBAAmB,EAAE,yBAAyB,EAAE,+BAA+B,EAAE,cAAc,EAAE,eAAe,EAAE,2BAA2B,CAAC;QACzO,eAAe,EAAE,CAAC,gBAAgB,EAAE,kCAAkC,EAAE,qBAAqB,EAAE,YAAY,EAAE,mBAAmB,EAAE,cAAc,EAAE,+BAA+B,EAAE,mBAAmB,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,EAAE,+BAA+B,EAAE,OAAO,EAAE,wBAAwB,CAAC;KACpT;CACF,CAAC;AAEF,MAAM,gBAAgB,GAAuB,EAAE,WAAW,EAAE,EAAE,EAAE,eAAe,EAAE,EAAE,EAAE,CAAC;AAEtF,SAAS,YAAY,CAAC,MAAc,EAAE,GAAW;IAC/C,OAAO,GAAG,oCAAmB,GAAG,MAAM,IAAI,GAAG,EAAE,CAAC,WAAW,EAAE,CAAC;AAChE,CAAC;AAED,SAAS,MAAM,CAAC,KAAc;IAC5B,IAAI,OAAO,KAAK,KAAK,SAAS;QAAE,OAAO,KAAK,CAAC;IAC7C,IAAI,OAAO,KAAK,KAAK,QAAQ;QAAE,OAAO,KAAK,CAAC,WAAW,EAAE,KAAK,MAAM,CAAC;IACrE,OAAO,KAAK,CAAC;AACf,CAAC;AAED,wEAAwE;AACxE,SAAgB,uBAAuB,CACrC,QAAsB,EACtB,SAAiB;IAEjB,MAAM,EAAE,WAAW,EAAE,eAAe,EAAE,GACpC,iBAAiB,CAAC,SAAsB,CAAC,IAAI,gBAAgB,CAAC;IAEhE,MAAM,MAAM,GAA4B,EAAE,CAAC;IAE3C,KAAK,MAAM,GAAG,IAAI,WAAW,EAAE,CAAC;QAC9B,MAAM,GAAG,GAAG,QAAQ,CAAC,YAAY,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,CAAC;QACnD,IAAI,GAAG,KAAK,SAAS;YAAE,MAAM,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;IACnD,CAAC;IAED,KAAK,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QAClC,MAAM,GAAG,GAAG,QAAQ,CAAC,YAAY,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,CAAC;QACnD,IAAI,GAAG,KAAK,SAAS;YAAE,MAAM,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;IAC3C,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,iDAAiD;AACjD,SAAgB,uBAAuB,CAAC,QAAsB;IAC5D,OAAO,uBAAuB,CAAC,QAAQ,EAAE,SAAS,CAAC,aAAa,CAA+B,CAAC;AAClG,CAAC;AAED,kDAAkD;AAClD,SAAgB,wBAAwB,CAAC,QAAsB;IAC7D,OAAO,uBAAuB,CAAC,QAAQ,EAAE,SAAS,CAAC,eAAe,CAAgC,CAAC;AACrG,CAAC;AAED,8CAA8C;AAC9C,SAAgB,oBAAoB,CAAC,QAAsB;IACzD,OAAO,uBAAuB,CAAC,QAAQ,EAAE,SAAS,CAAC,UAAU,CAA4B,CAAC;AAC5F,CAAC;AAED,qDAAqD;AACrD,SAAgB,2BAA2B,CAAC,QAAsB;IAChE,OAAO,uBAAuB,CAAC,QAAQ,EAAE,SAAS,CAAC,iBAAiB,CAA8B,CAAC;AACrG,CAAC","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport type { BuildEnvironment } from 'aws-cdk-lib/aws-codebuild';\nimport type { CodeBuildStepProps, CodePipelineProps, ShellStepProps } from 'aws-cdk-lib/pipelines';\nimport { CDK_METADATA_PREFIX } from './pipeline-types';\nimport type { MetaDataType } from './pipeline-types';\n\n/**\n * Type-safe namespace constants for metadata configuration.\n */\nconst NAMESPACE = {\n  SHELL_STEP: 'pipelines:shellstep',\n  CODE_BUILD_STEP: 'pipelines:codebuildstep',\n  BUILD_ENVIRONMENT: 'codebuild:buildenvironment',\n  CODE_PIPELINE: 'pipelines:codepipeline',\n} as const;\ntype Namespace = (typeof NAMESPACE)[keyof typeof NAMESPACE];\n\ninterface NamespaceKeyConfig {\n  booleanKeys: readonly string[];\n  passthroughKeys: readonly string[];\n}\n\nconst NAMESPACE_KEY_MAP: Record<Namespace, NamespaceKeyConfig> = {\n  [NAMESPACE.SHELL_STEP]: {\n    booleanKeys: [],\n    passthroughKeys: ['additionalInputs', 'commands', 'env', 'envFromCfnOutputs', 'input', 'installCommands', 'primaryOutputDirectory'],\n  },\n  [NAMESPACE.CODE_BUILD_STEP]: {\n    booleanKeys: [],\n    passthroughKeys: ['actionRole', 'additionalInputs', 'buildEnvironment', 'cache', 'commands', 'env', 'envFromCfnOutputs', 'fileSystemLocations', 'input', 'installCommands', 'logging', 'partialBuildSpec', 'primaryOutputDirectory', 'projectName', 'role', 'rolePolicyStatements', 'timeout'],\n  },\n  [NAMESPACE.BUILD_ENVIRONMENT]: {\n    booleanKeys: ['privileged'],\n    passthroughKeys: ['buildImage', 'certificate', 'computeType', 'dockerServer', 'environmentVariables', 'fleet'],\n  },\n  [NAMESPACE.CODE_PIPELINE]: {\n    booleanKeys: ['crossAccountKeys', 'dockerEnabledForSelfMutation', 'dockerEnabledForSynth', 'enableKeyRotation', 'publishAssetsInParallel', 'reuseCrossRegionSupportStacks', 'selfMutation', 'useChangeSets', 'usePipelineRoleForActions'],\n    passthroughKeys: ['artifactBucket', 'assetPublishingCodeBuildDefaults', 'cdkAssetsCliVersion', 'cliVersion', 'codeBuildDefaults', 'codePipeline', 'crossRegionReplicationBuckets', 'dockerCredentials', 'pipelineName', 'pipelineType', 'role', 'selfMutationCodeBuildDefaults', 'synth', 'synthCodeBuildDefaults'],\n  },\n};\n\nconst EMPTY_KEY_CONFIG: NamespaceKeyConfig = { booleanKeys: [], passthroughKeys: [] };\n\nfunction getCustomKey(prefix: string, key: string): string {\n  return `${CDK_METADATA_PREFIX}${prefix}:${key}`.toLowerCase();\n}\n\nfunction isTrue(value: unknown): boolean {\n  if (typeof value === 'boolean') return value;\n  if (typeof value === 'string') return value.toLowerCase() === 'true';\n  return false;\n}\n\n/** Extract CDK construct config from metadata for a given namespace. */\nexport function buildConfigFromMetadata(\n  metadata: MetaDataType,\n  namespace: string,\n): Record<string, unknown> {\n  const { booleanKeys, passthroughKeys } =\n    NAMESPACE_KEY_MAP[namespace as Namespace] ?? EMPTY_KEY_CONFIG;\n\n  const result: Record<string, unknown> = {};\n\n  for (const key of booleanKeys) {\n    const raw = metadata[getCustomKey(namespace, key)];\n    if (raw !== undefined) result[key] = isTrue(raw);\n  }\n\n  for (const key of passthroughKeys) {\n    const raw = metadata[getCustomKey(namespace, key)];\n    if (raw !== undefined) result[key] = raw;\n  }\n\n  return result;\n}\n\n/** Extract CodePipeline config from metadata. */\nexport function metadataForCodePipeline(metadata: MetaDataType): Partial<CodePipelineProps> {\n  return buildConfigFromMetadata(metadata, NAMESPACE.CODE_PIPELINE) as Partial<CodePipelineProps>;\n}\n\n/** Extract CodeBuildStep config from metadata. */\nexport function metadataForCodeBuildStep(metadata: MetaDataType): Partial<CodeBuildStepProps> {\n  return buildConfigFromMetadata(metadata, NAMESPACE.CODE_BUILD_STEP) as Partial<CodeBuildStepProps>;\n}\n\n/** Extract ShellStep config from metadata. */\nexport function metadataForShellStep(metadata: MetaDataType): Partial<ShellStepProps> {\n  return buildConfigFromMetadata(metadata, NAMESPACE.SHELL_STEP) as Partial<ShellStepProps>;\n}\n\n/** Extract BuildEnvironment config from metadata. */\nexport function metadataForBuildEnvironment(metadata: MetaDataType): Partial<BuildEnvironment> {\n  return buildConfigFromMetadata(metadata, NAMESPACE.BUILD_ENVIRONMENT) as Partial<BuildEnvironment>;\n}\n"]}