aws-ddk-core
Version:
The AWS DataOps Development Kit is an open source development framework for customers that build data workflows and modern data architecture on AWS.
261 lines (257 loc) • 42.5 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CICDPipelineStack = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const codestarnotifications = require("aws-cdk-lib/aws-codestarnotifications");
const sns = require("aws-cdk-lib/aws-sns");
const pipelines = require("aws-cdk-lib/pipelines");
const actions_1 = require("./actions");
const utils_1 = require("./utils");
const base_1 = require("../base");
const config_1 = require("../config");
/**
* Create a stack that contains DDK Continuous Integration and Delivery (CI/CD) pipeline.
The pipeline is based on
[CDK self-mutating pipeline](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.pipelines-readme.html)
but includes several DDK-specific features, including:
- Ability to configure some properties via JSON config e.g. manual approvals for application stages
- Defaults for source/synth - CodeCommit & cdk synth, with ability to override them
- Ability to connect to private artifactory to pull artifacts from at synth
- Security best practices - ensures pipeline buckets block non-SSL, and are KMS-encrypted with rotated keys
- Builder interface to avoid chunky constructor methods
The user should be able to reuse the pipeline in multiple DDK applications hoping to save LOC.
@example
const stack = new CICDPipelineStack(app, "dummy-pipeline", { environmentId: "dev", pipelineName: "dummy-pipeline" })
.addSourceAction({ repositoryName: "dummy-repository" })
.addSynthAction()
.buildPipeline()
.add_checks()
.addStage({ stageId: "dev", stage: devStage, manualApprovals: true })
.synth()
.add_notifications();
*/
class CICDPipelineStack extends base_1.BaseStack {
/**
* Creates a new CICD Pipeline stack.
*
* @param scope Parent of this stack, usually an `App` or a `Stage`, but could be any construct.
* @param id The construct ID of this stack. If `stackName` is not explicitly
* defined, this id (and any parent IDs) will be used to determine the
* physical ID of the stack.
* @param props Stack properties.
*/
constructor(scope, id, props) {
super(scope, id, props);
this.environmentId = props.environmentId;
this.pipelineName = props.pipelineName;
this.pipelineId = id;
const config = props.config ?? "./ddk.json";
this.config = new config_1.Configurator(this, config, this.environmentId);
this.cdkLanguage = props.cdkLanguage ?? "typescript";
}
/**
* Add source action.
*
* @param props Source action properties.
* @returns reference to this pipeline.
*/
addSourceAction(props) {
var branch = props.branch ?? "main";
this.sourceAction =
props.sourceAction ||
actions_1.CICDActions.getCodeCommitSourceAction(this, {
repositoryName: props.repositoryName,
branch: branch,
});
return this;
}
/**
* Build the pipeline structure.
* @param props Additional pipeline properties.
* @returns reference to this pipeline.
*/
buildPipeline(props = {}) {
if (this.synthAction === undefined) {
throw new Error("Pipeline cannot be built without a synth action.");
}
this.pipeline = new pipelines.CodePipeline(this, "DDKCodePipeline", {
synth: this.synthAction,
crossAccountKeys: true,
pipelineName: this.pipelineName,
...props,
});
return this;
}
/**
* Add synth action. During synth can connect and pull artifacts from a private artifactory.
* @param props Synth action properties.
* @returns reference to this pipeline.
*/
addSynthAction(props = {}) {
const languageInstallCommands = {
typescript: "npm install",
python: "pip install -r requirements.txt",
};
let languageInstallCommand = languageInstallCommands[this.cdkLanguage];
if (props.cdkLanguageCommandLineArguments) {
for (const [argument, value] of Object.entries(props.cdkLanguageCommandLineArguments)) {
languageInstallCommand += ` ${argument} ${value}`;
}
}
this.synthAction =
props.synthAction ||
actions_1.CICDActions.getSynthAction({
codePipelineSource: this.sourceAction,
cdkVersion: props.cdkVersion,
partition: this.partition,
region: this.region,
account: this.account,
env: props.env,
rolePolicyStatements: props.rolePolicyStatements,
codeartifactRepository: props.codeartifactRepository,
codeartifactDomain: props.codeartifactDomain,
codeartifactDomainOwner: props.codeartifactDomainOwner,
additionalInstallCommands: props.additionalInstallCommands
? [languageInstallCommand].concat(props.additionalInstallCommands)
: [languageInstallCommand],
});
return this;
}
/**
* Add application stage to the CICD pipeline. This stage deploys your application infrastructure.
* @param props Application stage properties.
* @returns reference to this pipeline.
*/
addStage(props) {
if (this.pipeline === undefined) {
throw new Error("`.buildPipeline()` needs to be called first before adding application stages to the pipeline.");
}
const manualApprovals = props.manualApprovals ?? this.config.getConfigAttribute("manual_approvals") ?? false;
if (manualApprovals) {
this.pipeline?.addStage(props.stage, {
pre: [new pipelines.ManualApprovalStep("PromoteTo" + utils_1.toTitleCase(props.stageId))],
});
}
else {
this.pipeline?.addStage(props.stage, {});
}
return this;
}
/**
* Add multiple application stages in parallel to the CICD pipeline.
* @param props Application wave properties.
* @returns reference to this pipeline.
*/
addWave(props) {
if (this.pipeline === undefined) {
throw new Error("`.buildPipeline()` needs to be called first before adding application stages to the pipeline.");
}
const manualApprovals = props.manualApprovals ?? this.config.getConfigAttribute("manual_approvals") ?? false;
var wave = new pipelines.Wave(props.stageId);
if (manualApprovals) {
wave.addPre(new pipelines.ManualApprovalStep("PromoteTo" + utils_1.toTitleCase(props.stageId)));
}
props.stages.forEach((stage) => {
wave.addStage(stage);
});
this.pipeline?.addWave(props.stageId, wave);
return this;
}
/**
* Add linting - cfn-nag, and bandit.
* @param props Security lint properties.
* @returns reference to this pipeline.
*/
addSecurityLintStage(props) {
if (this.sourceAction === undefined) {
throw new Error("Source Action Must Be configured before calling this method.");
}
if (this.pipeline?.cloudAssemblyFileSet === undefined) {
throw new Error("No cloudAssemblyFileSet configured, source action needs to be configured for this pipeline.");
}
var stageName = props.stageName ?? "SecurityLint";
var cloudAssemblyFileSet = props.cloudAssemblyFileSet ?? this.pipeline?.cloudAssemblyFileSet;
this.pipeline?.addWave(stageName, {
post: [
actions_1.CICDActions.getCfnNagAction(cloudAssemblyFileSet, "CFNNag", props.cfnNagFailBuild),
actions_1.CICDActions.getBanditAction(this.sourceAction),
],
});
return this;
}
/**
* Add test - e.g. pytest.
* @param props Test stage properties.
* @returns reference to this pipeline.
*/
addTestStage(props) {
var stageName = props.stageName ?? "Tests";
var cloudAssemblyFileSet = props.cloudAssemblyFileSet ?? this.pipeline?.cloudAssemblyFileSet;
var commands = props.commands ?? ["./test.sh"];
if (cloudAssemblyFileSet === undefined) {
throw new Error("No cloudAssemblyFileSet configured, source action needs to be configured for this pipeline.");
}
this.pipeline?.addWave(stageName || "Tests", {
post: [actions_1.CICDActions.getTestsAction(cloudAssemblyFileSet, commands)],
});
return this;
}
/**
* Add pipeline notifications.
* Create notification rule that sends events to the specified SNS topic.
* @param props Notification properties.
* @returns reference to this pipeline.
*/
addNotifications(props = {}) {
if (this.pipeline === undefined) {
throw new Error("`.buildPipeline()` needs to be called first before adding notifications to the pipeline.");
}
const topic = this.environmentId && this.config.getConfigAttribute("notifications_topic_arn")
? sns.Topic.fromTopicArn(this, "ExecutionFailedNotifications", this.config.getConfigAttribute("notifications_topic_arn"))
: new sns.Topic(this, "ExecutionFailedNotifications");
this.notificationRule =
props.notificationRule ??
new codestarnotifications.NotificationRule(this, "Notification", {
detailType: codestarnotifications.DetailType.BASIC,
events: ["codepipeline-pipeline-pipeline-execution-failed"],
source: this.pipeline?.pipeline,
targets: [topic],
});
return this;
}
/**
* Add checks to the pipeline (e.g. linting, security, tests...).
* @returns reference to this pipeline.
*/
addChecks() {
this.addSecurityLintStage({});
this.addTestStage({});
return this;
}
/**
* Add custom stage to the pipeline.
* @param props Properties for adding a custom stage.
* @returns reference to this pipeline.
*/
addCustomStage(props) {
this.pipeline?.addWave(props.stageName, {
post: props.steps,
});
return this;
}
/**
* Synthesize the pipeline.
* @returns reference to this pipeline.
*/
synth() {
this.pipeline?.buildPipeline();
this.pipelineKey = this.pipeline?.pipeline.artifactBucket.encryptionKey?.node.defaultChild;
this.pipelineKey.addPropertyOverride("EnableKeyRotation", true);
return this;
}
}
exports.CICDPipelineStack = CICDPipelineStack;
_a = JSII_RTTI_SYMBOL_1;
CICDPipelineStack[_a] = { fqn: "aws-ddk-core.CICDPipelineStack", version: "1.4.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipelines.js","sourceRoot":"","sources":["../../src/cicd/pipelines.ts"],"names":[],"mappings":";;;;;AAEA,+EAA+E;AAG/E,2CAA2C;AAC3C,mDAAmD;AAEnD,uCAAwC;AACxC,mCAAsC;AACtC,kCAAoD;AACpD,sCAAyC;AAgRzC;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,MAAa,iBAAkB,SAAQ,gBAAS;IAa9C;;;;;;;;OAQG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QAExB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,aAAa,CAAC;QACzC,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC;QACvC,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QAErB,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,IAAI,YAAY,CAAC;QAC5C,IAAI,CAAC,MAAM,GAAG,IAAI,qBAAY,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;QAEjE,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,IAAI,YAAY,CAAC;IACvD,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,KAAwB;QACtC,IAAI,MAAM,GAAG,KAAK,CAAC,MAAM,IAAI,MAAM,CAAC;QACpC,IAAI,CAAC,YAAY;YACf,KAAK,CAAC,YAAY;gBAClB,qBAAW,CAAC,yBAAyB,CAAC,IAAI,EAAE;oBAC1C,cAAc,EAAE,KAAK,CAAC,cAAc;oBACpC,MAAM,EAAE,MAAM;iBACf,CAAC,CAAC;QACL,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,aAAa,CAAC,QAAiC,EAAE;QAC/C,IAAI,IAAI,CAAC,WAAW,KAAK,SAAS,EAAE;YAClC,MAAM,IAAI,KAAK,CAAC,kDAAkD,CAAC,CAAC;SACrE;QACD,IAAI,CAAC,QAAQ,GAAG,IAAI,SAAS,CAAC,YAAY,CAAC,IAAI,EAAE,iBAAiB,EAAE;YAClE,KAAK,EAAE,IAAI,CAAC,WAAW;YACvB,gBAAgB,EAAE,IAAI;YACtB,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,GAAG,KAAK;SACT,CAAC,CAAC;QACH,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,QAA0B,EAAE;QACzC,MAAM,uBAAuB,GAAQ;YACnC,UAAU,EAAE,aAAa;YACzB,MAAM,EAAE,iCAAiC;SAC1C,CAAC;QAEF,IAAI,sBAAsB,GAAG,uBAAuB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACvE,IAAI,KAAK,CAAC,+BAA+B,EAAE;YACzC,KAAK,MAAM,CAAC,QAAQ,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,+BAA+B,CAAC,EAAE;gBACrF,sBAAsB,IAAI,IAAI,QAAQ,IAAI,KAAK,EAAE,CAAC;aACnD;SACF;QAED,IAAI,CAAC,WAAW;YACd,KAAK,CAAC,WAAW;gBACjB,qBAAW,CAAC,cAAc,CAAC;oBACzB,kBAAkB,EAAE,IAAI,CAAC,YAAY;oBACrC,UAAU,EAAE,KAAK,CAAC,UAAU;oBAC5B,SAAS,EAAE,IAAI,CAAC,SAAS;oBACzB,MAAM,EAAE,IAAI,CAAC,MAAM;oBACnB,OAAO,EAAE,IAAI,CAAC,OAAO;oBACrB,GAAG,EAAE,KAAK,CAAC,GAAG;oBACd,oBAAoB,EAAE,KAAK,CAAC,oBAAoB;oBAChD,sBAAsB,EAAE,KAAK,CAAC,sBAAsB;oBACpD,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;oBAC5C,uBAAuB,EAAE,KAAK,CAAC,uBAAuB;oBACtD,yBAAyB,EAAE,KAAK,CAAC,yBAAyB;wBACxD,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,yBAAyB,CAAC;wBAClE,CAAC,CAAC,CAAC,sBAAsB,CAAC;iBAC7B,CAAC,CAAC;QACL,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,QAAQ,CAAC,KAA+B;QACtC,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS,EAAE;YAC/B,MAAM,IAAI,KAAK,CAAC,+FAA+F,CAAC,CAAC;SAClH;QACD,MAAM,eAAe,GAAG,KAAK,CAAC,eAAe,IAAI,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAC;QAE7G,IAAI,eAAe,EAAE;YACnB,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,KAAK,CAAC,KAAK,EAAE;gBACnC,GAAG,EAAE,CAAC,IAAI,SAAS,CAAC,kBAAkB,CAAC,WAAW,GAAG,mBAAW,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC;aAClF,CAAC,CAAC;SACJ;aAAM;YACL,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;SAC1C;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,OAAO,CAAC,KAA8B;QACpC,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS,EAAE;YAC/B,MAAM,IAAI,KAAK,CAAC,+FAA+F,CAAC,CAAC;SAClH;QACD,MAAM,eAAe,GAAG,KAAK,CAAC,eAAe,IAAI,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAC;QAE7G,IAAI,IAAI,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC7C,IAAI,eAAe,EAAE;YACnB,IAAI,CAAC,MAAM,CAAC,IAAI,SAAS,CAAC,kBAAkB,CAAC,WAAW,GAAG,mBAAW,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;SACzF;QAED,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YAC7B,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACvB,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QAC5C,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,oBAAoB,CAAC,KAAgC;QACnD,IAAI,IAAI,CAAC,YAAY,KAAK,SAAS,EAAE;YACnC,MAAM,IAAI,KAAK,CAAC,8DAA8D,CAAC,CAAC;SACjF;QACD,IAAI,IAAI,CAAC,QAAQ,EAAE,oBAAoB,KAAK,SAAS,EAAE;YACrD,MAAM,IAAI,KAAK,CAAC,6FAA6F,CAAC,CAAC;SAChH;QAED,IAAI,SAAS,GAAG,KAAK,CAAC,SAAS,IAAI,cAAc,CAAC;QAClD,IAAI,oBAAoB,GAAG,KAAK,CAAC,oBAAoB,IAAI,IAAI,CAAC,QAAQ,EAAE,oBAAoB,CAAC;QAE7F,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,SAAS,EAAE;YAChC,IAAI,EAAE;gBACJ,qBAAW,CAAC,eAAe,CAAC,oBAAoB,EAAE,QAAQ,EAAE,KAAK,CAAC,eAAe,CAAC;gBAClF,qBAAW,CAAC,eAAe,CAAC,IAAI,CAAC,YAAY,CAAC;aAC/C;SACF,CAAC,CAAC;QAEH,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,YAAY,CAAC,KAAwB;QACnC,IAAI,SAAS,GAAG,KAAK,CAAC,SAAS,IAAI,OAAO,CAAC;QAC3C,IAAI,oBAAoB,GAAG,KAAK,CAAC,oBAAoB,IAAI,IAAI,CAAC,QAAQ,EAAE,oBAAoB,CAAC;QAC7F,IAAI,QAAQ,GAAG,KAAK,CAAC,QAAQ,IAAI,CAAC,WAAW,CAAC,CAAC;QAE/C,IAAI,oBAAoB,KAAK,SAAS,EAAE;YACtC,MAAM,IAAI,KAAK,CAAC,6FAA6F,CAAC,CAAC;SAChH;QAED,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,SAAS,IAAI,OAAO,EAAE;YAC3C,IAAI,EAAE,CAAC,qBAAW,CAAC,cAAc,CAAC,oBAAoB,EAAE,QAAQ,CAAC,CAAC;SACnE,CAAC,CAAC;QAEH,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;OAKG;IACH,gBAAgB,CAAC,QAA+B,EAAE;QAChD,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS,EAAE;YAC/B,MAAM,IAAI,KAAK,CAAC,0FAA0F,CAAC,CAAC;SAC7G;QAED,MAAM,KAAK,GACT,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,yBAAyB,CAAC;YAC7E,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,YAAY,CACpB,IAAI,EACJ,8BAA8B,EAC9B,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,yBAAyB,CAAC,CAC1D;YACH,CAAC,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,8BAA8B,CAAC,CAAC;QAC1D,IAAI,CAAC,gBAAgB;YACnB,KAAK,CAAC,gBAAgB;gBACtB,IAAI,qBAAqB,CAAC,gBAAgB,CAAC,IAAI,EAAE,cAAc,EAAE;oBAC/D,UAAU,EAAE,qBAAqB,CAAC,UAAU,CAAC,KAAK;oBAClD,MAAM,EAAE,CAAC,iDAAiD,CAAC;oBAC3D,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,QAAQ;oBAC/B,OAAO,EAAE,CAAC,KAAK,CAAC;iBACjB,CAAC,CAAC;QACL,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,SAAS;QACP,IAAI,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC;QAC9B,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;QACtB,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,KAA0B;QACvC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,KAAK,CAAC,SAAS,EAAE;YACtC,IAAI,EAAE,KAAK,CAAC,KAAK;SAClB,CAAC,CAAC;QAEH,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,IAAI,CAAC,QAAQ,EAAE,aAAa,EAAE,CAAC;QAC/B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,cAAc,CAAC,aAAa,EAAE,IAAI,CAAC,YAA0B,CAAC;QACzG,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAEhE,OAAO,IAAI,CAAC;IACd,CAAC;;AAxQH,8CAyQC","sourcesContent":["import * as cdk from \"aws-cdk-lib\";\nimport * as codepipeline from \"aws-cdk-lib/aws-codepipeline\";\nimport * as codestarnotifications from \"aws-cdk-lib/aws-codestarnotifications\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as kms from \"aws-cdk-lib/aws-kms\";\nimport * as sns from \"aws-cdk-lib/aws-sns\";\nimport * as pipelines from \"aws-cdk-lib/pipelines\";\nimport { Construct } from \"constructs\";\nimport { CICDActions } from \"./actions\";\nimport { toTitleCase } from \"./utils\";\nimport { BaseStack, BaseStackProps } from \"../base\";\nimport { Configurator } from \"../config\";\n\n/**\n * Properties for the source action.\n */\nexport interface SourceActionProps {\n  /**\n   * Override source action.\n   */\n  readonly sourceAction?: pipelines.CodePipelineSource;\n  /**\n   * Name of the SCM repository.\n   */\n  readonly repositoryName: string;\n  /**\n   * Branch of the SCM repository.\n   */\n  readonly branch?: string;\n}\n\n/**\n * Properties for the synth action.\n */\nexport interface SynthActionProps {\n  /**\n   * CDK versio to use during the synth action.\n   *\n   * @default \"latest\"\n   */\n  readonly cdkVersion?: string;\n  /**\n   * Name of the CodeArtifact repository to pull artifacts from.\n   */\n  readonly codeartifactRepository?: string;\n  /**\n   *  Name of the CodeArtifact domain.\n   */\n  readonly codeartifactDomain?: string;\n  /**\n   * CodeArtifact domain owner account.\n   */\n  readonly codeartifactDomainOwner?: string;\n  /**\n   * Environment variables to set.\n   */\n  readonly env?: { [key: string]: any };\n  /**\n   * Additional policies to add to the synth action role.\n   */\n  readonly rolePolicyStatements?: iam.PolicyStatement[];\n  /**\n   *  Override synth action.\n   */\n  readonly synthAction?: pipelines.CodeBuildStep;\n  /**\n   * Additional install commands.\n   */\n  readonly additionalInstallCommands?: string[];\n  /**\n   * Additional command line arguements to append to the install command of the `cdk_langauge` that is specified.\n   *\n   * @default - No command line arguments are appended\n   */\n  readonly cdkLanguageCommandLineArguments?: { [key: string]: string };\n}\n\n/**\n * Properties for adding an application stage.\n */\nexport interface AddApplicationStageProps {\n  /**\n   * Identifier of the stage.\n   */\n  readonly stageId: string;\n  /**\n   * Application stage instance.\n   */\n  readonly stage: cdk.Stage;\n  /**\n   * Configure manual approvals.\n   * @default false\n   */\n  readonly manualApprovals?: boolean;\n}\n\n/**\n * Properties for adding an application wave.\n */\nexport interface AddApplicationWaveProps {\n  /**\n   * Identifier of the wave.\n   */\n  readonly stageId: string;\n  /**\n   * Application stage instance.\n   */\n  readonly stages: cdk.Stage[];\n  /**\n   * Configure manual approvals.\n   * @default false\n   */\n  readonly manualApprovals?: boolean;\n}\n\n/**\n * Properties for adding a security lint stage.\n */\nexport interface AddSecurityLintStageProps {\n  /**\n   * Name of the stage.\n   */\n  readonly stageName?: string;\n  /**\n   * Cloud assembly file set producer.\n   */\n  readonly cloudAssemblyFileSet?: pipelines.IFileSetProducer;\n  /**\n   * Fail Codepipeline Build Action on failed results from CfnNag scan.\n   */\n  readonly cfnNagFailBuild?: boolean;\n}\n\n/**\n * Properties for adding a test stage.\n */\nexport interface AddTestStageProps {\n  /**\n   * Name of the stage.\n   */\n  readonly stageName?: string;\n  /**\n   * Cloud assembly file set.\n   */\n  readonly cloudAssemblyFileSet?: pipelines.IFileSetProducer;\n  /**\n   * Additional commands to run in the test.\n   * @default \"./test.sh\"\n   */\n  readonly commands?: string[];\n}\n\n/**\n * Properties for adding notifications.\n */\nexport interface AddNotificationsProps {\n  /**\n   * Override notification rule.\n   */\n  readonly notificationRule?: codestarnotifications.NotificationRule;\n}\n\n/**\n * Properties for adding a custom stage.\n */\nexport interface AddCustomStageProps {\n  /**\n   * Name of the stage.\n   */\n  readonly stageName: string;\n  /**\n   * Steps to add to this stage. List of Step objects.\n   *\n   * See [Documentation on aws_cdk.pipelines.Step](https://docs.aws.amazon.com/cdk/api/v1/python/aws_cdk.pipelines/Step.html)\n   * for more detail.\n   */\n  readonly steps: pipelines.Step[];\n}\n\n/**\n * CICD Pipeline Stack properties.\n */\nexport interface CICDPipelineStackProps extends BaseStackProps {\n  /**\n   * Name of the pipeline.\n   */\n  readonly pipelineName?: string;\n  /**\n   * Language of the CDK construct definitions.\n   *\n   * @default \"typescript\"\n   */\n  readonly cdkLanguage?: string;\n}\n\n/**\n * Additional properties for building the CodePipeline.\n */\nexport interface AdditionalPipelineProps {\n  /**\n   * Additional customizations to apply to the asset publishing CodeBuild projects\n   *\n   * @default - Only `codeBuildDefaults` are applied\n   */\n  readonly assetPublishingCodeBuildDefaults?: pipelines.CodeBuildOptions;\n  /**\n   * CDK CLI version to use in self-mutation and asset publishing steps\n   *\n   * @default latest version\n   */\n  readonly cliVersion?: string;\n  /**\n   * Customize the CodeBuild projects created for this pipeline\n   *\n   * @default - All projects run non-privileged build, SMALL instance, LinuxBuildImage.STANDARD_6_0\n   */\n  readonly codeBuildDefaults?: pipelines.CodeBuildOptions;\n  /**\n   * An existing Pipeline to be reused and built upon.\n   *\n   * @default - a new underlying pipeline is created.\n   */\n  readonly codePipeline?: codepipeline.Pipeline;\n  /**\n   * A list of credentials used to authenticate to Docker registries.\n   *\n   * Specify any credentials necessary within the pipeline to build, synth, update, or publish assets.\n   *\n   * @default []\n   */\n  readonly dockerCredentials?: pipelines.DockerCredential[];\n  /**\n   * Enable Docker for the self-mutate step\n   *\n   * @default false\n   */\n  readonly dockerEnabledForSelfMutation?: boolean;\n  /**\n   * Enable Docker for the 'synth' step\n   *\n   * @default false\n   */\n  readonly dockerEnabledForSynth?: boolean;\n  /**\n   * Publish assets in multiple CodeBuild projects\n\n   *\n   * @default true\n   */\n  readonly publishAssetsInParallel?: boolean;\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   * Whether the pipeline will update itself\n   *\n   * This needs to be set to `true` to allow the pipeline to reconfigure\n   * itself when assets or stages are being added to it, and `true` is the\n   * recommended setting.\n   *\n   * You can temporarily set this to `false` while you are iterating\n   * on the pipeline itself and prefer to deploy changes using `cdk deploy`.\n   *\n   * @default true\n   */\n  readonly selfMutation?: boolean;\n  /**\n   * Additional customizations to apply to the self mutation CodeBuild projects\n   *\n   * @default - Only `codeBuildDefaults` are applied\n   */\n  readonly selfMutationCodeBuildDefaults?: pipelines.CodeBuildOptions;\n  /**\n   * Additional customizations to apply to the synthesize CodeBuild projects\n   *\n   * @default - Only `codeBuildDefaults` are applied\n   */\n  readonly synthCodeBuildDefaults?: pipelines.CodeBuildOptions;\n}\n\n/**\n * Create a stack that contains DDK Continuous Integration and Delivery (CI/CD) pipeline.\n\n  The pipeline is based on\n  [CDK self-mutating pipeline](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.pipelines-readme.html)\n  but includes several DDK-specific features, including:\n\n  - Ability to configure some properties via JSON config e.g. manual approvals for application stages\n  - Defaults for source/synth - CodeCommit & cdk synth, with ability to override them\n  - Ability to connect to private artifactory to pull artifacts from at synth\n  - Security best practices - ensures pipeline buckets block non-SSL, and are KMS-encrypted with rotated keys\n  - Builder interface to avoid chunky constructor methods\n\n  The user should be able to reuse the pipeline in multiple DDK applications hoping to save LOC.\n\n  @example\n  const stack = new CICDPipelineStack(app, \"dummy-pipeline\", { environmentId: \"dev\", pipelineName: \"dummy-pipeline\" })\n    .addSourceAction({ repositoryName: \"dummy-repository\" })\n    .addSynthAction()\n    .buildPipeline()\n    .add_checks()\n    .addStage({ stageId: \"dev\", stage: devStage, manualApprovals: true })\n    .synth()\n    .add_notifications();\n */\nexport class CICDPipelineStack extends BaseStack {\n  readonly environmentId?: string;\n  readonly pipelineName?: string;\n  readonly pipelineId?: string;\n  readonly config: Configurator;\n  readonly cdkLanguage: string;\n\n  public notificationRule?: codestarnotifications.NotificationRule;\n  public pipeline?: pipelines.CodePipeline;\n  public pipelineKey?: kms.CfnKey;\n  public sourceAction?: pipelines.CodePipelineSource;\n  public synthAction?: pipelines.CodeBuildStep;\n\n  /**\n   * Creates a new CICD Pipeline stack.\n   *\n   * @param scope Parent of this stack, usually an `App` or a `Stage`, but could be any construct.\n   * @param id The construct ID of this stack. If `stackName` is not explicitly\n   * defined, this id (and any parent IDs) will be used to determine the\n   * physical ID of the stack.\n   * @param props Stack properties.\n   */\n  constructor(scope: Construct, id: string, props: CICDPipelineStackProps) {\n    super(scope, id, props);\n\n    this.environmentId = props.environmentId;\n    this.pipelineName = props.pipelineName;\n    this.pipelineId = id;\n\n    const config = props.config ?? \"./ddk.json\";\n    this.config = new Configurator(this, config, this.environmentId);\n\n    this.cdkLanguage = props.cdkLanguage ?? \"typescript\";\n  }\n\n  /**\n   * Add source action.\n   *\n   * @param props Source action properties.\n   * @returns reference to this pipeline.\n   */\n  addSourceAction(props: SourceActionProps) {\n    var branch = props.branch ?? \"main\";\n    this.sourceAction =\n      props.sourceAction ||\n      CICDActions.getCodeCommitSourceAction(this, {\n        repositoryName: props.repositoryName,\n        branch: branch,\n      });\n    return this;\n  }\n\n  /**\n   * Build the pipeline structure.\n   * @param props Additional pipeline properties.\n   * @returns reference to this pipeline.\n   */\n  buildPipeline(props: AdditionalPipelineProps = {}) {\n    if (this.synthAction === undefined) {\n      throw new Error(\"Pipeline cannot be built without a synth action.\");\n    }\n    this.pipeline = new pipelines.CodePipeline(this, \"DDKCodePipeline\", {\n      synth: this.synthAction,\n      crossAccountKeys: true,\n      pipelineName: this.pipelineName,\n      ...props,\n    });\n    return this;\n  }\n\n  /**\n   * Add synth action. During synth can connect and pull artifacts from a private artifactory.\n   * @param props Synth action properties.\n   * @returns reference to this pipeline.\n   */\n  addSynthAction(props: SynthActionProps = {}) {\n    const languageInstallCommands: any = {\n      typescript: \"npm install\",\n      python: \"pip install -r requirements.txt\",\n    };\n\n    let languageInstallCommand = languageInstallCommands[this.cdkLanguage];\n    if (props.cdkLanguageCommandLineArguments) {\n      for (const [argument, value] of Object.entries(props.cdkLanguageCommandLineArguments)) {\n        languageInstallCommand += ` ${argument} ${value}`;\n      }\n    }\n\n    this.synthAction =\n      props.synthAction ||\n      CICDActions.getSynthAction({\n        codePipelineSource: this.sourceAction,\n        cdkVersion: props.cdkVersion,\n        partition: this.partition,\n        region: this.region,\n        account: this.account,\n        env: props.env,\n        rolePolicyStatements: props.rolePolicyStatements,\n        codeartifactRepository: props.codeartifactRepository,\n        codeartifactDomain: props.codeartifactDomain,\n        codeartifactDomainOwner: props.codeartifactDomainOwner,\n        additionalInstallCommands: props.additionalInstallCommands\n          ? [languageInstallCommand].concat(props.additionalInstallCommands)\n          : [languageInstallCommand],\n      });\n    return this;\n  }\n\n  /**\n   * Add application stage to the CICD pipeline. This stage deploys your application infrastructure.\n   * @param props Application stage properties.\n   * @returns reference to this pipeline.\n   */\n  addStage(props: AddApplicationStageProps) {\n    if (this.pipeline === undefined) {\n      throw new Error(\"`.buildPipeline()` needs to be called first before adding application stages to the pipeline.\");\n    }\n    const manualApprovals = props.manualApprovals ?? this.config.getConfigAttribute(\"manual_approvals\") ?? false;\n\n    if (manualApprovals) {\n      this.pipeline?.addStage(props.stage, {\n        pre: [new pipelines.ManualApprovalStep(\"PromoteTo\" + toTitleCase(props.stageId))],\n      });\n    } else {\n      this.pipeline?.addStage(props.stage, {});\n    }\n\n    return this;\n  }\n\n  /**\n   * Add multiple application stages in parallel to the CICD pipeline.\n   * @param props Application wave properties.\n   * @returns reference to this pipeline.\n   */\n  addWave(props: AddApplicationWaveProps) {\n    if (this.pipeline === undefined) {\n      throw new Error(\"`.buildPipeline()` needs to be called first before adding application stages to the pipeline.\");\n    }\n    const manualApprovals = props.manualApprovals ?? this.config.getConfigAttribute(\"manual_approvals\") ?? false;\n\n    var wave = new pipelines.Wave(props.stageId);\n    if (manualApprovals) {\n      wave.addPre(new pipelines.ManualApprovalStep(\"PromoteTo\" + toTitleCase(props.stageId)));\n    }\n\n    props.stages.forEach((stage) => {\n      wave.addStage(stage);\n    });\n\n    this.pipeline?.addWave(props.stageId, wave);\n    return this;\n  }\n\n  /**\n   * Add linting - cfn-nag, and bandit.\n   * @param props Security lint properties.\n   * @returns reference to this pipeline.\n   */\n  addSecurityLintStage(props: AddSecurityLintStageProps) {\n    if (this.sourceAction === undefined) {\n      throw new Error(\"Source Action Must Be configured before calling this method.\");\n    }\n    if (this.pipeline?.cloudAssemblyFileSet === undefined) {\n      throw new Error(\"No cloudAssemblyFileSet configured, source action needs to be configured for this pipeline.\");\n    }\n\n    var stageName = props.stageName ?? \"SecurityLint\";\n    var cloudAssemblyFileSet = props.cloudAssemblyFileSet ?? this.pipeline?.cloudAssemblyFileSet;\n\n    this.pipeline?.addWave(stageName, {\n      post: [\n        CICDActions.getCfnNagAction(cloudAssemblyFileSet, \"CFNNag\", props.cfnNagFailBuild),\n        CICDActions.getBanditAction(this.sourceAction),\n      ],\n    });\n\n    return this;\n  }\n\n  /**\n   * Add test - e.g. pytest.\n   * @param props Test stage properties.\n   * @returns reference to this pipeline.\n   */\n  addTestStage(props: AddTestStageProps) {\n    var stageName = props.stageName ?? \"Tests\";\n    var cloudAssemblyFileSet = props.cloudAssemblyFileSet ?? this.pipeline?.cloudAssemblyFileSet;\n    var commands = props.commands ?? [\"./test.sh\"];\n\n    if (cloudAssemblyFileSet === undefined) {\n      throw new Error(\"No cloudAssemblyFileSet configured, source action needs to be configured for this pipeline.\");\n    }\n\n    this.pipeline?.addWave(stageName || \"Tests\", {\n      post: [CICDActions.getTestsAction(cloudAssemblyFileSet, commands)],\n    });\n\n    return this;\n  }\n\n  /**\n   * Add pipeline notifications.\n   * Create notification rule that sends events to the specified SNS topic.\n   * @param props Notification properties.\n   * @returns reference to this pipeline.\n   */\n  addNotifications(props: AddNotificationsProps = {}) {\n    if (this.pipeline === undefined) {\n      throw new Error(\"`.buildPipeline()` needs to be called first before adding notifications to the pipeline.\");\n    }\n\n    const topic =\n      this.environmentId && this.config.getConfigAttribute(\"notifications_topic_arn\")\n        ? sns.Topic.fromTopicArn(\n            this,\n            \"ExecutionFailedNotifications\",\n            this.config.getConfigAttribute(\"notifications_topic_arn\"),\n          )\n        : new sns.Topic(this, \"ExecutionFailedNotifications\");\n    this.notificationRule =\n      props.notificationRule ??\n      new codestarnotifications.NotificationRule(this, \"Notification\", {\n        detailType: codestarnotifications.DetailType.BASIC,\n        events: [\"codepipeline-pipeline-pipeline-execution-failed\"],\n        source: this.pipeline?.pipeline,\n        targets: [topic],\n      });\n    return this;\n  }\n\n  /**\n   * Add checks to the pipeline (e.g. linting, security, tests...).\n   * @returns reference to this pipeline.\n   */\n  addChecks() {\n    this.addSecurityLintStage({});\n    this.addTestStage({});\n    return this;\n  }\n\n  /**\n   * Add custom stage to the pipeline.\n   * @param props Properties for adding a custom stage.\n   * @returns reference to this pipeline.\n   */\n  addCustomStage(props: AddCustomStageProps) {\n    this.pipeline?.addWave(props.stageName, {\n      post: props.steps,\n    });\n\n    return this;\n  }\n\n  /**\n   * Synthesize the pipeline.\n   * @returns reference to this pipeline.\n   */\n  synth() {\n    this.pipeline?.buildPipeline();\n    this.pipelineKey = this.pipeline?.pipeline.artifactBucket.encryptionKey?.node.defaultChild as kms.CfnKey;\n    this.pipelineKey.addPropertyOverride(\"EnableKeyRotation\", true);\n\n    return this;\n  }\n}\n"]}