@cloudcamp/aws-runtime
Version:
CloudCamp - Launch faster by building scalable infrastructure in few lines of code.
334 lines • 48.9 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.WebService = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const app_1 = require("./app");
const _ = require("lodash");
const ec2 = require("aws-cdk-lib/aws-ec2");
const logs = require("aws-cdk-lib/aws-logs");
const ecs = require("aws-cdk-lib/aws-ecs");
const ecs_patterns = require("aws-cdk-lib/aws-ecs-patterns");
const cdk = require("aws-cdk-lib/core");
const elasticloadbalancingv2 = require("aws-cdk-lib/aws-elasticloadbalancingv2");
const chatbot = require("aws-cdk-lib/aws-chatbot");
const sns = require("aws-cdk-lib/aws-sns");
const cloudwatch = require("aws-cdk-lib/aws-cloudwatch");
const cloudwatch_actions = require("aws-cdk-lib/aws-cloudwatch-actions");
const subscriptions = require("aws-cdk-lib/aws-sns-subscriptions");
const applicationautoscaling = require("aws-cdk-lib/aws-applicationautoscaling");
const utils_1 = require("./utils");
const _1 = require(".");
const constructs_1 = require("constructs");
/**
* (experimental) A scalable web server running one or more docker containers behind a load balancer.
*
* `WebService` runs any web application behing a load balancers as docker
* containers. For example, this runs a web application as a single container
* exposed on port 8080:
*
* ```ts
* void 0;
* import { App, WebService } from "@cloudcamp/aws-runtime";
* let app = new App();
* void 'show';
* new WebService(app.production, "prod-web", {
* dockerfile: "../Dockerfile",
* port: 8080
* });
* ```
*
* @experimental
* @order 4
*/
class WebService extends constructs_1.Construct {
/**
* (experimental) Initialize a new web service.
*
* *Examples:*
*
* To use your own domain and serve traffic via SSL, use the `domain`
* and `ssl` properties:
* ```ts
* void 0;
* import { App, WebService } from "@cloudcamp/aws-runtime";
* let app = new App();
* void 'show';
*
* new WebService(app.production, "prod", {
* dockerfile: "../Dockerfile",
* domain: "example.com",
* ssl: true
* });
* ```
*
* See `{@link "command/domain/#domain-create" | domain:create}` and
* `{@link "command/cert/#cert-create" | cert:create}` for more information on
* setting up domains/SSL.
*
* @param scope the parent, i.e. a stack.
* @param id a unique identifier within the parent scope.
* @param props the properties of WebService.
* @experimental
* @remarks During initialization you can configure: Custom domains, SSL,
* machine configuration, health checks and the default number of instances.
* @topic Initialization
*/
constructor(scope, id, props) {
super(scope, id);
let appName = app_1.App.instance.configuration.name;
let vpc = ec2.Vpc.fromLookup(this, "vpc", {
vpcId: app_1.App.instance.configuration.vpcId,
});
let logGroup = new logs.LogGroup(this, "log-group", {
logGroupName: `/${appName}/webserver/${id}`,
retention: logs.RetentionDays.ONE_MONTH,
removalPolicy: cdk.RemovalPolicy.DESTROY,
});
let certificate = undefined;
if (props.domain) {
certificate = _1.Ref.getCertificate(this, props.domain + "-certificate", {
appName: app_1.App.instance.configuration.name,
name: props.domain,
});
}
this.fargateService =
new ecs_patterns.ApplicationLoadBalancedFargateService(this, "fargate-service", {
vpc: vpc,
cpu: props.cpu,
memoryLimitMiB: props.memory,
desiredCount: props.desiredCount,
assignPublicIp: true,
publicLoadBalancer: true,
domainName: props.domain,
certificate: certificate,
redirectHTTP: certificate ? true : false,
serviceName: id,
protocol: certificate
? elasticloadbalancingv2.ApplicationProtocol.HTTPS
: elasticloadbalancingv2.ApplicationProtocol.HTTP,
taskImageOptions: {
image: ecs.ContainerImage.fromAsset(path.dirname(props.dockerfile), {
file: path.basename(props.dockerfile),
// exclude is deprecated, but this seems to be just a
// side-effect of internal refactoring
// https://github.com/aws/aws-cdk/issues/10125
exclude: ["cdk.out"],
}),
containerPort: props.port || 80,
enableLogging: true,
logDriver: ecs.LogDriver.awsLogs({
streamPrefix: "ecs",
logGroup: logGroup,
}),
environment: props.environment,
},
});
if (props.healthCheckPath) {
this.fargateService.targetGroup.configureHealthCheck({
path: props.healthCheckPath,
port: (props.port || 80).toString(),
});
}
}
/**
* @experimental
*/
scaleOnSchedule(props) {
let task = this.fargateService.service.autoScaleTaskCount({
minCapacity: props.min,
maxCapacity: props.max,
});
for (let schedule of props.schedule) {
task.scaleOnSchedule(schedule.id, {
schedule: applicationautoscaling.Schedule.cron(schedule),
});
}
}
/**
* @experimental
*/
scaleOnMetric(props) {
let task = this.fargateService.service.autoScaleTaskCount({
minCapacity: props.min,
maxCapacity: props.max,
});
if (props.cpu !== undefined) {
task.scaleOnCpuUtilization("autoscale-cpu", {
targetUtilizationPercent: props.cpu,
});
}
if (props.memory !== undefined) {
task.scaleOnMemoryUtilization("autoscale-memory", {
targetUtilizationPercent: props.memory,
});
}
if (props.requestCount !== undefined) {
task.scaleOnRequestCount("autoscale-request-count", {
requestsPerTarget: props.requestCount,
targetGroup: this.fargateService.targetGroup,
});
}
}
/**
* @experimental
*/
alarms(props) {
var _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o;
props = utils_1.setDefaults(props, {
slack: undefined,
emails: [],
phones: [],
http5xx: {
duration: 1,
threshold: 1,
enabled: true,
},
http4xx: {
duration: 1,
threshold: 5,
enabled: true,
},
rejected: {
duration: 1,
threshold: 5,
enabled: true,
},
slow: {
duration: 1,
threshold: 5,
enabled: true,
},
});
let appName = app_1.App.instance.configuration.name;
let topic = new sns.Topic(this, "web-service-alarms-topic", {
displayName: "Web service Alarms Topic",
});
if (props.slack !== undefined) {
new chatbot.SlackChannelConfiguration(this, "slack-channel", {
slackChannelConfigurationName: "Slack Alarms Channel",
slackWorkspaceId: props.slack.workspaceId,
slackChannelId: props.slack.channelId,
notificationTopics: [topic],
loggingLevel: chatbot.LoggingLevel.INFO,
});
}
for (let email of props.email) {
topic.addSubscription(new subscriptions.EmailSubscription(email));
}
for (let sms of props.sms) {
topic.addSubscription(new subscriptions.SmsSubscription(sms));
}
if ((_b = props === null || props === void 0 ? void 0 : props.http5xx) === null || _b === void 0 ? void 0 : _b.enabled) {
this.addHttpAlarm("HTTP_5XX", `${appName}/${this.node.id}: HTTP 5XX threshold exceeded`, topic, (_c = props === null || props === void 0 ? void 0 : props.http5xx) === null || _c === void 0 ? void 0 : _c.threshold, (_d = props === null || props === void 0 ? void 0 : props.http5xx) === null || _d === void 0 ? void 0 : _d.duration);
}
if ((_e = props === null || props === void 0 ? void 0 : props.http4xx) === null || _e === void 0 ? void 0 : _e.enabled) {
this.addHttpAlarm("HTTP_4XX", `${appName}/${this.node.id}: HTTP 4XX threshold exceeded`, topic, (_f = props === null || props === void 0 ? void 0 : props.http4xx) === null || _f === void 0 ? void 0 : _f.threshold, (_g = props === null || props === void 0 ? void 0 : props.http4xx) === null || _g === void 0 ? void 0 : _g.duration);
}
if ((_h = props === null || props === void 0 ? void 0 : props.rejected) === null || _h === void 0 ? void 0 : _h.enabled) {
this.addRejectedAlarm(topic, (_j = props === null || props === void 0 ? void 0 : props.rejected) === null || _j === void 0 ? void 0 : _j.threshold, (_k = props === null || props === void 0 ? void 0 : props.rejected) === null || _k === void 0 ? void 0 : _k.duration);
}
if ((_l = props === null || props === void 0 ? void 0 : props.slow) === null || _l === void 0 ? void 0 : _l.enabled) {
this.addSlowAlarm(topic, (_m = props === null || props === void 0 ? void 0 : props.slow) === null || _m === void 0 ? void 0 : _m.threshold, (_o = props === null || props === void 0 ? void 0 : props.slow) === null || _o === void 0 ? void 0 : _o.duration);
}
}
addHttpAlarm(name, description, topic, threshold, period) {
let elbCode;
switch (name) {
case "HTTP_5XX":
elbCode = elasticloadbalancingv2.HttpCodeElb.ELB_5XX_COUNT;
break;
case "HTTP_4XX":
elbCode = elasticloadbalancingv2.HttpCodeElb.ELB_4XX_COUNT;
break;
}
let elbAlarm = new cloudwatch.Alarm(this, _.kebabCase(name + "-elb-alarm"), {
alarmName: name,
alarmDescription: description,
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
threshold: threshold,
evaluationPeriods: 1,
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
metric: this.fargateService.loadBalancer.metricHttpCodeElb(elbCode, {
period: cdk.Duration.minutes(period),
statistic: "Sum",
dimensionsMap: {
LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,
},
}),
});
elbAlarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));
elbAlarm.addOkAction(new cloudwatch_actions.SnsAction(topic));
let targetCode;
switch (name) {
case "HTTP_5XX":
targetCode = elasticloadbalancingv2.HttpCodeTarget.TARGET_5XX_COUNT;
break;
case "HTTP_4XX":
targetCode = elasticloadbalancingv2.HttpCodeTarget.TARGET_4XX_COUNT;
break;
}
let targetAlarm = new cloudwatch.Alarm(this, _.kebabCase(name + "-target-alarm"), {
alarmName: name,
alarmDescription: description,
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
threshold: threshold,
evaluationPeriods: 1,
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
metric: this.fargateService.loadBalancer.metricHttpCodeTarget(targetCode, {
period: cdk.Duration.minutes(period),
statistic: "Sum",
dimensionsMap: {
LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,
},
}),
});
targetAlarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));
targetAlarm.addOkAction(new cloudwatch_actions.SnsAction(topic));
}
addRejectedAlarm(topic, threshold, period) {
let appName = app_1.App.instance.configuration.name;
let alarm = new cloudwatch.Alarm(this, "rejected-connections-alarm", {
alarmName: "REJECTED",
alarmDescription: `${appName}/${this.node.id}: Rejected connections threshold exceeded`,
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
threshold: threshold,
evaluationPeriods: 1,
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
metric: this.fargateService.loadBalancer.metricRejectedConnectionCount({
period: cdk.Duration.minutes(period),
statistic: "Sum",
dimensionsMap: {
LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,
},
}),
});
alarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));
alarm.addOkAction(new cloudwatch_actions.SnsAction(topic));
}
addSlowAlarm(topic, threshold, period) {
let appName = app_1.App.instance.configuration.name;
let alarm = new cloudwatch.Alarm(this, "rejected-connections-alarm", {
alarmName: "REJECTED",
alarmDescription: `${appName}/${this.node.id}: Rejected connections threshold exceeded`,
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
threshold: threshold,
evaluationPeriods: 1,
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
metric: this.fargateService.loadBalancer.metricTargetResponseTime({
period: cdk.Duration.minutes(period),
statistic: "Sum",
dimensionsMap: {
LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,
},
}),
});
alarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));
alarm.addOkAction(new cloudwatch_actions.SnsAction(topic));
}
}
exports.WebService = WebService;
_a = JSII_RTTI_SYMBOL_1;
WebService[_a] = { fqn: "@cloudcamp/aws-runtime.WebService", version: "0.0.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webserver.js","sourceRoot":"","sources":["../src/webserver.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { App } from \"./app\";\nimport _ = require(\"lodash\");\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as logs from \"aws-cdk-lib/aws-logs\";\nimport * as ecs from \"aws-cdk-lib/aws-ecs\";\nimport * as ecs_patterns from \"aws-cdk-lib/aws-ecs-patterns\";\nimport * as cdk from \"aws-cdk-lib/core\";\nimport * as elasticloadbalancingv2 from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport * as chatbot from \"aws-cdk-lib/aws-chatbot\";\nimport * as sns from \"aws-cdk-lib/aws-sns\";\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\";\nimport * as cloudwatch_actions from \"aws-cdk-lib/aws-cloudwatch-actions\";\nimport * as subscriptions from \"aws-cdk-lib/aws-sns-subscriptions\";\nimport * as applicationautoscaling from \"aws-cdk-lib/aws-applicationautoscaling\";\nimport { setDefaults } from \"./utils\";\nimport { ICertificate } from \"aws-cdk-lib/aws-certificatemanager\";\nimport { Ref } from \".\";\nimport { Construct } from \"constructs\";\n\n// TODO add redirectHTTP\n// TODO add multiple domains https://jeremynagel.medium.com/adding-multiple-certificates-to-a-applicationloadbalancedfargateservice-with-cdk-adc877e2831d\nexport interface WebServiceProps {\n                                                   \n  readonly dockerfile: string;\n                                                                                \n  readonly port?: number;\n                                       \n  readonly environment?: {\n    [key: string]: string;\n  };\n                     \n  readonly domain?: string;\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            \n  readonly cpu?: number;\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          \n  readonly memory?: number;\n\n  readonly desiredCount?: number;\n  readonly healthCheckPath?: string;\n}\n\nexport interface AlarmConfiguration {\n  readonly duration?: number;\n  readonly threshold?: number;\n  readonly enabled?: boolean;\n}\n\nexport interface SlackConfiguration {\n  readonly workspaceId: string;\n  readonly channelId: string;\n}\n\nexport interface WebServiceAlarmProps {\n  readonly slack?: SlackConfiguration;\n  readonly email?: string[];\n  readonly sms?: string[];\n  readonly http5xx?: AlarmConfiguration;\n  readonly http4xx?: AlarmConfiguration;\n  readonly rejected?: AlarmConfiguration;\n  readonly slow?: AlarmConfiguration;\n}\n\nexport interface ScalingSchedule {\n  readonly id: string;\n                                                                                  \n  readonly minute?: string;\n                                                                              \n  readonly hour?: string;\n                                                                                                      \n  readonly day?: string;\n                                                                                \n  readonly month?: string;\n                                                                              \n  readonly year?: string;\n                                                                                                  \n  readonly weekDay?: string;\n}\n\nexport interface ScheduleScalingProps {\n  readonly min: number;\n  readonly max: number;\n  readonly schedule: ScalingSchedule[];\n}\n\nexport interface MetricScalingProps {\n  readonly min: number;\n  readonly max: number;\n  readonly cpu?: number;\n  readonly memory?: number;\n  readonly requestCount?: number;\n}\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              \nexport class WebService extends Construct {\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           \n  constructor(scope: Construct, id: string, props: WebServiceProps) {\n    super(scope, id);\n\n    let appName = App.instance.configuration.name;\n\n    let vpc = ec2.Vpc.fromLookup(this, \"vpc\", {\n      vpcId: App.instance.configuration.vpcId,\n    });\n\n    let logGroup = new logs.LogGroup(this, \"log-group\", {\n      logGroupName: `/${appName}/webserver/${id}`,\n      retention: logs.RetentionDays.ONE_MONTH,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n    });\n\n    let certificate: ICertificate | undefined = undefined;\n\n    if (props.domain) {\n      certificate = Ref.getCertificate(this, props.domain! + \"-certificate\", {\n        appName: App.instance.configuration.name,\n        name: props.domain!,\n      });\n    }\n\n    this.fargateService =\n      new ecs_patterns.ApplicationLoadBalancedFargateService(\n        this,\n        \"fargate-service\",\n        {\n          vpc: vpc,\n          cpu: props.cpu,\n          memoryLimitMiB: props.memory,\n          desiredCount: props.desiredCount,\n          assignPublicIp: true,\n          publicLoadBalancer: true,\n          domainName: props.domain,\n          certificate: certificate,\n          redirectHTTP: certificate ? true : false,\n          serviceName: id,\n          protocol: certificate\n            ? elasticloadbalancingv2.ApplicationProtocol.HTTPS\n            : elasticloadbalancingv2.ApplicationProtocol.HTTP,\n          taskImageOptions: {\n            image: ecs.ContainerImage.fromAsset(\n              path.dirname(props.dockerfile),\n              {\n                file: path.basename(props.dockerfile),\n\n                // exclude is deprecated, but this seems to be just a\n                // side-effect of internal refactoring\n                // https://github.com/aws/aws-cdk/issues/10125\n                exclude: [\"cdk.out\"],\n              }\n            ),\n            containerPort: props.port || 80,\n            enableLogging: true,\n            logDriver: ecs.LogDriver.awsLogs({\n              streamPrefix: \"ecs\",\n              logGroup: logGroup,\n            }),\n            environment: props.environment,\n          },\n        }\n      );\n\n    if (props.healthCheckPath) {\n      this.fargateService.targetGroup.configureHealthCheck({\n        path: props.healthCheckPath,\n        port: (props.port || 80).toString(),\n      });\n    }\n  }\n\n  fargateService: ecs_patterns.ApplicationLoadBalancedFargateService;\n\n  scaleOnSchedule(props: ScheduleScalingProps) {\n    let task = this.fargateService.service.autoScaleTaskCount({\n      minCapacity: props.min,\n      maxCapacity: props.max,\n    });\n    for (let schedule of props.schedule) {\n      task.scaleOnSchedule(schedule.id, {\n        schedule: applicationautoscaling.Schedule.cron(schedule),\n      });\n    }\n  }\n\n  scaleOnMetric(props: MetricScalingProps) {\n    let task = this.fargateService.service.autoScaleTaskCount({\n      minCapacity: props.min,\n      maxCapacity: props.max,\n    });\n    if (props.cpu !== undefined) {\n      task.scaleOnCpuUtilization(\"autoscale-cpu\", {\n        targetUtilizationPercent: props.cpu,\n      });\n    }\n\n    if (props.memory !== undefined) {\n      task.scaleOnMemoryUtilization(\"autoscale-memory\", {\n        targetUtilizationPercent: props.memory,\n      });\n    }\n\n    if (props.requestCount !== undefined) {\n      task.scaleOnRequestCount(\"autoscale-request-count\", {\n        requestsPerTarget: props.requestCount,\n        targetGroup: this.fargateService.targetGroup,\n      });\n    }\n  }\n\n  alarms(props?: WebServiceAlarmProps) {\n    props = setDefaults(props, {\n      slack: undefined,\n      emails: [],\n      phones: [],\n      http5xx: {\n        duration: 1,\n        threshold: 1,\n        enabled: true,\n      },\n      http4xx: {\n        duration: 1,\n        threshold: 5,\n        enabled: true,\n      },\n      rejected: {\n        duration: 1,\n        threshold: 5,\n        enabled: true,\n      },\n      slow: {\n        duration: 1,\n        threshold: 5,\n        enabled: true,\n      },\n    });\n\n    let appName = App.instance.configuration.name;\n\n    let topic = new sns.Topic(this, \"web-service-alarms-topic\", {\n      displayName: \"Web service Alarms Topic\",\n    });\n\n    if (props.slack !== undefined) {\n      new chatbot.SlackChannelConfiguration(this, \"slack-channel\", {\n        slackChannelConfigurationName: \"Slack Alarms Channel\",\n        slackWorkspaceId: props.slack.workspaceId,\n        slackChannelId: props.slack.channelId,\n        notificationTopics: [topic],\n        loggingLevel: chatbot.LoggingLevel.INFO, // TODO should be ERROR?\n      });\n    }\n\n    for (let email of props.email as string[]) {\n      topic.addSubscription(new subscriptions.EmailSubscription(email));\n    }\n\n    for (let sms of props.sms as string[]) {\n      topic.addSubscription(new subscriptions.SmsSubscription(sms));\n    }\n\n    if (props?.http5xx?.enabled) {\n      this.addHttpAlarm(\n        \"HTTP_5XX\",\n        `${appName}/${this.node.id}: HTTP 5XX threshold exceeded`,\n        topic,\n        props?.http5xx?.threshold as number,\n        props?.http5xx?.duration as number\n      );\n    }\n\n    if (props?.http4xx?.enabled) {\n      this.addHttpAlarm(\n        \"HTTP_4XX\",\n        `${appName}/${this.node.id}: HTTP 4XX threshold exceeded`,\n        topic,\n        props?.http4xx?.threshold as number,\n        props?.http4xx?.duration as number\n      );\n    }\n\n    if (props?.rejected?.enabled) {\n      this.addRejectedAlarm(\n        topic,\n        props?.rejected?.threshold as number,\n        props?.rejected?.duration as number\n      );\n    }\n\n    if (props?.slow?.enabled) {\n      this.addSlowAlarm(\n        topic,\n        props?.slow?.threshold as number,\n        props?.slow?.duration as number\n      );\n    }\n  }\n\n  private addHttpAlarm(\n    name: \"HTTP_5XX\" | \"HTTP_4XX\",\n    description: string,\n    topic: sns.ITopic,\n    threshold: number,\n    period: number\n  ) {\n    let elbCode: elasticloadbalancingv2.HttpCodeElb;\n    switch (name) {\n      case \"HTTP_5XX\":\n        elbCode = elasticloadbalancingv2.HttpCodeElb.ELB_5XX_COUNT;\n        break;\n      case \"HTTP_4XX\":\n        elbCode = elasticloadbalancingv2.HttpCodeElb.ELB_4XX_COUNT;\n        break;\n    }\n\n    let elbAlarm = new cloudwatch.Alarm(\n      this,\n      _.kebabCase(name + \"-elb-alarm\"),\n      {\n        alarmName: name,\n        alarmDescription: description,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        threshold: threshold,\n        evaluationPeriods: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n        metric: this.fargateService.loadBalancer.metricHttpCodeElb(elbCode, {\n          period: cdk.Duration.minutes(period),\n          statistic: \"Sum\",\n          dimensionsMap: {\n            LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,\n          },\n        }),\n      }\n    );\n    elbAlarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));\n    elbAlarm.addOkAction(new cloudwatch_actions.SnsAction(topic));\n\n    let targetCode: elasticloadbalancingv2.HttpCodeTarget;\n    switch (name) {\n      case \"HTTP_5XX\":\n        targetCode = elasticloadbalancingv2.HttpCodeTarget.TARGET_5XX_COUNT;\n        break;\n      case \"HTTP_4XX\":\n        targetCode = elasticloadbalancingv2.HttpCodeTarget.TARGET_4XX_COUNT;\n        break;\n    }\n\n    let targetAlarm = new cloudwatch.Alarm(\n      this,\n      _.kebabCase(name + \"-target-alarm\"),\n      {\n        alarmName: name,\n        alarmDescription: description,\n        comparisonOperator:\n          cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        threshold: threshold,\n        evaluationPeriods: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n        metric: this.fargateService.loadBalancer.metricHttpCodeTarget(\n          targetCode,\n          {\n            period: cdk.Duration.minutes(period),\n            statistic: \"Sum\",\n            dimensionsMap: {\n              LoadBalancer:\n                this.fargateService.loadBalancer.loadBalancerFullName,\n            },\n          }\n        ),\n      }\n    );\n    targetAlarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));\n    targetAlarm.addOkAction(new cloudwatch_actions.SnsAction(topic));\n  }\n\n  private addRejectedAlarm(\n    topic: sns.ITopic,\n    threshold: number,\n    period: number\n  ) {\n    let appName = App.instance.configuration.name;\n    let alarm = new cloudwatch.Alarm(this, \"rejected-connections-alarm\", {\n      alarmName: \"REJECTED\",\n      alarmDescription: `${appName}/${this.node.id}: Rejected connections threshold exceeded`,\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      threshold: threshold,\n      evaluationPeriods: 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      metric: this.fargateService.loadBalancer.metricRejectedConnectionCount({\n        period: cdk.Duration.minutes(period),\n        statistic: \"Sum\",\n        dimensionsMap: {\n          LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,\n        },\n      }),\n    });\n    alarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));\n    alarm.addOkAction(new cloudwatch_actions.SnsAction(topic));\n  }\n\n  private addSlowAlarm(topic: sns.ITopic, threshold: number, period: number) {\n    let appName = App.instance.configuration.name;\n    let alarm = new cloudwatch.Alarm(this, \"rejected-connections-alarm\", {\n      alarmName: \"REJECTED\",\n      alarmDescription: `${appName}/${this.node.id}: Rejected connections threshold exceeded`,\n      comparisonOperator:\n        cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n      threshold: threshold,\n      evaluationPeriods: 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      metric: this.fargateService.loadBalancer.metricTargetResponseTime({\n        period: cdk.Duration.minutes(period),\n        statistic: \"Sum\",\n        dimensionsMap: {\n          LoadBalancer: this.fargateService.loadBalancer.loadBalancerFullName,\n        },\n      }),\n    });\n    alarm.addAlarmAction(new cloudwatch_actions.SnsAction(topic));\n    alarm.addOkAction(new cloudwatch_actions.SnsAction(topic));\n  }\n}\n"]}