UNPKG

aws-delivlib

Version:

A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.

98 lines 14 kB
"use strict"; 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"]}