UNPKG

@cdklabs/cdk-ecs-codedeploy

Version:

CDK Constructs for performing ECS Deployments with CodeDeploy

158 lines 28 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.ApplicationLoadBalancedCodeDeployedFargateService = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_cloudwatch_1 = require("aws-cdk-lib/aws-cloudwatch"); const aws_codedeploy_1 = require("aws-cdk-lib/aws-codedeploy"); const aws_ecs_1 = require("aws-cdk-lib/aws-ecs"); const aws_ecs_patterns_1 = require("aws-cdk-lib/aws-ecs-patterns"); const aws_elasticloadbalancingv2_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const aws_synthetics_1 = require("aws-cdk-lib/aws-synthetics"); const api_canary_1 = require("../api-canary"); const ecs_deployment_1 = require("../ecs-deployment"); /** * A Fargate service running on an ECS cluster fronted by an application load balancer and deployed by CodeDeploy. */ class ApplicationLoadBalancedCodeDeployedFargateService extends aws_ecs_patterns_1.ApplicationLoadBalancedFargateService { /** * Constructs a new instance of the ApplicationLoadBalancedCodeDeployedFargateService class. */ constructor(scope, id, props) { super(scope, id, { ...props, deploymentController: { type: aws_ecs_1.DeploymentControllerType.CODE_DEPLOY, }, }); if (props.deregistrationDelay) { this.targetGroup.setAttribute('deregistration_delay.timeout_seconds', props.deregistrationDelay.toSeconds().toString()); } if (props.targetHealthCheck) { this.targetGroup.configureHealthCheck(props.targetHealthCheck); } this.accessLogBucket = props.accessLogBucket ?? new aws_s3_1.Bucket(this, 'AccessLogBucket', { encryption: aws_s3_1.BucketEncryption.S3_MANAGED, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, enforceSSL: true, }); this.loadBalancer.logAccessLogs(this.accessLogBucket, props.accessLogPrefix); const alarms = []; if (props.responseTimeAlarmThreshold) { const responseTimeAlarm = new aws_cloudwatch_1.Alarm(this, 'ResponseTimeAlarm', { metric: this.loadBalancer.metrics.targetResponseTime({ period: aws_cdk_lib_1.Duration.minutes(1), statistic: 'p95', }), evaluationPeriods: 2, threshold: props.responseTimeAlarmThreshold.toSeconds(), comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD, }); alarms.push(responseTimeAlarm); } const protocol = props.protocol ?? (props.certificate ? aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTPS : aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTP); const testHostName = props.domainName ? props.domainName : this.loadBalancer.loadBalancerDnsName; if (props.apiTestSteps?.length) { this.apiCanary = new api_canary_1.ApiCanary(this, 'Canary', { baseUrl: `${protocol.toLowerCase()}://${testHostName}`, durationAlarmThreshold: props.apiCanaryTimeout, schedule: aws_synthetics_1.Schedule.rate(props.apiCanarySchedule ?? aws_cdk_lib_1.Duration.minutes(5)), threadCount: props.apiCanaryThreadCount, steps: props.apiTestSteps, }); this.apiCanary.node.addDependency(this.service); alarms.push(this.apiCanary.successAlarm); if (this.apiCanary.durationAlarm) { alarms.push(this.apiCanary.durationAlarm); } } if (alarms.length > 0) { this.healthAlarm = new aws_cloudwatch_1.CompositeAlarm(this, 'HealthAlarm', { alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(...alarms), }); } else { this.healthAlarm = undefined; } let testPort; if (props.testPort) { testPort = props.testPort; } else if (props.listenerPort) { testPort = props.listenerPort + 1; } else if (protocol === aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTP) { testPort = 8080; } else if (protocol === aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTPS) { testPort = 8443; } else { throw new Error('Unable to determine port for test listener'); } let certificates; if (props.certificate) { certificates = [props.certificate]; } this.testListener = this.loadBalancer.addListener('TestListener', { protocol, port: testPort, open: props.openListener ?? true, sslPolicy: props.sslPolicy, certificates: certificates, }); this.greenTargetGroup = new aws_elasticloadbalancingv2_1.ApplicationTargetGroup(this, 'GreenTargetGroup', { vpc: this.cluster.vpc, port: testPort, protocol: props.targetProtocol ?? aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTP, protocolVersion: props.protocolVersion, deregistrationDelay: props.deregistrationDelay, healthCheck: props.targetHealthCheck, targetType: aws_elasticloadbalancingv2_1.TargetType.IP, }); this.listener.node.addDependency(this.greenTargetGroup); this.testListener.addTargetGroups('ECS', { targetGroups: [this.greenTargetGroup], }); this.application = new aws_codedeploy_1.EcsApplication(this, 'Application', { applicationName: props.applicationName, }); this.deploymentGroup = new aws_codedeploy_1.EcsDeploymentGroup(this, 'DeploymentGroup', { application: this.application, deploymentGroupName: props.deploymentGroupName, alarms: this.healthAlarm ? [this.healthAlarm] : undefined, service: this.service, blueGreenDeploymentConfig: { blueTargetGroup: this.targetGroup, greenTargetGroup: this.greenTargetGroup, listener: this.listener, testListener: this.testListener, terminationWaitTime: props.terminationWaitTime ?? aws_cdk_lib_1.Duration.minutes(10), }, deploymentConfig: props.deploymentConfig ?? aws_codedeploy_1.EcsDeploymentConfig.ALL_AT_ONCE, autoRollback: { stoppedDeployment: true, }, }); this.deployment = new ecs_deployment_1.EcsDeployment({ deploymentGroup: this.deploymentGroup, timeout: props.deploymentTimeout ?? aws_cdk_lib_1.Duration.minutes(60), targetService: { taskDefinition: this.taskDefinition, containerName: this.taskDefinition.defaultContainer.containerName, containerPort: this.taskDefinition.defaultContainer.containerPort, }, hooks: props.hooks, }); } addServiceAsTarget(service) { super.addServiceAsTarget(service); } } exports.ApplicationLoadBalancedCodeDeployedFargateService = ApplicationLoadBalancedCodeDeployedFargateService; _a = JSII_RTTI_SYMBOL_1; ApplicationLoadBalancedCodeDeployedFargateService[_a] = { fqn: "@cdklabs/cdk-ecs-codedeploy.ApplicationLoadBalancedCodeDeployedFargateService", version: "0.0.421" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application-load-balanced-codedeployed-fargate-service.js","sourceRoot":"","sources":["../../src/ecs-patterns/application-load-balanced-codedeployed-fargate-service.ts"],"names":[],"mappings":";;;;;AAAA,6CAAuC;AACvC,+DAA0G;AAC1G,+DAA2H;AAC3H,iDAA4E;AAC5E,mEAAiI;AACjI,uFAAmJ;AACnJ,+CAA0F;AAC1F,+DAAsD;AAEtD,8CAAuD;AAEvD,sDAAkD;AAwHlD;;GAEG;AACH,MAAa,iDAAkD,SAAQ,wDAAqC;IAyC1G;;OAEG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6D;QACrG,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE;YACf,GAAG,KAAK;YACR,oBAAoB,EAAE;gBACpB,IAAI,EAAE,kCAAwB,CAAC,WAAW;aAC3C;SACF,CAAC,CAAC;QAEH,IAAI,KAAK,CAAC,mBAAmB,EAAE,CAAC;YAC9B,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,sCAAsC,EAAE,KAAK,CAAC,mBAAmB,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC1H,CAAC;QACD,IAAI,KAAK,CAAC,iBAAiB,EAAE,CAAC;YAC5B,IAAI,CAAC,WAAW,CAAC,oBAAoB,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;QACjE,CAAC;QACD,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC,eAAe;YAC1C,IAAI,eAAM,CAAC,IAAI,EAAE,iBAAiB,EAAE;gBAClC,UAAU,EAAE,yBAAgB,CAAC,UAAU;gBACvC,iBAAiB,EAAE,0BAAiB,CAAC,SAAS;gBAC9C,UAAU,EAAE,IAAI;aACjB,CAAC,CAAC;QAEL,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,IAAI,CAAC,eAAe,EAAE,KAAK,CAAC,eAAe,CAAC,CAAC;QAE7E,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,IAAI,KAAK,CAAC,0BAA0B,EAAE,CAAC;YACrC,MAAM,iBAAiB,GAAG,IAAI,sBAAK,CAAC,IAAI,EAAE,mBAAmB,EAAE;gBAC7D,MAAM,EAAE,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,kBAAkB,CAAC;oBACnD,MAAM,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;oBAC3B,SAAS,EAAE,KAAK;iBACjB,CAAC;gBACF,iBAAiB,EAAE,CAAC;gBACpB,SAAS,EAAE,KAAK,CAAC,0BAA0B,CAAC,SAAS,EAAE;gBACvD,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;aAC9D,CAAC,CAAC;YACH,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACjC,CAAC;QAED,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,gDAAmB,CAAC,KAAK,CAAC,CAAC,CAAC,gDAAmB,CAAC,IAAI,CAAC,CAAC;QAE9G,MAAM,YAAY,GAAG,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,mBAAmB,CAAC;QAEjG,IAAI,KAAK,CAAC,YAAY,EAAE,MAAM,EAAE,CAAC;YAC/B,IAAI,CAAC,SAAS,GAAG,IAAI,sBAAS,CAAC,IAAI,EAAE,QAAQ,EAAE;gBAC7C,OAAO,EAAE,GAAG,QAAQ,CAAC,WAAW,EAAE,MAAM,YAAY,EAAE;gBACtD,sBAAsB,EAAE,KAAK,CAAC,gBAAgB;gBAC9C,QAAQ,EAAE,yBAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACvE,WAAW,EAAE,KAAK,CAAC,oBAAoB;gBACvC,KAAK,EAAE,KAAK,CAAC,YAAY;aAC1B,CAAC,CAAC;YAEH,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAEhD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;YACzC,IAAI,IAAI,CAAC,SAAS,CAAC,aAAa,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC;QACD,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,IAAI,CAAC,WAAW,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,aAAa,EAAE;gBACzD,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,GAAG,MAAM,CAAC;aACtC,CAAC,CAAC;QACL,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,WAAW,GAAG,SAAS,CAAC;QAC/B,CAAC;QAED,IAAI,QAAgB,CAAC;QACrB,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;YACnB,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;QAC5B,CAAC;aAAM,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;YAC9B,QAAQ,GAAG,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC;QACpC,CAAC;aAAM,IAAI,QAAQ,KAAK,gDAAmB,CAAC,IAAI,EAAE,CAAC;YACjD,QAAQ,GAAG,IAAI,CAAC;QAClB,CAAC;aAAM,IAAI,QAAQ,KAAK,gDAAmB,CAAC,KAAK,EAAE,CAAC;YAClD,QAAQ,GAAG,IAAI,CAAC;QAClB,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;QAChE,CAAC;QAED,IAAI,YAAY,CAAC;QACjB,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;YACtB,YAAY,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,cAAc,EAAE;YAChE,QAAQ;YACR,IAAI,EAAE,QAAQ;YACd,IAAI,EAAE,KAAK,CAAC,YAAY,IAAI,IAAI;YAChC,SAAS,EAAE,KAAK,CAAC,SAAS;YAC1B,YAAY,EAAE,YAAY;SAC3B,CAAC,CAAC;QAEH,IAAI,CAAC,gBAAgB,GAAG,IAAI,mDAAsB,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC3E,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG;YACrB,IAAI,EAAE,QAAQ;YACd,QAAQ,EAAE,KAAK,CAAC,cAAc,IAAI,gDAAmB,CAAC,IAAI;YAC1D,eAAe,EAAE,KAAK,CAAC,eAAe;YACtC,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;YAC9C,WAAW,EAAE,KAAK,CAAC,iBAAiB;YACpC,UAAU,EAAE,uCAAU,CAAC,EAAE;SAC1B,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAExD,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,KAAK,EAAE;YACvC,YAAY,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC;SACtC,CAAC,CAAC;QAEH,IAAI,CAAC,WAAW,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,aAAa,EAAE;YACzD,eAAe,EAAE,KAAK,CAAC,eAAe;SACvC,CAAC,CAAC;QAEH,IAAI,CAAC,eAAe,GAAG,IAAI,mCAAkB,CAAC,IAAI,EAAE,iBAAiB,EAAE;YACrE,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;YAC9C,MAAM,EAAE,IAAI,CAAC,WAAW,CAAA,CAAC,CAAA,CAAC,IAAI,CAAC,WAAW,CAAC,CAAA,CAAC,CAAA,SAAS;YACrD,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,yBAAyB,EAAE;gBACzB,eAAe,EAAE,IAAI,CAAC,WAAW;gBACjC,gBAAgB,EAAE,IAAI,CAAC,gBAAgB;gBACvC,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,YAAY,EAAE,IAAI,CAAC,YAAY;gBAC/B,mBAAmB,EAAE,KAAK,CAAC,mBAAmB,IAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;aACvE;YACD,gBAAgB,EAAE,KAAK,CAAC,gBAAgB,IAAI,oCAAmB,CAAC,WAAW;YAC3E,YAAY,EAAE;gBACZ,iBAAiB,EAAE,IAAI;aACxB;SACF,CAAC,CAAC;QACH,IAAI,CAAC,UAAU,GAAG,IAAI,8BAAa,CAAC;YAClC,eAAe,EAAE,IAAI,CAAC,eAAe;YACrC,OAAO,EAAE,KAAK,CAAC,iBAAiB,IAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACxD,aAAa,EAAE;gBACb,cAAc,EAAE,IAAI,CAAC,cAAc;gBACnC,aAAa,EAAE,IAAI,CAAC,cAAc,CAAC,gBAAiB,CAAC,aAAa;gBAClE,aAAa,EAAE,IAAI,CAAC,cAAc,CAAC,gBAAiB,CAAC,aAAa;aACnE;YACD,KAAK,EAAE,KAAK,CAAC,KAAK;SACnB,CAAC,CAAC;IACL,CAAC;IAES,kBAAkB,CAAC,OAAoB;QAC/C,KAAK,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;;AA1LH,8GA2LC","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport { Alarm, AlarmRule, ComparisonOperator, CompositeAlarm, IAlarm } from 'aws-cdk-lib/aws-cloudwatch';\nimport { EcsApplication, EcsDeploymentConfig, EcsDeploymentGroup, IEcsDeploymentConfig } from 'aws-cdk-lib/aws-codedeploy';\nimport { BaseService, DeploymentControllerType } from 'aws-cdk-lib/aws-ecs';\nimport { ApplicationLoadBalancedFargateService, ApplicationLoadBalancedFargateServiceProps } from 'aws-cdk-lib/aws-ecs-patterns';\nimport { ApplicationListener, ApplicationProtocol, ApplicationTargetGroup, HealthCheck, TargetType } from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport { BlockPublicAccess, Bucket, BucketEncryption, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Schedule } from 'aws-cdk-lib/aws-synthetics';\nimport { Construct } from 'constructs';\nimport { ApiTestStep, ApiCanary } from '../api-canary';\nimport { AppSpecHooks } from '../ecs-appspec';\nimport { EcsDeployment } from '../ecs-deployment';\n\n/**\n * The properties for the ApplicationLoadBalancedCodeDeployedFargateService service.\n */\nexport interface ApplicationLoadBalancedCodeDeployedFargateServiceProps extends ApplicationLoadBalancedFargateServiceProps {\n  /**\n   * The timeout for a CodeDeploy deployment.\n   *\n   * @default - 60 minutes\n   */\n  readonly deploymentTimeout?: Duration;\n\n  /**\n   * The time to wait before terminating the original (blue) task set.\n   *\n   * @default - 10 minutes\n   */\n  readonly terminationWaitTime?: Duration;\n\n  /**\n   * The deployment configuration to use for the deployment group.\n   *\n   * @default - EcsDeploymentConfig.ALL_AT_ONCE\n   */\n  readonly deploymentConfig?: IEcsDeploymentConfig;\n\n  /**\n   * The amount of time for ELB to wait before changing the state of a deregistering target\n   * from 'draining' to 'unused'.\n   *\n   * @default - 300 seconds\n   */\n  readonly deregistrationDelay?: Duration;\n\n  /**\n   * The healthcheck to configure on the Application Load Balancer target groups.\n   *\n   * @default - no health check is configured\n   */\n  readonly targetHealthCheck?: HealthCheck;\n\n  /**\n   * The bucket to use for access logs from the Application Load Balancer.\n   *\n   * @default - a new S3 bucket will be created\n   */\n  readonly accessLogBucket?: IBucket;\n\n  /**\n   * The prefix to use for access logs from the Application Load Balancer.\n   *\n   * @default - none\n   */\n  readonly accessLogPrefix?: string;\n\n  /**\n   * The threshold for response time alarm.\n   *\n   * @default - no alarm will be created\n   */\n  readonly responseTimeAlarmThreshold?: Duration;\n\n  /**\n   * The number of threads to run concurrently for the synthetic test.\n   *\n   * @default - 20\n   */\n  readonly apiCanaryThreadCount?: number;\n\n  /**\n   * The frequency for running the api canaries.\n   *\n   * @default - 5 minutes\n   */\n  readonly apiCanarySchedule?: Duration;\n\n  /**\n   * The threshold for how long a api canary can take to run.\n   *\n   * @default - no alarm is created for test duration\n   */\n  readonly apiCanaryTimeout?: Duration;\n\n  /**\n   * The steps to run in the canary.\n   *\n   * @default - no synthetic test will be created\n   */\n  readonly apiTestSteps?: ApiTestStep[];\n\n  /**\n   * The port to use for test traffic on the listener\n   *\n   * @default - listenerPort + 1\n   */\n  readonly testPort?: number;\n\n  /**\n   * Optional lifecycle hooks\n   *\n   * @default - no lifecycle hooks\n   */\n  readonly hooks?: AppSpecHooks;\n\n  /**\n   * The physical, human-readable name of the CodeDeploy Application.\n   *\n   * @default an auto-generated name will be used\n   */\n  readonly applicationName?: string;\n\n  /**\n   * The physical, human-readable name of the CodeDeploy Deployment Group.\n   *\n   * @default An auto-generated name will be used.\n   */\n  readonly deploymentGroupName?: string;\n}\n\n/**\n * A Fargate service running on an ECS cluster fronted by an application load balancer and deployed by CodeDeploy.\n */\nexport class ApplicationLoadBalancedCodeDeployedFargateService extends ApplicationLoadBalancedFargateService {\n  /**\n   * Composite alarm for monitoring health of service.\n   */\n  healthAlarm?: IAlarm;\n\n  /**\n   * API Canary for the service.\n   */\n  apiCanary?: ApiCanary;\n\n  /**\n   * Test listener to use for CodeDeploy deployments.\n   */\n  testListener: ApplicationListener;\n\n  /**\n   * Test target group to use for CodeDeploy deployments.\n   */\n  greenTargetGroup: ApplicationTargetGroup;\n\n  /**\n   * S3 Bucket used for access logs.\n   */\n  accessLogBucket: IBucket;\n\n  /**\n   * CodeDeploy application for this service.\n   */\n  application: EcsApplication;\n\n  /**\n   * CodeDeploy deployment group for this service.\n   */\n  deploymentGroup: EcsDeploymentGroup;\n\n  /**\n   * CodeDeploy deployment for this service.\n   */\n  deployment: EcsDeployment;\n\n  /**\n   * Constructs a new instance of the ApplicationLoadBalancedCodeDeployedFargateService class.\n   */\n  constructor(scope: Construct, id: string, props: ApplicationLoadBalancedCodeDeployedFargateServiceProps) {\n    super(scope, id, {\n      ...props,\n      deploymentController: {\n        type: DeploymentControllerType.CODE_DEPLOY,\n      },\n    });\n\n    if (props.deregistrationDelay) {\n      this.targetGroup.setAttribute('deregistration_delay.timeout_seconds', props.deregistrationDelay.toSeconds().toString());\n    }\n    if (props.targetHealthCheck) {\n      this.targetGroup.configureHealthCheck(props.targetHealthCheck);\n    }\n    this.accessLogBucket = props.accessLogBucket ??\n      new Bucket(this, 'AccessLogBucket', {\n        encryption: BucketEncryption.S3_MANAGED,\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        enforceSSL: true,\n      });\n\n    this.loadBalancer.logAccessLogs(this.accessLogBucket, props.accessLogPrefix);\n\n    const alarms: IAlarm[] = [];\n    if (props.responseTimeAlarmThreshold) {\n      const responseTimeAlarm = new Alarm(this, 'ResponseTimeAlarm', {\n        metric: this.loadBalancer.metrics.targetResponseTime({\n          period: Duration.minutes(1),\n          statistic: 'p95',\n        }),\n        evaluationPeriods: 2,\n        threshold: props.responseTimeAlarmThreshold.toSeconds(),\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n      });\n      alarms.push(responseTimeAlarm);\n    }\n\n    const protocol = props.protocol ?? (props.certificate ? ApplicationProtocol.HTTPS : ApplicationProtocol.HTTP);\n\n    const testHostName = props.domainName ? props.domainName : this.loadBalancer.loadBalancerDnsName;\n\n    if (props.apiTestSteps?.length) {\n      this.apiCanary = new ApiCanary(this, 'Canary', {\n        baseUrl: `${protocol.toLowerCase()}://${testHostName}`,\n        durationAlarmThreshold: props.apiCanaryTimeout,\n        schedule: Schedule.rate(props.apiCanarySchedule ?? Duration.minutes(5)),\n        threadCount: props.apiCanaryThreadCount,\n        steps: props.apiTestSteps,\n      });\n\n      this.apiCanary.node.addDependency(this.service);\n\n      alarms.push(this.apiCanary.successAlarm);\n      if (this.apiCanary.durationAlarm) {\n        alarms.push(this.apiCanary.durationAlarm);\n      }\n    }\n    if (alarms.length > 0) {\n      this.healthAlarm = new CompositeAlarm(this, 'HealthAlarm', {\n        alarmRule: AlarmRule.anyOf(...alarms),\n      });\n    } else {\n      this.healthAlarm = undefined;\n    }\n\n    let testPort: number;\n    if (props.testPort) {\n      testPort = props.testPort;\n    } else if (props.listenerPort) {\n      testPort = props.listenerPort + 1;\n    } else if (protocol === ApplicationProtocol.HTTP) {\n      testPort = 8080;\n    } else if (protocol === ApplicationProtocol.HTTPS) {\n      testPort = 8443;\n    } else {\n      throw new Error('Unable to determine port for test listener');\n    }\n\n    let certificates;\n    if (props.certificate) {\n      certificates = [props.certificate];\n    }\n\n    this.testListener = this.loadBalancer.addListener('TestListener', {\n      protocol,\n      port: testPort,\n      open: props.openListener ?? true,\n      sslPolicy: props.sslPolicy,\n      certificates: certificates,\n    });\n\n    this.greenTargetGroup = new ApplicationTargetGroup(this, 'GreenTargetGroup', {\n      vpc: this.cluster.vpc,\n      port: testPort,\n      protocol: props.targetProtocol ?? ApplicationProtocol.HTTP,\n      protocolVersion: props.protocolVersion,\n      deregistrationDelay: props.deregistrationDelay,\n      healthCheck: props.targetHealthCheck,\n      targetType: TargetType.IP,\n    });\n\n    this.listener.node.addDependency(this.greenTargetGroup);\n\n    this.testListener.addTargetGroups('ECS', {\n      targetGroups: [this.greenTargetGroup],\n    });\n\n    this.application = new EcsApplication(this, 'Application', {\n      applicationName: props.applicationName,\n    });\n\n    this.deploymentGroup = new EcsDeploymentGroup(this, 'DeploymentGroup', {\n      application: this.application,\n      deploymentGroupName: props.deploymentGroupName,\n      alarms: this.healthAlarm?[this.healthAlarm]:undefined,\n      service: this.service,\n      blueGreenDeploymentConfig: {\n        blueTargetGroup: this.targetGroup,\n        greenTargetGroup: this.greenTargetGroup,\n        listener: this.listener,\n        testListener: this.testListener,\n        terminationWaitTime: props.terminationWaitTime ?? Duration.minutes(10),\n      },\n      deploymentConfig: props.deploymentConfig ?? EcsDeploymentConfig.ALL_AT_ONCE,\n      autoRollback: {\n        stoppedDeployment: true,\n      },\n    });\n    this.deployment = new EcsDeployment({\n      deploymentGroup: this.deploymentGroup,\n      timeout: props.deploymentTimeout ?? Duration.minutes(60),\n      targetService: {\n        taskDefinition: this.taskDefinition,\n        containerName: this.taskDefinition.defaultContainer!.containerName,\n        containerPort: this.taskDefinition.defaultContainer!.containerPort,\n      },\n      hooks: props.hooks,\n    });\n  }\n\n  protected addServiceAsTarget(service: BaseService) {\n    super.addServiceAsTarget(service);\n  }\n}\n"]}