1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Pipeline = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const notifications = require("@aws-cdk/aws-codestarnotifications");
|
8 | const events = require("@aws-cdk/aws-events");
|
9 | const iam = require("@aws-cdk/aws-iam");
|
10 | const kms = require("@aws-cdk/aws-kms");
|
11 | const s3 = require("@aws-cdk/aws-s3");
|
12 | const core_1 = require("@aws-cdk/core");
|
13 | const action_1 = require("./action");
|
14 | const codepipeline_generated_1 = require("./codepipeline.generated");
|
15 | const cross_region_support_stack_1 = require("./private/cross-region-support-stack");
|
16 | const full_action_descriptor_1 = require("./private/full-action-descriptor");
|
17 | const rich_action_1 = require("./private/rich-action");
|
18 | const stage_1 = require("./private/stage");
|
19 | const validation_1 = require("./private/validation");
|
20 | class PipelineBase extends core_1.Resource {
|
21 | /**
|
22 | * Defines an event rule triggered by this CodePipeline.
|
23 | *
|
24 | * @param id Identifier for this event handler.
|
25 | * @param options Additional options to pass to the event rule.
|
26 | */
|
27 | onEvent(id, options = {}) {
|
28 | const rule = new events.Rule(this, id, options);
|
29 | rule.addTarget(options.target);
|
30 | rule.addEventPattern({
|
31 | source: ['aws.codepipeline'],
|
32 | resources: [this.pipelineArn],
|
33 | });
|
34 | return rule;
|
35 | }
|
36 | /**
|
37 | * Defines an event rule triggered by the "CodePipeline Pipeline Execution
|
38 | * State Change" event emitted from this pipeline.
|
39 | *
|
40 | * @param id Identifier for this event handler.
|
41 | * @param options Additional options to pass to the event rule.
|
42 | */
|
43 | onStateChange(id, options = {}) {
|
44 | const rule = this.onEvent(id, options);
|
45 | rule.addEventPattern({
|
46 | detailType: ['CodePipeline Pipeline Execution State Change'],
|
47 | });
|
48 | return rule;
|
49 | }
|
50 | bindAsNotificationRuleSource(_scope) {
|
51 | return {
|
52 | sourceArn: this.pipelineArn,
|
53 | };
|
54 | }
|
55 | notifyOn(id, target, options) {
|
56 | return new notifications.NotificationRule(this, id, {
|
57 | ...options,
|
58 | source: this,
|
59 | targets: [target],
|
60 | });
|
61 | }
|
62 | notifyOnExecutionStateChange(id, target, options) {
|
63 | return this.notifyOn(id, target, {
|
64 | ...options,
|
65 | events: [
|
66 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_FAILED,
|
67 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_CANCELED,
|
68 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_STARTED,
|
69 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_RESUMED,
|
70 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_SUCCEEDED,
|
71 | action_1.PipelineNotificationEvents.PIPELINE_EXECUTION_SUPERSEDED,
|
72 | ],
|
73 | });
|
74 | }
|
75 | notifyOnAnyStageStateChange(id, target, options) {
|
76 | return this.notifyOn(id, target, {
|
77 | ...options,
|
78 | events: [
|
79 | action_1.PipelineNotificationEvents.STAGE_EXECUTION_CANCELED,
|
80 | action_1.PipelineNotificationEvents.STAGE_EXECUTION_FAILED,
|
81 | action_1.PipelineNotificationEvents.STAGE_EXECUTION_RESUMED,
|
82 | action_1.PipelineNotificationEvents.STAGE_EXECUTION_STARTED,
|
83 | action_1.PipelineNotificationEvents.STAGE_EXECUTION_SUCCEEDED,
|
84 | ],
|
85 | });
|
86 | }
|
87 | notifyOnAnyActionStateChange(id, target, options) {
|
88 | return this.notifyOn(id, target, {
|
89 | ...options,
|
90 | events: [
|
91 | action_1.PipelineNotificationEvents.ACTION_EXECUTION_CANCELED,
|
92 | action_1.PipelineNotificationEvents.ACTION_EXECUTION_FAILED,
|
93 | action_1.PipelineNotificationEvents.ACTION_EXECUTION_STARTED,
|
94 | action_1.PipelineNotificationEvents.ACTION_EXECUTION_SUCCEEDED,
|
95 | ],
|
96 | });
|
97 | }
|
98 | notifyOnAnyManualApprovalStateChange(id, target, options) {
|
99 | return this.notifyOn(id, target, {
|
100 | ...options,
|
101 | events: [
|
102 | action_1.PipelineNotificationEvents.MANUAL_APPROVAL_FAILED,
|
103 | action_1.PipelineNotificationEvents.MANUAL_APPROVAL_NEEDED,
|
104 | action_1.PipelineNotificationEvents.MANUAL_APPROVAL_SUCCEEDED,
|
105 | ],
|
106 | });
|
107 | }
|
108 | }
|
109 | /**
|
110 | * An AWS CodePipeline pipeline with its associated IAM role and S3 bucket.
|
111 | *
|
112 | * @example
|
113 | * // create a pipeline
|
114 | * import * as codecommit from '@aws-cdk/aws-codecommit';
|
115 | *
|
116 | * const pipeline = new codepipeline.Pipeline(this, 'Pipeline');
|
117 | *
|
118 | * // add a stage
|
119 | * const sourceStage = pipeline.addStage({ stageName: 'Source' });
|
120 | *
|
121 | * // add a source action to the stage
|
122 | * declare const repo: codecommit.Repository;
|
123 | * declare const sourceArtifact: codepipeline.Artifact;
|
124 | * sourceStage.addAction(new codepipeline_actions.CodeCommitSourceAction({
|
125 | * actionName: 'Source',
|
126 | * output: sourceArtifact,
|
127 | * repository: repo,
|
128 | * }));
|
129 | *
|
130 | * // ... add more stages
|
131 | */
|
132 | class Pipeline extends PipelineBase {
|
133 | constructor(scope, id, props = {}) {
|
134 | super(scope, id, {
|
135 | physicalName: props.pipelineName,
|
136 | });
|
137 | this._stages = new Array();
|
138 | this._crossRegionSupport = {};
|
139 | this._crossAccountSupport = {};
|
140 | try {
|
141 | jsiiDeprecationWarnings._aws_cdk_aws_codepipeline_PipelineProps(props);
|
142 | }
|
143 | catch (error) {
|
144 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
145 | Error.captureStackTrace(error, Pipeline);
|
146 | }
|
147 | throw error;
|
148 | }
|
149 | validation_1.validateName('Pipeline', this.physicalName);
|
150 | // only one of artifactBucket and crossRegionReplicationBuckets can be supplied
|
151 | if (props.artifactBucket && props.crossRegionReplicationBuckets) {
|
152 | throw new Error('Only one of artifactBucket and crossRegionReplicationBuckets can be specified!');
|
153 | }
|
154 | // @deprecated(v2): switch to default false
|
155 | this.crossAccountKeys = props.crossAccountKeys ?? true;
|
156 | this.enableKeyRotation = props.enableKeyRotation;
|
157 | // Cross account keys must be set for key rotation to be enabled
|
158 | if (this.enableKeyRotation && !this.crossAccountKeys) {
|
159 | throw new Error("Setting 'enableKeyRotation' to true also requires 'crossAccountKeys' to be enabled");
|
160 | }
|
161 | this.reuseCrossRegionSupportStacks = props.reuseCrossRegionSupportStacks ?? true;
|
162 | // If a bucket has been provided, use it - otherwise, create a bucket.
|
163 | let propsBucket = this.getArtifactBucketFromProps(props);
|
164 | if (!propsBucket) {
|
165 | let encryptionKey;
|
166 | if (this.crossAccountKeys) {
|
167 | encryptionKey = new kms.Key(this, 'ArtifactsBucketEncryptionKey', {
|
168 | // remove the key - there is a grace period of a few days before it's gone for good,
|
169 | // that should be enough for any emergency access to the bucket artifacts
|
170 | removalPolicy: core_1.RemovalPolicy.DESTROY,
|
171 | enableKeyRotation: this.enableKeyRotation,
|
172 | });
|
173 | // add an alias to make finding the key in the console easier
|
174 | new kms.Alias(this, 'ArtifactsBucketEncryptionKeyAlias', {
|
175 | aliasName: this.generateNameForDefaultBucketKeyAlias(),
|
176 | targetKey: encryptionKey,
|
177 | removalPolicy: core_1.RemovalPolicy.DESTROY,
|
178 | });
|
179 | }
|
180 | propsBucket = new s3.Bucket(this, 'ArtifactsBucket', {
|
181 | bucketName: core_1.PhysicalName.GENERATE_IF_NEEDED,
|
182 | encryptionKey,
|
183 | encryption: encryptionKey ? s3.BucketEncryption.KMS : s3.BucketEncryption.KMS_MANAGED,
|
184 | enforceSSL: true,
|
185 | blockPublicAccess: new s3.BlockPublicAccess(s3.BlockPublicAccess.BLOCK_ALL),
|
186 | removalPolicy: core_1.RemovalPolicy.RETAIN,
|
187 | });
|
188 | }
|
189 | this.artifactBucket = propsBucket;
|
190 | // If a role has been provided, use it - otherwise, create a role.
|
191 | this.role = props.role || new iam.Role(this, 'Role', {
|
192 | assumedBy: new iam.ServicePrincipal('codepipeline.amazonaws.com'),
|
193 | });
|
194 | this.codePipeline = new codepipeline_generated_1.CfnPipeline(this, 'Resource', {
|
195 | artifactStore: core_1.Lazy.any({ produce: () => this.renderArtifactStoreProperty() }),
|
196 | artifactStores: core_1.Lazy.any({ produce: () => this.renderArtifactStoresProperty() }),
|
197 | stages: core_1.Lazy.any({ produce: () => this.renderStages() }),
|
198 | disableInboundStageTransitions: core_1.Lazy.any({ produce: () => this.renderDisabledTransitions() }, { omitEmptyArray: true }),
|
199 | roleArn: this.role.roleArn,
|
200 | restartExecutionOnUpdate: props && props.restartExecutionOnUpdate,
|
201 | name: this.physicalName,
|
202 | });
|
203 | // this will produce a DependsOn for both the role and the policy resources.
|
204 | this.codePipeline.node.addDependency(this.role);
|
205 | this.artifactBucket.grantReadWrite(this.role);
|
206 | this.pipelineName = this.getResourceNameAttribute(this.codePipeline.ref);
|
207 | this.pipelineVersion = this.codePipeline.attrVersion;
|
208 | this.crossRegionBucketsPassed = !!props.crossRegionReplicationBuckets;
|
209 | for (const [region, replicationBucket] of Object.entries(props.crossRegionReplicationBuckets || {})) {
|
210 | this._crossRegionSupport[region] = {
|
211 | replicationBucket,
|
212 | stack: core_1.Stack.of(replicationBucket),
|
213 | };
|
214 | }
|
215 | // Does not expose a Fn::GetAtt for the ARN so we'll have to make it ourselves
|
216 | this.pipelineArn = core_1.Stack.of(this).formatArn({
|
217 | service: 'codepipeline',
|
218 | resource: this.pipelineName,
|
219 | });
|
220 | for (const stage of props.stages || []) {
|
221 | this.addStage(stage);
|
222 | }
|
223 | }
|
224 | /**
|
225 | * Import a pipeline into this app.
|
226 | *
|
227 | * @param scope the scope into which to import this pipeline
|
228 | * @param id the logical ID of the returned pipeline construct
|
229 | * @param pipelineArn The ARN of the pipeline (e.g. `arn:aws:codepipeline:us-east-1:123456789012:MyDemoPipeline`)
|
230 | */
|
231 | static fromPipelineArn(scope, id, pipelineArn) {
|
232 | class Import extends PipelineBase {
|
233 | constructor() {
|
234 | super(...arguments);
|
235 | this.pipelineName = core_1.Stack.of(scope).splitArn(pipelineArn, core_1.ArnFormat.SLASH_RESOURCE_NAME).resource;
|
236 | this.pipelineArn = pipelineArn;
|
237 | }
|
238 | }
|
239 | return new Import(scope, id);
|
240 | }
|
241 | /**
|
242 | * Creates a new Stage, and adds it to this Pipeline.
|
243 | *
|
244 | * @param props the creation properties of the new Stage
|
245 | * @returns the newly created Stage
|
246 | */
|
247 | addStage(props) {
|
248 | try {
|
249 | jsiiDeprecationWarnings._aws_cdk_aws_codepipeline_StageOptions(props);
|
250 | }
|
251 | catch (error) {
|
252 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
253 | Error.captureStackTrace(error, this.addStage);
|
254 | }
|
255 | throw error;
|
256 | }
|
257 | // check for duplicate Stages and names
|
258 | if (this._stages.find(s => s.stageName === props.stageName)) {
|
259 | throw new Error(`Stage with duplicate name '${props.stageName}' added to the Pipeline`);
|
260 | }
|
261 | const stage = new stage_1.Stage(props, this);
|
262 | const index = props.placement
|
263 | ? this.calculateInsertIndexFromPlacement(props.placement)
|
264 | : this.stageCount;
|
265 | this._stages.splice(index, 0, stage);
|
266 | return stage;
|
267 | }
|
268 | /**
|
269 | * Adds a statement to the pipeline role.
|
270 | */
|
271 | addToRolePolicy(statement) {
|
272 | this.role.addToPrincipalPolicy(statement);
|
273 | }
|
274 | /**
|
275 | * Get the number of Stages in this Pipeline.
|
276 | */
|
277 | get stageCount() {
|
278 | return this._stages.length;
|
279 | }
|
280 | /**
|
281 | * Returns the stages that comprise the pipeline.
|
282 | *
|
283 | * **Note**: the returned array is a defensive copy,
|
284 | * so adding elements to it has no effect.
|
285 | * Instead, use the {@link addStage} method if you want to add more stages
|
286 | * to the pipeline.
|
287 | */
|
288 | get stages() {
|
289 | return this._stages.slice();
|
290 | }
|
291 | /**
|
292 | * Access one of the pipeline's stages by stage name
|
293 | */
|
294 | stage(stageName) {
|
295 | for (const stage of this._stages) {
|
296 | if (stage.stageName === stageName) {
|
297 | return stage;
|
298 | }
|
299 | }
|
300 | throw new Error(`Pipeline does not contain a stage named '${stageName}'. Available stages: ${this._stages.map(s => s.stageName).join(', ')}`);
|
301 | }
|
302 | /**
|
303 | * Returns all of the {@link CrossRegionSupportStack}s that were generated automatically
|
304 | * when dealing with Actions that reside in a different region than the Pipeline itself.
|
305 | *
|
306 | */
|
307 | get crossRegionSupport() {
|
308 | const ret = {};
|
309 | Object.keys(this._crossRegionSupport).forEach((key) => {
|
310 | ret[key] = this._crossRegionSupport[key];
|
311 | });
|
312 | return ret;
|
313 | }
|
314 | /** @internal */
|
315 | _attachActionToPipeline(stage, action, actionScope) {
|
316 | const richAction = new rich_action_1.RichAction(action, this);
|
317 | // handle cross-region actions here
|
318 | const crossRegionInfo = this.ensureReplicationResourcesExistFor(richAction);
|
319 | // get the role for the given action, handling if it's cross-account
|
320 | const actionRole = this.getRoleForAction(stage, richAction, actionScope);
|
321 | // // CodePipeline Variables
|
322 | validation_1.validateNamespaceName(richAction.actionProperties.variablesNamespace);
|
323 | // bind the Action (type h4x)
|
324 | const actionConfig = richAction.bind(actionScope, stage, {
|
325 | role: actionRole ? actionRole : this.role,
|
326 | bucket: crossRegionInfo.artifactBucket,
|
327 | });
|
328 | return new full_action_descriptor_1.FullActionDescriptor({
|
329 | // must be 'action', not 'richAction',
|
330 | // as those are returned by the IStage.actions property,
|
331 | // and it's important customers of Pipeline get the same instance
|
332 | // back as they added to the pipeline
|
333 | action,
|
334 | actionConfig,
|
335 | actionRole,
|
336 | actionRegion: crossRegionInfo.region,
|
337 | });
|
338 | }
|
339 | /**
|
340 | * Validate the pipeline structure
|
341 | *
|
342 | * Validation happens according to the rules documented at
|
343 | *
|
344 | * https://docs.aws.amazon.com/codepipeline/latest/userguide/reference-pipeline-structure.html#pipeline-requirements
|
345 | * @override
|
346 | */
|
347 | validate() {
|
348 | return [
|
349 | ...this.validateSourceActionLocations(),
|
350 | ...this.validateHasStages(),
|
351 | ...this.validateStages(),
|
352 | ...this.validateArtifacts(),
|
353 | ];
|
354 | }
|
355 | ensureReplicationResourcesExistFor(action) {
|
356 | if (!action.isCrossRegion) {
|
357 | return {
|
358 | artifactBucket: this.artifactBucket,
|
359 | };
|
360 | }
|
361 | // The action has a specific region,
|
362 | // require the pipeline to have a known region as well.
|
363 | this.requireRegion();
|
364 | // source actions have to be in the same region as the pipeline
|
365 | if (action.actionProperties.category === action_1.ActionCategory.SOURCE) {
|
366 | throw new Error(`Source action '${action.actionProperties.actionName}' must be in the same region as the pipeline`);
|
367 | }
|
368 | // check whether we already have a bucket in that region,
|
369 | // either passed from the outside or previously created
|
370 | const crossRegionSupport = this.obtainCrossRegionSupportFor(action);
|
371 | // the stack containing the replication bucket must be deployed before the pipeline
|
372 | core_1.Stack.of(this).addDependency(crossRegionSupport.stack);
|
373 | // The Pipeline role must be able to replicate to that bucket
|
374 | crossRegionSupport.replicationBucket.grantReadWrite(this.role);
|
375 | return {
|
376 | artifactBucket: crossRegionSupport.replicationBucket,
|
377 | region: action.effectiveRegion,
|
378 | };
|
379 | }
|
380 | /**
|
381 | * Get or create the cross-region support construct for the given action
|
382 | */
|
383 | obtainCrossRegionSupportFor(action) {
|
384 | // this method is never called for non cross-region actions
|
385 | const actionRegion = action.effectiveRegion;
|
386 | let crossRegionSupport = this._crossRegionSupport[actionRegion];
|
387 | if (!crossRegionSupport) {
|
388 | // we need to create scaffolding resources for this region
|
389 | const otherStack = action.resourceStack;
|
390 | crossRegionSupport = this.createSupportResourcesForRegion(otherStack, actionRegion);
|
391 | this._crossRegionSupport[actionRegion] = crossRegionSupport;
|
392 | }
|
393 | return crossRegionSupport;
|
394 | }
|
395 | createSupportResourcesForRegion(otherStack, actionRegion) {
|
396 | // if we have a stack from the resource passed - use that!
|
397 | if (otherStack) {
|
398 | // check if the stack doesn't have this magic construct already
|
399 | const id = `CrossRegionReplicationSupport-d823f1d8-a990-4e5c-be18-4ac698532e65-${actionRegion}`;
|
400 | let crossRegionSupportConstruct = otherStack.node.tryFindChild(id);
|
401 | if (!crossRegionSupportConstruct) {
|
402 | crossRegionSupportConstruct = new cross_region_support_stack_1.CrossRegionSupportConstruct(otherStack, id, {
|
403 | createKmsKey: this.crossAccountKeys,
|
404 | enableKeyRotation: this.enableKeyRotation,
|
405 | });
|
406 | }
|
407 | return {
|
408 | replicationBucket: crossRegionSupportConstruct.replicationBucket,
|
409 | stack: otherStack,
|
410 | };
|
411 | }
|
412 | // otherwise - create a stack with the resources needed for replication across regions
|
413 | const pipelineStack = core_1.Stack.of(this);
|
414 | const pipelineAccount = pipelineStack.account;
|
415 | if (core_1.Token.isUnresolved(pipelineAccount)) {
|
416 | throw new Error("You need to specify an explicit account when using CodePipeline's cross-region support");
|
417 | }
|
418 | const app = this.supportScope();
|
419 | const supportStackId = `cross-region-stack-${this.reuseCrossRegionSupportStacks ? pipelineAccount : pipelineStack.stackName}:${actionRegion}`;
|
420 | let supportStack = app.node.tryFindChild(supportStackId);
|
421 | if (!supportStack) {
|
422 | supportStack = new cross_region_support_stack_1.CrossRegionSupportStack(app, supportStackId, {
|
423 | pipelineStackName: pipelineStack.stackName,
|
424 | region: actionRegion,
|
425 | account: pipelineAccount,
|
426 | synthesizer: this.getCrossRegionSupportSynthesizer(),
|
427 | createKmsKey: this.crossAccountKeys,
|
428 | enableKeyRotation: this.enableKeyRotation,
|
429 | });
|
430 | }
|
431 | return {
|
432 | stack: supportStack,
|
433 | replicationBucket: supportStack.replicationBucket,
|
434 | };
|
435 | }
|
436 | getCrossRegionSupportSynthesizer() {
|
437 | if (this.stack.synthesizer instanceof core_1.DefaultStackSynthesizer) {
|
438 | // if we have the new synthesizer,
|
439 | // we need a bootstrapless copy of it,
|
440 | // because we don't want to require bootstrapping the environment
|
441 | // of the pipeline account in this replication region
|
442 | return new core_1.BootstraplessSynthesizer({
|
443 | deployRoleArn: this.stack.synthesizer.deployRoleArn,
|
444 | cloudFormationExecutionRoleArn: this.stack.synthesizer.cloudFormationExecutionRoleArn,
|
445 | });
|
446 | }
|
447 | else {
|
448 | // any other synthesizer: just return undefined
|
449 | // (ie., use the default based on the context settings)
|
450 | return undefined;
|
451 | }
|
452 | }
|
453 | generateNameForDefaultBucketKeyAlias() {
|
454 | const prefix = 'alias/codepipeline-';
|
455 | const maxAliasLength = 256;
|
456 | const uniqueId = core_1.Names.uniqueId(this);
|
457 | // take the last 256 - (prefix length) characters of uniqueId
|
458 | const startIndex = Math.max(0, uniqueId.length - (maxAliasLength - prefix.length));
|
459 | return prefix + uniqueId.substring(startIndex).toLowerCase();
|
460 | }
|
461 | /**
|
462 | * Gets the role used for this action,
|
463 | * including handling the case when the action is supposed to be cross-account.
|
464 | *
|
465 | * @param stage the stage the action belongs to
|
466 | * @param action the action to return/create a role for
|
467 | * @param actionScope the scope, unique to the action, to create new resources in
|
468 | */
|
469 | getRoleForAction(stage, action, actionScope) {
|
470 | const pipelineStack = core_1.Stack.of(this);
|
471 | let actionRole = this.getRoleFromActionPropsOrGenerateIfCrossAccount(stage, action);
|
472 | if (!actionRole && this.isAwsOwned(action)) {
|
473 | // generate a Role for this specific Action
|
474 | actionRole = new iam.Role(actionScope, 'CodePipelineActionRole', {
|
475 | assumedBy: new iam.AccountPrincipal(pipelineStack.account),
|
476 | });
|
477 | }
|
478 | // the pipeline role needs assumeRole permissions to the action role
|
479 | const grant = actionRole?.grantAssumeRole(this.role);
|
480 | grant?.applyBefore(this.codePipeline);
|
481 | return actionRole;
|
482 | }
|
483 | getRoleFromActionPropsOrGenerateIfCrossAccount(stage, action) {
|
484 | const pipelineStack = core_1.Stack.of(this);
|
485 | // if we have a cross-account action, the pipeline's bucket must have a KMS key
|
486 | // (otherwise we can't configure cross-account trust policies)
|
487 | if (action.isCrossAccount) {
|
488 | const artifactBucket = this.ensureReplicationResourcesExistFor(action).artifactBucket;
|
489 | if (!artifactBucket.encryptionKey) {
|
490 | throw new Error(`Artifact Bucket must have a KMS Key to add cross-account action '${action.actionProperties.actionName}' ` +
|
491 | `(pipeline account: '${renderEnvDimension(this.env.account)}', action account: '${renderEnvDimension(action.effectiveAccount)}'). ` +
|
492 | 'Create Pipeline with \'crossAccountKeys: true\' (or pass an existing Bucket with a key)');
|
493 | }
|
494 | }
|
495 | // if a Role has been passed explicitly, always use it
|
496 | // (even if the backing resource is from a different account -
|
497 | // this is how the user can override our default support logic)
|
498 | if (action.actionProperties.role) {
|
499 | if (this.isAwsOwned(action)) {
|
500 | // the role has to be deployed before the pipeline
|
501 | // (our magical cross-stack dependencies will not work,
|
502 | // because the role might be from a different environment),
|
503 | // but _only_ if it's a new Role -
|
504 | // an imported Role should not add the dependency
|
505 | if (action.actionProperties.role instanceof iam.Role) {
|
506 | const roleStack = core_1.Stack.of(action.actionProperties.role);
|
507 | pipelineStack.addDependency(roleStack);
|
508 | }
|
509 | return action.actionProperties.role;
|
510 | }
|
511 | else {
|
512 | // ...except if the Action is not owned by 'AWS',
|
513 | // as that would be rejected by CodePipeline at deploy time
|
514 | throw new Error("Specifying a Role is not supported for actions with an owner different than 'AWS' - " +
|
515 | `got '${action.actionProperties.owner}' (Action: '${action.actionProperties.actionName}' in Stage: '${stage.stageName}')`);
|
516 | }
|
517 | }
|
518 | // if we don't have a Role passed,
|
519 | // and the action is cross-account,
|
520 | // generate a Role in that other account stack
|
521 | const otherAccountStack = this.getOtherStackIfActionIsCrossAccount(action);
|
522 | if (!otherAccountStack) {
|
523 | return undefined;
|
524 | }
|
525 | // generate a role in the other stack, that the Pipeline will assume for executing this action
|
526 | const ret = new iam.Role(otherAccountStack, `${core_1.Names.uniqueId(this)}-${stage.stageName}-${action.actionProperties.actionName}-ActionRole`, {
|
527 | assumedBy: new iam.AccountPrincipal(pipelineStack.account),
|
528 | roleName: core_1.PhysicalName.GENERATE_IF_NEEDED,
|
529 | });
|
530 | // the other stack with the role has to be deployed before the pipeline stack
|
531 | // (CodePipeline verifies you can assume the action Role on creation)
|
532 | pipelineStack.addDependency(otherAccountStack);
|
533 | return ret;
|
534 | }
|
535 | /**
|
536 | * Returns the Stack this Action belongs to if this is a cross-account Action.
|
537 | * If this Action is not cross-account (i.e., it lives in the same account as the Pipeline),
|
538 | * it returns undefined.
|
539 | *
|
540 | * @param action the Action to return the Stack for
|
541 | */
|
542 | getOtherStackIfActionIsCrossAccount(action) {
|
543 | const targetAccount = action.actionProperties.resource
|
544 | ? action.actionProperties.resource.env.account
|
545 | : action.actionProperties.account;
|
546 | if (targetAccount === undefined) {
|
547 | // if the account of the Action is not specified,
|
548 | // then it defaults to the same account the pipeline itself is in
|
549 | return undefined;
|
550 | }
|
551 | // check whether the action's account is a static string
|
552 | if (core_1.Token.isUnresolved(targetAccount)) {
|
553 | if (core_1.Token.isUnresolved(this.env.account)) {
|
554 | // the pipeline is also env-agnostic, so that's fine
|
555 | return undefined;
|
556 | }
|
557 | else {
|
558 | throw new Error(`The 'account' property must be a concrete value (action: '${action.actionProperties.actionName}')`);
|
559 | }
|
560 | }
|
561 | // At this point, we know that the action's account is a static string.
|
562 | // In this case, the pipeline's account must also be a static string.
|
563 | if (core_1.Token.isUnresolved(this.env.account)) {
|
564 | throw new Error('Pipeline stack which uses cross-environment actions must have an explicitly set account');
|
565 | }
|
566 | // at this point, we know that both the Pipeline's account,
|
567 | // and the action-backing resource's account are static strings
|
568 | // if they are identical - nothing to do (the action is not cross-account)
|
569 | if (this.env.account === targetAccount) {
|
570 | return undefined;
|
571 | }
|
572 | // at this point, we know that the action is certainly cross-account,
|
573 | // so we need to return a Stack in its account to create the helper Role in
|
574 | const candidateActionResourceStack = action.actionProperties.resource
|
575 | ? core_1.Stack.of(action.actionProperties.resource)
|
576 | : undefined;
|
577 | if (candidateActionResourceStack?.account === targetAccount) {
|
578 | // we always use the "latest" action-backing resource's Stack for this account,
|
579 | // even if a different one was used earlier
|
580 | this._crossAccountSupport[targetAccount] = candidateActionResourceStack;
|
581 | return candidateActionResourceStack;
|
582 | }
|
583 | let targetAccountStack = this._crossAccountSupport[targetAccount];
|
584 | if (!targetAccountStack) {
|
585 | const stackId = `cross-account-support-stack-${targetAccount}`;
|
586 | const app = this.supportScope();
|
587 | targetAccountStack = app.node.tryFindChild(stackId);
|
588 | if (!targetAccountStack) {
|
589 | const actionRegion = action.actionProperties.resource
|
590 | ? action.actionProperties.resource.env.region
|
591 | : action.actionProperties.region;
|
592 | const pipelineStack = core_1.Stack.of(this);
|
593 | targetAccountStack = new core_1.Stack(app, stackId, {
|
594 | stackName: `${pipelineStack.stackName}-support-${targetAccount}`,
|
595 | env: {
|
596 | account: targetAccount,
|
597 | region: actionRegion ?? pipelineStack.region,
|
598 | },
|
599 | });
|
600 | }
|
601 | this._crossAccountSupport[targetAccount] = targetAccountStack;
|
602 | }
|
603 | return targetAccountStack;
|
604 | }
|
605 | isAwsOwned(action) {
|
606 | const owner = action.actionProperties.owner;
|
607 | return !owner || owner === 'AWS';
|
608 | }
|
609 | getArtifactBucketFromProps(props) {
|
610 | if (props.artifactBucket) {
|
611 | return props.artifactBucket;
|
612 | }
|
613 | if (props.crossRegionReplicationBuckets) {
|
614 | const pipelineRegion = this.requireRegion();
|
615 | return props.crossRegionReplicationBuckets[pipelineRegion];
|
616 | }
|
617 | return undefined;
|
618 | }
|
619 | calculateInsertIndexFromPlacement(placement) {
|
620 | // check if at most one placement property was provided
|
621 | const providedPlacementProps = ['rightBefore', 'justAfter', 'atIndex']
|
622 | .filter((prop) => placement[prop] !== undefined);
|
623 | if (providedPlacementProps.length > 1) {
|
624 | throw new Error('Error adding Stage to the Pipeline: ' +
|
625 | 'you can only provide at most one placement property, but ' +
|
626 | `'${providedPlacementProps.join(', ')}' were given`);
|
627 | }
|
628 | if (placement.rightBefore !== undefined) {
|
629 | const targetIndex = this.findStageIndex(placement.rightBefore);
|
630 | if (targetIndex === -1) {
|
631 | throw new Error('Error adding Stage to the Pipeline: ' +
|
632 | `the requested Stage to add it before, '${placement.rightBefore.stageName}', was not found`);
|
633 | }
|
634 | return targetIndex;
|
635 | }
|
636 | if (placement.justAfter !== undefined) {
|
637 | const targetIndex = this.findStageIndex(placement.justAfter);
|
638 | if (targetIndex === -1) {
|
639 | throw new Error('Error adding Stage to the Pipeline: ' +
|
640 | `the requested Stage to add it after, '${placement.justAfter.stageName}', was not found`);
|
641 | }
|
642 | return targetIndex + 1;
|
643 | }
|
644 | return this.stageCount;
|
645 | }
|
646 | findStageIndex(targetStage) {
|
647 | return this._stages.findIndex(stage => stage === targetStage);
|
648 | }
|
649 | validateSourceActionLocations() {
|
650 | const errors = new Array();
|
651 | let firstStage = true;
|
652 | for (const stage of this._stages) {
|
653 | const onlySourceActionsPermitted = firstStage;
|
654 | for (const action of stage.actionDescriptors) {
|
655 | errors.push(...validation_1.validateSourceAction(onlySourceActionsPermitted, action.category, action.actionName, stage.stageName));
|
656 | }
|
657 | firstStage = false;
|
658 | }
|
659 | return errors;
|
660 | }
|
661 | validateHasStages() {
|
662 | if (this.stageCount < 2) {
|
663 | return ['Pipeline must have at least two stages'];
|
664 | }
|
665 | return [];
|
666 | }
|
667 | validateStages() {
|
668 | const ret = new Array();
|
669 | for (const stage of this._stages) {
|
670 | ret.push(...stage.validate());
|
671 | }
|
672 | return ret;
|
673 | }
|
674 | validateArtifacts() {
|
675 | const ret = new Array();
|
676 | const producers = {};
|
677 | const firstConsumers = {};
|
678 | for (const [stageIndex, stage] of enumerate(this._stages)) {
|
679 | // For every output artifact, get the producer
|
680 | for (const action of stage.actionDescriptors) {
|
681 | const actionLoc = new PipelineLocation(stageIndex, stage, action);
|
682 | for (const outputArtifact of action.outputs) {
|
683 | // output Artifacts always have a name set
|
684 | const name = outputArtifact.artifactName;
|
685 | if (producers[name]) {
|
686 | ret.push(`Both Actions '${producers[name].actionName}' and '${action.actionName}' are producting Artifact '${name}'. Every artifact can only be produced once.`);
|
687 | continue;
|
688 | }
|
689 | producers[name] = actionLoc;
|
690 | }
|
691 | // For every input artifact, get the first consumer
|
692 | for (const inputArtifact of action.inputs) {
|
693 | const name = inputArtifact.artifactName;
|
694 | if (!name) {
|
695 | ret.push(`Action '${action.actionName}' is using an unnamed input Artifact, which is not being produced in this pipeline`);
|
696 | continue;
|
697 | }
|
698 | firstConsumers[name] = firstConsumers[name] ? firstConsumers[name].first(actionLoc) : actionLoc;
|
699 | }
|
700 | }
|
701 | }
|
702 | // Now validate that every input artifact is produced before it's
|
703 | // being consumed.
|
704 | for (const [artifactName, consumerLoc] of Object.entries(firstConsumers)) {
|
705 | const producerLoc = producers[artifactName];
|
706 | if (!producerLoc) {
|
707 | ret.push(`Action '${consumerLoc.actionName}' is using input Artifact '${artifactName}', which is not being produced in this pipeline`);
|
708 | continue;
|
709 | }
|
710 | if (consumerLoc.beforeOrEqual(producerLoc)) {
|
711 | ret.push(`${consumerLoc} is consuming input Artifact '${artifactName}' before it is being produced at ${producerLoc}`);
|
712 | }
|
713 | }
|
714 | return ret;
|
715 | }
|
716 | renderArtifactStoresProperty() {
|
717 | if (!this.crossRegion) {
|
718 | return undefined;
|
719 | }
|
720 | // add the Pipeline's artifact store
|
721 | const primaryRegion = this.requireRegion();
|
722 | this._crossRegionSupport[primaryRegion] = {
|
723 | replicationBucket: this.artifactBucket,
|
724 | stack: core_1.Stack.of(this),
|
725 | };
|
726 | return Object.entries(this._crossRegionSupport).map(([region, support]) => ({
|
727 | region,
|
728 | artifactStore: this.renderArtifactStore(support.replicationBucket),
|
729 | }));
|
730 | }
|
731 | renderArtifactStoreProperty() {
|
732 | if (this.crossRegion) {
|
733 | return undefined;
|
734 | }
|
735 | return this.renderPrimaryArtifactStore();
|
736 | }
|
737 | renderPrimaryArtifactStore() {
|
738 | return this.renderArtifactStore(this.artifactBucket);
|
739 | }
|
740 | renderArtifactStore(bucket) {
|
741 | let encryptionKey;
|
742 | const bucketKey = bucket.encryptionKey;
|
743 | if (bucketKey) {
|
744 | encryptionKey = {
|
745 | type: 'KMS',
|
746 | id: bucketKey.keyArn,
|
747 | };
|
748 | }
|
749 | return {
|
750 | type: 'S3',
|
751 | location: bucket.bucketName,
|
752 | encryptionKey,
|
753 | };
|
754 | }
|
755 | get crossRegion() {
|
756 | if (this.crossRegionBucketsPassed) {
|
757 | return true;
|
758 | }
|
759 | return this._stages.some(stage => stage.actionDescriptors.some(action => action.region !== undefined));
|
760 | }
|
761 | renderStages() {
|
762 | return this._stages.map(stage => stage.render());
|
763 | }
|
764 | renderDisabledTransitions() {
|
765 | return this._stages
|
766 | .filter(stage => !stage.transitionToEnabled)
|
767 | .map(stage => ({
|
768 | reason: stage.transitionDisabledReason,
|
769 | stageName: stage.stageName,
|
770 | }));
|
771 | }
|
772 | requireRegion() {
|
773 | const region = this.env.region;
|
774 | if (core_1.Token.isUnresolved(region)) {
|
775 | throw new Error('Pipeline stack which uses cross-environment actions must have an explicitly set region');
|
776 | }
|
777 | return region;
|
778 | }
|
779 | supportScope() {
|
780 | const scope = core_1.Stage.of(this);
|
781 | if (!scope) {
|
782 | throw new Error('Pipeline stack which uses cross-environment actions must be part of a CDK App or Stage');
|
783 | }
|
784 | return scope;
|
785 | }
|
786 | }
|
787 | exports.Pipeline = Pipeline;
|
788 | _a = JSII_RTTI_SYMBOL_1;
|
789 | Pipeline[_a] = { fqn: "@aws-cdk/aws-codepipeline.Pipeline", version: "1.204.0" };
|
790 | function enumerate(xs) {
|
791 | const ret = new Array();
|
792 | for (let i = 0; i < xs.length; i++) {
|
793 | ret.push([i, xs[i]]);
|
794 | }
|
795 | return ret;
|
796 | }
|
797 | class PipelineLocation {
|
798 | constructor(stageIndex, stage, action) {
|
799 | this.stageIndex = stageIndex;
|
800 | this.stage = stage;
|
801 | this.action = action;
|
802 | }
|
803 | get stageName() {
|
804 | return this.stage.stageName;
|
805 | }
|
806 | get actionName() {
|
807 | return this.action.actionName;
|
808 | }
|
809 | /**
|
810 | * Returns whether a is before or the same order as b
|
811 | */
|
812 | beforeOrEqual(rhs) {
|
813 | if (this.stageIndex !== rhs.stageIndex) {
|
814 | return rhs.stageIndex < rhs.stageIndex;
|
815 | }
|
816 | return this.action.runOrder <= rhs.action.runOrder;
|
817 | }
|
818 | /**
|
819 | * Returns the first location between this and the other one
|
820 | */
|
821 | first(rhs) {
|
822 | return this.beforeOrEqual(rhs) ? this : rhs;
|
823 | }
|
824 | toString() {
|
825 | // runOrders are 1-based, so make the stageIndex also 1-based otherwise it's going to be confusing.
|
826 | return `Stage ${this.stageIndex + 1} Action ${this.action.runOrder} ('${this.stageName}'/'${this.actionName}')`;
|
827 | }
|
828 | }
|
829 | /**
|
830 | * Render an env dimension without showing the ugly stringified tokens
|
831 | */
|
832 | function renderEnvDimension(s) {
|
833 | return core_1.Token.isUnresolved(s) ? '(current)' : s;
|
834 | }
|
835 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as notifications from '@aws-cdk/aws-codestarnotifications';\nimport * as events from '@aws-cdk/aws-events';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as kms from '@aws-cdk/aws-kms';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport {\n  ArnFormat,\n  BootstraplessSynthesizer,\n  DefaultStackSynthesizer,\n  IStackSynthesizer,\n  Lazy,\n  Names,\n  PhysicalName,\n  RemovalPolicy,\n  Resource,\n  Stack,\n  Stage as CdkStage,\n  Token,\n} from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { ActionCategory, IAction, IPipeline, IStage, PipelineNotificationEvents, PipelineNotifyOnOptions } from './action';\nimport { CfnPipeline } from './codepipeline.generated';\nimport { CrossRegionSupportConstruct, CrossRegionSupportStack } from './private/cross-region-support-stack';\nimport { FullActionDescriptor } from './private/full-action-descriptor';\nimport { RichAction } from './private/rich-action';\nimport { Stage } from './private/stage';\nimport { validateName, validateNamespaceName, validateSourceAction } from './private/validation';\n\n// keep this import separate from other imports to reduce chance for merge conflicts with v2-main\n// eslint-disable-next-line no-duplicate-imports, import/order\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\n/**\n * Allows you to control where to place a new Stage when it's added to the Pipeline.\n * Note that you can provide only one of the below properties -\n * specifying more than one will result in a validation error.\n *\n * @see #rightBefore\n * @see #justAfter\n */\nexport interface StagePlacement {\n  /**\n   * Inserts the new Stage as a parent of the given Stage\n   * (changing its current parent Stage, if it had one).\n   */\n  readonly rightBefore?: IStage;\n\n  /**\n   * Inserts the new Stage as a child of the given Stage\n   * (changing its current child Stage, if it had one).\n   */\n  readonly justAfter?: IStage;\n}\n\n/**\n * Construction properties of a Pipeline Stage.\n */\nexport interface StageProps {\n  /**\n   * The physical, human-readable name to assign to this Pipeline Stage.\n   */\n  readonly stageName: string;\n\n  /**\n   * The list of Actions to create this Stage with.\n   * You can always add more Actions later by calling {@link IStage#addAction}.\n   */\n  readonly actions?: IAction[];\n\n  /**\n   * Whether to enable transition to this stage.\n   *\n   * @default true\n   */\n  readonly transitionToEnabled?: boolean;\n\n  /**\n   * The reason for disabling transition to this stage. Only applicable\n   * if `transitionToEnabled` is set to `false`.\n   *\n   * @default 'Transition disabled'\n   */\n  readonly transitionDisabledReason?: string;\n}\n\nexport interface StageOptions extends StageProps {\n  readonly placement?: StagePlacement;\n}\n\nexport interface PipelineProps {\n  /**\n   * The S3 bucket used by this Pipeline to store artifacts.\n   *\n   * @default - A new S3 bucket will be created.\n   */\n  readonly artifactBucket?: s3.IBucket;\n\n  /**\n   * The IAM role to be assumed by this Pipeline.\n   *\n   * @default a new IAM role will be created.\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * Indicates whether to rerun the AWS CodePipeline pipeline after you update it.\n   *\n   * @default false\n   */\n  readonly restartExecutionOnUpdate?: boolean;\n\n  /**\n   * Name of the pipeline.\n   *\n   * @default - AWS CloudFormation generates an ID and uses that for the pipeline name.\n   */\n  readonly pipelineName?: string;\n\n  /**\n   * A map of region to S3 bucket name used for cross-region CodePipeline.\n   * For every Action that you specify targeting a different region than the Pipeline itself,\n   * if you don't provide an explicit Bucket for that region using this property,\n   * the construct will automatically create a Stack containing an S3 Bucket in that region.\n   *\n   * @default - None.\n   */\n  readonly crossRegionReplicationBuckets?: { [region: string]: s3.IBucket };\n\n  /**\n   * The list of Stages, in order,\n   * to create this Pipeline with.\n   * You can always add more Stages later by calling {@link Pipeline#addStage}.\n   *\n   * @default - None.\n   */\n  readonly stages?: StageProps[];\n\n  /**\n   * Create KMS keys for cross-account deployments.\n   *\n   * This controls whether the pipeline is enabled for cross-account deployments.\n   *\n   * By default cross-account deployments are enabled, but this feature requires\n   * that KMS Customer Master Keys are created which have a cost of $1/month.\n   *\n   * If you do not need cross-account deployments, you can set this to `false` to\n   * not create those keys and save on that cost (the artifact bucket will be\n   * encrypted with an AWS-managed key). However, cross-account deployments will\n   * no longer be possible.\n   *\n   * @default true\n   */\n  readonly crossAccountKeys?: boolean;\n\n  /**\n   * Enable KMS key rotation for the generated KMS keys.\n   *\n   * By default KMS key rotation is disabled, but will add an additional $1/month\n   * for each year the key exists when enabled.\n   *\n   * @default - false (key rotation is disabled)\n   */\n  readonly enableKeyRotation?: boolean;\n\n  /**\n   * Reuse the same cross region support stack for all pipelines in the App.\n   *\n   * @default - true (Use the same support stack for all pipelines in App)\n   */\n  readonly reuseCrossRegionSupportStacks?: boolean;\n}\n\nabstract class PipelineBase extends Resource implements IPipeline {\n  public abstract readonly pipelineName: string;\n  public abstract readonly pipelineArn: string;\n\n  /**\n   * Defines an event rule triggered by this CodePipeline.\n   *\n   * @param id Identifier for this event handler.\n   * @param options Additional options to pass to the event rule.\n   */\n  public onEvent(id: string, options: events.OnEventOptions = {}): events.Rule {\n    const rule = new events.Rule(this, id, options);\n    rule.addTarget(options.target);\n    rule.addEventPattern({\n      source: ['aws.codepipeline'],\n      resources: [this.pipelineArn],\n    });\n    return rule;\n  }\n\n  /**\n   * Defines an event rule triggered by the \"CodePipeline Pipeline Execution\n   * State Change\" event emitted from this pipeline.\n   *\n   * @param id Identifier for this event handler.\n   * @param options Additional options to pass to the event rule.\n   */\n  public onStateChange(id: string, options: events.OnEventOptions = {}): events.Rule {\n    const rule = this.onEvent(id, options);\n    rule.addEventPattern({\n      detailType: ['CodePipeline Pipeline Execution State Change'],\n    });\n    return rule;\n  }\n\n  public bindAsNotificationRuleSource(_scope: Construct): notifications.NotificationRuleSourceConfig {\n    return {\n      sourceArn: this.pipelineArn,\n    };\n  }\n\n  public notifyOn(\n    id: string,\n    target: notifications.INotificationRuleTarget,\n    options: PipelineNotifyOnOptions,\n  ): notifications.INotificationRule {\n    return new notifications.NotificationRule(this, id, {\n      ...options,\n      source: this,\n      targets: [target],\n    });\n  }\n\n  public notifyOnExecutionStateChange(\n    id: string,\n    target: notifications.INotificationRuleTarget,\n    options?: notifications.NotificationRuleOptions,\n  ): notifications.INotificationRule {\n    return this.notifyOn(id, target, {\n      ...options,\n      events: [\n        PipelineNotificationEvents.PIPELINE_EXECUTION_FAILED,\n        PipelineNotificationEvents.PIPELINE_EXECUTION_CANCELED,\n        PipelineNotificationEvents.PIPELINE_EXECUTION_STARTED,\n        PipelineNotificationEvents.PIPELINE_EXECUTION_RESUMED,\n        PipelineNotificationEvents.PIPELINE_EXECUTION_SUCCEEDED,\n        PipelineNotificationEvents.PIPELINE_EXECUTION_SUPERSEDED,\n      ],\n    });\n  }\n\n  public notifyOnAnyStageStateChange(\n    id: string,\n    target: notifications.INotificationRuleTarget,\n    options?: notifications.NotificationRuleOptions,\n  ): notifications.INotificationRule {\n    return this.notifyOn(id, target, {\n      ...options,\n      events: [\n        PipelineNotificationEvents.STAGE_EXECUTION_CANCELED,\n        PipelineNotificationEvents.STAGE_EXECUTION_FAILED,\n        PipelineNotificationEvents.STAGE_EXECUTION_RESUMED,\n        PipelineNotificationEvents.STAGE_EXECUTION_STARTED,\n        PipelineNotificationEvents.STAGE_EXECUTION_SUCCEEDED,\n      ],\n    });\n  }\n\n  public notifyOnAnyActionStateChange(\n    id: string,\n    target: notifications.INotificationRuleTarget,\n    options?: notifications.NotificationRuleOptions,\n  ): notifications.INotificationRule {\n    return this.notifyOn(id, target, {\n      ...options,\n      events: [\n        PipelineNotificationEvents.ACTION_EXECUTION_CANCELED,\n        PipelineNotificationEvents.ACTION_EXECUTION_FAILED,\n        PipelineNotificationEvents.ACTION_EXECUTION_STARTED,\n        PipelineNotificationEvents.ACTION_EXECUTION_SUCCEEDED,\n      ],\n    });\n  }\n\n  public notifyOnAnyManualApprovalStateChange(\n    id: string,\n    target: notifications.INotificationRuleTarget,\n    options?: notifications.NotificationRuleOptions,\n  ): notifications.INotificationRule {\n    return this.notifyOn(id, target, {\n      ...options,\n      events: [\n        PipelineNotificationEvents.MANUAL_APPROVAL_FAILED,\n        PipelineNotificationEvents.MANUAL_APPROVAL_NEEDED,\n        PipelineNotificationEvents.MANUAL_APPROVAL_SUCCEEDED,\n      ],\n    });\n  }\n}\n\n/**\n * An AWS CodePipeline pipeline with its associated IAM role and S3 bucket.\n *\n * @example\n * // create a pipeline\n * import * as codecommit from '@aws-cdk/aws-codecommit';\n *\n * const pipeline = new codepipeline.Pipeline(this, 'Pipeline');\n *\n * // add a stage\n * const sourceStage = pipeline.addStage({ stageName: 'Source' });\n *\n * // add a source action to the stage\n * declare const repo: codecommit.Repository;\n * declare const sourceArtifact: codepipeline.Artifact;\n * sourceStage.addAction(new codepipeline_actions.CodeCommitSourceAction({\n *   actionName: 'Source',\n *   output: sourceArtifact,\n *   repository: repo,\n * }));\n *\n * // ... add more stages\n */\nexport class Pipeline extends PipelineBase {\n  /**\n   * Import a pipeline into this app.\n   *\n   * @param scope the scope into which to import this pipeline\n   * @param id the logical ID of the returned pipeline construct\n   * @param pipelineArn The ARN of the pipeline (e.g. `arn:aws:codepipeline:us-east-1:123456789012:MyDemoPipeline`)\n   */\n  public static fromPipelineArn(scope: Construct, id: string, pipelineArn: string): IPipeline {\n    class Import extends PipelineBase {\n      public readonly pipelineName = Stack.of(scope).splitArn(pipelineArn, ArnFormat.SLASH_RESOURCE_NAME).resource;\n      public readonly pipelineArn = pipelineArn;\n    }\n\n    return new Import(scope, id);\n  }\n\n  /**\n   * The IAM role AWS CodePipeline will use to perform actions or assume roles for actions with\n   * a more specific IAM role.\n   */\n  public readonly role: iam.IRole;\n\n  /**\n   * ARN of this pipeline\n   */\n  public readonly pipelineArn: string;\n\n  /**\n   * The name of the pipeline\n   */\n  public readonly pipelineName: string;\n\n  /**\n   * The version of the pipeline\n   *\n   * @attribute\n   */\n  public readonly pipelineVersion: string;\n\n  /**\n   * Bucket used to store output artifacts\n   */\n  public readonly artifactBucket: s3.IBucket;\n\n  private readonly _stages = new Array<Stage>();\n  private readonly crossRegionBucketsPassed: boolean;\n  private readonly _crossRegionSupport: { [region: string]: CrossRegionSupport } = {};\n  private readonly _crossAccountSupport: { [account: string]: Stack } = {};\n  private readonly crossAccountKeys: boolean;\n  private readonly enableKeyRotation?: boolean;\n  private readonly reuseCrossRegionSupportStacks: boolean;\n  private readonly codePipeline: CfnPipeline;\n\n  constructor(scope: Construct, id: string, props: PipelineProps = {}) {\n    super(scope, id, {\n      physicalName: props.pipelineName,\n    });\n\n    validateName('Pipeline', this.physicalName);\n\n    // only one of artifactBucket and crossRegionReplicationBuckets can be supplied\n    if (props.artifactBucket && props.crossRegionReplicationBuckets) {\n      throw new Error('Only one of artifactBucket and crossRegionReplicationBuckets can be specified!');\n    }\n\n    // @deprecated(v2): switch to default false\n    this.crossAccountKeys = props.crossAccountKeys ?? true;\n    this.enableKeyRotation = props.enableKeyRotation;\n\n    // Cross account keys must be set for key rotation to be enabled\n    if (this.enableKeyRotation && !this.crossAccountKeys) {\n      throw new Error(\"Setting 'enableKeyRotation' to true also requires 'crossAccountKeys' to be enabled\");\n    }\n\n    this.reuseCrossRegionSupportStacks = props.reuseCrossRegionSupportStacks ?? true;\n\n    // If a bucket has been provided, use it - otherwise, create a bucket.\n    let propsBucket = this.getArtifactBucketFromProps(props);\n\n    if (!propsBucket) {\n      let encryptionKey;\n\n      if (this.crossAccountKeys) {\n        encryptionKey = new kms.Key(this, 'ArtifactsBucketEncryptionKey', {\n          // remove the key - there is a grace period of a few days before it's gone for good,\n          // that should be enough for any emergency access to the bucket artifacts\n          removalPolicy: RemovalPolicy.DESTROY,\n          enableKeyRotation: this.enableKeyRotation,\n        });\n        // add an alias to make finding the key in the console easier\n        new kms.Alias(this, 'ArtifactsBucketEncryptionKeyAlias', {\n          aliasName: this.generateNameForDefaultBucketKeyAlias(),\n          targetKey: encryptionKey,\n          removalPolicy: RemovalPolicy.DESTROY, // destroy the alias along with the key\n        });\n      }\n\n      propsBucket = new s3.Bucket(this, 'ArtifactsBucket', {\n        bucketName: PhysicalName.GENERATE_IF_NEEDED,\n        encryptionKey,\n        encryption: encryptionKey ? s3.BucketEncryption.KMS : s3.BucketEncryption.KMS_MANAGED,\n        enforceSSL: true,\n        blockPublicAccess: new s3.BlockPublicAccess(s3.BlockPublicAccess.BLOCK_ALL),\n        removalPolicy: RemovalPolicy.RETAIN,\n      });\n    }\n    this.artifactBucket = propsBucket;\n\n    // If a role has been provided, use it - otherwise, create a role.\n    this.role = props.role || new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('codepipeline.amazonaws.com'),\n    });\n\n    this.codePipeline = new CfnPipeline(this, 'Resource', {\n      artifactStore: Lazy.any({ produce: () => this.renderArtifactStoreProperty() }),\n      artifactStores: Lazy.any({ produce: () => this.renderArtifactStoresProperty() }),\n      stages: Lazy.any({ produce: () => this.renderStages() }),\n      disableInboundStageTransitions: Lazy.any({ produce: () => this.renderDisabledTransitions() }, { omitEmptyArray: true }),\n      roleArn: this.role.roleArn,\n      restartExecutionOnUpdate: props && props.restartExecutionOnUpdate,\n      name: this.physicalName,\n    });\n\n    // this will produce a DependsOn for both the role and the policy resources.\n    this.codePipeline.node.addDependency(this.role);\n\n    this.artifactBucket.grantReadWrite(this.role);\n    this.pipelineName = this.getResourceNameAttribute(this.codePipeline.ref);\n    this.pipelineVersion = this.codePipeline.attrVersion;\n    this.crossRegionBucketsPassed = !!props.crossRegionReplicationBuckets;\n\n    for (const [region, replicationBucket] of Object.entries(props.crossRegionReplicationBuckets || {})) {\n      this._crossRegionSupport[region] = {\n        replicationBucket,\n        stack: Stack.of(replicationBucket),\n      };\n    }\n\n    // Does not expose a Fn::GetAtt for the ARN so we'll have to make it ourselves\n    this.pipelineArn = Stack.of(this).formatArn({\n      service: 'codepipeline',\n      resource: this.pipelineName,\n    });\n\n    for (const stage of props.stages || []) {\n      this.addStage(stage);\n    }\n  }\n\n  /**\n   * Creates a new Stage, and adds it to this Pipeline.\n   *\n   * @param props the creation properties of the new Stage\n   * @returns the newly created Stage\n   */\n  public addStage(props: StageOptions): IStage {\n    // check for duplicate Stages and names\n    if (this._stages.find(s => s.stageName === props.stageName)) {\n      throw new Error(`Stage with duplicate name '${props.stageName}' added to the Pipeline`);\n    }\n\n    const stage = new Stage(props, this);\n\n    const index = props.placement\n      ? this.calculateInsertIndexFromPlacement(props.placement)\n      : this.stageCount;\n\n    this._stages.splice(index, 0, stage);\n\n    return stage;\n  }\n\n  /**\n   * Adds a statement to the pipeline role.\n   */\n  public addToRolePolicy(statement: iam.PolicyStatement) {\n    this.role.addToPrincipalPolicy(statement);\n  }\n\n  /**\n   * Get the number of Stages in this Pipeline.\n   */\n  public get stageCount(): number {\n    return this._stages.length;\n  }\n\n  /**\n   * Returns the stages that comprise the pipeline.\n   *\n   * **Note**: the returned array is a defensive copy,\n   * so adding elements to it has no effect.\n   * Instead, use the {@link addStage} method if you want to add more stages\n   * to the pipeline.\n   */\n  public get stages(): IStage[] {\n    return this._stages.slice();\n  }\n\n  /**\n   * Access one of the pipeline's stages by stage name\n   */\n  public stage(stageName: string): IStage {\n    for (const stage of this._stages) {\n      if (stage.stageName === stageName) {\n        return stage;\n      }\n    }\n    throw new Error(`Pipeline does not contain a stage named '${stageName}'. Available stages: ${this._stages.map(s => s.stageName).join(', ')}`);\n  }\n\n  /**\n   * Returns all of the {@link CrossRegionSupportStack}s that were generated automatically\n   * when dealing with Actions that reside in a different region than the Pipeline itself.\n   *\n   */\n  public get crossRegionSupport(): { [region: string]: CrossRegionSupport } {\n    const ret: { [region: string]: CrossRegionSupport } = {};\n    Object.keys(this._crossRegionSupport).forEach((key) => {\n      ret[key] = this._crossRegionSupport[key];\n    });\n    return ret;\n  }\n\n  /** @internal */\n  public _attachActionToPipeline(stage: Stage, action: IAction, actionScope: Construct): FullActionDescriptor {\n    const richAction = new RichAction(action, this);\n\n    // handle cross-region actions here\n    const crossRegionInfo = this.ensureReplicationResourcesExistFor(richAction);\n\n    // get the role for the given action, handling if it's cross-account\n    const actionRole = this.getRoleForAction(stage, richAction, actionScope);\n\n    // // CodePipeline Variables\n    validateNamespaceName(richAction.actionProperties.variablesNamespace);\n\n    // bind the Action (type h4x)\n    const actionConfig = richAction.bind(actionScope as CoreConstruct, stage, {\n      role: actionRole ? actionRole : this.role,\n      bucket: crossRegionInfo.artifactBucket,\n    });\n\n    return new FullActionDescriptor({\n      // must be 'action', not 'richAction',\n      // as those are returned by the IStage.actions property,\n      // and it's important customers of Pipeline get the same instance\n      // back as they added to the pipeline\n      action,\n      actionConfig,\n      actionRole,\n      actionRegion: crossRegionInfo.region,\n    });\n  }\n\n  /**\n   * Validate the pipeline structure\n   *\n   * Validation happens according to the rules documented at\n   *\n   * https://docs.aws.amazon.com/codepipeline/latest/userguide/reference-pipeline-structure.html#pipeline-requirements\n   * @override\n   */\n  protected validate(): string[] {\n    return [\n      ...this.validateSourceActionLocations(),\n      ...this.validateHasStages(),\n      ...this.validateStages(),\n      ...this.validateArtifacts(),\n    ];\n  }\n\n  private ensureReplicationResourcesExistFor(action: RichAction): CrossRegionInfo {\n    if (!action.isCrossRegion) {\n      return {\n        artifactBucket: this.artifactBucket,\n      };\n    }\n\n    // The action has a specific region,\n    // require the pipeline to have a known region as well.\n    this.requireRegion();\n\n    // source actions have to be in the same region as the pipeline\n    if (action.actionProperties.category === ActionCategory.SOURCE) {\n      throw new Error(`Source action '${action.actionProperties.actionName}' must be in the same region as the pipeline`);\n    }\n\n    // check whether we already have a bucket in that region,\n    // either passed from the outside or previously created\n    const crossRegionSupport = this.obtainCrossRegionSupportFor(action);\n\n    // the stack containing the replication bucket must be deployed before the pipeline\n    Stack.of(this).addDependency(crossRegionSupport.stack);\n    // The Pipeline role must be able to replicate to that bucket\n    crossRegionSupport.replicationBucket.grantReadWrite(this.role);\n\n    return {\n      artifactBucket: crossRegionSupport.replicationBucket,\n      region: action.effectiveRegion,\n    };\n  }\n\n  /**\n   * Get or create the cross-region support construct for the given action\n   */\n  private obtainCrossRegionSupportFor(action: RichAction) {\n    // this method is never called for non cross-region actions\n    const actionRegion = action.effectiveRegion!;\n    let crossRegionSupport = this._crossRegionSupport[actionRegion];\n    if (!crossRegionSupport) {\n      // we need to create scaffolding resources for this region\n      const otherStack = action.resourceStack;\n      crossRegionSupport = this.createSupportResourcesForRegion(otherStack, actionRegion);\n      this._crossRegionSupport[actionRegion] = crossRegionSupport;\n    }\n    return crossRegionSupport;\n  }\n\n  private createSupportResourcesForRegion(otherStack: Stack | undefined, actionRegion: string): CrossRegionSupport {\n    // if we have a stack from the resource passed - use that!\n    if (otherStack) {\n      // check if the stack doesn't have this magic construct already\n      const id = `CrossRegionReplicationSupport-d823f1d8-a990-4e5c-be18-4ac698532e65-${actionRegion}`;\n      let crossRegionSupportConstruct = otherStack.node.tryFindChild(id) as CrossRegionSupportConstruct;\n      if (!crossRegionSupportConstruct) {\n        crossRegionSupportConstruct = new CrossRegionSupportConstruct(otherStack, id, {\n          createKmsKey: this.crossAccountKeys,\n          enableKeyRotation: this.enableKeyRotation,\n        });\n      }\n\n      return {\n        replicationBucket: crossRegionSupportConstruct.replicationBucket,\n        stack: otherStack,\n      };\n    }\n\n    // otherwise - create a stack with the resources needed for replication across regions\n    const pipelineStack = Stack.of(this);\n    const pipelineAccount = pipelineStack.account;\n    if (Token.isUnresolved(pipelineAccount)) {\n      throw new Error(\"You need to specify an explicit account when using CodePipeline's cross-region support\");\n    }\n\n    const app = this.supportScope();\n    const supportStackId = `cross-region-stack-${this.reuseCrossRegionSupportStacks ? pipelineAccount : pipelineStack.stackName}:${actionRegion}`;\n    let supportStack = app.node.tryFindChild(supportStackId) as CrossRegionSupportStack;\n    if (!supportStack) {\n      supportStack = new CrossRegionSupportStack(app, supportStackId, {\n        pipelineStackName: pipelineStack.stackName,\n        region: actionRegion,\n        account: pipelineAccount,\n        synthesizer: this.getCrossRegionSupportSynthesizer(),\n        createKmsKey: this.crossAccountKeys,\n        enableKeyRotation: this.enableKeyRotation,\n      });\n    }\n\n    return {\n      stack: supportStack,\n      replicationBucket: supportStack.replicationBucket,\n    };\n  }\n\n  private getCrossRegionSupportSynthesizer(): IStackSynthesizer | undefined {\n    if (this.stack.synthesizer instanceof DefaultStackSynthesizer) {\n      // if we have the new synthesizer,\n      // we need a bootstrapless copy of it,\n      // because we don't want to require bootstrapping the environment\n      // of the pipeline account in this replication region\n      return new BootstraplessSynthesizer({\n        deployRoleArn: this.stack.synthesizer.deployRoleArn,\n        cloudFormationExecutionRoleArn: this.stack.synthesizer.cloudFormationExecutionRoleArn,\n      });\n    } else {\n      // any other synthesizer: just return undefined\n      // (ie., use the default based on the context settings)\n      return undefined;\n    }\n  }\n\n  private generateNameForDefaultBucketKeyAlias(): string {\n    const prefix = 'alias/codepipeline-';\n    const maxAliasLength = 256;\n    const uniqueId = Names.uniqueId(this);\n    // take the last 256 - (prefix length) characters of uniqueId\n    const startIndex = Math.max(0, uniqueId.length - (maxAliasLength - prefix.length));\n    return prefix + uniqueId.substring(startIndex).toLowerCase();\n  }\n\n  /**\n   * Gets the role used for this action,\n   * including handling the case when the action is supposed to be cross-account.\n   *\n   * @param stage the stage the action belongs to\n   * @param action the action to return/create a role for\n   * @param actionScope the scope, unique to the action, to create new resources in\n   */\n  private getRoleForAction(stage: Stage, action: RichAction, actionScope: Construct): iam.IRole | undefined {\n    const pipelineStack = Stack.of(this);\n\n    let actionRole = this.getRoleFromActionPropsOrGenerateIfCrossAccount(stage, action);\n\n    if (!actionRole && this.isAwsOwned(action)) {\n      // generate a Role for this specific Action\n      actionRole = new iam.Role(actionScope, 'CodePipelineActionRole', {\n        assumedBy: new iam.AccountPrincipal(pipelineStack.account),\n      });\n    }\n\n    // the pipeline role needs assumeRole permissions to the action role\n    const grant = actionRole?.grantAssumeRole(this.role);\n    grant?.applyBefore(this.codePipeline);\n    return actionRole;\n  }\n\n  private getRoleFromActionPropsOrGenerateIfCrossAccount(stage: Stage, action: RichAction): iam.IRole | undefined {\n    const pipelineStack = Stack.of(this);\n\n    // if we have a cross-account action, the pipeline's bucket must have a KMS key\n    // (otherwise we can't configure cross-account trust policies)\n    if (action.isCrossAccount) {\n      const artifactBucket = this.ensureReplicationResourcesExistFor(action).artifactBucket;\n      if (!artifactBucket.encryptionKey) {\n        throw new Error(\n          `Artifact Bucket must have a KMS Key to add cross-account action '${action.actionProperties.actionName}' ` +\n          `(pipeline account: '${renderEnvDimension(this.env.account)}', action account: '${renderEnvDimension(action.effectiveAccount)}'). ` +\n          'Create Pipeline with \\'crossAccountKeys: true\\' (or pass an existing Bucket with a key)',\n        );\n      }\n    }\n\n    // if a Role has been passed explicitly, always use it\n    // (even if the backing resource is from a different account -\n    // this is how the user can override our default support logic)\n    if (action.actionProperties.role) {\n      if (this.isAwsOwned(action)) {\n        // the role has to be deployed before the pipeline\n        // (our magical cross-stack dependencies will not work,\n        // because the role might be from a different environment),\n        // but _only_ if it's a new Role -\n        // an imported Role should not add the dependency\n        if (action.actionProperties.role instanceof iam.Role) {\n          const roleStack = Stack.of(action.actionProperties.role);\n          pipelineStack.addDependency(roleStack);\n        }\n\n        return action.actionProperties.role;\n      } else {\n        // ...except if the Action is not owned by 'AWS',\n        // as that would be rejected by CodePipeline at deploy time\n        throw new Error(\"Specifying a Role is not supported for actions with an owner different than 'AWS' - \" +\n          `got '${action.actionProperties.owner}' (Action: '${action.actionProperties.actionName}' in Stage: '${stage.stageName}')`);\n      }\n    }\n\n    // if we don't have a Role passed,\n    // and the action is cross-account,\n    // generate a Role in that other account stack\n    const otherAccountStack = this.getOtherStackIfActionIsCrossAccount(action);\n    if (!otherAccountStack) {\n      return undefined;\n    }\n\n    // generate a role in the other stack, that the Pipeline will assume for executing this action\n    const ret = new iam.Role(otherAccountStack,\n      `${Names.uniqueId(this)}-${stage.stageName}-${action.actionProperties.actionName}-ActionRole`, {\n        assumedBy: new iam.AccountPrincipal(pipelineStack.account),\n        roleName: PhysicalName.GENERATE_IF_NEEDED,\n      });\n    // the other stack with the role has to be deployed before the pipeline stack\n    // (CodePipeline verifies you can assume the action Role on creation)\n    pipelineStack.addDependency(otherAccountStack);\n\n    return ret;\n  }\n\n  /**\n   * Returns the Stack this Action belongs to if this is a cross-account Action.\n   * If this Action is not cross-account (i.e., it lives in the same account as the Pipeline),\n   * it returns undefined.\n   *\n   * @param action the Action to return the Stack for\n   */\n  private getOtherStackIfActionIsCrossAccount(action: IAction): Stack | undefined {\n    const targetAccount = action.actionProperties.resource\n      ? action.actionProperties.resource.env.account\n      : action.actionProperties.account;\n\n    if (targetAccount === undefined) {\n      // if the account of the Action is not specified,\n      // then it defaults to the same account the pipeline itself is in\n      return undefined;\n    }\n\n    // check whether the action's account is a static string\n    if (Token.isUnresolved(targetAccount)) {\n      if (Token.isUnresolved(this.env.account)) {\n        // the pipeline is also env-agnostic, so that's fine\n        return undefined;\n      } else {\n        throw new Error(`The 'account' property must be a concrete value (action: '${action.actionProperties.actionName}')`);\n      }\n    }\n\n    // At this point, we know that the action's account is a static string.\n    // In this case, the pipeline's account must also be a static string.\n    if (Token.isUnresolved(this.env.account)) {\n      throw new Error('Pipeline stack which uses cross-environment actions must have an explicitly set account');\n    }\n\n    // at this point, we know that both the Pipeline's account,\n    // and the action-backing resource's account are static strings\n\n    // if they are identical - nothing to do (the action is not cross-account)\n    if (this.env.account === targetAccount) {\n      return undefined;\n    }\n\n    // at this point, we know that the action is certainly cross-account,\n    // so we need to return a Stack in its account to create the helper Role in\n\n    const candidateActionResourceStack = action.actionProperties.resource\n      ? Stack.of(action.actionProperties.resource)\n      : undefined;\n    if (candidateActionResourceStack?.account === targetAccount) {\n      // we always use the \"latest\" action-backing resource's Stack for this account,\n      // even if a different one was used earlier\n      this._crossAccountSupport[targetAccount] = candidateActionResourceStack;\n      return candidateActionResourceStack;\n    }\n\n    let targetAccountStack: Stack | undefined = this._crossAccountSupport[targetAccount];\n    if (!targetAccountStack) {\n      const stackId = `cross-account-support-stack-${targetAccount}`;\n      const app = this.supportScope();\n      targetAccountStack = app.node.tryFindChild(stackId) as Stack;\n      if (!targetAccountStack) {\n        const actionRegion = action.actionProperties.resource\n          ? action.actionProperties.resource.env.region\n          : action.actionProperties.region;\n        const pipelineStack = Stack.of(this);\n        targetAccountStack = new Stack(app, stackId, {\n          stackName: `${pipelineStack.stackName}-support-${targetAccount}`,\n          env: {\n            account: targetAccount,\n            region: actionRegion ?? pipelineStack.region,\n          },\n        });\n      }\n      this._crossAccountSupport[targetAccount] = targetAccountStack;\n    }\n    return targetAccountStack;\n  }\n\n  private isAwsOwned(action: IAction) {\n    const owner = action.actionProperties.owner;\n    return !owner || owner === 'AWS';\n  }\n\n  private getArtifactBucketFromProps(props: PipelineProps): s3.IBucket | undefined {\n    if (props.artifactBucket) {\n      return props.artifactBucket;\n    }\n    if (props.crossRegionReplicationBuckets) {\n      const pipelineRegion = this.requireRegion();\n      return props.crossRegionReplicationBuckets[pipelineRegion];\n    }\n    return undefined;\n  }\n\n  private calculateInsertIndexFromPlacement(placement: StagePlacement): number {\n    // check if at most one placement property was provided\n    const providedPlacementProps = ['rightBefore', 'justAfter', 'atIndex']\n      .filter((prop) => (placement as any)[prop] !== undefined);\n    if (providedPlacementProps.length > 1) {\n      throw new Error('Error adding Stage to the Pipeline: ' +\n        'you can only provide at most one placement property, but ' +\n        `'${providedPlacementProps.join(', ')}' were given`);\n    }\n\n    if (placement.rightBefore !== undefined) {\n      const targetIndex = this.findStageIndex(placement.rightBefore);\n      if (targetIndex === -1) {\n        throw new Error('Error adding Stage to the Pipeline: ' +\n          `the requested Stage to add it before, '${placement.rightBefore.stageName}', was not found`);\n      }\n      return targetIndex;\n    }\n\n    if (placement.justAfter !== undefined) {\n      const targetIndex = this.findStageIndex(placement.justAfter);\n      if (targetIndex === -1) {\n        throw new Error('Error adding Stage to the Pipeline: ' +\n          `the requested Stage to add it after, '${placement.justAfter.stageName}', was not found`);\n      }\n      return targetIndex + 1;\n    }\n\n    return this.stageCount;\n  }\n\n  private findStageIndex(targetStage: IStage) {\n    return this._stages.findIndex(stage => stage === targetStage);\n  }\n\n  private validateSourceActionLocations(): string[] {\n    const errors = new Array<string>();\n    let firstStage = true;\n    for (const stage of this._stages) {\n      const onlySourceActionsPermitted = firstStage;\n      for (const action of stage.actionDescriptors) {\n        errors.push(...validateSourceAction(onlySourceActionsPermitted, action.category, action.actionName, stage.stageName));\n      }\n      firstStage = false;\n    }\n    return errors;\n  }\n\n  private validateHasStages(): string[] {\n    if (this.stageCount < 2) {\n      return ['Pipeline must have at least two stages'];\n    }\n    return [];\n  }\n\n  private validateStages(): string[] {\n    const ret = new Array<string>();\n    for (const stage of this._stages) {\n      ret.push(...stage.validate());\n    }\n    return ret;\n  }\n\n  private validateArtifacts(): string[] {\n    const ret = new Array<string>();\n\n    const producers: Record<string, PipelineLocation> = {};\n    const firstConsumers: Record<string, PipelineLocation> = {};\n\n    for (const [stageIndex, stage] of enumerate(this._stages)) {\n      // For every output artifact, get the producer\n      for (const action of stage.actionDescriptors) {\n        const actionLoc = new PipelineLocation(stageIndex, stage, action);\n\n        for (const outputArtifact of action.outputs) {\n          // output Artifacts always have a name set\n          const name = outputArtifact.artifactName!;\n          if (producers[name]) {\n            ret.push(`Both Actions '${producers[name].actionName}' and '${action.actionName}' are producting Artifact '${name}'. Every artifact can only be produced once.`);\n            continue;\n          }\n\n          producers[name] = actionLoc;\n        }\n\n        // For every input artifact, get the first consumer\n        for (const inputArtifact of action.inputs) {\n          const name = inputArtifact.artifactName;\n          if (!name) {\n            ret.push(`Action '${action.actionName}' is using an unnamed input Artifact, which is not being produced in this pipeline`);\n            continue;\n          }\n\n          firstConsumers[name] = firstConsumers[name] ? firstConsumers[name].first(actionLoc) : actionLoc;\n        }\n      }\n    }\n\n    // Now validate that every input artifact is produced before it's\n    // being consumed.\n    for (const [artifactName, consumerLoc] of Object.entries(firstConsumers)) {\n      const producerLoc = producers[artifactName];\n      if (!producerLoc) {\n        ret.push(`Action '${consumerLoc.actionName}' is using input Artifact '${artifactName}', which is not being produced in this pipeline`);\n        continue;\n      }\n\n      if (consumerLoc.beforeOrEqual(producerLoc)) {\n        ret.push(`${consumerLoc} is consuming input Artifact '${artifactName}' before it is being produced at ${producerLoc}`);\n      }\n    }\n\n    return ret;\n  }\n\n  private renderArtifactStoresProperty(): CfnPipeline.ArtifactStoreMapProperty[] | undefined {\n    if (!this.crossRegion) { return undefined; }\n\n    // add the Pipeline's artifact store\n    const primaryRegion = this.requireRegion();\n    this._crossRegionSupport[primaryRegion] = {\n      replicationBucket: this.artifactBucket,\n      stack: Stack.of(this),\n    };\n\n    return Object.entries(this._crossRegionSupport).map(([region, support]) => ({\n      region,\n      artifactStore: this.renderArtifactStore(support.replicationBucket),\n    }));\n  }\n\n  private renderArtifactStoreProperty(): CfnPipeline.ArtifactStoreProperty | undefined {\n    if (this.crossRegion) { return undefined; }\n    return this.renderPrimaryArtifactStore();\n  }\n\n  private renderPrimaryArtifactStore(): CfnPipeline.ArtifactStoreProperty {\n    return this.renderArtifactStore(this.artifactBucket);\n  }\n\n  private renderArtifactStore(bucket: s3.IBucket): CfnPipeline.ArtifactStoreProperty {\n    let encryptionKey: CfnPipeline.EncryptionKeyProperty | undefined;\n    const bucketKey = bucket.encryptionKey;\n    if (bucketKey) {\n      encryptionKey = {\n        type: 'KMS',\n        id: bucketKey.keyArn,\n      };\n    }\n\n    return {\n      type: 'S3',\n      location: bucket.bucketName,\n      encryptionKey,\n    };\n  }\n\n  private get crossRegion(): boolean {\n    if (this.crossRegionBucketsPassed) { return true; }\n    return this._stages.some(stage => stage.actionDescriptors.some(action => action.region !== undefined));\n  }\n\n  private renderStages(): CfnPipeline.StageDeclarationProperty[] {\n    return this._stages.map(stage => stage.render());\n  }\n\n  private renderDisabledTransitions(): CfnPipeline.StageTransitionProperty[] {\n    return this._stages\n      .filter(stage => !stage.transitionToEnabled)\n      .map(stage => ({\n        reason: stage.transitionDisabledReason,\n        stageName: stage.stageName,\n      }));\n  }\n\n  private requireRegion(): string {\n    const region = this.env.region;\n    if (Token.isUnresolved(region)) {\n      throw new Error('Pipeline stack which uses cross-environment actions must have an explicitly set region');\n    }\n    return region;\n  }\n\n  private supportScope(): CdkStage {\n    const scope = CdkStage.of(this);\n    if (!scope) {\n      throw new Error('Pipeline stack which uses cross-environment actions must be part of a CDK App or Stage');\n    }\n    return scope;\n  }\n}\n\n/**\n * An interface representing resources generated in order to support\n * the cross-region capabilities of CodePipeline.\n * You get instances of this interface from the {@link Pipeline#crossRegionSupport} property.\n *\n */\nexport interface CrossRegionSupport {\n  /**\n   * The Stack that has been created to house the replication Bucket\n   * required for this  region.\n   */\n  readonly stack: Stack;\n\n  /**\n   * The replication Bucket used by CodePipeline to operate in this region.\n   * Belongs to {@link stack}.\n   */\n  readonly replicationBucket: s3.IBucket;\n}\n\ninterface CrossRegionInfo {\n  readonly artifactBucket: s3.IBucket;\n\n  readonly region?: string;\n}\n\nfunction enumerate<A>(xs: A[]): Array<[number, A]> {\n  const ret = new Array<[number, A]>();\n  for (let i = 0; i < xs.length; i++) {\n    ret.push([i, xs[i]]);\n  }\n  return ret;\n}\n\nclass PipelineLocation {\n  constructor(private readonly stageIndex: number, private readonly stage: IStage, private readonly action: FullActionDescriptor) {\n  }\n\n  public get stageName() {\n    return this.stage.stageName;\n  }\n\n  public get actionName() {\n    return this.action.actionName;\n  }\n\n  /**\n   * Returns whether a is before or the same order as b\n   */\n  public beforeOrEqual(rhs: PipelineLocation) {\n    if (this.stageIndex !== rhs.stageIndex) { return rhs.stageIndex < rhs.stageIndex; }\n    return this.action.runOrder <= rhs.action.runOrder;\n  }\n\n  /**\n   * Returns the first location between this and the other one\n   */\n  public first(rhs: PipelineLocation) {\n    return this.beforeOrEqual(rhs) ? this : rhs;\n  }\n\n  public toString() {\n    // runOrders are 1-based, so make the stageIndex also 1-based otherwise it's going to be confusing.\n    return `Stage ${this.stageIndex + 1} Action ${this.action.runOrder} ('${this.stageName}'/'${this.actionName}')`;\n  }\n}\n\n/**\n * Render an env dimension without showing the ugly stringified tokens\n */\nfunction renderEnvDimension(s: string | undefined) {\n  return Token.isUnresolved(s) ? '(current)' : s;\n}\n"]} |
\ | No newline at end of file |