@pipeline-builder/pipeline-core 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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,292 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
5
+ if (k2 === undefined) k2 = k;
6
+ var desc = Object.getOwnPropertyDescriptor(m, k);
7
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
8
+ desc = { enumerable: true, get: function() { return m[k]; } };
9
+ }
10
+ Object.defineProperty(o, k2, desc);
11
+ }) : (function(o, m, k, k2) {
12
+ if (k2 === undefined) k2 = k;
13
+ o[k2] = m[k];
14
+ }));
15
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
16
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
17
+ }) : function(o, v) {
18
+ o["default"] = v;
19
+ });
20
+ var __importStar = (this && this.__importStar) || (function () {
21
+ var ownKeys = function(o) {
22
+ ownKeys = Object.getOwnPropertyNames || function (o) {
23
+ var ar = [];
24
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
25
+ return ar;
26
+ };
27
+ return ownKeys(o);
28
+ };
29
+ return function (mod) {
30
+ if (mod && mod.__esModule) return mod;
31
+ var result = {};
32
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
33
+ __setModuleDefault(result, mod);
34
+ return result;
35
+ };
36
+ })();
37
+ Object.defineProperty(exports, "__esModule", { value: true });
38
+ exports.PipelineBuilder = void 0;
39
+ const api_core_1 = require("@pipeline-builder/api-core");
40
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
41
+ const cloudwatch = __importStar(require("aws-cdk-lib/aws-cloudwatch"));
42
+ const aws_codepipeline_1 = require("aws-cdk-lib/aws-codepipeline");
43
+ const events = __importStar(require("aws-cdk-lib/aws-events"));
44
+ const targets = __importStar(require("aws-cdk-lib/aws-events-targets"));
45
+ const kms = __importStar(require("aws-cdk-lib/aws-kms"));
46
+ const sns = __importStar(require("aws-cdk-lib/aws-sns"));
47
+ const pipelines_1 = require("aws-cdk-lib/pipelines");
48
+ const constructs_1 = require("constructs");
49
+ const pipeline_configuration_1 = require("./pipeline-configuration");
50
+ const plugin_lookup_1 = require("./plugin-lookup");
51
+ const source_builder_1 = require("./source-builder");
52
+ const stage_builder_1 = require("./stage-builder");
53
+ const app_config_1 = require("../config/app-config");
54
+ const artifact_manager_1 = require("../core/artifact-manager");
55
+ const id_generator_1 = require("../core/id-generator");
56
+ const metadata_builder_1 = require("../core/metadata-builder");
57
+ const network_1 = require("../core/network");
58
+ const pipeline_helpers_1 = require("../core/pipeline-helpers");
59
+ const pipeline_types_1 = require("../core/pipeline-types");
60
+ const role_1 = require("../core/role");
61
+ const security_group_1 = require("../core/security-group");
62
+ const PIPELINE_EVENT_MAP = {
63
+ FAILED: aws_codepipeline_1.PipelineNotificationEvents.PIPELINE_EXECUTION_FAILED,
64
+ SUCCEEDED: aws_codepipeline_1.PipelineNotificationEvents.PIPELINE_EXECUTION_SUCCEEDED,
65
+ STARTED: aws_codepipeline_1.PipelineNotificationEvents.PIPELINE_EXECUTION_STARTED,
66
+ CANCELED: aws_codepipeline_1.PipelineNotificationEvents.PIPELINE_EXECUTION_CANCELED,
67
+ SUPERSEDED: aws_codepipeline_1.PipelineNotificationEvents.PIPELINE_EXECUTION_SUPERSEDED,
68
+ };
69
+ function parseNotificationEvents(value) {
70
+ if (Array.isArray(value))
71
+ return value;
72
+ if (typeof value === 'string')
73
+ return value.split(',').map(s => s.trim());
74
+ return ['FAILED', 'SUCCEEDED'];
75
+ }
76
+ /**
77
+ * CDK construct that creates and configures a CodePipeline for continuous deployment.
78
+ *
79
+ * Features:
80
+ * - Multi-source support (S3, GitHub, CodeStar)
81
+ * - Plugin-based build steps
82
+ * - Metadata-driven configuration
83
+ * - Automatic tagging
84
+ * - Automatic sanitization of project and organization names
85
+ *
86
+ * @example
87
+ * ```typescript
88
+ * new PipelineBuilder(this, 'MyPipeline', {
89
+ * project: 'my-app',
90
+ * organization: 'my-org',
91
+ * synth: {
92
+ * source: {
93
+ * type: 'github',
94
+ * options: { repo: 'owner/repo', branch: 'main' }
95
+ * },
96
+ * plugin: { name: 'synth' }
97
+ * }
98
+ * });
99
+ * ```
100
+ */
101
+ class PipelineBuilder extends constructs_1.Construct {
102
+ pipeline;
103
+ config;
104
+ constructor(scope, id, props) {
105
+ super(scope, id);
106
+ // Use PipelineConfiguration for all business logic (validation, sanitization, metadata merging)
107
+ this.config = new pipeline_configuration_1.PipelineConfiguration(props);
108
+ const serverConfig = app_config_1.Config.get('server');
109
+ const awsConfig = app_config_1.Config.get('aws');
110
+ const uniqueId = new id_generator_1.UniqueId();
111
+ const pluginLookup = new plugin_lookup_1.PluginLookup(this, uniqueId.generate('plugin:lookup'), {
112
+ organization: this.config.organization,
113
+ project: this.config.project,
114
+ platformUrl: serverConfig.platformUrl,
115
+ uniqueId,
116
+ orgId: props.orgId,
117
+ runtime: awsConfig.lambda.runtime,
118
+ timeout: awsConfig.lambda.timeout,
119
+ reservedConcurrentExecutions: awsConfig.lambda.reservedConcurrentExecutions,
120
+ });
121
+ // Create source and build step
122
+ const sourceBuilder = new source_builder_1.SourceBuilder(this, this.config);
123
+ const source = sourceBuilder.create(uniqueId);
124
+ // RESOLVED_SYNTH_PLUGIN=true (CodePipeline): resolve plugin via custom resource Lambda
125
+ // RESOLVED_SYNTH_PLUGIN=false (default/CLI): use fallback with pipeline-manager synth commands
126
+ const plugin = awsConfig.resolvedSynthPlugin
127
+ ? pluginLookup.plugin(this.config.plugin)
128
+ : pluginLookup.fallbackSynth();
129
+ const defaultComputeType = awsConfig.codeBuild.computeType;
130
+ const artifactManager = new artifact_manager_1.ArtifactManager();
131
+ const synthAlias = this.config.plugin.alias ?? this.config.plugin.name;
132
+ const synth = (0, pipeline_helpers_1.createCodeBuildStep)({
133
+ ...this.config.synthCustomization,
134
+ id: uniqueId.generate('cdk:synth'),
135
+ uniqueId,
136
+ plugin,
137
+ input: source,
138
+ metadata: this.config.metadata.merged,
139
+ network: this.config.network,
140
+ scope: this,
141
+ defaultComputeType,
142
+ artifactManager,
143
+ stageName: 'no-stage',
144
+ stageAlias: 'no-stage-alias',
145
+ pluginAlias: `${synthAlias}-alias`,
146
+ orgId: props.orgId,
147
+ });
148
+ // Resolve pipeline-level defaults into codeBuildDefaults
149
+ // Build the per-org platform secret name for CodeBuild env vars
150
+ const platformSecretName = props.orgId
151
+ ? app_config_1.CoreConstants.secretPath(props.orgId, 'platform')
152
+ : undefined;
153
+ const codeBuildDefaults = this.resolveDefaults(this.config.defaults, uniqueId, props.pipelineId, platformSecretName, serverConfig.platformUrl);
154
+ // Resolve IAM role if explicitly provided; otherwise let CDK auto-create
155
+ // the pipeline role with the correct codepipeline.amazonaws.com principal.
156
+ if (props.role?.type === 'codeBuildDefault') {
157
+ (0, api_core_1.createLogger)('PipelineBuilder').warn('codeBuildDefault role type uses codebuild.amazonaws.com trust principal — ' +
158
+ 'this is not suitable as the pipeline-level role. Consider using roleArn/roleName ' +
159
+ 'or omitting the role to let CDK auto-create one with codepipeline.amazonaws.com.');
160
+ }
161
+ const role = props.role
162
+ ? (0, role_1.resolveRole)(this, uniqueId, props.role)
163
+ : undefined;
164
+ // Create CodePipeline construct
165
+ this.pipeline = new pipelines_1.CodePipeline(this, uniqueId.generate('pipelines:codepipeline'), {
166
+ ...(codeBuildDefaults && { codeBuildDefaults }),
167
+ ...(role && { role }),
168
+ pipelineType: aws_codepipeline_1.PipelineType.V2,
169
+ pipelineName: this.config.pipelineName,
170
+ synth,
171
+ ...(0, metadata_builder_1.metadataForCodePipeline)(this.config.metadata.merged),
172
+ });
173
+ // Add stages as waves via StageBuilder
174
+ if (props.stages) {
175
+ const stageBuilder = new stage_builder_1.StageBuilder({
176
+ scope: this,
177
+ pluginLookup,
178
+ uniqueId,
179
+ globalMetadata: this.config.metadata.merged,
180
+ defaultComputeType,
181
+ artifactManager,
182
+ orgId: props.orgId,
183
+ });
184
+ stageBuilder.addStages(this.pipeline, props.stages);
185
+ }
186
+ // ── Tags ──
187
+ aws_cdk_lib_1.Tags.of(this.pipeline).add('project', this.config.project);
188
+ aws_cdk_lib_1.Tags.of(this.pipeline).add('organization', this.config.organization);
189
+ if (props.tags) {
190
+ for (const [key, value] of Object.entries(props.tags)) {
191
+ aws_cdk_lib_1.Tags.of(this.pipeline).add(key, value);
192
+ }
193
+ }
194
+ // Build the internal pipeline before accessing its properties
195
+ this.pipeline.buildPipeline();
196
+ const cdkPipeline = this.pipeline.pipeline;
197
+ const meta = this.config.metadata.merged;
198
+ // ── SNS Notifications ──
199
+ const notificationTopicArn = meta[pipeline_types_1.MetadataKeys.NOTIFICATION_TOPIC_ARN];
200
+ if (typeof notificationTopicArn === 'string') {
201
+ const topic = sns.Topic.fromTopicArn(this, 'NotificationTopic', notificationTopicArn);
202
+ const notificationEvents = parseNotificationEvents(meta[pipeline_types_1.MetadataKeys.NOTIFICATION_EVENTS])
203
+ .map(e => PIPELINE_EVENT_MAP[e.toUpperCase()])
204
+ .filter(Boolean);
205
+ if (notificationEvents.length > 0) {
206
+ cdkPipeline.notifyOn('PipelineNotification', topic, { events: notificationEvents });
207
+ }
208
+ }
209
+ // ── Scheduled Execution ──
210
+ if (props.synth.source.options?.trigger === pipeline_types_1.TriggerType.SCHEDULE || props.schedule) {
211
+ const expr = props.schedule || props.synth.source.options?.schedule || 'rate(1 day)';
212
+ new events.Rule(this, 'ScheduleRule', {
213
+ schedule: events.Schedule.expression(expr),
214
+ targets: [new targets.CodePipeline(cdkPipeline)],
215
+ });
216
+ }
217
+ // ── Execution Event Tracking (forward pipeline state changes to SNS) ──
218
+ if (meta[pipeline_types_1.MetadataKeys.ENABLE_EXECUTION_EVENTS] && typeof notificationTopicArn === 'string') {
219
+ new events.Rule(this, 'ExecutionEventRule', {
220
+ eventPattern: {
221
+ source: ['aws.codepipeline'],
222
+ detailType: ['CodePipeline Pipeline Execution State Change'],
223
+ resources: [cdkPipeline.pipelineArn],
224
+ },
225
+ targets: [new targets.SnsTopic(sns.Topic.fromTopicArn(this, 'ExecutionEventTopic', notificationTopicArn))],
226
+ });
227
+ }
228
+ // ── Artifact Encryption (KMS key) ──
229
+ const kmsKeyArn = this.config.metadata.merged[pipeline_types_1.MetadataKeys.KMS_KEY_ARN];
230
+ if (typeof kmsKeyArn === 'string') {
231
+ const key = kms.Key.fromKeyArn(this, 'ArtifactKey', kmsKeyArn);
232
+ aws_cdk_lib_1.Tags.of(key).add('pipeline', this.config.pipelineName);
233
+ }
234
+ // ── Pipeline Metrics & Alarms ──
235
+ const enableMetrics = this.config.metadata.merged[pipeline_types_1.MetadataKeys.ENABLE_METRICS];
236
+ if (enableMetrics) {
237
+ new cloudwatch.Alarm(this, 'PipelineFailureAlarm', {
238
+ metric: new cloudwatch.Metric({
239
+ namespace: 'AWS/CodePipeline',
240
+ metricName: 'FailedPipelineExecutionCount',
241
+ dimensionsMap: { PipelineName: this.config.pipelineName },
242
+ statistic: 'Sum',
243
+ period: aws_cdk_lib_1.Duration.minutes(5),
244
+ }),
245
+ threshold: 1,
246
+ evaluationPeriods: 1,
247
+ alarmDescription: `Pipeline ${this.config.pipelineName} execution failed`,
248
+ comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
249
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
250
+ });
251
+ }
252
+ }
253
+ /**
254
+ * Resolves CodeBuildDefaults into the shape expected by CDK's codeBuildDefaults.
255
+ * Combines network config, security groups, and pipeline-level environment variables
256
+ * (PIPELINE_ID, EXECUTION_ID, PLATFORM_BASE_URL) available to all CodeBuild actions.
257
+ */
258
+ resolveDefaults(defaults, id, pipelineId, platformSecretName, platformUrl) {
259
+ const networkProps = defaults?.network
260
+ ? (0, network_1.resolveNetwork)(this, id, defaults.network)
261
+ : undefined;
262
+ const standaloneSecurityGroups = defaults?.securityGroups
263
+ ? (0, security_group_1.resolveSecurityGroup)(this, id, defaults.securityGroups)
264
+ : undefined;
265
+ // Pipeline-level env vars available to all CodeBuild actions
266
+ // Note: #{codepipeline.*} resolved variables must go through CodeBuildStep.env
267
+ // (action-level), not buildEnvironment.environmentVariables (project-level).
268
+ const pipelineEnvVars = {
269
+ PLATFORM_BASE_URL: { value: platformUrl },
270
+ ...(pipelineId && { PIPELINE_ID: { value: pipelineId } }),
271
+ ...(platformSecretName && { PLATFORM_SECRET_NAME: { value: platformSecretName } }),
272
+ // Enable plugin resolution via custom resource Lambda inside CodePipeline
273
+ RESOLVED_SYNTH_PLUGIN: { value: 'true' },
274
+ // Propagate TLS verification setting so all CodeBuild steps can reach
275
+ // the platform API when using self-signed certificates
276
+ ...(process.env.NODE_TLS_REJECT_UNAUTHORIZED === '0' && {
277
+ NODE_TLS_REJECT_UNAUTHORIZED: { value: '0' },
278
+ }),
279
+ };
280
+ const securityGroups = [
281
+ ...(networkProps?.securityGroups ?? []),
282
+ ...(standaloneSecurityGroups ?? []),
283
+ ];
284
+ return {
285
+ ...(networkProps && { vpc: networkProps.vpc, subnetSelection: networkProps.subnetSelection }),
286
+ ...(securityGroups.length > 0 && { securityGroups }),
287
+ buildEnvironment: { environmentVariables: pipelineEnvVars },
288
+ };
289
+ }
290
+ }
291
+ exports.PipelineBuilder = PipelineBuilder;
292
+ //# 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    const uniqueId = new UniqueId();\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    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    });\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    // Add stages as waves via StageBuilder\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      });\n      stageBuilder.addStages(this.pipeline, props.stages);\n    }\n\n    // ── Tags ──\n    Tags.of(this.pipeline).add('project', this.config.project);\n    Tags.of(this.pipeline).add('organization', this.config.organization);\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"]}
@@ -0,0 +1,72 @@
1
+ import type { BuilderProps } from './pipeline-builder';
2
+ import type { CodeCommitOptions, CodeStarOptions, GitHubOptions, S3Options } from './source-types';
3
+ import type { PluginOptions, StageOptions, StepCustomization } from './step-types';
4
+ import type { CodeBuildDefaults, NetworkConfig } from '../core/network-types';
5
+ import type { MetaDataType, SourceType } from '../core/pipeline-types';
6
+ /**
7
+ * Validated and processed pipeline configuration (business logic layer).
8
+ * This class handles all non-CDK logic: validation, sanitization, and metadata merging.
9
+ * It can be tested independently without CDK dependencies.
10
+ *
11
+ * ## Metadata merge chain (last wins)
12
+ *
13
+ * 1. `BuilderProps.global` — base metadata inherited by all steps
14
+ * 2. `BuilderProps.defaults.metadata` — pipeline-level CodeBuild defaults metadata
15
+ * 3. `BuilderProps.synth.metadata` or `StageStepOptions.metadata` — per-step metadata
16
+ * 4. `plugin.metadata` — plugin's own metadata (merged in `createCodeBuildStep`)
17
+ * 5. `metadataForXxx()` — extracted CDK props spread last into construct calls
18
+ *
19
+ * Steps 1–3 are merged here into `this.metadata.merged`.
20
+ * Step 4 happens in `createCodeBuildStep` via `merge(metadata, plugin.metadata)`.
21
+ * Step 5 is spread last in the CDK constructor call, giving metadata full override authority.
22
+ */
23
+ export declare class PipelineConfiguration {
24
+ readonly project: string;
25
+ readonly organization: string;
26
+ readonly pipelineName: string;
27
+ readonly metadata: {
28
+ readonly global: MetaDataType;
29
+ readonly synth: MetaDataType;
30
+ readonly merged: MetaDataType;
31
+ };
32
+ readonly source: SourceType;
33
+ readonly plugin: PluginOptions;
34
+ readonly network: NetworkConfig | undefined;
35
+ readonly defaults: CodeBuildDefaults | undefined;
36
+ readonly synthCustomization: StepCustomization;
37
+ readonly stages: StageOptions[] | undefined;
38
+ constructor(props: BuilderProps);
39
+ /**
40
+ * Validates BuilderProps to ensure all required fields are present.
41
+ * Throws an error with detailed messages if validation fails.
42
+ */
43
+ private validateProps;
44
+ /**
45
+ * Extracts S3 source options with defaults applied.
46
+ *
47
+ * @returns S3 options with `objectKey` defaulting to `'source.zip'` and `trigger` defaulting to `NONE`
48
+ * @throws {Error} If the configured source type is not `s3`
49
+ */
50
+ getS3Options(): Required<Pick<S3Options, 'bucketName' | 'objectKey' | 'trigger'>> & Omit<S3Options, 'bucketName' | 'objectKey' | 'trigger'>;
51
+ /**
52
+ * Extracts GitHub source options with defaults applied.
53
+ *
54
+ * @returns GitHub options with `branch` defaulting to `'main'` and `trigger` defaulting to `NONE`
55
+ * @throws {Error} If the configured source type is not `github`
56
+ */
57
+ getGitHubOptions(): Required<Pick<GitHubOptions, 'repo' | 'branch' | 'trigger'>> & Omit<GitHubOptions, 'repo' | 'branch' | 'trigger'>;
58
+ /**
59
+ * Extracts CodeStar source options with defaults applied.
60
+ *
61
+ * @returns CodeStar options with `branch` defaulting to `'main'`, `trigger` defaulting to `NONE`, and `codeBuildCloneOutput` defaulting to `false`
62
+ * @throws {Error} If the configured source type is not `codestar`
63
+ */
64
+ getCodeStarOptions(): Required<Pick<CodeStarOptions, 'repo' | 'branch' | 'trigger' | 'codeBuildCloneOutput' | 'connectionArn'>>;
65
+ /**
66
+ * Extracts CodeCommit source options with defaults applied.
67
+ *
68
+ * @returns CodeCommit options with `branch` defaulting to `'main'` and `trigger` defaulting to `NONE`
69
+ * @throws {Error} If the configured source type is not `codecommit`
70
+ */
71
+ getCodeCommitOptions(): Required<Pick<CodeCommitOptions, 'repositoryName' | 'branch' | 'trigger'>>;
72
+ }