@liflig/cdk 3.22.18 → 3.23.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,5 @@
1
1
  import * as cloudwatchActions from "aws-cdk-lib/aws-cloudwatch-actions";
2
+ import * as lambda from "aws-cdk-lib/aws-lambda";
2
3
  import type * as secretsmanager from "aws-cdk-lib/aws-secretsmanager";
3
4
  import * as sns from "aws-cdk-lib/aws-sns";
4
5
  import * as constructs from "constructs";
@@ -15,11 +16,15 @@ export interface SlackAlarmProps {
15
16
  slackWebhookUrlSecret: secretsmanager.ISecret;
16
17
  }
17
18
  /**
18
- * SNS Topic that can be used to action alarms, with a Lambda
19
+ * Creates
20
+ * - SNS Topic that can be used to action alarms, with a Lambda
19
21
  * that will send a message to Slack for the alarm.
22
+ * - A reusable Lambda that accepts CloudWatch Logs subscription events and posts
23
+ * formatted error messages to Slack.
20
24
  */
21
25
  export declare class SlackAlarm extends constructs.Construct {
22
26
  readonly alarmTopic: sns.Topic;
23
27
  readonly snsAction: cloudwatchActions.SnsAction;
28
+ readonly logHandler: lambda.Function;
24
29
  constructor(scope: constructs.Construct, id: string, props: SlackAlarmProps);
25
30
  }
@@ -7,21 +7,25 @@ import { Effect, PolicyStatement } from "aws-cdk-lib/aws-iam";
7
7
  import * as lambda from "aws-cdk-lib/aws-lambda";
8
8
  import * as sns from "aws-cdk-lib/aws-sns";
9
9
  import * as constructs from "constructs";
10
- const __filename = fileURLToPath(import.meta.url);
11
- const __dirname = path.dirname(__filename);
10
+ const __file = fileURLToPath(import.meta.url);
11
+ const __dir = path.dirname(__file);
12
12
  /**
13
- * SNS Topic that can be used to action alarms, with a Lambda
13
+ * Creates
14
+ * - SNS Topic that can be used to action alarms, with a Lambda
14
15
  * that will send a message to Slack for the alarm.
16
+ * - A reusable Lambda that accepts CloudWatch Logs subscription events and posts
17
+ * formatted error messages to Slack.
15
18
  */
16
19
  export class SlackAlarm extends constructs.Construct {
17
20
  alarmTopic;
18
21
  snsAction;
22
+ logHandler;
19
23
  constructor(scope, id, props) {
20
24
  super(scope, id);
21
25
  this.alarmTopic = new sns.Topic(this, "Topic");
22
26
  this.snsAction = new cloudwatchActions.SnsAction(this.alarmTopic);
23
27
  const slackLambda = new lambda.Function(this, "Function", {
24
- code: lambda.Code.fromAsset(path.join(__dirname, "../../assets/slack-alarm-lambda")),
28
+ code: lambda.Code.fromAsset(path.join(__dir, "../../assets/slack-alarm-lambda")),
25
29
  description: "Receives CloudWatch Alarms through SNS and sends a formatted version to Slack",
26
30
  handler: "index.handler",
27
31
  memorySize: 128,
@@ -33,6 +37,20 @@ export class SlackAlarm extends constructs.Construct {
33
37
  ENVIRONMENT_NAME: props.envName,
34
38
  },
35
39
  });
40
+ this.logHandler = new lambda.Function(this, "LogHandler", {
41
+ code: lambda.Code.fromAsset(path.join(__dir, "../../assets", "slack-error-log-handler-lambda")),
42
+ description: "Receives CloudWatch Logs subscription events and sends formatted errors to Slack",
43
+ handler: "index.handler",
44
+ memorySize: 128,
45
+ runtime: lambda.Runtime.PYTHON_3_14,
46
+ timeout: Duration.seconds(10),
47
+ environment: {
48
+ SLACK_URL_SECRET_NAME: props.slackWebhookUrlSecret.secretName,
49
+ PROJECT_NAME: props.projectName,
50
+ ENVIRONMENT_NAME: props.envName,
51
+ },
52
+ });
53
+ props.slackWebhookUrlSecret.grantRead(this.logHandler);
36
54
  props.slackWebhookUrlSecret.grantRead(slackLambda);
37
55
  slackLambda.addPermission("InvokePermission", {
38
56
  action: "lambda:InvokeFunction",
@@ -51,4 +69,4 @@ export class SlackAlarm extends constructs.Construct {
51
69
  });
52
70
  }
53
71
  }
54
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stYWxhcm0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWxhcm1zL3NsYWNrLWFsYXJtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxJQUFJLE1BQU0sV0FBVyxDQUFBO0FBQ2pDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxVQUFVLENBQUE7QUFDeEMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUN0QyxPQUFPLEtBQUssaUJBQWlCLE1BQU0sb0NBQW9DLENBQUE7QUFDdkUsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEVBQUUsTUFBTSxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFBO0FBQzdELE9BQU8sS0FBSyxNQUFNLE1BQU0sd0JBQXdCLENBQUE7QUFFaEQsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEtBQUssVUFBVSxNQUFNLFlBQVksQ0FBQTtBQUV4QyxNQUFNLFVBQVUsR0FBRyxhQUFhLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQTtBQUNqRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxDQUFBO0FBZTFDOzs7R0FHRztBQUNILE1BQU0sT0FBTyxVQUFXLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDbEMsVUFBVSxDQUFXO0lBQ3JCLFNBQVMsQ0FBNkI7SUFFdEQsWUFBWSxLQUEyQixFQUFFLEVBQVUsRUFBRSxLQUFzQjtRQUN6RSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxHQUFHLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQTtRQUU5QyxJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksaUJBQWlCLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQTtRQUVqRSxNQUFNLFdBQVcsR0FBRyxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUN4RCxJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQ3pCLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLGlDQUFpQyxDQUFDLENBQ3hEO1lBQ0QsV0FBVyxFQUNULCtFQUErRTtZQUNqRixPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsR0FBRztZQUNmLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1lBQzVCLFdBQVcsRUFBRTtnQkFDWCxxQkFBcUIsRUFBRSxLQUFLLENBQUMscUJBQXFCLENBQUMsVUFBVTtnQkFDN0QsWUFBWSxFQUFFLEtBQUssQ0FBQyxXQUFXO2dCQUMvQixnQkFBZ0IsRUFBRSxLQUFLLENBQUMsT0FBTzthQUNoQztTQUNGLENBQUMsQ0FBQTtRQUVGLEtBQUssQ0FBQyxxQkFBcUIsQ0FBQyxTQUFTLENBQUMsV0FBVyxDQUFDLENBQUE7UUFFbEQsV0FBVyxDQUFDLGFBQWEsQ0FBQyxrQkFBa0IsRUFBRTtZQUM1QyxNQUFNLEVBQUUsdUJBQXVCO1lBQy9CLFNBQVMsRUFBRSxJQUFJLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxtQkFBbUIsQ0FBQztZQUN4RCxTQUFTLEVBQUUsSUFBSSxDQUFDLFVBQVUsQ0FBQyxRQUFRO1NBQ3BDLENBQUMsQ0FBQTtRQUNGLFdBQVcsQ0FBQyxlQUFlLENBQ3pCLElBQUksZUFBZSxDQUFDO1lBQ2xCLE9BQU8sRUFBRSxDQUFDLDJCQUEyQixDQUFDO1lBQ3RDLE1BQU0sRUFBRSxNQUFNLENBQUMsS0FBSztZQUNwQixTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUM7U0FDakIsQ0FBQyxDQUNILENBQUE7UUFFRCxJQUFJLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxFQUFFLGNBQWMsRUFBRTtZQUN6QyxRQUFRLEVBQUUsV0FBVyxDQUFDLFdBQVc7WUFDakMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxNQUFNO1lBQ3pDLEtBQUssRUFBRSxJQUFJLENBQUMsVUFBVTtTQUN2QixDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJub2RlOnBhdGhcIlxuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gXCJub2RlOnVybFwiXG5pbXBvcnQgeyBEdXJhdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBjbG91ZHdhdGNoQWN0aW9ucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gtYWN0aW9uc1wiXG5pbXBvcnQgKiBhcyBpYW0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIlxuaW1wb3J0IHsgRWZmZWN0LCBQb2xpY3lTdGF0ZW1lbnQgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0IHR5cGUgKiBhcyBzZWNyZXRzbWFuYWdlciBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNlY3JldHNtYW5hZ2VyXCJcbmltcG9ydCAqIGFzIHNucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNuc1wiXG5pbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcblxuY29uc3QgX19maWxlbmFtZSA9IGZpbGVVUkxUb1BhdGgoaW1wb3J0Lm1ldGEudXJsKVxuY29uc3QgX19kaXJuYW1lID0gcGF0aC5kaXJuYW1lKF9fZmlsZW5hbWUpXG5cbmV4cG9ydCBpbnRlcmZhY2UgU2xhY2tBbGFybVByb3BzIHtcbiAgcHJvamVjdE5hbWU6IHN0cmluZ1xuICBlbnZOYW1lOiBzdHJpbmdcbiAgLyoqXG4gICAqIEEgcGxhaW50ZXh0IHNlY3JldCBjb250YWluaW5nIHRoZSBVUkwgb2YgYSBTbGFjayBpbmNvbWluZyB3ZWJob29rLlxuICAgKiBUaGUgd2ViaG9vayBzaG91bGQgYmUgY3JlYXRlZCB0aHJvdWdoIGEgU2xhY2sgYXBwLCBhbmQgb25seSBhbGxvd3MgcG9zdGluZyB0byBvbmUgc3BlY2lmaWMgU2xhY2sgY2hhbm5lbC5cbiAgICogU2VlIFNsYWNrJ3Mgb2ZmaWNpYWwgZG9jdW1lbnRhdGlvbiAoZS5nLiwgaHR0cHM6Ly9hcGkuc2xhY2suY29tL21lc3NhZ2luZy93ZWJob29rcykgZm9yIG1vcmUgZGV0YWlscy5cbiAgICpcbiAgICogTk9URTogSW5jb21pbmcgd2ViaG9va3MgY3JlYXRlZCB0aHJvdWdoIGxlZ2FjeSBjdXN0b20gaW50ZWdyYXRpb25zIGluIFNsYWNrIGFyZSBub3Qgc3VwcG9ydGVkLlxuICAgKi9cbiAgc2xhY2tXZWJob29rVXJsU2VjcmV0OiBzZWNyZXRzbWFuYWdlci5JU2VjcmV0XG59XG5cbi8qKlxuICogU05TIFRvcGljIHRoYXQgY2FuIGJlIHVzZWQgdG8gYWN0aW9uIGFsYXJtcywgd2l0aCBhIExhbWJkYVxuICogdGhhdCB3aWxsIHNlbmQgYSBtZXNzYWdlIHRvIFNsYWNrIGZvciB0aGUgYWxhcm0uXG4gKi9cbmV4cG9ydCBjbGFzcyBTbGFja0FsYXJtIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgYWxhcm1Ub3BpYzogc25zLlRvcGljXG4gIHB1YmxpYyByZWFkb25seSBzbnNBY3Rpb246IGNsb3Vkd2F0Y2hBY3Rpb25zLlNuc0FjdGlvblxuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFNsYWNrQWxhcm1Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuYWxhcm1Ub3BpYyA9IG5ldyBzbnMuVG9waWModGhpcywgXCJUb3BpY1wiKVxuXG4gICAgdGhpcy5zbnNBY3Rpb24gPSBuZXcgY2xvdWR3YXRjaEFjdGlvbnMuU25zQWN0aW9uKHRoaXMuYWxhcm1Ub3BpYylcblxuICAgIGNvbnN0IHNsYWNrTGFtYmRhID0gbmV3IGxhbWJkYS5GdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChcbiAgICAgICAgcGF0aC5qb2luKF9fZGlybmFtZSwgXCIuLi8uLi9hc3NldHMvc2xhY2stYWxhcm0tbGFtYmRhXCIpLFxuICAgICAgKSxcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICBcIlJlY2VpdmVzIENsb3VkV2F0Y2ggQWxhcm1zIHRocm91Z2ggU05TIGFuZCBzZW5kcyBhIGZvcm1hdHRlZCB2ZXJzaW9uIHRvIFNsYWNrXCIsXG4gICAgICBoYW5kbGVyOiBcImluZGV4LmhhbmRsZXJcIixcbiAgICAgIG1lbW9yeVNpemU6IDEyOCxcbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLlBZVEhPTl8zXzEzLFxuICAgICAgdGltZW91dDogRHVyYXRpb24uc2Vjb25kcyg2KSxcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIFNMQUNLX1VSTF9TRUNSRVRfTkFNRTogcHJvcHMuc2xhY2tXZWJob29rVXJsU2VjcmV0LnNlY3JldE5hbWUsXG4gICAgICAgIFBST0pFQ1RfTkFNRTogcHJvcHMucHJvamVjdE5hbWUsXG4gICAgICAgIEVOVklST05NRU5UX05BTUU6IHByb3BzLmVudk5hbWUsXG4gICAgICB9LFxuICAgIH0pXG5cbiAgICBwcm9wcy5zbGFja1dlYmhvb2tVcmxTZWNyZXQuZ3JhbnRSZWFkKHNsYWNrTGFtYmRhKVxuXG4gICAgc2xhY2tMYW1iZGEuYWRkUGVybWlzc2lvbihcIkludm9rZVBlcm1pc3Npb25cIiwge1xuICAgICAgYWN0aW9uOiBcImxhbWJkYTpJbnZva2VGdW5jdGlvblwiLFxuICAgICAgcHJpbmNpcGFsOiBuZXcgaWFtLlNlcnZpY2VQcmluY2lwYWwoXCJzbnMuYW1hem9uYXdzLmNvbVwiKSxcbiAgICAgIHNvdXJjZUFybjogdGhpcy5hbGFybVRvcGljLnRvcGljQXJuLFxuICAgIH0pXG4gICAgc2xhY2tMYW1iZGEuYWRkVG9Sb2xlUG9saWN5KFxuICAgICAgbmV3IFBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgIGFjdGlvbnM6IFtcImNsb3Vkd2F0Y2g6RGVzY3JpYmVBbGFybXNcIl0sXG4gICAgICAgIGVmZmVjdDogRWZmZWN0LkFMTE9XLFxuICAgICAgICByZXNvdXJjZXM6IFtcIipcIl0sXG4gICAgICB9KSxcbiAgICApXG5cbiAgICBuZXcgc25zLlN1YnNjcmlwdGlvbih0aGlzLCBcIlN1YnNjcmlwdGlvblwiLCB7XG4gICAgICBlbmRwb2ludDogc2xhY2tMYW1iZGEuZnVuY3Rpb25Bcm4sXG4gICAgICBwcm90b2NvbDogc25zLlN1YnNjcmlwdGlvblByb3RvY29sLkxBTUJEQSxcbiAgICAgIHRvcGljOiB0aGlzLmFsYXJtVG9waWMsXG4gICAgfSlcbiAgfVxufVxuIl19
72
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slack-alarm.js","sourceRoot":"","sources":["../../src/alarms/slack-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport { Duration } from \"aws-cdk-lib\"\nimport * as cloudwatchActions from \"aws-cdk-lib/aws-cloudwatch-actions\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport { Effect, PolicyStatement } from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport type * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as sns from \"aws-cdk-lib/aws-sns\"\nimport * as constructs from \"constructs\"\n\nconst __file = fileURLToPath(import.meta.url)\nconst __dir = path.dirname(__file)\n\nexport interface SlackAlarmProps {\n  projectName: string\n  envName: string\n  /**\n   * A plaintext secret containing the URL of a Slack incoming webhook.\n   * The webhook should be created through a Slack app, and only allows posting to one specific Slack channel.\n   * See Slack's official documentation (e.g., https://api.slack.com/messaging/webhooks) for more details.\n   *\n   * NOTE: Incoming webhooks created through legacy custom integrations in Slack are not supported.\n   */\n  slackWebhookUrlSecret: secretsmanager.ISecret\n}\n\n/**\n * Creates\n * - SNS Topic that can be used to action alarms, with a Lambda\n * that will send a message to Slack for the alarm.\n * - A reusable Lambda that accepts CloudWatch Logs subscription events and posts\n * formatted error messages to Slack.\n */\nexport class SlackAlarm extends constructs.Construct {\n  public readonly alarmTopic: sns.Topic\n  public readonly snsAction: cloudwatchActions.SnsAction\n  public readonly logHandler: lambda.Function\n\n  constructor(scope: constructs.Construct, id: string, props: SlackAlarmProps) {\n    super(scope, id)\n\n    this.alarmTopic = new sns.Topic(this, \"Topic\")\n\n    this.snsAction = new cloudwatchActions.SnsAction(this.alarmTopic)\n\n    const slackLambda = new lambda.Function(this, \"Function\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dir, \"../../assets/slack-alarm-lambda\"),\n      ),\n      description:\n        \"Receives CloudWatch Alarms through SNS and sends a formatted version to Slack\",\n      handler: \"index.handler\",\n      memorySize: 128,\n      runtime: lambda.Runtime.PYTHON_3_13,\n      timeout: Duration.seconds(6),\n      environment: {\n        SLACK_URL_SECRET_NAME: props.slackWebhookUrlSecret.secretName,\n        PROJECT_NAME: props.projectName,\n        ENVIRONMENT_NAME: props.envName,\n      },\n    })\n\n    this.logHandler = new lambda.Function(this, \"LogHandler\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dir, \"../../assets\", \"slack-error-log-handler-lambda\"),\n      ),\n      description:\n        \"Receives CloudWatch Logs subscription events and sends formatted errors to Slack\",\n      handler: \"index.handler\",\n      memorySize: 128,\n      runtime: lambda.Runtime.PYTHON_3_14,\n      timeout: Duration.seconds(10),\n      environment: {\n        SLACK_URL_SECRET_NAME: props.slackWebhookUrlSecret.secretName,\n        PROJECT_NAME: props.projectName,\n        ENVIRONMENT_NAME: props.envName,\n      },\n    })\n\n    props.slackWebhookUrlSecret.grantRead(this.logHandler)\n    props.slackWebhookUrlSecret.grantRead(slackLambda)\n\n    slackLambda.addPermission(\"InvokePermission\", {\n      action: \"lambda:InvokeFunction\",\n      principal: new iam.ServicePrincipal(\"sns.amazonaws.com\"),\n      sourceArn: this.alarmTopic.topicArn,\n    })\n    slackLambda.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\"cloudwatch:DescribeAlarms\"],\n        effect: Effect.ALLOW,\n        resources: [\"*\"],\n      }),\n    )\n\n    new sns.Subscription(this, \"Subscription\", {\n      endpoint: slackLambda.functionArn,\n      protocol: sns.SubscriptionProtocol.LAMBDA,\n      topic: this.alarmTopic,\n    })\n  }\n}\n"]}
@@ -1,10 +1,107 @@
1
1
  import * as cdk from "aws-cdk-lib";
2
+ import type * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
2
3
  import * as ec2 from "aws-cdk-lib/aws-ec2";
3
4
  import * as ecs from "aws-cdk-lib/aws-ecs";
4
5
  import * as elb from "aws-cdk-lib/aws-elasticloadbalancingv2";
6
+ import type * as lambda from "aws-cdk-lib/aws-lambda";
5
7
  import * as logs from "aws-cdk-lib/aws-logs";
6
8
  import * as constructs from "constructs";
7
- import type { Parameter } from "../configure-parameters/configure-parameters";
9
+ import { ServiceAlarms } from "../alarms";
10
+ import type { Parameter } from "../configure-parameters";
11
+ /**
12
+ * Configure service alarms.
13
+ *
14
+ * Alarms are enabled by default when a config object with `alarmAction` and
15
+ * `warningAction` is provided. To explicitly disable automatic alarms use
16
+ * `{ enabled: false }`.
17
+ */
18
+ export type ServiceAlarmsConfig = {
19
+ enabled: false;
20
+ } | {
21
+ alarmAction: cloudwatch.IAlarmAction;
22
+ warningAction: cloudwatch.IAlarmAction;
23
+ loadBalancerFullName: string;
24
+ /** Optional Lambda function that will receive
25
+ * forwarded log events */
26
+ logHandler?: lambda.IFunction;
27
+ /**
28
+ * Individual alarm configuration overrides defaults.
29
+ */
30
+ jsonErrorAlarm?: {
31
+ /**
32
+ * @default true
33
+ */
34
+ enabled?: boolean;
35
+ alarmDescription?: string;
36
+ enableOkAlarm?: boolean;
37
+ action?: cloudwatch.IAlarmAction;
38
+ };
39
+ uncaughtJavaExceptionAlarm?: {
40
+ /**
41
+ * @default false
42
+ */
43
+ enabled?: boolean;
44
+ alarmDescription?: string;
45
+ enableOkAlarm?: boolean;
46
+ action?: cloudwatch.IAlarmAction;
47
+ };
48
+ targetHealthAlarm?: {
49
+ /**
50
+ * @default true
51
+ */
52
+ enabled?: boolean;
53
+ action?: cloudwatch.IAlarmAction;
54
+ period?: cdk.Duration;
55
+ evaluationPeriods?: number;
56
+ threshold?: number;
57
+ description?: string;
58
+ };
59
+ tooMany5xxResponsesFromTargetsAlarm?: {
60
+ /**
61
+ * @default true
62
+ */
63
+ enabled?: boolean;
64
+ action?: cloudwatch.IAlarmAction;
65
+ period?: cdk.Duration;
66
+ evaluationPeriods?: number;
67
+ threshold?: number;
68
+ description?: string;
69
+ };
70
+ targetResponseTimeAlarm?: {
71
+ /**
72
+ * @default true
73
+ */
74
+ enabled?: boolean;
75
+ action?: cloudwatch.IAlarmAction;
76
+ period?: cdk.Duration;
77
+ evaluationPeriods?: number;
78
+ threshold?: cdk.Duration;
79
+ description?: string;
80
+ };
81
+ single5xxResponseAlarm?: {
82
+ /**
83
+ * @default true
84
+ */
85
+ enabled?: boolean;
86
+ /**
87
+ * An action to use for CloudWatch alarm state changes instead of the default action
88
+ */
89
+ action?: cloudwatch.IAlarmAction;
90
+ /**
91
+ * @default 60 seconds
92
+ */
93
+ period?: cdk.Duration;
94
+ /**
95
+ * @default 1
96
+ */
97
+ evaluationPeriods?: number;
98
+ /**
99
+ * @default 1
100
+ */
101
+ threshold?: number;
102
+ description?: string;
103
+ };
104
+ };
8
105
  export interface FargateServiceProps {
9
106
  serviceName: string;
10
107
  vpc: ec2.IVpc;
@@ -66,6 +163,18 @@ export interface FargateServiceProps {
66
163
  * @default false
67
164
  */
68
165
  enableCircuitBreaker?: boolean;
166
+ /**
167
+ * Either
168
+ * - `{ enabled: false }` to disable all alarms.
169
+ * - object with required `alarmAction`, `warningAction`, and `loadBalancerFullName`.
170
+ *
171
+ * When enabled, the construct will:
172
+ * - (By default) add a log-based JSON error alarm (enabled by default).
173
+ * - (By default) add target-group related alarms (target health, 5xx responses, response time)
174
+ * when a target group is present.
175
+ * - (Opt-in) optionally enable the uncaught Java exception alarm
176
+ */
177
+ alarms: ServiceAlarmsConfig;
69
178
  }
70
179
  export declare class FargateService extends constructs.Construct {
71
180
  readonly fargateService: ecs.FargateService;
@@ -73,5 +182,6 @@ export declare class FargateService extends constructs.Construct {
73
182
  readonly taskDefinition: ecs.TaskDefinition;
74
183
  readonly targetGroup: elb.ApplicationTargetGroup | undefined;
75
184
  readonly logGroup: logs.LogGroup;
185
+ readonly serviceAlarms: ServiceAlarms | undefined;
76
186
  constructor(scope: constructs.Construct, id: string, props: FargateServiceProps);
77
187
  }
@@ -5,6 +5,7 @@ import * as ecs from "aws-cdk-lib/aws-ecs";
5
5
  import * as elb from "aws-cdk-lib/aws-elasticloadbalancingv2";
6
6
  import * as logs from "aws-cdk-lib/aws-logs";
7
7
  import * as constructs from "constructs";
8
+ import { ServiceAlarms } from "../alarms";
8
9
  import { ConfigureParameters } from "../configure-parameters";
9
10
  export class FargateService extends constructs.Construct {
10
11
  fargateService;
@@ -12,6 +13,7 @@ export class FargateService extends constructs.Construct {
12
13
  taskDefinition;
13
14
  targetGroup;
14
15
  logGroup;
16
+ serviceAlarms;
15
17
  constructor(scope, id, props) {
16
18
  super(scope, id);
17
19
  /**
@@ -37,7 +39,7 @@ export class FargateService extends constructs.Construct {
37
39
  });
38
40
  parameters.grantRead(this.taskDefinition.taskRole);
39
41
  const port = props.containerPort ?? 8080;
40
- const container = this.taskDefinition.addContainer("Container", {
42
+ this.taskDefinition.addContainer("Container", {
41
43
  logging: ecs.LogDriver.awsLogs({
42
44
  logGroup: this.logGroup,
43
45
  streamPrefix: "ecs",
@@ -111,6 +113,48 @@ export class FargateService extends constructs.Construct {
111
113
  ...props.overrideHealthCheck,
112
114
  });
113
115
  }
116
+ // Require explicit alarmAction and warningAction when enabled.
117
+ if (props.alarms && "alarmAction" in props.alarms) {
118
+ const alarms = props.alarms;
119
+ this.serviceAlarms = new ServiceAlarms(this, "Alarms", {
120
+ serviceName: props.serviceName,
121
+ alarmAction: alarms.alarmAction,
122
+ warningAction: alarms.warningAction,
123
+ logHandler: alarms.logHandler,
124
+ });
125
+ // Enabled by default (opt-out).
126
+ const jsonErrorOverrides = alarms.jsonErrorAlarm;
127
+ if (jsonErrorOverrides?.enabled ?? true) {
128
+ this.serviceAlarms.addJsonErrorAlarm({
129
+ logGroup: this.logGroup,
130
+ alarmDescription: jsonErrorOverrides?.alarmDescription,
131
+ enableOkAlarm: jsonErrorOverrides?.enableOkAlarm,
132
+ action: jsonErrorOverrides?.action,
133
+ });
134
+ }
135
+ // Not enabled by default (opt-in).
136
+ const javaExceptionOverrides = alarms.uncaughtJavaExceptionAlarm;
137
+ if (javaExceptionOverrides?.enabled) {
138
+ this.serviceAlarms.addUncaughtJavaExceptionAlarm({
139
+ logGroup: this.logGroup,
140
+ alarmDescription: javaExceptionOverrides.alarmDescription,
141
+ enableOkAlarm: javaExceptionOverrides.enableOkAlarm,
142
+ action: javaExceptionOverrides.action,
143
+ enabled: true,
144
+ });
145
+ }
146
+ if (!props.skipTargetGroup) {
147
+ // All alarms enabled by-default (opt-out individually)
148
+ this.serviceAlarms.addTargetGroupAlarms({
149
+ targetGroupFullName: this.targetGroup.targetGroupFullName,
150
+ loadBalancerFullName: alarms.loadBalancerFullName,
151
+ targetHealthAlarm: alarms.targetHealthAlarm,
152
+ single5xxResponseAlarm: alarms.single5xxResponseAlarm,
153
+ tooMany5xxResponsesFromTargetsAlarm: alarms.tooMany5xxResponsesFromTargetsAlarm,
154
+ targetResponseTimeAlarm: alarms.targetResponseTimeAlarm,
155
+ });
156
+ }
157
+ }
114
158
  }
115
159
  }
116
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-service.js","sourceRoot":"","sources":["../../src/ecs/fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport { Duration } from \"aws-cdk-lib\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport type { CfnService } from \"aws-cdk-lib/aws-ecs\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\nimport { ConfigureParameters } from \"../configure-parameters\"\nimport type { Parameter } from \"../configure-parameters/configure-parameters\"\n\nexport interface FargateServiceProps {\n  serviceName: string\n  vpc: ec2.IVpc\n  cluster: ecs.ICluster\n  desiredCount: number\n  ecsImage: ecs.ContainerImage\n  portMappings?: ecs.PortMapping[]\n  containerHealthCheck?: ecs.HealthCheck\n  /**\n   * @default 256\n   */\n  cpu?: number\n  /**\n   * @default undefined\n   */\n  runtimePlatform?: ecs.RuntimePlatform\n  /**\n   * @default 512\n   */\n  memoryLimitMiB?: number\n  /**\n   * @default 2 weeks\n   */\n  logsRetention?: logs.RetentionDays\n  /**\n   * @default AwsLogDriverMode.BLOCKING\n   */\n  logDriverMode?: ecs.AwsLogDriverMode\n  /**\n   * @default 15 seconds\n   */\n  deregistrationDelay?: cdk.Duration\n  /**\n   * @default 8080\n   */\n  containerPort?: number\n  /**\n   * @default 60 seconds\n   */\n  healthCheckGracePeriod?: cdk.Duration\n  /**\n   * Use this as workaround when adding the service to a load balancer after\n   * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'\n   * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842\n   */\n  skipHealthCheckGracePeriod?: boolean\n  parameters?: Parameter[]\n  overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>\n  overrideHealthCheck?: Partial<elb.HealthCheck>\n  overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>\n  overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>\n  secrets?: Record<string, ecs.Secret>\n  environment?: Record<string, string>\n  /**\n   * @default false\n   */\n  skipTargetGroup?: boolean\n  /**\n   * @default false\n   */\n  enableCircuitBreaker?: boolean\n}\n\nexport class FargateService extends constructs.Construct {\n  public readonly fargateService: ecs.FargateService\n  public readonly securityGroup: ec2.SecurityGroup\n  public readonly taskDefinition: ecs.TaskDefinition\n  public readonly targetGroup: elb.ApplicationTargetGroup | undefined\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: FargateServiceProps,\n  ) {\n    super(scope, id)\n\n    /**\n     * Set this flag to disable this stack creating a completely new service and attempting replace when enabling circuit breakers\n     * Mitigating the deployment error: 'a service with the name <...> already exists'\n     * See: github.com/aws/aws-cdk/pull/22467\n     */\n    this.node.setContext(\n      \"@aws-cdk/aws-ecs:disableExplicitDeploymentControllerForCircuitBreaker\",\n      true,\n    )\n\n    const parameters = new ConfigureParameters(this, {\n      ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${\n        this.node.addr\n      }/parameters`,\n      parameters: props.parameters ?? [],\n    })\n\n    this.logGroup = new logs.LogGroup(this, \"LogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.TWO_WEEKS,\n    })\n\n    this.securityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: props.vpc,\n    })\n\n    this.taskDefinition = new ecs.FargateTaskDefinition(\n      this,\n      \"TaskDefinition\",\n      {\n        cpu: props.cpu ?? 256,\n        memoryLimitMiB: props.memoryLimitMiB ?? 512,\n        runtimePlatform: props.runtimePlatform,\n      },\n    )\n\n    parameters.grantRead(this.taskDefinition.taskRole)\n\n    const port = props.containerPort ?? 8080\n    const container = this.taskDefinition.addContainer(\"Container\", {\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: this.logGroup,\n        streamPrefix: \"ecs\",\n        datetimeFormat: \"%Y-%m-%dT%H:%M:%S\",\n        mode: props.logDriverMode ?? ecs.AwsLogDriverMode.BLOCKING,\n      }),\n      image: props.ecsImage,\n      secrets: props.secrets,\n      healthCheck: props.containerHealthCheck,\n      portMappings: props.portMappings ?? [\n        {\n          containerPort: port,\n          hostPort: port,\n        },\n      ],\n      environment: {\n        SSM_PREFIX: parameters.ssmPrefix,\n        // Not read by the application, only used to help with redeployments.\n        PARAMS_HASH: parameters.hashValue,\n        ...(props.environment ?? {}),\n      },\n      linuxParameters: new ecs.LinuxParameters(this, \"LinuxParameters\", {\n        initProcessEnabled: true,\n      }),\n      ...props.overrideContainerProps,\n    })\n\n    const enableCircuitBreaker = props.enableCircuitBreaker ?? false\n\n    this.fargateService = new ecs.FargateService(this, \"Service\", {\n      serviceName: props.serviceName,\n      vpcSubnets: {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      taskDefinition: this.taskDefinition,\n      cluster: props.cluster,\n      minHealthyPercent: 100,\n      healthCheckGracePeriod: props.healthCheckGracePeriod,\n      desiredCount: props.desiredCount,\n      assignPublicIp: true,\n      securityGroups: [this.securityGroup],\n      platformVersion: ecs.FargatePlatformVersion.VERSION1_4,\n      enableExecuteCommand: true,\n      circuitBreaker: enableCircuitBreaker\n        ? {\n            enable: true,\n            rollback: true,\n          }\n        : undefined,\n      ...props.overrideFargateServiceProps,\n    })\n\n    if (props.skipHealthCheckGracePeriod) {\n      ;(\n        this.fargateService.node.defaultChild as CfnService\n      ).healthCheckGracePeriodSeconds = undefined\n    }\n\n    for (const param of parameters.parameters) {\n      this.fargateService.node.addDependency(param)\n    }\n\n    if (!props.skipTargetGroup) {\n      this.targetGroup = new elb.ApplicationTargetGroup(this, \"TargetGroup\", {\n        protocol: elb.ApplicationProtocol.HTTP,\n        port: port,\n        vpc: props.vpc,\n        targetType: elb.TargetType.IP,\n        targets: [this.fargateService],\n        deregistrationDelay:\n          props.deregistrationDelay ?? cdk.Duration.seconds(15),\n        ...props.overrideTargetGroupProps,\n      })\n\n      this.targetGroup.configureHealthCheck({\n        interval: Duration.seconds(10),\n        path: \"/health\",\n        healthyThresholdCount: 2,\n        ...props.overrideHealthCheck,\n      })\n    }\n  }\n}\n"]}
160
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-service.js","sourceRoot":"","sources":["../../src/ecs/fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport { Duration } from \"aws-cdk-lib\"\nimport type * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport type { CfnService } from \"aws-cdk-lib/aws-ecs\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport type * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\nimport { ServiceAlarms } from \"../alarms\"\nimport type { Parameter } from \"../configure-parameters\"\nimport { ConfigureParameters } from \"../configure-parameters\"\n\n/**\n * Configure service alarms.\n *\n * Alarms are enabled by default when a config object with `alarmAction` and\n * `warningAction` is provided. To explicitly disable automatic alarms use\n * `{ enabled: false }`.\n */\nexport type ServiceAlarmsConfig =\n  | { enabled: false }\n  | {\n      alarmAction: cloudwatch.IAlarmAction\n      warningAction: cloudwatch.IAlarmAction\n      loadBalancerFullName: string\n      /** Optional Lambda function that will receive\n       * forwarded log events */\n      logHandler?: lambda.IFunction\n      /**\n       * Individual alarm configuration overrides defaults.\n       */\n      jsonErrorAlarm?: {\n        /**\n         * @default true\n         */\n        enabled?: boolean\n        alarmDescription?: string\n        enableOkAlarm?: boolean\n        action?: cloudwatch.IAlarmAction\n      }\n      uncaughtJavaExceptionAlarm?: {\n        /**\n         * @default false\n         */\n        enabled?: boolean\n        alarmDescription?: string\n        enableOkAlarm?: boolean\n        action?: cloudwatch.IAlarmAction\n      }\n      targetHealthAlarm?: {\n        /**\n         * @default true\n         */\n        enabled?: boolean\n        action?: cloudwatch.IAlarmAction\n        period?: cdk.Duration\n        evaluationPeriods?: number\n        threshold?: number\n        description?: string\n      }\n      tooMany5xxResponsesFromTargetsAlarm?: {\n        /**\n         * @default true\n         */\n        enabled?: boolean\n        action?: cloudwatch.IAlarmAction\n        period?: cdk.Duration\n        evaluationPeriods?: number\n        threshold?: number\n        description?: string\n      }\n      targetResponseTimeAlarm?: {\n        /**\n         * @default true\n         */\n        enabled?: boolean\n        action?: cloudwatch.IAlarmAction\n        period?: cdk.Duration\n        evaluationPeriods?: number\n        threshold?: cdk.Duration\n        description?: string\n      }\n      single5xxResponseAlarm?: {\n        /**\n         * @default true\n         */\n        enabled?: boolean\n        /**\n         * An action to use for CloudWatch alarm state changes instead of the default action\n         */\n        action?: cloudwatch.IAlarmAction\n        /**\n         * @default 60 seconds\n         */\n        period?: cdk.Duration\n        /**\n         * @default 1\n         */\n        evaluationPeriods?: number\n        /**\n         * @default 1\n         */\n        threshold?: number\n        description?: string\n      }\n    }\n\nexport interface FargateServiceProps {\n  serviceName: string\n  vpc: ec2.IVpc\n  cluster: ecs.ICluster\n  desiredCount: number\n  ecsImage: ecs.ContainerImage\n  portMappings?: ecs.PortMapping[]\n  containerHealthCheck?: ecs.HealthCheck\n  /**\n   * @default 256\n   */\n  cpu?: number\n  /**\n   * @default undefined\n   */\n  runtimePlatform?: ecs.RuntimePlatform\n  /**\n   * @default 512\n   */\n  memoryLimitMiB?: number\n  /**\n   * @default 2 weeks\n   */\n  logsRetention?: logs.RetentionDays\n  /**\n   * @default AwsLogDriverMode.BLOCKING\n   */\n  logDriverMode?: ecs.AwsLogDriverMode\n  /**\n   * @default 15 seconds\n   */\n  deregistrationDelay?: cdk.Duration\n  /**\n   * @default 8080\n   */\n  containerPort?: number\n  /**\n   * @default 60 seconds\n   */\n  healthCheckGracePeriod?: cdk.Duration\n  /**\n   * Use this as workaround when adding the service to a load balancer after\n   * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'\n   * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842\n   */\n  skipHealthCheckGracePeriod?: boolean\n  parameters?: Parameter[]\n  overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>\n  overrideHealthCheck?: Partial<elb.HealthCheck>\n  overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>\n  overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>\n  secrets?: Record<string, ecs.Secret>\n  environment?: Record<string, string>\n  /**\n   * @default false\n   */\n  skipTargetGroup?: boolean\n  /**\n   * @default false\n   */\n  enableCircuitBreaker?: boolean\n  /**\n   *  Either\n   *  - `{ enabled: false }` to disable all alarms.\n   *  - object with required `alarmAction`, `warningAction`, and `loadBalancerFullName`.\n   *\n   *  When enabled, the construct will:\n   *  - (By default) add a log-based JSON error alarm (enabled by default).\n   *  - (By default) add target-group related alarms (target health, 5xx responses, response time)\n   *  when a target group is present.\n   *  - (Opt-in) optionally enable the uncaught Java exception alarm\n   */\n  alarms: ServiceAlarmsConfig\n}\n\nexport class FargateService extends constructs.Construct {\n  public readonly fargateService: ecs.FargateService\n  public readonly securityGroup: ec2.SecurityGroup\n  public readonly taskDefinition: ecs.TaskDefinition\n  public readonly targetGroup: elb.ApplicationTargetGroup | undefined\n  public readonly logGroup: logs.LogGroup\n  public readonly serviceAlarms: ServiceAlarms | undefined\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: FargateServiceProps,\n  ) {\n    super(scope, id)\n\n    /**\n     * Set this flag to disable this stack creating a completely new service and attempting replace when enabling circuit breakers\n     * Mitigating the deployment error: 'a service with the name <...> already exists'\n     * See: github.com/aws/aws-cdk/pull/22467\n     */\n    this.node.setContext(\n      \"@aws-cdk/aws-ecs:disableExplicitDeploymentControllerForCircuitBreaker\",\n      true,\n    )\n\n    const parameters = new ConfigureParameters(this, {\n      ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${\n        this.node.addr\n      }/parameters`,\n      parameters: props.parameters ?? [],\n    })\n\n    this.logGroup = new logs.LogGroup(this, \"LogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.TWO_WEEKS,\n    })\n\n    this.securityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: props.vpc,\n    })\n\n    this.taskDefinition = new ecs.FargateTaskDefinition(\n      this,\n      \"TaskDefinition\",\n      {\n        cpu: props.cpu ?? 256,\n        memoryLimitMiB: props.memoryLimitMiB ?? 512,\n        runtimePlatform: props.runtimePlatform,\n      },\n    )\n\n    parameters.grantRead(this.taskDefinition.taskRole)\n\n    const port = props.containerPort ?? 8080\n    this.taskDefinition.addContainer(\"Container\", {\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: this.logGroup,\n        streamPrefix: \"ecs\",\n        datetimeFormat: \"%Y-%m-%dT%H:%M:%S\",\n        mode: props.logDriverMode ?? ecs.AwsLogDriverMode.BLOCKING,\n      }),\n      image: props.ecsImage,\n      secrets: props.secrets,\n      healthCheck: props.containerHealthCheck,\n      portMappings: props.portMappings ?? [\n        {\n          containerPort: port,\n          hostPort: port,\n        },\n      ],\n      environment: {\n        SSM_PREFIX: parameters.ssmPrefix,\n        // Not read by the application, only used to help with redeployments.\n        PARAMS_HASH: parameters.hashValue,\n        ...(props.environment ?? {}),\n      },\n      linuxParameters: new ecs.LinuxParameters(this, \"LinuxParameters\", {\n        initProcessEnabled: true,\n      }),\n      ...props.overrideContainerProps,\n    })\n\n    const enableCircuitBreaker = props.enableCircuitBreaker ?? false\n\n    this.fargateService = new ecs.FargateService(this, \"Service\", {\n      serviceName: props.serviceName,\n      vpcSubnets: {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      taskDefinition: this.taskDefinition,\n      cluster: props.cluster,\n      minHealthyPercent: 100,\n      healthCheckGracePeriod: props.healthCheckGracePeriod,\n      desiredCount: props.desiredCount,\n      assignPublicIp: true,\n      securityGroups: [this.securityGroup],\n      platformVersion: ecs.FargatePlatformVersion.VERSION1_4,\n      enableExecuteCommand: true,\n      circuitBreaker: enableCircuitBreaker\n        ? {\n            enable: true,\n            rollback: true,\n          }\n        : undefined,\n      ...props.overrideFargateServiceProps,\n    })\n\n    if (props.skipHealthCheckGracePeriod) {\n      ;(\n        this.fargateService.node.defaultChild as CfnService\n      ).healthCheckGracePeriodSeconds = undefined\n    }\n\n    for (const param of parameters.parameters) {\n      this.fargateService.node.addDependency(param)\n    }\n\n    if (!props.skipTargetGroup) {\n      this.targetGroup = new elb.ApplicationTargetGroup(this, \"TargetGroup\", {\n        protocol: elb.ApplicationProtocol.HTTP,\n        port: port,\n        vpc: props.vpc,\n        targetType: elb.TargetType.IP,\n        targets: [this.fargateService],\n        deregistrationDelay:\n          props.deregistrationDelay ?? cdk.Duration.seconds(15),\n        ...props.overrideTargetGroupProps,\n      })\n\n      this.targetGroup.configureHealthCheck({\n        interval: Duration.seconds(10),\n        path: \"/health\",\n        healthyThresholdCount: 2,\n        ...props.overrideHealthCheck,\n      })\n    }\n\n    // Require explicit alarmAction and warningAction when enabled.\n    if (props.alarms && \"alarmAction\" in props.alarms) {\n      const alarms = props.alarms\n\n      this.serviceAlarms = new ServiceAlarms(this, \"Alarms\", {\n        serviceName: props.serviceName,\n        alarmAction: alarms.alarmAction,\n        warningAction: alarms.warningAction,\n        logHandler: alarms.logHandler,\n      })\n\n      // Enabled by default (opt-out).\n      const jsonErrorOverrides = alarms.jsonErrorAlarm\n      if (jsonErrorOverrides?.enabled ?? true) {\n        this.serviceAlarms.addJsonErrorAlarm({\n          logGroup: this.logGroup,\n          alarmDescription: jsonErrorOverrides?.alarmDescription,\n          enableOkAlarm: jsonErrorOverrides?.enableOkAlarm,\n          action: jsonErrorOverrides?.action,\n        })\n      }\n\n      // Not enabled by default (opt-in).\n      const javaExceptionOverrides = alarms.uncaughtJavaExceptionAlarm\n      if (javaExceptionOverrides?.enabled) {\n        this.serviceAlarms.addUncaughtJavaExceptionAlarm({\n          logGroup: this.logGroup,\n          alarmDescription: javaExceptionOverrides.alarmDescription,\n          enableOkAlarm: javaExceptionOverrides.enableOkAlarm,\n          action: javaExceptionOverrides.action,\n          enabled: true,\n        })\n      }\n\n      if (!props.skipTargetGroup) {\n        // All alarms enabled by-default (opt-out individually)\n        this.serviceAlarms.addTargetGroupAlarms({\n          targetGroupFullName: this.targetGroup!.targetGroupFullName,\n          loadBalancerFullName: alarms.loadBalancerFullName,\n          targetHealthAlarm: alarms.targetHealthAlarm,\n          single5xxResponseAlarm: alarms.single5xxResponseAlarm,\n          tooMany5xxResponsesFromTargetsAlarm:\n            alarms.tooMany5xxResponsesFromTargetsAlarm,\n          targetResponseTimeAlarm: alarms.targetResponseTimeAlarm,\n        })\n      }\n    }\n  }\n}\n"]}
@@ -38,7 +38,7 @@ export class LoadBalancer extends constructs.Construct {
38
38
  targetType: elb.TargetType.INSTANCE,
39
39
  });
40
40
  this.httpsListener = this.loadBalancer.addListener("HttpsListener", {
41
- sslPolicy: elb.SslPolicy.TLS12,
41
+ sslPolicy: elb.SslPolicy.RECOMMENDED_TLS,
42
42
  protocol: elb.ApplicationProtocol.HTTPS,
43
43
  port: 443,
44
44
  certificates: props.certificates,
@@ -57,4 +57,4 @@ export class LoadBalancer extends constructs.Construct {
57
57
  this.loadBalancer.logAccessLogs(this.accessLogsBucket);
58
58
  }
59
59
  }
60
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC1iYWxhbmNlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sb2FkLWJhbGFuY2VyL2xvYWQtYmFsYW5jZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLEdBQUcsTUFBTSxhQUFhLENBQUE7QUFFbEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEtBQUssR0FBRyxNQUFNLHdDQUF3QyxDQUFBO0FBQzdELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQTtBQUN2RSxPQUFPLEtBQUssRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBQ3hDLE9BQU8sS0FBSyxVQUFVLE1BQU0sWUFBWSxDQUFBO0FBU3hDLE1BQU0sT0FBTyxZQUFhLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDcEMsWUFBWSxDQUE2QjtJQUN6QyxhQUFhLENBQXlCO0lBQ3RDLGdCQUFnQixDQUFXO0lBRTNDLFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQXdCO1FBRXhCLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLEdBQUcsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUFFO1lBQ3hFLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLFVBQVUsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQztnQkFDbEMsVUFBVSxFQUFFLEdBQUcsQ0FBQyxVQUFVLENBQUMsTUFBTTthQUNsQyxDQUFDO1lBQ0YsR0FBRyxLQUFLLENBQUMseUJBQXlCO1NBQ25DLENBQUMsQ0FBQTtRQUVGLElBQUksQ0FBQyxZQUFZO2FBQ2QsV0FBVyxDQUFDLGNBQWMsRUFBRTtZQUMzQixJQUFJLEVBQUUsRUFBRTtTQUNULENBQUM7YUFDRCxTQUFTLENBQUMsZUFBZSxFQUFFO1lBQzFCLE1BQU0sRUFBRSxjQUFjLENBQUMsUUFBUSxDQUFDO2dCQUM5QixJQUFJLEVBQUUsS0FBSztnQkFDWCxRQUFRLEVBQUUsT0FBTztnQkFDakIsU0FBUyxFQUFFLElBQUk7YUFDaEIsQ0FBQztTQUNILENBQUMsQ0FBQTtRQUVKLG9EQUFvRDtRQUNwRCw0REFBNEQ7UUFDNUQsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLEdBQUcsQ0FBQyxzQkFBc0IsQ0FDdkQsSUFBSSxFQUNKLG9CQUFvQixFQUNwQjtZQUNFLFFBQVEsRUFBRSxHQUFHLENBQUMsbUJBQW1CLENBQUMsSUFBSTtZQUN0QyxJQUFJLEVBQUUsRUFBRTtZQUNSLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLFVBQVUsRUFBRSxHQUFHLENBQUMsVUFBVSxDQUFDLFFBQVE7U0FDcEMsQ0FDRixDQUFBO1FBRUQsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxlQUFlLEVBQUU7WUFDbEUsU0FBUyxFQUFFLEdBQUcsQ0FBQyxTQUFTLENBQUMsS0FBSztZQUM5QixRQUFRLEVBQUUsR0FBRyxDQUFDLG1CQUFtQixDQUFDLEtBQUs7WUFDdkMsSUFBSSxFQUFFLEdBQUc7WUFDVCxZQUFZLEVBQUUsS0FBSyxDQUFDLFlBQVk7WUFDaEMsbUJBQW1CLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztZQUN6QyxHQUFHLEtBQUssQ0FBQywwQkFBMEI7U0FDcEMsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsa0JBQWtCLEVBQUU7WUFDOUQsVUFBVSxFQUFFLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVO1lBQzFDLGlCQUFpQixFQUFFLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQyxTQUFTO1lBQ2pELGNBQWMsRUFBRTtnQkFDZDtvQkFDRSxVQUFVLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2lCQUNsQzthQUNGO1NBQ0YsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLFlBQVksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLENBQUE7SUFDeEQsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgdHlwZSAqIGFzIGNlcnRpZmljYXRlbWFuYWdlciBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNlcnRpZmljYXRlbWFuYWdlclwiXG5pbXBvcnQgKiBhcyBlYzIgZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzJcIlxuaW1wb3J0ICogYXMgZWxiIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgeyBMaXN0ZW5lckFjdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgKiBhcyBzMyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXMzXCJcbmltcG9ydCAqIGFzIGNvbnN0cnVjdHMgZnJvbSBcImNvbnN0cnVjdHNcIlxuXG5leHBvcnQgaW50ZXJmYWNlIExvYWRCYWxhbmNlclByb3BzIHtcbiAgY2VydGlmaWNhdGVzOiBjZXJ0aWZpY2F0ZW1hbmFnZXIuSUNlcnRpZmljYXRlW11cbiAgdnBjOiBlYzIuSVZwY1xuICBvdmVycmlkZUxvYWRCYWxhbmNlclByb3BzPzogUGFydGlhbDxlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJQcm9wcz5cbiAgb3ZlcnJpZGVIdHRwc0xpc3RlbmVyUHJvcHM/OiBQYXJ0aWFsPGVsYi5CYXNlQXBwbGljYXRpb25MaXN0ZW5lclByb3BzPlxufVxuXG5leHBvcnQgY2xhc3MgTG9hZEJhbGFuY2VyIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgbG9hZEJhbGFuY2VyOiBlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJcbiAgcHVibGljIHJlYWRvbmx5IGh0dHBzTGlzdGVuZXI6IGVsYi5BcHBsaWNhdGlvbkxpc3RlbmVyXG4gIHB1YmxpYyByZWFkb25seSBhY2Nlc3NMb2dzQnVja2V0OiBzMy5CdWNrZXRcblxuICBjb25zdHJ1Y3RvcihcbiAgICBzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogTG9hZEJhbGFuY2VyUHJvcHMsXG4gICkge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyID0gbmV3IGVsYi5BcHBsaWNhdGlvbkxvYWRCYWxhbmNlcih0aGlzLCBcIkxvYWRCYWxhbmNlclwiLCB7XG4gICAgICB2cGM6IHByb3BzLnZwYyxcbiAgICAgIGludGVybmV0RmFjaW5nOiB0cnVlLFxuICAgICAgdnBjU3VibmV0czogcHJvcHMudnBjLnNlbGVjdFN1Ym5ldHMoe1xuICAgICAgICBzdWJuZXRUeXBlOiBlYzIuU3VibmV0VHlwZS5QVUJMSUMsXG4gICAgICB9KSxcbiAgICAgIC4uLnByb3BzLm92ZXJyaWRlTG9hZEJhbGFuY2VyUHJvcHMsXG4gICAgfSlcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyXG4gICAgICAuYWRkTGlzdGVuZXIoXCJIdHRwTGlzdGVuZXJcIiwge1xuICAgICAgICBwb3J0OiA4MCxcbiAgICAgIH0pXG4gICAgICAuYWRkQWN0aW9uKFwiSHR0cHNSZWRpcmVjdFwiLCB7XG4gICAgICAgIGFjdGlvbjogTGlzdGVuZXJBY3Rpb24ucmVkaXJlY3Qoe1xuICAgICAgICAgIHBvcnQ6IFwiNDQzXCIsXG4gICAgICAgICAgcHJvdG9jb2w6IFwiSFRUUFNcIixcbiAgICAgICAgICBwZXJtYW5lbnQ6IHRydWUsXG4gICAgICAgIH0pLFxuICAgICAgfSlcblxuICAgIC8vIFRoZSBMb2FkIEJhbGFuY2VyIHJlcXVpcmUgYSBkZWZhdWx0IHRhcmdldCBncm91cC5cbiAgICAvLyBXZSB3aWxsIG5vdCBjb25uZWN0IGFueXRoaW5nIHRvIHRoZSBkZWZhdWx0IHRhcmdldCBncm91cC5cbiAgICBjb25zdCBkZWZhdWx0VGFyZ2V0R3JvdXAgPSBuZXcgZWxiLkFwcGxpY2F0aW9uVGFyZ2V0R3JvdXAoXG4gICAgICB0aGlzLFxuICAgICAgXCJEZWZhdWx0VGFyZ2V0R3JvdXBcIixcbiAgICAgIHtcbiAgICAgICAgcHJvdG9jb2w6IGVsYi5BcHBsaWNhdGlvblByb3RvY29sLkhUVFAsXG4gICAgICAgIHBvcnQ6IDgwLFxuICAgICAgICB2cGM6IHByb3BzLnZwYyxcbiAgICAgICAgdGFyZ2V0VHlwZTogZWxiLlRhcmdldFR5cGUuSU5TVEFOQ0UsXG4gICAgICB9LFxuICAgIClcblxuICAgIHRoaXMuaHR0cHNMaXN0ZW5lciA9IHRoaXMubG9hZEJhbGFuY2VyLmFkZExpc3RlbmVyKFwiSHR0cHNMaXN0ZW5lclwiLCB7XG4gICAgICBzc2xQb2xpY3k6IGVsYi5Tc2xQb2xpY3kuVExTMTIsXG4gICAgICBwcm90b2NvbDogZWxiLkFwcGxpY2F0aW9uUHJvdG9jb2wuSFRUUFMsXG4gICAgICBwb3J0OiA0NDMsXG4gICAgICBjZXJ0aWZpY2F0ZXM6IHByb3BzLmNlcnRpZmljYXRlcyxcbiAgICAgIGRlZmF1bHRUYXJnZXRHcm91cHM6IFtkZWZhdWx0VGFyZ2V0R3JvdXBdLFxuICAgICAgLi4ucHJvcHMub3ZlcnJpZGVIdHRwc0xpc3RlbmVyUHJvcHMsXG4gICAgfSlcblxuICAgIHRoaXMuYWNjZXNzTG9nc0J1Y2tldCA9IG5ldyBzMy5CdWNrZXQodGhpcywgXCJBY2Nlc3NMb2dzQnVja2V0XCIsIHtcbiAgICAgIGVuY3J5cHRpb246IHMzLkJ1Y2tldEVuY3J5cHRpb24uUzNfTUFOQUdFRCxcbiAgICAgIGJsb2NrUHVibGljQWNjZXNzOiBzMy5CbG9ja1B1YmxpY0FjY2Vzcy5CTE9DS19BTEwsXG4gICAgICBsaWZlY3ljbGVSdWxlczogW1xuICAgICAgICB7XG4gICAgICAgICAgZXhwaXJhdGlvbjogY2RrLkR1cmF0aW9uLmRheXMoMzApLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9KVxuXG4gICAgdGhpcy5sb2FkQmFsYW5jZXIubG9nQWNjZXNzTG9ncyh0aGlzLmFjY2Vzc0xvZ3NCdWNrZXQpXG4gIH1cbn1cbiJdfQ==
60
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC1iYWxhbmNlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sb2FkLWJhbGFuY2VyL2xvYWQtYmFsYW5jZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLEdBQUcsTUFBTSxhQUFhLENBQUE7QUFFbEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEtBQUssR0FBRyxNQUFNLHdDQUF3QyxDQUFBO0FBQzdELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQTtBQUN2RSxPQUFPLEtBQUssRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBQ3hDLE9BQU8sS0FBSyxVQUFVLE1BQU0sWUFBWSxDQUFBO0FBU3hDLE1BQU0sT0FBTyxZQUFhLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDcEMsWUFBWSxDQUE2QjtJQUN6QyxhQUFhLENBQXlCO0lBQ3RDLGdCQUFnQixDQUFXO0lBRTNDLFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQXdCO1FBRXhCLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLEdBQUcsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUFFO1lBQ3hFLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLFVBQVUsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQztnQkFDbEMsVUFBVSxFQUFFLEdBQUcsQ0FBQyxVQUFVLENBQUMsTUFBTTthQUNsQyxDQUFDO1lBQ0YsR0FBRyxLQUFLLENBQUMseUJBQXlCO1NBQ25DLENBQUMsQ0FBQTtRQUVGLElBQUksQ0FBQyxZQUFZO2FBQ2QsV0FBVyxDQUFDLGNBQWMsRUFBRTtZQUMzQixJQUFJLEVBQUUsRUFBRTtTQUNULENBQUM7YUFDRCxTQUFTLENBQUMsZUFBZSxFQUFFO1lBQzFCLE1BQU0sRUFBRSxjQUFjLENBQUMsUUFBUSxDQUFDO2dCQUM5QixJQUFJLEVBQUUsS0FBSztnQkFDWCxRQUFRLEVBQUUsT0FBTztnQkFDakIsU0FBUyxFQUFFLElBQUk7YUFDaEIsQ0FBQztTQUNILENBQUMsQ0FBQTtRQUVKLG9EQUFvRDtRQUNwRCw0REFBNEQ7UUFDNUQsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLEdBQUcsQ0FBQyxzQkFBc0IsQ0FDdkQsSUFBSSxFQUNKLG9CQUFvQixFQUNwQjtZQUNFLFFBQVEsRUFBRSxHQUFHLENBQUMsbUJBQW1CLENBQUMsSUFBSTtZQUN0QyxJQUFJLEVBQUUsRUFBRTtZQUNSLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLFVBQVUsRUFBRSxHQUFHLENBQUMsVUFBVSxDQUFDLFFBQVE7U0FDcEMsQ0FDRixDQUFBO1FBRUQsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxlQUFlLEVBQUU7WUFDbEUsU0FBUyxFQUFFLEdBQUcsQ0FBQyxTQUFTLENBQUMsZUFBZTtZQUN4QyxRQUFRLEVBQUUsR0FBRyxDQUFDLG1CQUFtQixDQUFDLEtBQUs7WUFDdkMsSUFBSSxFQUFFLEdBQUc7WUFDVCxZQUFZLEVBQUUsS0FBSyxDQUFDLFlBQVk7WUFDaEMsbUJBQW1CLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztZQUN6QyxHQUFHLEtBQUssQ0FBQywwQkFBMEI7U0FDcEMsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsa0JBQWtCLEVBQUU7WUFDOUQsVUFBVSxFQUFFLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVO1lBQzFDLGlCQUFpQixFQUFFLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQyxTQUFTO1lBQ2pELGNBQWMsRUFBRTtnQkFDZDtvQkFDRSxVQUFVLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2lCQUNsQzthQUNGO1NBQ0YsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLFlBQVksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLENBQUE7SUFDeEQsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgdHlwZSAqIGFzIGNlcnRpZmljYXRlbWFuYWdlciBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNlcnRpZmljYXRlbWFuYWdlclwiXG5pbXBvcnQgKiBhcyBlYzIgZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzJcIlxuaW1wb3J0ICogYXMgZWxiIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgeyBMaXN0ZW5lckFjdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgKiBhcyBzMyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXMzXCJcbmltcG9ydCAqIGFzIGNvbnN0cnVjdHMgZnJvbSBcImNvbnN0cnVjdHNcIlxuXG5leHBvcnQgaW50ZXJmYWNlIExvYWRCYWxhbmNlclByb3BzIHtcbiAgY2VydGlmaWNhdGVzOiBjZXJ0aWZpY2F0ZW1hbmFnZXIuSUNlcnRpZmljYXRlW11cbiAgdnBjOiBlYzIuSVZwY1xuICBvdmVycmlkZUxvYWRCYWxhbmNlclByb3BzPzogUGFydGlhbDxlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJQcm9wcz5cbiAgb3ZlcnJpZGVIdHRwc0xpc3RlbmVyUHJvcHM/OiBQYXJ0aWFsPGVsYi5CYXNlQXBwbGljYXRpb25MaXN0ZW5lclByb3BzPlxufVxuXG5leHBvcnQgY2xhc3MgTG9hZEJhbGFuY2VyIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgbG9hZEJhbGFuY2VyOiBlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJcbiAgcHVibGljIHJlYWRvbmx5IGh0dHBzTGlzdGVuZXI6IGVsYi5BcHBsaWNhdGlvbkxpc3RlbmVyXG4gIHB1YmxpYyByZWFkb25seSBhY2Nlc3NMb2dzQnVja2V0OiBzMy5CdWNrZXRcblxuICBjb25zdHJ1Y3RvcihcbiAgICBzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogTG9hZEJhbGFuY2VyUHJvcHMsXG4gICkge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyID0gbmV3IGVsYi5BcHBsaWNhdGlvbkxvYWRCYWxhbmNlcih0aGlzLCBcIkxvYWRCYWxhbmNlclwiLCB7XG4gICAgICB2cGM6IHByb3BzLnZwYyxcbiAgICAgIGludGVybmV0RmFjaW5nOiB0cnVlLFxuICAgICAgdnBjU3VibmV0czogcHJvcHMudnBjLnNlbGVjdFN1Ym5ldHMoe1xuICAgICAgICBzdWJuZXRUeXBlOiBlYzIuU3VibmV0VHlwZS5QVUJMSUMsXG4gICAgICB9KSxcbiAgICAgIC4uLnByb3BzLm92ZXJyaWRlTG9hZEJhbGFuY2VyUHJvcHMsXG4gICAgfSlcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyXG4gICAgICAuYWRkTGlzdGVuZXIoXCJIdHRwTGlzdGVuZXJcIiwge1xuICAgICAgICBwb3J0OiA4MCxcbiAgICAgIH0pXG4gICAgICAuYWRkQWN0aW9uKFwiSHR0cHNSZWRpcmVjdFwiLCB7XG4gICAgICAgIGFjdGlvbjogTGlzdGVuZXJBY3Rpb24ucmVkaXJlY3Qoe1xuICAgICAgICAgIHBvcnQ6IFwiNDQzXCIsXG4gICAgICAgICAgcHJvdG9jb2w6IFwiSFRUUFNcIixcbiAgICAgICAgICBwZXJtYW5lbnQ6IHRydWUsXG4gICAgICAgIH0pLFxuICAgICAgfSlcblxuICAgIC8vIFRoZSBMb2FkIEJhbGFuY2VyIHJlcXVpcmUgYSBkZWZhdWx0IHRhcmdldCBncm91cC5cbiAgICAvLyBXZSB3aWxsIG5vdCBjb25uZWN0IGFueXRoaW5nIHRvIHRoZSBkZWZhdWx0IHRhcmdldCBncm91cC5cbiAgICBjb25zdCBkZWZhdWx0VGFyZ2V0R3JvdXAgPSBuZXcgZWxiLkFwcGxpY2F0aW9uVGFyZ2V0R3JvdXAoXG4gICAgICB0aGlzLFxuICAgICAgXCJEZWZhdWx0VGFyZ2V0R3JvdXBcIixcbiAgICAgIHtcbiAgICAgICAgcHJvdG9jb2w6IGVsYi5BcHBsaWNhdGlvblByb3RvY29sLkhUVFAsXG4gICAgICAgIHBvcnQ6IDgwLFxuICAgICAgICB2cGM6IHByb3BzLnZwYyxcbiAgICAgICAgdGFyZ2V0VHlwZTogZWxiLlRhcmdldFR5cGUuSU5TVEFOQ0UsXG4gICAgICB9LFxuICAgIClcblxuICAgIHRoaXMuaHR0cHNMaXN0ZW5lciA9IHRoaXMubG9hZEJhbGFuY2VyLmFkZExpc3RlbmVyKFwiSHR0cHNMaXN0ZW5lclwiLCB7XG4gICAgICBzc2xQb2xpY3k6IGVsYi5Tc2xQb2xpY3kuUkVDT01NRU5ERURfVExTLFxuICAgICAgcHJvdG9jb2w6IGVsYi5BcHBsaWNhdGlvblByb3RvY29sLkhUVFBTLFxuICAgICAgcG9ydDogNDQzLFxuICAgICAgY2VydGlmaWNhdGVzOiBwcm9wcy5jZXJ0aWZpY2F0ZXMsXG4gICAgICBkZWZhdWx0VGFyZ2V0R3JvdXBzOiBbZGVmYXVsdFRhcmdldEdyb3VwXSxcbiAgICAgIC4uLnByb3BzLm92ZXJyaWRlSHR0cHNMaXN0ZW5lclByb3BzLFxuICAgIH0pXG5cbiAgICB0aGlzLmFjY2Vzc0xvZ3NCdWNrZXQgPSBuZXcgczMuQnVja2V0KHRoaXMsIFwiQWNjZXNzTG9nc0J1Y2tldFwiLCB7XG4gICAgICBlbmNyeXB0aW9uOiBzMy5CdWNrZXRFbmNyeXB0aW9uLlMzX01BTkFHRUQsXG4gICAgICBibG9ja1B1YmxpY0FjY2VzczogczMuQmxvY2tQdWJsaWNBY2Nlc3MuQkxPQ0tfQUxMLFxuICAgICAgbGlmZWN5Y2xlUnVsZXM6IFtcbiAgICAgICAge1xuICAgICAgICAgIGV4cGlyYXRpb246IGNkay5EdXJhdGlvbi5kYXlzKDMwKSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSlcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyLmxvZ0FjY2Vzc0xvZ3ModGhpcy5hY2Nlc3NMb2dzQnVja2V0KVxuICB9XG59XG4iXX0=
@@ -1,8 +1,81 @@
1
1
  import * as cdk from "aws-cdk-lib";
2
+ import type * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
2
3
  import * as ec2 from "aws-cdk-lib/aws-ec2";
3
4
  import * as rds from "aws-cdk-lib/aws-rds";
4
5
  import type * as sm from "aws-cdk-lib/aws-secretsmanager";
5
6
  import * as constructs from "constructs";
7
+ /**
8
+ * Configure database alarms.
9
+ *
10
+ * Alarms are enabled by default (when you supply an `alarmAction` and
11
+ * `warningAction`). To explicitly disable automatic alarms use
12
+ * `{ enabled: false }`.
13
+ */
14
+ export type DatabaseAlarmsConfig = {
15
+ enabled: false;
16
+ } | {
17
+ /**
18
+ * When alarms are enabled, both actions are required.
19
+ */
20
+ alarmAction: cloudwatch.IAlarmAction;
21
+ warningAction: cloudwatch.IAlarmAction;
22
+ /**
23
+ * CPU credits alarm config
24
+ */
25
+ cpuCreditsAlarm?: {
26
+ /**
27
+ * @default true if instance type is burstable
28
+ */
29
+ enabled?: boolean;
30
+ /** Whether to attach OK actions for this alarm. @default true */
31
+ enableOkAlarm?: boolean;
32
+ action?: cloudwatch.IAlarmAction;
33
+ /** @default 10% of maximum earned credits for instance type */
34
+ threshold?: number;
35
+ appendToAlarmDescription?: string;
36
+ };
37
+ /**
38
+ * Storage space alarm overrides
39
+ */
40
+ storageSpaceAlarms?: {
41
+ /**
42
+ * Set to `false` to disable all storage space alarms (both low and critically low).
43
+ * @default true
44
+ */
45
+ enabled?: boolean;
46
+ lowStorageSpaceAlarm?: {
47
+ action?: cloudwatch.IAlarmAction;
48
+ /** Whether to attach OK actions for this alarm. @default true */
49
+ enableOkAlarm?: boolean;
50
+ /** @default 25% of allocated storage */
51
+ threshold?: cdk.Size;
52
+ };
53
+ criticallyLowStorageSpaceAlarm?: {
54
+ action?: cloudwatch.IAlarmAction;
55
+ /** Whether to attach OK actions for this alarm. @default true */
56
+ enableOkAlarm?: boolean;
57
+ /** @default 5% of allocated storage */
58
+ threshold?: cdk.Size;
59
+ };
60
+ appendToAlarmDescription?: string;
61
+ };
62
+ /**
63
+ * CPU utilization alarm overrides.
64
+ */
65
+ cpuUtilizationAlarm?: {
66
+ enabled?: boolean;
67
+ /** Whether to attach OK actions for this alarm. @default true */
68
+ enableOkAlarm?: boolean;
69
+ action?: cloudwatch.IAlarmAction;
70
+ /** @default 80 */
71
+ threshold?: number;
72
+ /** @default 5 */
73
+ evaluationPeriods?: number;
74
+ /** @default 2 minutes */
75
+ period?: cdk.Duration;
76
+ appendToAlarmDescription?: string;
77
+ };
78
+ };
6
79
  export interface DatabaseProps extends cdk.StackProps {
7
80
  vpc: ec2.IVpc;
8
81
  engine: rds.IInstanceEngine;
@@ -37,6 +110,16 @@ export interface DatabaseProps extends cdk.StackProps {
37
110
  */
38
111
  usePublicSubnets?: boolean;
39
112
  overrideDbOptions?: Partial<rds.DatabaseInstanceSourceProps>;
113
+ /**
114
+ * Configure database alarms.
115
+ *
116
+ * This property is required and must be one of two shapes:
117
+ * - `{ enabled: false }` to explicitly disable automatic alarms
118
+ * - `{ alarmAction, warningAction }` to enable alarms and provide both channels
119
+ *
120
+ * Default: enabled.
121
+ */
122
+ alarms: DatabaseAlarmsConfig;
40
123
  }
41
124
  export declare class Database extends constructs.Construct {
42
125
  readonly secret: sm.ISecret;