aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
98 lines • 14 kB
JavaScript
;
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.ChangeController = void 0;
const path = __importStar(require("path"));
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
/**
* Controls enabling and disabling a CodePipeline promotion into a particular stage based on "blocking" windows that are
* configured in an iCal document stored in an S3 bucket. If the document is not present or the bucket does not exist,
* the transition will be disabled.
*/
class ChangeController extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
let changeControlBucket = props.changeControlBucket;
let ownBucket;
if (!changeControlBucket) {
changeControlBucket = ownBucket = new aws_cdk_lib_1.aws_s3.Bucket(this, 'Calendar', {
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
versioned: true,
});
}
// const changeControlBucket = props.changeControlBucket || new s3.Bucket(this, 'Bucket', { versioned: true });
const changeControlObjectKey = props.changeControlObjectKey || 'change-control.ics';
const fn = new aws_cdk_lib_1.aws_lambda_nodejs.NodejsFunction(this, 'Function', {
description: `Enforces a Change Control Policy into CodePipeline's ${props.pipelineStage.stageName} stage`,
entry: path.join(__dirname, 'change-control-lambda', 'index.ts'),
runtime: aws_cdk_lib_1.aws_lambda.Runtime.NODEJS_20_X,
environment: {
// CAPITAL punishment 👌🏻
CHANGE_CONTROL_BUCKET_NAME: changeControlBucket.bucketName,
CHANGE_CONTROL_OBJECT_KEY: changeControlObjectKey,
PIPELINE_NAME: props.pipelineStage.pipeline.pipelineName,
STAGE_NAME: props.pipelineStage.stageName,
},
timeout: aws_cdk_lib_1.Duration.seconds(300),
});
fn.addToRolePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
resources: [`${props.pipelineStage.pipeline.pipelineArn}/${props.pipelineStage.stageName}`],
actions: ['codepipeline:EnableStageTransition', 'codepipeline:DisableStageTransition'],
}));
changeControlBucket.grantRead(fn, props.changeControlObjectKey);
if (ownBucket) {
ownBucket.addObjectCreatedNotification(new aws_cdk_lib_1.aws_s3_notifications.LambdaDestination(fn), {
prefix: changeControlObjectKey,
});
}
this.failureAlarm = new aws_cdk_lib_1.aws_cloudwatch.Alarm(this, 'Failed', {
metric: fn.metricErrors({
period: aws_cdk_lib_1.Duration.seconds(300),
}),
threshold: 1,
datapointsToAlarm: 1,
evaluationPeriods: 1,
});
const schedule = props.schedule || aws_cdk_lib_1.aws_events.Schedule.expression('rate(15 minutes)');
// Run this on a schedule
new aws_cdk_lib_1.aws_events.Rule(this, 'Rule', {
// tslint:disable-next-line:max-line-length
description: `Run the change controller for promotions into ${props.pipelineStage.pipeline.pipelineName}'s ${props.pipelineStage.stageName} on a ${schedule} schedule`,
schedule,
targets: [new aws_cdk_lib_1.aws_events_targets.LambdaFunction(fn)],
});
if (props.createOutputs !== false) {
new aws_cdk_lib_1.CfnOutput(this, 'ChangeControlBucketKey', {
value: changeControlObjectKey,
});
new aws_cdk_lib_1.CfnOutput(this, 'ChangeControlBucket', {
value: changeControlBucket.bucketName,
});
}
}
}
exports.ChangeController = ChangeController;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"change-controller.js","sourceRoot":"","sources":["change-controller.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  CfnOutput, Duration, RemovalPolicy,\n  aws_cloudwatch as cloudwatch,\n  aws_codepipeline as cp,\n  aws_events as events,\n  aws_events_targets as events_targets,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_s3 as s3,\n  aws_s3_notifications as s3_notifications,\n  aws_lambda_nodejs as nodejs,\n} from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\n\nexport interface ChangeControllerProps {\n  /**\n   * The bucket in which the ChangeControl iCal document will be stored.\n   *\n   * @default a new versioned bucket will be provisioned.\n   */\n  changeControlBucket?: s3.IBucket;\n\n  /**\n   * The key in which the iCal fille will be stored.\n   *\n   * @default 'change-control.ical'\n   */\n  changeControlObjectKey?: string;\n\n  /**\n   * Name of the stage\n   */\n  pipelineStage: cp.IStage;\n\n  /**\n   * Schedule to run the change controller on\n   *\n   * @default once every 15 minutes\n   */\n  schedule?: events.Schedule;\n\n  /**\n   * Whether to create outputs to inform of the S3 bucket name and keys where the change control calendar should be\n   * stored.\n   *\n   * @defaults true\n   */\n  createOutputs?: boolean;\n}\n\n/**\n * Controls enabling and disabling a CodePipeline promotion into a particular stage based on \"blocking\" windows that are\n * configured in an iCal document stored in an S3 bucket. If the document is not present or the bucket does not exist,\n * the transition will be disabled.\n */\nexport class ChangeController extends Construct {\n  /**\n   * The alarm that will fire in case the change controller has failed.\n   */\n  public readonly failureAlarm: cloudwatch.Alarm;\n\n  constructor(scope: Construct, id: string, props: ChangeControllerProps) {\n    super(scope, id);\n\n    let changeControlBucket = props.changeControlBucket;\n    let ownBucket: s3.Bucket | undefined;\n\n    if (!changeControlBucket) {\n      changeControlBucket = ownBucket = new s3.Bucket(this, 'Calendar', {\n        removalPolicy: RemovalPolicy.DESTROY,\n        versioned: true,\n      });\n    }\n\n    // const changeControlBucket = props.changeControlBucket || new s3.Bucket(this, 'Bucket', { versioned: true });\n    const changeControlObjectKey = props.changeControlObjectKey || 'change-control.ics';\n\n    const fn = new nodejs.NodejsFunction(this, 'Function', {\n      description: `Enforces a Change Control Policy into CodePipeline's ${props.pipelineStage.stageName} stage`,\n      entry: path.join(__dirname, 'change-control-lambda', 'index.ts'),\n      runtime: lambda.Runtime.NODEJS_20_X,\n      environment: {\n        // CAPITAL punishment 👌🏻\n        CHANGE_CONTROL_BUCKET_NAME: changeControlBucket.bucketName,\n        CHANGE_CONTROL_OBJECT_KEY: changeControlObjectKey,\n        PIPELINE_NAME: props.pipelineStage.pipeline.pipelineName,\n        STAGE_NAME: props.pipelineStage.stageName,\n      },\n      timeout: Duration.seconds(300),\n    });\n\n    fn.addToRolePolicy(new iam.PolicyStatement({\n      resources: [`${props.pipelineStage.pipeline.pipelineArn}/${props.pipelineStage.stageName}`],\n      actions: ['codepipeline:EnableStageTransition', 'codepipeline:DisableStageTransition'],\n    }));\n\n    changeControlBucket.grantRead(fn, props.changeControlObjectKey);\n\n    if (ownBucket) {\n      ownBucket.addObjectCreatedNotification(new s3_notifications.LambdaDestination(fn), {\n        prefix: changeControlObjectKey,\n      });\n    }\n\n    this.failureAlarm = new cloudwatch.Alarm(this, 'Failed', {\n      metric: fn.metricErrors({\n        period: Duration.seconds(300),\n      }),\n      threshold: 1,\n      datapointsToAlarm: 1,\n      evaluationPeriods: 1,\n    });\n\n    const schedule = props.schedule || events.Schedule.expression('rate(15 minutes)');\n\n    // Run this on a schedule\n    new events.Rule(this, 'Rule', {\n      // tslint:disable-next-line:max-line-length\n      description: `Run the change controller for promotions into ${props.pipelineStage.pipeline.pipelineName}'s ${props.pipelineStage.stageName} on a ${schedule} schedule`,\n      schedule,\n      targets: [new events_targets.LambdaFunction(fn)],\n    });\n\n    if (props.createOutputs !== false) {\n      new CfnOutput(this, 'ChangeControlBucketKey', {\n        value: changeControlObjectKey,\n      });\n\n      new CfnOutput(this, 'ChangeControlBucket', {\n        value: changeControlBucket.bucketName,\n      });\n    }\n  }\n}\n"]}