@liflig/cdk 2.16.0 → 2.17.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.
Files changed (44) hide show
  1. package/assets/pipeline-slack-notification-lambda/index.py +23 -7
  2. package/assets/slack-alarm-lambda/index.py +70 -36
  3. package/lib/alarms/database-alarms.js +1 -1
  4. package/lib/alarms/service-alarms.js +1 -1
  5. package/lib/alarms/ses-alarms.js +1 -1
  6. package/lib/alarms/slack-alarm.d.ts +9 -2
  7. package/lib/alarms/slack-alarm.js +4 -4
  8. package/lib/bin/cdk-create-snapshots.js +1 -1
  9. package/lib/bin/fetch-pipeline-variables.js +1 -1
  10. package/lib/build-artifacts/index.js +1 -1
  11. package/lib/cdk-deploy/start-deploy-handler.js +1 -1
  12. package/lib/cdk-deploy/status-handler.js +1 -1
  13. package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +1 -1
  14. package/lib/cdk-pipelines/liflig-cdk-pipeline.js +1 -1
  15. package/lib/cdk-pipelines/slack-notification.d.ts +7 -16
  16. package/lib/cdk-pipelines/slack-notification.js +6 -7
  17. package/lib/cdk-pipelines/variables.js +1 -1
  18. package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.js +1 -1
  19. package/lib/configure-parameters/configure-parameters.js +1 -1
  20. package/lib/ecs/cluster.js +1 -1
  21. package/lib/ecs/fargate-service.js +1 -1
  22. package/lib/ecs/listener-rule.js +1 -1
  23. package/lib/ecs-update-image/ecs-update-image.js +1 -1
  24. package/lib/ecs-update-image/start-deploy-handler.js +1 -1
  25. package/lib/ecs-update-image/status-handler.js +1 -1
  26. package/lib/ecs-update-image/tag.js +1 -1
  27. package/lib/feature-flags.js +1 -1
  28. package/lib/hosted-zone-with-param.js +1 -1
  29. package/lib/pipelines/deploy-env.js +1 -1
  30. package/lib/pipelines/pipeline.js +1 -1
  31. package/lib/platform/platform.js +1 -1
  32. package/lib/ses/configurationsetsnsdestination/handler.js +1 -1
  33. package/lib/ses/configurationsetsnsdestination/index.js +1 -1
  34. package/lib/ses/sesdomain/handler.js +1 -1
  35. package/lib/ses/sesdomain/index.js +1 -1
  36. package/lib/ses/sesverifyemail/handler.js +1 -1
  37. package/lib/snapshots.js +1 -1
  38. package/lib/ssm-parameter-backed-resource.js +1 -1
  39. package/lib/tags.js +1 -1
  40. package/lib/utils.js +1 -1
  41. package/lib/webapp/monitor.js +1 -1
  42. package/lib/webapp/security-headers.js +1 -1
  43. package/lib/webapp/webapp.js +1 -1
  44. package/package.json +13 -13
@@ -10,10 +10,10 @@ import boto3
10
10
 
11
11
  client = boto3.client("codepipeline")
12
12
  s3 = boto3.client("s3")
13
+ secrets_manager = boto3.client("secretsmanager")
13
14
 
14
15
  ACCOUNT_FRIENDLY_NAME = os.getenv("ACCOUNT_FRIENDLY_NAME", None)
15
- SLACK_URL = os.getenv("SLACK_URL", None)
16
- SLACK_CHANNEL = os.getenv("SLACK_CHANNEL", None)
16
+ SLACK_URL_SECRET_NAME = os.getenv("SLACK_URL_SECRET_NAME", None)
17
17
  NOTIFICATION_LEVEL = os.getenv("NOTIFICATION_LEVEL", "WARN")
18
18
 
19
19
  # Example event:
@@ -64,6 +64,16 @@ class TriggerMetadata(t.TypedDict):
64
64
  vcs: TriggerMetadataVcs
65
65
 
66
66
 
67
+ def get_masked_slack_webhook_url(slack_webhook_url: str):
68
+ """
69
+ Return a string that masks the final path segment of a Slack webhook URL.
70
+ The URL is typically formatted as such: https://hooks.slack.com/services/T00000000/B00000000/XXXXXXXXXXXXXXXXXXXXXXXX
71
+ """
72
+ trimmed_url = slack_webhook_url.rstrip("/")
73
+ [*url, final_path_segment] = trimmed_url.split("/")
74
+ return "/".join(url + [len(final_path_segment) * "*"])
75
+
76
+
67
77
  def get_previous_pipeline_execution(
68
78
  pipeline_name: str, execution_id: str
69
79
  ) -> dict | None:
@@ -185,6 +195,13 @@ def get_footer_text(ci_metadata: TriggerMetadata) -> str:
185
195
  return footer_text
186
196
 
187
197
 
198
+ def get_secret(secret):
199
+ try:
200
+ return secrets_manager.get_secret_value(SecretId=secret)["SecretString"]
201
+ except Exception as e:
202
+ raise Exception(f"Error retrieving secret: {e}")
203
+
204
+
188
205
  def handler(event, context):
189
206
 
190
207
  print("Event: " + json.dumps(event))
@@ -267,17 +284,16 @@ def handler(event, context):
267
284
  ]
268
285
 
269
286
  slack_message = {
270
- "channel": SLACK_CHANNEL,
271
287
  "attachments": attachments,
272
- "username": "Liflig CDK Pipelines",
273
- "icon_emoji": ":traffic_light:",
274
288
  }
275
289
 
276
- req = Request(SLACK_URL, json.dumps(slack_message).encode("utf-8"))
290
+ slack_url = get_secret(SLACK_URL_SECRET_NAME)
291
+
292
+ req = Request(slack_url, json.dumps(slack_message).encode("utf-8"))
293
+ print(f"Posting message to Slack URL {get_masked_slack_webhook_url(slack_url)}")
277
294
  try:
278
295
  response = urlopen(req)
279
296
  response.read()
280
- print(f"Message posted to: {slack_message['channel']}")
281
297
  except HTTPError as e:
282
298
  raise Exception(f"Request to slack failed: {e.code} {e.reason}")
283
299
  except URLError as e:
@@ -1,68 +1,102 @@
1
1
  import json
2
2
  import os
3
- import boto3
4
3
  from urllib.request import Request, urlopen
5
4
  from urllib.error import URLError, HTTPError
6
5
 
7
- SLACK_URL = os.getenv('SLACK_URL', None)
8
- SLACK_CHANNEL = os.getenv('SLACK_CHANNEL', None)
9
- PROJECT_NAME = os.getenv('PROJECT_NAME', 'undefined')
10
- ENVIRONMENT_NAME = os.getenv('ENVIRONMENT_NAME', 'undefined')
6
+ import boto3
7
+
8
+ secrets_manager = boto3.client("secretsmanager")
9
+
10
+ SLACK_URL_SECRET_NAME = os.getenv("SLACK_URL_SECRET_NAME", None)
11
+ PROJECT_NAME = os.getenv("PROJECT_NAME", "undefined")
12
+ ENVIRONMENT_NAME = os.getenv("ENVIRONMENT_NAME", "undefined")
11
13
 
12
14
 
13
15
  def handler(event, context):
14
16
  print("Event: " + json.dumps(event))
15
- message = json.loads(event['Records'][0]['Sns']['Message'])
16
- region = event['Records'][0]['Sns']['TopicArn'].split(':')[3]
17
+ message = json.loads(event["Records"][0]["Sns"]["Message"])
18
+ region = event["Records"][0]["Sns"]["TopicArn"].split(":")[3]
17
19
 
18
20
  return send_slack_notification(message, region)
19
21
 
20
22
 
23
+ def get_masked_slack_webhook_url(slack_webhook_url: str):
24
+ """
25
+ Return a string that masks the final path segment of a Slack webhook URL.
26
+ The URL is typically formatted as such: https://hooks.slack.com/services/T00000000/B00000000/XXXXXXXXXXXXXXXXXXXXXXXX
27
+ """
28
+ trimmed_url = slack_webhook_url.rstrip("/")
29
+ [*url, final_path_segment] = trimmed_url.split("/")
30
+ return "/".join(url + [len(final_path_segment) * "*"])
31
+
32
+
33
+ def get_secret(secret):
34
+ try:
35
+ return secrets_manager.get_secret_value(SecretId=secret)["SecretString"]
36
+ except Exception as e:
37
+ raise Exception(f"Error retrieving secret: {e}")
38
+
39
+
21
40
  def send_slack_notification(message, region):
22
41
  alarm_emojis = {
23
42
  "ALARM": ":rotating_light:",
24
43
  "INSUFFICIENT_DATA": ":warning:",
25
- "OK": ":white_check_mark:"
44
+ "OK": ":white_check_mark:",
26
45
  }
27
- if (message['NewStateValue'] == "ALARM"):
46
+ if message["NewStateValue"] == "ALARM":
28
47
  color = "danger"
29
48
  else:
30
49
  color = "good"
31
50
  alarm_description = message["AlarmDescription"] or "Alarm is missing description"
32
- attachments = [{
33
- 'color': color,
34
- 'title_link': "https://console.aws.amazon.com/cloudwatch/home?region=" + region + "#alarm:alarmFilter=ANY;name=" + message['AlarmName'],
35
- 'fallback': f"{alarm_emojis.get(message['NewStateValue'], '')} {message['AlarmName']}: {alarm_description}",
36
- 'fields': [
37
- {'title': 'Alarm Name',
38
- 'value': message['AlarmName'], 'short': False},
39
- {'title': 'Alarm Description',
40
- 'value': alarm_description, 'short': False},
41
- {'title': 'Account',
42
- 'value': message['AWSAccountId'], 'short': True},
43
- {'title': 'Region', 'value': region, 'short': True},
44
- {'title': 'Project', 'value': PROJECT_NAME, 'short': True},
45
- {'title': 'Environment', 'value': ENVIRONMENT_NAME, 'short': True},
46
- {'title': 'State Transition',
47
- 'value': message.get('OldStateValue', 'Unknown') + ' -> ' + message['NewStateValue'], 'short': False},
48
- {'title': 'Link to Alarm', 'value': "https://console.aws.amazon.com/cloudwatch/home?region=" +
49
- region + "#alarm:alarmFilter=ANY;name=" + message['AlarmName'], "short": False},
50
- ]
51
-
52
- }]
51
+ attachments = [
52
+ {
53
+ "color": color,
54
+ "title_link": "https://console.aws.amazon.com/cloudwatch/home?region="
55
+ + region
56
+ + "#alarm:alarmFilter=ANY;name="
57
+ + message["AlarmName"],
58
+ "fallback": f"{alarm_emojis.get(message['NewStateValue'], '')} {message['AlarmName']}: {alarm_description}",
59
+ "fields": [
60
+ {"title": "Alarm Name", "value": message["AlarmName"], "short": False},
61
+ {
62
+ "title": "Alarm Description",
63
+ "value": alarm_description,
64
+ "short": False,
65
+ },
66
+ {"title": "Account", "value": message["AWSAccountId"], "short": True},
67
+ {"title": "Region", "value": region, "short": True},
68
+ {"title": "Project", "value": PROJECT_NAME, "short": True},
69
+ {"title": "Environment", "value": ENVIRONMENT_NAME, "short": True},
70
+ {
71
+ "title": "State Transition",
72
+ "value": message.get("OldStateValue", "Unknown")
73
+ + " -> "
74
+ + message["NewStateValue"],
75
+ "short": False,
76
+ },
77
+ {
78
+ "title": "Link to Alarm",
79
+ "value": "https://console.aws.amazon.com/cloudwatch/home?region="
80
+ + region
81
+ + "#alarm:alarmFilter=ANY;name="
82
+ + message["AlarmName"],
83
+ "short": False,
84
+ },
85
+ ],
86
+ }
87
+ ]
53
88
 
54
89
  slackMessage = {
55
- 'channel': SLACK_CHANNEL,
56
- 'attachments': attachments,
57
- 'username': 'CloudWatch-Notifier',
58
- 'icon_emoji': ':traffic_light:'
90
+ "attachments": attachments,
59
91
  }
60
92
 
61
- req = Request(SLACK_URL, json.dumps(slackMessage).encode('utf-8'))
93
+ slack_url = get_secret(SLACK_URL_SECRET_NAME)
94
+
95
+ req = Request(slack_url, json.dumps(slackMessage).encode("utf-8"))
96
+ print(f"Posting message to Slack URL {get_masked_slack_webhook_url(slack_url)}")
62
97
  try:
63
98
  response = urlopen(req)
64
99
  response.read()
65
- return "Message posted to: " + slackMessage['channel']
66
100
  except HTTPError as e:
67
101
  raise Exception(f"Request to slack failed: {e.code} {e.reason}")
68
102
  except URLError as e:
@@ -168,4 +168,4 @@ class DatabaseAlarms extends constructs.Construct {
168
168
  }
169
169
  }
170
170
  exports.DatabaseAlarms = DatabaseAlarms;
171
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database-alarms.js","sourceRoot":"","sources":["../../src/alarms/database-alarms.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\n\nexport interface DatabaseAlarmsProps {\n  /**\n   * The default action to use for CloudWatch alarm state changes\n   */\n  action: cloudwatch.IAlarmAction\n  instanceIdentifier: string\n  instanceType: ec2.InstanceType\n  allocatedStorage: cdk.Size\n}\n\n// Based on https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/burstable-credits-baseline-concepts.html#earning-CPU-credits\nconst cpuCreditBalanceByInstanceType: {\n  [instanceType: string]: number\n} = {\n  \"t2.nano\": 72,\n  \"t2.micro\": 144,\n  \"t2.small\": 288,\n  \"t2.medium\": 576,\n  \"t2.large\": 864,\n  \"t2.xlarge\": 1296,\n  \"t2.2xlarge\": 1958.4,\n  \"t3.nano\": 144,\n  \"t3.micro\": 288,\n  \"t3.small\": 576,\n  \"t3.medium\": 576,\n  \"t3.large\": 864,\n  \"t3.xlarge\": 2304,\n  \"t3.2xlarge\": 4608,\n  \"t3a.nano\": 144,\n  \"t3a.micro\": 288,\n  \"t3a.small\": 576,\n  \"t3a.medium\": 576,\n  \"t3a.large\": 864,\n  \"t3a.xlarge\": 2304,\n  \"t3a.2xlarge\": 4608,\n  \"t4g.nano\": 144,\n  \"t4g.micro\": 288,\n  \"t4g.small\": 576,\n  \"t4g.medium\": 576,\n  \"t4g.large\": 864,\n  \"t4g.xlarge\": 2304,\n  \"t4g.2xlarge\": 4608,\n}\n\nexport class DatabaseAlarms extends constructs.Construct {\n  private readonly action: cloudwatch.IAlarmAction\n  private readonly databaseInstanceIdentifier: string\n  private readonly instanceType: ec2.InstanceType\n  private readonly allocatedStorage: cdk.Size\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: DatabaseAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.action = props.action\n    this.databaseInstanceIdentifier = props.instanceIdentifier\n    this.instanceType = props.instanceType\n    this.allocatedStorage = props.allocatedStorage\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers if the CPU credit balance for\n   * a burstable instance breach a certain threshold.\n   *\n   * NOTE: This alarm is only applicable for burstable instances, and a balance of 0 credits will only have performance\n   * implications for T2 instances. T3 and T4g instances will instead cost more for prolonged high CPU utilization after\n   * the balance is depleted.\n   */\n  addCpuCreditsAlarm(\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    props?: {\n      /**\n       * An action to use for CloudWatch alarm state changes instead of the default action\n       */\n      action?: cloudwatch.IAlarmAction\n      /**\n       * The CloudWatch Alarm will change its state to ALARM if the number of CPU credits drops below this threshold.\n       *\n       * See https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/burstable-credits-baseline-concepts.html#earning-CPU-credits for an overview of maximum CPU credits for various instance types.\n       *\n       * @default 10% of the maximum earned CPU credits for the instance type.\n       */\n      threshold?: number\n    },\n  ): void {\n    if (\n      ![\"t2.\", \"t3.\", \"t4g.\"].some((s) =>\n        this.instanceType.toString().startsWith(s),\n      )\n    ) {\n      throw new Error(\n        \"CPU credits are only relevant for burstable instance types.\",\n      )\n    }\n\n    const defaultThreshold =\n      cpuCreditBalanceByInstanceType[this.instanceType.toString()] * 0.1\n    const threshold = props?.threshold ?? defaultThreshold\n    if (!threshold) {\n      throw new Error(\n        `No threshold supplied, and unable to determine a default value for instance type '${this.instanceType.toString()}'`,\n      )\n    }\n    new cloudwatch.Metric({\n      metricName: \"CPUCreditBalance\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    })\n      .createAlarm(this, \"CreditsAlarm\", {\n        alarmDescription: `Less than ${threshold} CPU credits remaining for RDS database '${\n          this.databaseInstanceIdentifier\n        }'. ${\n          this.instanceType.toString().startsWith(\"t2.\")\n            ? \"If this reaches 0, the instance will be limited to a baseline CPU utilization.\"\n            : \"If the balance is depleted, AWS adds additional charges.\"\n        }.`,\n        comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: threshold,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      })\n      .addAlarmAction(this.action)\n  }\n\n  /**\n   * Sets up two CloudWatch Alarms for monitoring disk storage space:\n   * 1) one that triggers if the available disk storage space is low.\n   * 2) one that triggers if the available disk storage space is critcally low.\n   *\n   * You may want to use different alarm actions for the two alarms, e.g., one can be\n   * categorized as a \"warning\", while the other one can be considered an \"alarm\".\n   */\n  addStorageSpaceAlarms(props?: {\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    lowStorageSpaceAlarm?: {\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 25% of the allocated storage.\n       */\n      threshold?: cdk.Size\n    }\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    criticallyLowStorageSpaceAlarm?: {\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 5% of the allocated storage.\n       */\n      threshold?: cdk.Size\n    }\n  }): void {\n    const lowStorageSpaceAlarm = new cloudwatch.Metric({\n      metricName: \"FreeStorageSpace\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"LowStorageSpaceAlarm\", {\n      alarmDescription: `Low storage space available on RDS database '${this.databaseInstanceIdentifier}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold:\n        props?.lowStorageSpaceAlarm?.threshold?.toBytes() ??\n        this.allocatedStorage.toBytes() * 0.25,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props?.lowStorageSpaceAlarm?.enabled ?? true) {\n      lowStorageSpaceAlarm.addAlarmAction(\n        props?.lowStorageSpaceAlarm?.action || this.action,\n      )\n      lowStorageSpaceAlarm.addOkAction(\n        props?.lowStorageSpaceAlarm?.action || this.action,\n      )\n    }\n\n    const criticallyLowStorageSpaceAlarm = new cloudwatch.Metric({\n      metricName: \"FreeStorageSpace\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"CriticallyLowStorageSpaceAlarm\", {\n      alarmDescription: `Critically low storage space available on RDS database '${this.databaseInstanceIdentifier}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold:\n        props?.criticallyLowStorageSpaceAlarm?.threshold?.toBytes() ??\n        this.allocatedStorage.toBytes() * 0.05,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props?.criticallyLowStorageSpaceAlarm?.enabled ?? true) {\n      criticallyLowStorageSpaceAlarm.addAlarmAction(\n        props?.criticallyLowStorageSpaceAlarm?.action || this.action,\n      )\n      criticallyLowStorageSpaceAlarm.addOkAction(\n        props?.criticallyLowStorageSpaceAlarm?.action || this.action,\n      )\n    }\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers if the average CPU utilization for\n   * the RDS instance exceeds a given threshold.\n   */\n  addCpuUtilizationAlarm(\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    props?: {\n      /**\n       * An action to use for CloudWatch alarm state changes instead of the default action\n       */\n      action?: cloudwatch.IAlarmAction\n      /**\n       * The threshold defined as a percentage that determines if CPU utilization should trigger an alarm or not.\n       * @default 80\n       */\n      threshold?: number\n      /**\n       * @default 5\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 2 minutes\n       */\n      period?: cdk.Duration\n    },\n  ): void {\n    const alarm = new cloudwatch.Metric({\n      metricName: \"CPUUtilization\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Average\",\n      period: props?.period ?? cdk.Duration.minutes(2),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"CpuUtilizationAlarm\", {\n      alarmDescription: `RDS database '${this.databaseInstanceIdentifier}' has a higher than expected CPU utilization.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: props?.evaluationPeriods ?? 5,\n      threshold: props?.threshold ?? 80,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    alarm.addAlarmAction(props?.action ?? this.action)\n    alarm.addOkAction(props?.action ?? this.action)\n  }\n}\n"]}
171
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database-alarms.js","sourceRoot":"","sources":["../../src/alarms/database-alarms.ts"],"names":[],"mappings":";;;AAAA,yCAAwC;AACxC,mCAAkC;AAClC,yDAAwD;AAaxD,4HAA4H;AAC5H,MAAM,8BAA8B,GAEhC;IACF,SAAS,EAAE,EAAE;IACb,UAAU,EAAE,GAAG;IACf,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,GAAG;IAChB,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,IAAI;IACjB,YAAY,EAAE,MAAM;IACpB,SAAS,EAAE,GAAG;IACd,UAAU,EAAE,GAAG;IACf,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,GAAG;IAChB,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,IAAI;IACjB,YAAY,EAAE,IAAI;IAClB,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,GAAG;IAChB,WAAW,EAAE,GAAG;IAChB,YAAY,EAAE,GAAG;IACjB,WAAW,EAAE,GAAG;IAChB,YAAY,EAAE,IAAI;IAClB,aAAa,EAAE,IAAI;IACnB,UAAU,EAAE,GAAG;IACf,WAAW,EAAE,GAAG;IAChB,WAAW,EAAE,GAAG;IAChB,YAAY,EAAE,GAAG;IACjB,WAAW,EAAE,GAAG;IAChB,YAAY,EAAE,IAAI;IAClB,aAAa,EAAE,IAAI;CACpB,CAAA;AAED,MAAa,cAAe,SAAQ,UAAU,CAAC,SAAS;IAMtD,YACE,KAA2B,EAC3B,EAAU,EACV,KAA0B;QAE1B,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAA;QAC1B,IAAI,CAAC,0BAA0B,GAAG,KAAK,CAAC,kBAAkB,CAAA;QAC1D,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,YAAY,CAAA;QACtC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,gBAAgB,CAAA;IAChD,CAAC;IAED;;;;;;;OAOG;IACH,kBAAkB;IAChB;;;;OAIG;IACH,KAaC;;QAED,IACE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CACjC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC,CAC3C,EACD,CAAC;YACD,MAAM,IAAI,KAAK,CACb,6DAA6D,CAC9D,CAAA;QACH,CAAC;QAED,MAAM,gBAAgB,GACpB,8BAA8B,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC,GAAG,GAAG,CAAA;QACpE,MAAM,SAAS,GAAG,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,SAAS,mCAAI,gBAAgB,CAAA;QACtD,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,MAAM,IAAI,KAAK,CACb,qFAAqF,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,GAAG,CACrH,CAAA;QACH,CAAC;QACD,IAAI,UAAU,CAAC,MAAM,CAAC;YACpB,UAAU,EAAE,kBAAkB;YAC9B,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;YACpB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC/B,aAAa,EAAE;gBACb,oBAAoB,EAAE,IAAI,CAAC,0BAA0B;aACtD;SACF,CAAC;aACC,WAAW,CAAC,IAAI,EAAE,cAAc,EAAE;YACjC,gBAAgB,EAAE,aAAa,SAAS,4CACtC,IAAI,CAAC,0BACP,MACE,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC;gBAC5C,CAAC,CAAC,gFAAgF;gBAClF,CAAC,CAAC,0DACN,GAAG;YACH,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,mBAAmB;YACrE,iBAAiB,EAAE,CAAC;YACpB,SAAS,EAAE,SAAS;YACpB,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC;aACD,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;IAChC,CAAC;IAED;;;;;;;OAOG;IACH,qBAAqB,CAAC,KAuCrB;;QACC,MAAM,oBAAoB,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YACjD,UAAU,EAAE,kBAAkB;YAC9B,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;YACpB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC/B,aAAa,EAAE;gBACb,oBAAoB,EAAE,IAAI,CAAC,0BAA0B;aACtD;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,sBAAsB,EAAE;YAC3C,gBAAgB,EAAE,gDAAgD,IAAI,CAAC,0BAA0B,IAAI;YACrG,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,mBAAmB;YACrE,iBAAiB,EAAE,CAAC;YACpB,SAAS,EACP,MAAA,MAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,oBAAoB,0CAAE,SAAS,0CAAE,OAAO,EAAE,mCACjD,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,GAAG,IAAI;YACxC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC,CAAA;QACF,IAAI,MAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,oBAAoB,0CAAE,OAAO,mCAAI,IAAI,EAAE,CAAC;YACjD,oBAAoB,CAAC,cAAc,CACjC,CAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,oBAAoB,0CAAE,MAAM,KAAI,IAAI,CAAC,MAAM,CACnD,CAAA;YACD,oBAAoB,CAAC,WAAW,CAC9B,CAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,oBAAoB,0CAAE,MAAM,KAAI,IAAI,CAAC,MAAM,CACnD,CAAA;QACH,CAAC;QAED,MAAM,8BAA8B,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YAC3D,UAAU,EAAE,kBAAkB;YAC9B,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;YACpB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC/B,aAAa,EAAE;gBACb,oBAAoB,EAAE,IAAI,CAAC,0BAA0B;aACtD;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,gCAAgC,EAAE;YACrD,gBAAgB,EAAE,2DAA2D,IAAI,CAAC,0BAA0B,IAAI;YAChH,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,mBAAmB;YACrE,iBAAiB,EAAE,CAAC;YACpB,SAAS,EACP,MAAA,MAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,8BAA8B,0CAAE,SAAS,0CAAE,OAAO,EAAE,mCAC3D,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,GAAG,IAAI;YACxC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC,CAAA;QACF,IAAI,MAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,8BAA8B,0CAAE,OAAO,mCAAI,IAAI,EAAE,CAAC;YAC3D,8BAA8B,CAAC,cAAc,CAC3C,CAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,8BAA8B,0CAAE,MAAM,KAAI,IAAI,CAAC,MAAM,CAC7D,CAAA;YACD,8BAA8B,CAAC,WAAW,CACxC,CAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,8BAA8B,0CAAE,MAAM,KAAI,IAAI,CAAC,MAAM,CAC7D,CAAA;QACH,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,sBAAsB;IACpB;;;;OAIG;IACH,KAkBC;;QAED,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YAClC,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,SAAS;YACpB,SAAS,EAAE,SAAS;YACpB,MAAM,EAAE,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,mCAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAChD,aAAa,EAAE;gBACb,oBAAoB,EAAE,IAAI,CAAC,0BAA0B;aACtD;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,qBAAqB,EAAE;YAC1C,gBAAgB,EAAE,iBAAiB,IAAI,CAAC,0BAA0B,+CAA+C;YACjH,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,sBAAsB;YACxE,iBAAiB,EAAE,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,iBAAiB,mCAAI,CAAC;YAChD,SAAS,EAAE,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,SAAS,mCAAI,EAAE;YACjC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC,CAAA;QACF,KAAK,CAAC,cAAc,CAAC,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CAAC,CAAA;QAClD,KAAK,CAAC,WAAW,CAAC,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CAAC,CAAA;IACjD,CAAC;CACF;AAhPD,wCAgPC","sourcesContent":["import * as constructs from \"constructs\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\n\nexport interface DatabaseAlarmsProps {\n  /**\n   * The default action to use for CloudWatch alarm state changes\n   */\n  action: cloudwatch.IAlarmAction\n  instanceIdentifier: string\n  instanceType: ec2.InstanceType\n  allocatedStorage: cdk.Size\n}\n\n// Based on https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/burstable-credits-baseline-concepts.html#earning-CPU-credits\nconst cpuCreditBalanceByInstanceType: {\n  [instanceType: string]: number\n} = {\n  \"t2.nano\": 72,\n  \"t2.micro\": 144,\n  \"t2.small\": 288,\n  \"t2.medium\": 576,\n  \"t2.large\": 864,\n  \"t2.xlarge\": 1296,\n  \"t2.2xlarge\": 1958.4,\n  \"t3.nano\": 144,\n  \"t3.micro\": 288,\n  \"t3.small\": 576,\n  \"t3.medium\": 576,\n  \"t3.large\": 864,\n  \"t3.xlarge\": 2304,\n  \"t3.2xlarge\": 4608,\n  \"t3a.nano\": 144,\n  \"t3a.micro\": 288,\n  \"t3a.small\": 576,\n  \"t3a.medium\": 576,\n  \"t3a.large\": 864,\n  \"t3a.xlarge\": 2304,\n  \"t3a.2xlarge\": 4608,\n  \"t4g.nano\": 144,\n  \"t4g.micro\": 288,\n  \"t4g.small\": 576,\n  \"t4g.medium\": 576,\n  \"t4g.large\": 864,\n  \"t4g.xlarge\": 2304,\n  \"t4g.2xlarge\": 4608,\n}\n\nexport class DatabaseAlarms extends constructs.Construct {\n  private readonly action: cloudwatch.IAlarmAction\n  private readonly databaseInstanceIdentifier: string\n  private readonly instanceType: ec2.InstanceType\n  private readonly allocatedStorage: cdk.Size\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: DatabaseAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.action = props.action\n    this.databaseInstanceIdentifier = props.instanceIdentifier\n    this.instanceType = props.instanceType\n    this.allocatedStorage = props.allocatedStorage\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers if the CPU credit balance for\n   * a burstable instance breach a certain threshold.\n   *\n   * NOTE: This alarm is only applicable for burstable instances, and a balance of 0 credits will only have performance\n   * implications for T2 instances. T3 and T4g instances will instead cost more for prolonged high CPU utilization after\n   * the balance is depleted.\n   */\n  addCpuCreditsAlarm(\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    props?: {\n      /**\n       * An action to use for CloudWatch alarm state changes instead of the default action\n       */\n      action?: cloudwatch.IAlarmAction\n      /**\n       * The CloudWatch Alarm will change its state to ALARM if the number of CPU credits drops below this threshold.\n       *\n       * See https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/burstable-credits-baseline-concepts.html#earning-CPU-credits for an overview of maximum CPU credits for various instance types.\n       *\n       * @default 10% of the maximum earned CPU credits for the instance type.\n       */\n      threshold?: number\n    },\n  ): void {\n    if (\n      ![\"t2.\", \"t3.\", \"t4g.\"].some((s) =>\n        this.instanceType.toString().startsWith(s),\n      )\n    ) {\n      throw new Error(\n        \"CPU credits are only relevant for burstable instance types.\",\n      )\n    }\n\n    const defaultThreshold =\n      cpuCreditBalanceByInstanceType[this.instanceType.toString()] * 0.1\n    const threshold = props?.threshold ?? defaultThreshold\n    if (!threshold) {\n      throw new Error(\n        `No threshold supplied, and unable to determine a default value for instance type '${this.instanceType.toString()}'`,\n      )\n    }\n    new cloudwatch.Metric({\n      metricName: \"CPUCreditBalance\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    })\n      .createAlarm(this, \"CreditsAlarm\", {\n        alarmDescription: `Less than ${threshold} CPU credits remaining for RDS database '${\n          this.databaseInstanceIdentifier\n        }'. ${\n          this.instanceType.toString().startsWith(\"t2.\")\n            ? \"If this reaches 0, the instance will be limited to a baseline CPU utilization.\"\n            : \"If the balance is depleted, AWS adds additional charges.\"\n        }.`,\n        comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: threshold,\n        treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n      })\n      .addAlarmAction(this.action)\n  }\n\n  /**\n   * Sets up two CloudWatch Alarms for monitoring disk storage space:\n   * 1) one that triggers if the available disk storage space is low.\n   * 2) one that triggers if the available disk storage space is critcally low.\n   *\n   * You may want to use different alarm actions for the two alarms, e.g., one can be\n   * categorized as a \"warning\", while the other one can be considered an \"alarm\".\n   */\n  addStorageSpaceAlarms(props?: {\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    lowStorageSpaceAlarm?: {\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 25% of the allocated storage.\n       */\n      threshold?: cdk.Size\n    }\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    criticallyLowStorageSpaceAlarm?: {\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 5% of the allocated storage.\n       */\n      threshold?: cdk.Size\n    }\n  }): void {\n    const lowStorageSpaceAlarm = new cloudwatch.Metric({\n      metricName: \"FreeStorageSpace\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"LowStorageSpaceAlarm\", {\n      alarmDescription: `Low storage space available on RDS database '${this.databaseInstanceIdentifier}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold:\n        props?.lowStorageSpaceAlarm?.threshold?.toBytes() ??\n        this.allocatedStorage.toBytes() * 0.25,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props?.lowStorageSpaceAlarm?.enabled ?? true) {\n      lowStorageSpaceAlarm.addAlarmAction(\n        props?.lowStorageSpaceAlarm?.action || this.action,\n      )\n      lowStorageSpaceAlarm.addOkAction(\n        props?.lowStorageSpaceAlarm?.action || this.action,\n      )\n    }\n\n    const criticallyLowStorageSpaceAlarm = new cloudwatch.Metric({\n      metricName: \"FreeStorageSpace\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Minimum\",\n      period: cdk.Duration.minutes(5),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"CriticallyLowStorageSpaceAlarm\", {\n      alarmDescription: `Critically low storage space available on RDS database '${this.databaseInstanceIdentifier}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold:\n        props?.criticallyLowStorageSpaceAlarm?.threshold?.toBytes() ??\n        this.allocatedStorage.toBytes() * 0.05,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props?.criticallyLowStorageSpaceAlarm?.enabled ?? true) {\n      criticallyLowStorageSpaceAlarm.addAlarmAction(\n        props?.criticallyLowStorageSpaceAlarm?.action || this.action,\n      )\n      criticallyLowStorageSpaceAlarm.addOkAction(\n        props?.criticallyLowStorageSpaceAlarm?.action || this.action,\n      )\n    }\n  }\n\n  /**\n   * Sets up a CloudWatch Alarm that triggers if the average CPU utilization for\n   * the RDS instance exceeds a given threshold.\n   */\n  addCpuUtilizationAlarm(\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    props?: {\n      /**\n       * An action to use for CloudWatch alarm state changes instead of the default action\n       */\n      action?: cloudwatch.IAlarmAction\n      /**\n       * The threshold defined as a percentage that determines if CPU utilization should trigger an alarm or not.\n       * @default 80\n       */\n      threshold?: number\n      /**\n       * @default 5\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 2 minutes\n       */\n      period?: cdk.Duration\n    },\n  ): void {\n    const alarm = new cloudwatch.Metric({\n      metricName: \"CPUUtilization\",\n      namespace: \"AWS/RDS\",\n      statistic: \"Average\",\n      period: props?.period ?? cdk.Duration.minutes(2),\n      dimensionsMap: {\n        DBInstanceIdentifier: this.databaseInstanceIdentifier,\n      },\n    }).createAlarm(this, \"CpuUtilizationAlarm\", {\n      alarmDescription: `RDS database '${this.databaseInstanceIdentifier}' has a higher than expected CPU utilization.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: props?.evaluationPeriods ?? 5,\n      threshold: props?.threshold ?? 80,\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    alarm.addAlarmAction(props?.action ?? this.action)\n    alarm.addOkAction(props?.action ?? this.action)\n  }\n}\n"]}
@@ -145,4 +145,4 @@ class ServiceAlarms extends constructs.Construct {
145
145
  }
146
146
  }
147
147
  exports.ServiceAlarms = ServiceAlarms;
148
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service-alarms.js","sourceRoot":"","sources":["../../src/alarms/service-alarms.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\n\nexport interface ServiceAlarmsProps extends cdk.StackProps {\n  /**\n   * The default action to use for CloudWatch alarm state changes\n   */\n  action: cloudwatch.IAlarmAction\n  /**\n   * The name of the ECS service.\n   */\n  serviceName: string\n}\n\n/**\n * Various alarms and monitoring.\n *\n * By itself no alarms is created. Use the methods available\n * to add alarms.\n *\n * See SlackAlarm construct for SNS Action.\n */\nexport class ServiceAlarms extends constructs.Construct {\n  private readonly action: cloudwatch.IAlarmAction\n  private readonly serviceName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ServiceAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.action = props.action\n    this.serviceName = props.serviceName\n  }\n\n  /**\n   * For logs stored as JSON, monitor log entries logged\n   * with level ERROR or higher, as well as any requests\n   * that causes 500 for logging with liflig-logging.\n   */\n  addJsonErrorAlarm(props: {\n    logGroup: logs.ILogGroup\n    alarmDescription?: string\n    /**\n     * Set to `false` to stop the alarm from sending OK events.\n     * @default true\n     * */\n    enableOkAction?: boolean\n    /**\n     * An action to use for CloudWatch alarm state changes instead of the default action\n     */\n    action?: cloudwatch.IAlarmAction\n  }): void {\n    const errorMetricFilter = props.logGroup.addMetricFilter(\n      \"ErrorMetricFilter\",\n      {\n        filterPattern: logs.FilterPattern.any(\n          logs.FilterPattern.stringValue(\"$.level\", \"=\", \"ERROR\"),\n          // FATAL covers some applications we run that uses log4j or\n          // other libraries. It is not existent in slf4j.\n          logs.FilterPattern.stringValue(\"$.level\", \"=\", \"FATAL\"),\n          logs.FilterPattern.stringValue(\n            // For liflig-logging.\n            \"$.requestInfo.status.code\",\n            \"=\",\n            \"INTERNAL_SERVER_ERROR\",\n          ),\n        ),\n        metricName: \"Errors\",\n        metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${\n          this.serviceName\n        }/Errors`,\n      },\n    )\n\n    const errorAlarm = errorMetricFilter\n      .metric()\n      .with({\n        statistic: \"Sum\",\n        period: cdk.Duration.seconds(60),\n      })\n      .createAlarm(this, \"ErrorLogAlarm\", {\n        alarmDescription:\n          props.alarmDescription ?? `${this.serviceName} logged an error`,\n        evaluationPeriods: 1,\n        threshold: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      })\n\n    errorAlarm.addAlarmAction(props.action ?? this.action)\n    if (props.enableOkAction ?? true) {\n      errorAlarm.addOkAction(props.action ?? this.action)\n    }\n  }\n\n  /**\n   * Sets up three CloudWatch Alarms for monitoring an ECS service behind a target group:\n   * 1) one that triggers if the target is responding with too many 5xx errors.\n   * 2) one that triggers if the 95% percentile of response times from the target is too high.\n   * 3) one that triggers if there are no healthy targets or if the load balancer fails to connect to targets.\n   */\n  addTargetGroupAlarms(props: {\n    /**\n     * The full name of the target group.\n     */\n    targetGroupFullName: string\n    /**\n     * The full name of the application load balancer.\n     */\n    loadBalancerFullName: string\n    /**\n     * Configuration for a composite alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    targetHealthAlarm?: {\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     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    tooMany5xxResponsesFromTargetsAlarm?: {\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 3\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 10\n       */\n      threshold?: number\n      description?: string\n    }\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    targetResponseTimeAlarm?: {\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 5 minutes\n       */\n      period?: cdk.Duration\n      /**\n       * @default 1\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 500 milliseconds\n       */\n      threshold?: cdk.Duration\n      description?: string\n    }\n  }): void {\n    const targetConnectionErrorAlarm = new cloudwatch.Metric({\n      metricName: \"TargetConnectionErrorCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Sum\",\n      period: props.targetHealthAlarm?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"ConnectionAlarm\", {\n      actionsEnabled: true,\n      alarmDescription: `Load balancer is failing to connect to target(s) in ECS service '${this.serviceName}'.`,\n      evaluationPeriods: props.targetHealthAlarm?.evaluationPeriods ?? 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n\n    const healthAlarm = new cloudwatch.Metric({\n      metricName: \"HealthyHostCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Minimum\",\n      period: props.targetHealthAlarm?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"HealthAlarm\", {\n      alarmDescription: `There are no healthy target(s) in ECS service '${this.serviceName}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: props.targetHealthAlarm?.evaluationPeriods ?? 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.BREACHING,\n    })\n\n    const targetHealthAlarm = new cloudwatch.CompositeAlarm(\n      this,\n      \"TargetHealthAlarm\",\n      {\n        alarmRule: cdk.aws_cloudwatch.AlarmRule.anyOf(\n          cdk.aws_cloudwatch.AlarmRule.fromAlarm(\n            targetConnectionErrorAlarm,\n            cloudwatch.AlarmState.ALARM,\n          ),\n          cdk.aws_cloudwatch.AlarmRule.fromAlarm(\n            healthAlarm,\n            cloudwatch.AlarmState.ALARM,\n          ),\n        ),\n        alarmDescription:\n          props.targetHealthAlarm?.description ??\n          `The load balancer is either receiving bad health checks from or is unable to connect to target(s) in ECS service '${this.serviceName}'`,\n        actionsEnabled: false,\n      },\n    )\n    if (props.targetHealthAlarm?.enabled ?? true) {\n      targetHealthAlarm.addAlarmAction(\n        props.targetHealthAlarm?.action ?? this.action,\n      )\n      targetHealthAlarm.addOkAction(\n        props.targetHealthAlarm?.action ?? this.action,\n      )\n    }\n\n    const tooMany5xxResponsesFromTargetsAlarm = new cloudwatch.Metric({\n      metricName: \"HTTPCode_Target_5XX_Count\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Sum\",\n      period:\n        props.tooMany5xxResponsesFromTargetsAlarm?.period ??\n        cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"AlbTargets5xxAlarm\", {\n      actionsEnabled: true,\n      alarmDescription:\n        props.tooMany5xxResponsesFromTargetsAlarm?.description ??\n        `Load balancer received too many 5XX responses from target(s) in ECS service '${this.serviceName}'.`,\n      evaluationPeriods:\n        props.tooMany5xxResponsesFromTargetsAlarm?.evaluationPeriods ?? 3,\n      threshold: props.tooMany5xxResponsesFromTargetsAlarm?.threshold ?? 10,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n    if (props.tooMany5xxResponsesFromTargetsAlarm?.enabled ?? true) {\n      tooMany5xxResponsesFromTargetsAlarm.addAlarmAction(\n        props.tooMany5xxResponsesFromTargetsAlarm?.action ?? this.action,\n      )\n      tooMany5xxResponsesFromTargetsAlarm.addOkAction(\n        props.tooMany5xxResponsesFromTargetsAlarm?.action ?? this.action,\n      )\n    }\n\n    const targetResponseTimeAlarm = new cloudwatch.Metric({\n      metricName: \"TargetResponseTime\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"p95\",\n      period: props.targetResponseTimeAlarm?.period ?? cdk.Duration.minutes(5),\n      dimensionsMap: {\n        LoadBalancer: props.loadBalancerFullName,\n        TargetGroup: props.targetGroupFullName,\n      },\n    }).createAlarm(this, \"TargetResponseTimeAlarm\", {\n      alarmDescription:\n        props.targetResponseTimeAlarm?.description ??\n        `5% of responses from ECS service '${\n          this.serviceName\n        }' are taking longer than the expected duration of ${(\n          props.targetResponseTimeAlarm?.threshold ?? cdk.Duration.millis(500)\n        ).toMilliseconds()} ms.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: props.targetResponseTimeAlarm?.evaluationPeriods ?? 1,\n      threshold: (\n        props.targetResponseTimeAlarm?.threshold ?? cdk.Duration.millis(500)\n      ).toSeconds({ integral: false }),\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props.targetResponseTimeAlarm?.enabled ?? true) {\n      targetResponseTimeAlarm.addAlarmAction(\n        props.targetResponseTimeAlarm?.action ?? this.action,\n      )\n      targetResponseTimeAlarm.addOkAction(\n        props.targetResponseTimeAlarm?.action ?? this.action,\n      )\n    }\n  }\n}\n"]}
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service-alarms.js","sourceRoot":"","sources":["../../src/alarms/service-alarms.ts"],"names":[],"mappings":";;;AAAA,mCAAkC;AAClC,yDAAwD;AACxD,6CAA4C;AAC5C,yCAAwC;AAaxC;;;;;;;GAOG;AACH,MAAa,aAAc,SAAQ,UAAU,CAAC,SAAS;IAIrD,YACE,KAA2B,EAC3B,EAAU,EACV,KAAyB;QAEzB,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAA;QAC1B,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAA;IACtC,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,KAYjB;;QACC,MAAM,iBAAiB,GAAG,KAAK,CAAC,QAAQ,CAAC,eAAe,CACtD,mBAAmB,EACnB;YACE,aAAa,EAAE,IAAI,CAAC,aAAa,CAAC,GAAG,CACnC,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC;YACvD,2DAA2D;YAC3D,gDAAgD;YAChD,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,EACvD,IAAI,CAAC,aAAa,CAAC,WAAW;YAC5B,sBAAsB;YACtB,2BAA2B,EAC3B,GAAG,EACH,uBAAuB,CACxB,CACF;YACD,UAAU,EAAE,QAAQ;YACpB,eAAe,EAAE,SAAS,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,IACpD,IAAI,CAAC,WACP,SAAS;SACV,CACF,CAAA;QAED,MAAM,UAAU,GAAG,iBAAiB;aACjC,MAAM,EAAE;aACR,IAAI,CAAC;YACJ,SAAS,EAAE,KAAK;YAChB,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SACjC,CAAC;aACD,WAAW,CAAC,IAAI,EAAE,eAAe,EAAE;YAClC,gBAAgB,EACd,MAAA,KAAK,CAAC,gBAAgB,mCAAI,GAAG,IAAI,CAAC,WAAW,kBAAkB;YACjE,iBAAiB,EAAE,CAAC;YACpB,SAAS,EAAE,CAAC;YACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,aAAa;SAC5D,CAAC,CAAA;QAEJ,UAAU,CAAC,cAAc,CAAC,MAAA,KAAK,CAAC,MAAM,mCAAI,IAAI,CAAC,MAAM,CAAC,CAAA;QACtD,IAAI,MAAA,KAAK,CAAC,cAAc,mCAAI,IAAI,EAAE,CAAC;YACjC,UAAU,CAAC,WAAW,CAAC,MAAA,KAAK,CAAC,MAAM,mCAAI,IAAI,CAAC,MAAM,CAAC,CAAA;QACrD,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,oBAAoB,CAAC,KA6FpB;;QACC,MAAM,0BAA0B,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YACvD,UAAU,EAAE,4BAA4B;YACxC,SAAS,EAAE,oBAAoB;YAC/B,SAAS,EAAE,KAAK;YAChB,MAAM,EAAE,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,MAAM,mCAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACnE,aAAa,EAAE;gBACb,WAAW,EAAE,KAAK,CAAC,mBAAmB;gBACtC,YAAY,EAAE,KAAK,CAAC,oBAAoB;aACzC;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,iBAAiB,EAAE;YACtC,cAAc,EAAE,IAAI;YACpB,gBAAgB,EAAE,oEAAoE,IAAI,CAAC,WAAW,IAAI;YAC1G,iBAAiB,EAAE,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,iBAAiB,mCAAI,CAAC;YAClE,SAAS,EAAE,CAAC;YACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,aAAa;SAC5D,CAAC,CAAA;QAEF,MAAM,WAAW,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YACxC,UAAU,EAAE,kBAAkB;YAC9B,SAAS,EAAE,oBAAoB;YAC/B,SAAS,EAAE,SAAS;YACpB,MAAM,EAAE,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,MAAM,mCAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACnE,aAAa,EAAE;gBACb,WAAW,EAAE,KAAK,CAAC,mBAAmB;gBACtC,YAAY,EAAE,KAAK,CAAC,oBAAoB;aACzC;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,aAAa,EAAE;YAClC,gBAAgB,EAAE,kDAAkD,IAAI,CAAC,WAAW,IAAI;YACxF,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,mBAAmB;YACrE,iBAAiB,EAAE,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,iBAAiB,mCAAI,CAAC;YAClE,SAAS,EAAE,CAAC;YACZ,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,SAAS;SACxD,CAAC,CAAA;QAEF,MAAM,iBAAiB,GAAG,IAAI,UAAU,CAAC,cAAc,CACrD,IAAI,EACJ,mBAAmB,EACnB;YACE,SAAS,EAAE,GAAG,CAAC,cAAc,CAAC,SAAS,CAAC,KAAK,CAC3C,GAAG,CAAC,cAAc,CAAC,SAAS,CAAC,SAAS,CACpC,0BAA0B,EAC1B,UAAU,CAAC,UAAU,CAAC,KAAK,CAC5B,EACD,GAAG,CAAC,cAAc,CAAC,SAAS,CAAC,SAAS,CACpC,WAAW,EACX,UAAU,CAAC,UAAU,CAAC,KAAK,CAC5B,CACF;YACD,gBAAgB,EACd,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,WAAW,mCACpC,qHAAqH,IAAI,CAAC,WAAW,GAAG;YAC1I,cAAc,EAAE,KAAK;SACtB,CACF,CAAA;QACD,IAAI,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,OAAO,mCAAI,IAAI,EAAE,CAAC;YAC7C,iBAAiB,CAAC,cAAc,CAC9B,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CAC/C,CAAA;YACD,iBAAiB,CAAC,WAAW,CAC3B,MAAA,MAAA,KAAK,CAAC,iBAAiB,0CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CAC/C,CAAA;QACH,CAAC;QAED,MAAM,mCAAmC,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YAChE,UAAU,EAAE,2BAA2B;YACvC,SAAS,EAAE,oBAAoB;YAC/B,SAAS,EAAE,KAAK;YAChB,MAAM,EACJ,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,MAAM,mCACjD,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1B,aAAa,EAAE;gBACb,WAAW,EAAE,KAAK,CAAC,mBAAmB;gBACtC,YAAY,EAAE,KAAK,CAAC,oBAAoB;aACzC;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,oBAAoB,EAAE;YACzC,cAAc,EAAE,IAAI;YACpB,gBAAgB,EACd,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,WAAW,mCACtD,gFAAgF,IAAI,CAAC,WAAW,IAAI;YACtG,iBAAiB,EACf,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,iBAAiB,mCAAI,CAAC;YACnE,SAAS,EAAE,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,SAAS,mCAAI,EAAE;YACrE,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,aAAa;SAC5D,CAAC,CAAA;QACF,IAAI,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,OAAO,mCAAI,IAAI,EAAE,CAAC;YAC/D,mCAAmC,CAAC,cAAc,CAChD,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CACjE,CAAA;YACD,mCAAmC,CAAC,WAAW,CAC7C,MAAA,MAAA,KAAK,CAAC,mCAAmC,0CAAE,MAAM,mCAAI,IAAI,CAAC,MAAM,CACjE,CAAA;QACH,CAAC;QAED,MAAM,uBAAuB,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC;YACpD,UAAU,EAAE,oBAAoB;YAChC,SAAS,EAAE,oBAAoB;YAC/B,SAAS,EAAE,KAAK;YAChB,MAAM,EAAE,MAAA,MAAA,KAAK,CAAC,uBAAuB,0CAAE,MAAM,mCAAI,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YACxE,aAAa,EAAE;gBACb,YAAY,EAAE,KAAK,CAAC,oBAAoB;gBACxC,WAAW,EAAE,KAAK,CAAC,mBAAmB;aACvC;SACF,CAAC,CAAC,WAAW,CAAC,IAAI,EAAE,yBAAyB,EAAE;YAC9C,gBAAgB,EACd,MAAA,MAAA,KAAK,CAAC,uBAAuB,0CAAE,WAAW,mCAC1C,qCACE,IAAI,CAAC,WACP,qDAAqD,CACnD,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,SAAS,qCAAI,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CACrE,CAAC,cAAc,EAAE,MAAM;YAC1B,kBAAkB,EAAE,UAAU,CAAC,kBAAkB,CAAC,sBAAsB;YACxE,iBAAiB,EAAE,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,iBAAiB,qCAAI,CAAC;YACxE,SAAS,EAAE,CACT,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,SAAS,qCAAI,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CACrE,CAAC,SAAS,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;YAChC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB,CAAC,MAAM;SACrD,CAAC,CAAA;QACF,IAAI,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,OAAO,qCAAI,IAAI,EAAE,CAAC;YACnD,uBAAuB,CAAC,cAAc,CACpC,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,MAAM,qCAAI,IAAI,CAAC,MAAM,CACrD,CAAA;YACD,uBAAuB,CAAC,WAAW,CACjC,OAAA,OAAA,KAAK,CAAC,uBAAuB,4CAAE,MAAM,qCAAI,IAAI,CAAC,MAAM,CACrD,CAAA;QACH,CAAC;IACH,CAAC;CACF;AA7SD,sCA6SC","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\n\nexport interface ServiceAlarmsProps extends cdk.StackProps {\n  /**\n   * The default action to use for CloudWatch alarm state changes\n   */\n  action: cloudwatch.IAlarmAction\n  /**\n   * The name of the ECS service.\n   */\n  serviceName: string\n}\n\n/**\n * Various alarms and monitoring.\n *\n * By itself no alarms is created. Use the methods available\n * to add alarms.\n *\n * See SlackAlarm construct for SNS Action.\n */\nexport class ServiceAlarms extends constructs.Construct {\n  private readonly action: cloudwatch.IAlarmAction\n  private readonly serviceName: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: ServiceAlarmsProps,\n  ) {\n    super(scope, id)\n\n    this.action = props.action\n    this.serviceName = props.serviceName\n  }\n\n  /**\n   * For logs stored as JSON, monitor log entries logged\n   * with level ERROR or higher, as well as any requests\n   * that causes 500 for logging with liflig-logging.\n   */\n  addJsonErrorAlarm(props: {\n    logGroup: logs.ILogGroup\n    alarmDescription?: string\n    /**\n     * Set to `false` to stop the alarm from sending OK events.\n     * @default true\n     * */\n    enableOkAction?: boolean\n    /**\n     * An action to use for CloudWatch alarm state changes instead of the default action\n     */\n    action?: cloudwatch.IAlarmAction\n  }): void {\n    const errorMetricFilter = props.logGroup.addMetricFilter(\n      \"ErrorMetricFilter\",\n      {\n        filterPattern: logs.FilterPattern.any(\n          logs.FilterPattern.stringValue(\"$.level\", \"=\", \"ERROR\"),\n          // FATAL covers some applications we run that uses log4j or\n          // other libraries. It is not existent in slf4j.\n          logs.FilterPattern.stringValue(\"$.level\", \"=\", \"FATAL\"),\n          logs.FilterPattern.stringValue(\n            // For liflig-logging.\n            \"$.requestInfo.status.code\",\n            \"=\",\n            \"INTERNAL_SERVER_ERROR\",\n          ),\n        ),\n        metricName: \"Errors\",\n        metricNamespace: `stack/${cdk.Stack.of(this).stackName}/${\n          this.serviceName\n        }/Errors`,\n      },\n    )\n\n    const errorAlarm = errorMetricFilter\n      .metric()\n      .with({\n        statistic: \"Sum\",\n        period: cdk.Duration.seconds(60),\n      })\n      .createAlarm(this, \"ErrorLogAlarm\", {\n        alarmDescription:\n          props.alarmDescription ?? `${this.serviceName} logged an error`,\n        evaluationPeriods: 1,\n        threshold: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      })\n\n    errorAlarm.addAlarmAction(props.action ?? this.action)\n    if (props.enableOkAction ?? true) {\n      errorAlarm.addOkAction(props.action ?? this.action)\n    }\n  }\n\n  /**\n   * Sets up three CloudWatch Alarms for monitoring an ECS service behind a target group:\n   * 1) one that triggers if the target is responding with too many 5xx errors.\n   * 2) one that triggers if the 95% percentile of response times from the target is too high.\n   * 3) one that triggers if there are no healthy targets or if the load balancer fails to connect to targets.\n   */\n  addTargetGroupAlarms(props: {\n    /**\n     * The full name of the target group.\n     */\n    targetGroupFullName: string\n    /**\n     * The full name of the application load balancer.\n     */\n    loadBalancerFullName: string\n    /**\n     * Configuration for a composite alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    targetHealthAlarm?: {\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     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    tooMany5xxResponsesFromTargetsAlarm?: {\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 3\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 10\n       */\n      threshold?: number\n      description?: string\n    }\n    /**\n     * Configuration for an alarm.\n     *\n     * @default Configured with sane defaults.\n     */\n    targetResponseTimeAlarm?: {\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 5 minutes\n       */\n      period?: cdk.Duration\n      /**\n       * @default 1\n       */\n      evaluationPeriods?: number\n      /**\n       * @default 500 milliseconds\n       */\n      threshold?: cdk.Duration\n      description?: string\n    }\n  }): void {\n    const targetConnectionErrorAlarm = new cloudwatch.Metric({\n      metricName: \"TargetConnectionErrorCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Sum\",\n      period: props.targetHealthAlarm?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"ConnectionAlarm\", {\n      actionsEnabled: true,\n      alarmDescription: `Load balancer is failing to connect to target(s) in ECS service '${this.serviceName}'.`,\n      evaluationPeriods: props.targetHealthAlarm?.evaluationPeriods ?? 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n\n    const healthAlarm = new cloudwatch.Metric({\n      metricName: \"HealthyHostCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Minimum\",\n      period: props.targetHealthAlarm?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"HealthAlarm\", {\n      alarmDescription: `There are no healthy target(s) in ECS service '${this.serviceName}'.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: props.targetHealthAlarm?.evaluationPeriods ?? 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.BREACHING,\n    })\n\n    const targetHealthAlarm = new cloudwatch.CompositeAlarm(\n      this,\n      \"TargetHealthAlarm\",\n      {\n        alarmRule: cdk.aws_cloudwatch.AlarmRule.anyOf(\n          cdk.aws_cloudwatch.AlarmRule.fromAlarm(\n            targetConnectionErrorAlarm,\n            cloudwatch.AlarmState.ALARM,\n          ),\n          cdk.aws_cloudwatch.AlarmRule.fromAlarm(\n            healthAlarm,\n            cloudwatch.AlarmState.ALARM,\n          ),\n        ),\n        alarmDescription:\n          props.targetHealthAlarm?.description ??\n          `The load balancer is either receiving bad health checks from or is unable to connect to target(s) in ECS service '${this.serviceName}'`,\n        actionsEnabled: false,\n      },\n    )\n    if (props.targetHealthAlarm?.enabled ?? true) {\n      targetHealthAlarm.addAlarmAction(\n        props.targetHealthAlarm?.action ?? this.action,\n      )\n      targetHealthAlarm.addOkAction(\n        props.targetHealthAlarm?.action ?? this.action,\n      )\n    }\n\n    const tooMany5xxResponsesFromTargetsAlarm = new cloudwatch.Metric({\n      metricName: \"HTTPCode_Target_5XX_Count\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Sum\",\n      period:\n        props.tooMany5xxResponsesFromTargetsAlarm?.period ??\n        cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"AlbTargets5xxAlarm\", {\n      actionsEnabled: true,\n      alarmDescription:\n        props.tooMany5xxResponsesFromTargetsAlarm?.description ??\n        `Load balancer received too many 5XX responses from target(s) in ECS service '${this.serviceName}'.`,\n      evaluationPeriods:\n        props.tooMany5xxResponsesFromTargetsAlarm?.evaluationPeriods ?? 3,\n      threshold: props.tooMany5xxResponsesFromTargetsAlarm?.threshold ?? 10,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n    if (props.tooMany5xxResponsesFromTargetsAlarm?.enabled ?? true) {\n      tooMany5xxResponsesFromTargetsAlarm.addAlarmAction(\n        props.tooMany5xxResponsesFromTargetsAlarm?.action ?? this.action,\n      )\n      tooMany5xxResponsesFromTargetsAlarm.addOkAction(\n        props.tooMany5xxResponsesFromTargetsAlarm?.action ?? this.action,\n      )\n    }\n\n    const targetResponseTimeAlarm = new cloudwatch.Metric({\n      metricName: \"TargetResponseTime\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"p95\",\n      period: props.targetResponseTimeAlarm?.period ?? cdk.Duration.minutes(5),\n      dimensionsMap: {\n        LoadBalancer: props.loadBalancerFullName,\n        TargetGroup: props.targetGroupFullName,\n      },\n    }).createAlarm(this, \"TargetResponseTimeAlarm\", {\n      alarmDescription:\n        props.targetResponseTimeAlarm?.description ??\n        `5% of responses from ECS service '${\n          this.serviceName\n        }' are taking longer than the expected duration of ${(\n          props.targetResponseTimeAlarm?.threshold ?? cdk.Duration.millis(500)\n        ).toMilliseconds()} ms.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      evaluationPeriods: props.targetResponseTimeAlarm?.evaluationPeriods ?? 1,\n      threshold: (\n        props.targetResponseTimeAlarm?.threshold ?? cdk.Duration.millis(500)\n      ).toSeconds({ integral: false }),\n      treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n    })\n    if (props.targetResponseTimeAlarm?.enabled ?? true) {\n      targetResponseTimeAlarm.addAlarmAction(\n        props.targetResponseTimeAlarm?.action ?? this.action,\n      )\n      targetResponseTimeAlarm.addOkAction(\n        props.targetResponseTimeAlarm?.action ?? this.action,\n      )\n    }\n  }\n}\n"]}
@@ -46,4 +46,4 @@ class SesAlarms extends constructs.Construct {
46
46
  }
47
47
  }
48
48
  exports.SesAlarms = SesAlarms;
49
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VzLWFsYXJtcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hbGFybXMvc2VzLWFsYXJtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxtQ0FBa0M7QUFDbEMseURBQXdEO0FBQ3hELHlDQUF3QztBQTJEeEM7OztHQUdHO0FBQ0gsTUFBYSxTQUFVLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFHakQsWUFBWSxLQUEyQixFQUFFLEVBQVUsRUFBRSxLQUFxQjs7UUFDeEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUVoQixJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQyxNQUFNLENBQUE7UUFFMUIsTUFBTSxvQkFBb0IsR0FBRyxJQUFJLFVBQVUsQ0FBQyxNQUFNLENBQUM7WUFDakQsVUFBVSxFQUFFLHVCQUF1QjtZQUNuQyxTQUFTLEVBQUUsU0FBUztZQUNwQixTQUFTLEVBQUUsU0FBUztZQUNwQixNQUFNLEVBQUUsTUFBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxvQkFBb0IsMENBQUUsTUFBTSxtQ0FBSSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7U0FDeEUsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsc0JBQXNCLEVBQUU7WUFDM0MsZ0JBQWdCLEVBQUUsK0JBQ2hCLE1BQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsb0JBQW9CLDBDQUFFLFNBQVMsbUNBQUksR0FDNUMsR0FBRztZQUNILGtCQUFrQixFQUFFLFVBQVUsQ0FBQyxrQkFBa0IsQ0FBQyxzQkFBc0I7WUFDeEUsaUJBQWlCLEVBQUUsQ0FBQztZQUNwQixnQkFBZ0IsRUFBRSxVQUFVLENBQUMsZ0JBQWdCLENBQUMsTUFBTTtZQUNwRCxTQUFTLEVBQUUsQ0FBQyxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLG9CQUFvQiwwQ0FBRSxTQUFTLG1DQUFJLEdBQUcsQ0FBQyxHQUFHLEdBQUc7U0FDakUsQ0FBQyxDQUFBO1FBRUYsSUFBSSxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLG9CQUFvQiwwQ0FBRSxPQUFPLG1DQUFJLElBQUksRUFBRTtZQUNoRCxvQkFBb0IsQ0FBQyxjQUFjLENBQ2pDLENBQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsb0JBQW9CLDBDQUFFLE1BQU0sS0FBSSxJQUFJLENBQUMsTUFBTSxDQUNuRCxDQUFBO1NBQ0Y7UUFFRCxNQUFNLHNCQUFzQixHQUFHLElBQUksVUFBVSxDQUFDLE1BQU0sQ0FBQztZQUNuRCxVQUFVLEVBQUUsMEJBQTBCO1lBQ3RDLFNBQVMsRUFBRSxTQUFTO1lBQ3BCLFNBQVMsRUFBRSxTQUFTO1lBQ3BCLE1BQU0sRUFBRSxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLGtCQUFrQiwwQ0FBRSxNQUFNLG1DQUFJLEdBQUcsQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztTQUN0RSxDQUFDLENBQUMsV0FBVyxDQUFDLElBQUksRUFBRSx3QkFBd0IsRUFBRTtZQUM3QyxnQkFBZ0IsRUFBRSxrQ0FDaEIsTUFBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxrQkFBa0IsMENBQUUsU0FBUyxtQ0FBSSxJQUMxQyxHQUFHO1lBQ0gsa0JBQWtCLEVBQUUsVUFBVSxDQUFDLGtCQUFrQixDQUFDLHNCQUFzQjtZQUN4RSxpQkFBaUIsRUFBRSxDQUFDO1lBQ3BCLFNBQVMsRUFBRSxDQUFDLE1BQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsa0JBQWtCLDBDQUFFLFNBQVMsbUNBQUksSUFBSSxDQUFDLEdBQUcsR0FBRztZQUMvRCxnQkFBZ0IsRUFBRSxVQUFVLENBQUMsZ0JBQWdCLENBQUMsTUFBTTtTQUNyRCxDQUFDLENBQUE7UUFFRixJQUFJLE1BQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsa0JBQWtCLDBDQUFFLE9BQU8sbUNBQUksSUFBSSxFQUFFO1lBQzlDLHNCQUFzQixDQUFDLGNBQWMsQ0FDbkMsQ0FBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxrQkFBa0IsMENBQUUsTUFBTSxLQUFJLElBQUksQ0FBQyxNQUFNLENBQ2pELENBQUE7U0FDRjtJQUNILENBQUM7Q0FDRjtBQWxERCw4QkFrREMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjZGsgZnJvbSBcImF3cy1jZGstbGliXCJcbmltcG9ydCAqIGFzIGNsb3Vkd2F0Y2ggZnJvbSBcImF3cy1jZGstbGliL2F3cy1jbG91ZHdhdGNoXCJcbmltcG9ydCAqIGFzIGNvbnN0cnVjdHMgZnJvbSBcImNvbnN0cnVjdHNcIlxuXG5leHBvcnQgaW50ZXJmYWNlIFNlc0FsYXJtc1Byb3BzIGV4dGVuZHMgY2RrLlN0YWNrUHJvcHMge1xuICAvKipcbiAgICogVGhlIGRlZmF1bHQgYWN0aW9uIHRvIHVzZSBmb3IgQ2xvdWRXYXRjaCBhbGFybSBzdGF0ZSBjaGFuZ2VzXG4gICAqL1xuICBhY3Rpb246IGNsb3Vkd2F0Y2guSUFsYXJtQWN0aW9uXG4gIC8qKlxuICAgKiBDb25maWd1cmF0aW9uIGZvciBhbiBhbGFybSBmb3IgaGlnaCByYXRlIGJvdW5jZWQgbWVzc2FnZXMuXG4gICAqXG4gICAqIEBkZWZhdWx0IENvbmZpZ3VyZWQgd2l0aCByZWFzb25hYmxlIGRlZmF1bHRzLlxuICAgKi9cbiAgYm91bmNlZE1lc3NhZ2VzQWxhcm0/OiB7XG4gICAgLyoqXG4gICAgICogQGRlZmF1bHQgdHJ1ZVxuICAgICAqL1xuICAgIGVuYWJsZWQ/OiBib29sZWFuXG4gICAgLyoqXG4gICAgICogQW4gYWN0aW9uIHRvIHVzZSBmb3IgQ2xvdWRXYXRjaCBhbGFybSBzdGF0ZSBjaGFuZ2VzIGluc3RlYWQgb2YgdGhlIGRlZmF1bHQgYWN0aW9uXG4gICAgICovXG4gICAgYWN0aW9uPzogY2xvdWR3YXRjaC5JQWxhcm1BY3Rpb25cbiAgICAvKipcbiAgICAgKiBAZGVmYXVsdCAxMCBtaW51dGVzXG4gICAgICovXG4gICAgcGVyaW9kPzogY2RrLkR1cmF0aW9uXG4gICAgLyoqXG4gICAgICogVGhyZXNob2xkIHZhbHVlIGZvciBhbGFybSBhcyBhIHBlcmNlbnRcbiAgICAgKiBAZGVmYXVsdCAyLjUoJSlcbiAgICAgKiA1JSBpcyB0aGUgdGhyZXNob2xkIGF0IHdoaWNoIEFXUyBjb25zaWRlcnMgcHV0dGluZyBhbiBhY2NvdW50IHVuZGVyIHJldmlld1xuICAgICAqL1xuICAgIHRocmVzaG9sZD86IG51bWJlclxuICB9XG4gIC8qKlxuICAgKiBDb25maWd1cmF0aW9uIGZvciBhbiBhbGFybSBmb3IgaGlnaCBjb21wbGFpbnQgcmF0ZS5cbiAgICpcbiAgICogQGRlZmF1bHQgQ29uZmlndXJlZCB3aXRoIHNhbmUgZGVmYXVsdHMuXG4gICAqL1xuICBjb21wbGFpbnRSYXRlQWxhcm0/OiB7XG4gICAgLyoqXG4gICAgICogQGRlZmF1bHQgdHJ1ZVxuICAgICAqL1xuICAgIGVuYWJsZWQ/OiBib29sZWFuXG4gICAgLyoqXG4gICAgICogQW4gYWN0aW9uIHRvIHVzZSBmb3IgQ2xvdWRXYXRjaCBhbGFybSBzdGF0ZSBjaGFuZ2VzIGluc3RlYWQgb2YgdGhlIGRlZmF1bHQgYWN0aW9uXG4gICAgICovXG4gICAgYWN0aW9uPzogY2xvdWR3YXRjaC5JQWxhcm1BY3Rpb25cbiAgICAvKipcbiAgICAgKiBAZGVmYXVsdCAxMCBtaW51dGVzXG4gICAgICovXG4gICAgcGVyaW9kPzogY2RrLkR1cmF0aW9uXG4gICAgLyoqXG4gICAgICogVGhyZXNob2xkIHZhbHVlIGZvciBhbGFybSBhcyBhIHBlcmNlbnRcbiAgICAgKiBAZGVmYXVsdCAwLjA1KCUpXG4gICAgICogMC4xMCUgaXMgdGhlIHRocmVzaG9sZCBhdCB3aGljaCBBV1MgY29uc2lkZXJzIHB1dHRpbmcgYW4gYWNjb3VudCB1bmRlciByZXZpZXdcbiAgICAgKi9cbiAgICB0aHJlc2hvbGQ/OiBudW1iZXJcbiAgfVxufVxuXG4vKipcbiAqXG4gKiBDb25zdHJ1Y3QgdGhhdCBjb25maWd1cmVzIHZhcmlvdXMgc2Vuc2libGUgQ2xvdWRXYXRjaCBhbGFybXMgZm9yIEFXUyBTRVNcbiAqL1xuZXhwb3J0IGNsYXNzIFNlc0FsYXJtcyBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgcHJpdmF0ZSByZWFkb25seSBhY3Rpb246IGNsb3Vkd2F0Y2guSUFsYXJtQWN0aW9uXG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogU2VzQWxhcm1zUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmFjdGlvbiA9IHByb3BzLmFjdGlvblxuXG4gICAgY29uc3QgYm91bmNlZE1lc3NhZ2VzQWxhcm0gPSBuZXcgY2xvdWR3YXRjaC5NZXRyaWMoe1xuICAgICAgbWV0cmljTmFtZTogXCJSZXB1dGF0aW9uLkJvdW5jZVJhdGVcIixcbiAgICAgIG5hbWVzcGFjZTogXCJBV1MvU0VTXCIsXG4gICAgICBzdGF0aXN0aWM6IFwiTWF4aW11bVwiLFxuICAgICAgcGVyaW9kOiBwcm9wcz8uYm91bmNlZE1lc3NhZ2VzQWxhcm0/LnBlcmlvZCA/PyBjZGsuRHVyYXRpb24ubWludXRlcygxMCksXG4gICAgfSkuY3JlYXRlQWxhcm0odGhpcywgXCJCb3VuY2VkTWVzc2FnZXNBbGFybVwiLCB7XG4gICAgICBhbGFybURlc2NyaXB0aW9uOiBgVGhlIFNFUyBib3VuY2UgcmF0ZSBpcyBvdmVyICR7XG4gICAgICAgIHByb3BzPy5ib3VuY2VkTWVzc2FnZXNBbGFybT8udGhyZXNob2xkID8/IDIuNVxuICAgICAgfSVgLFxuICAgICAgY29tcGFyaXNvbk9wZXJhdG9yOiBjbG91ZHdhdGNoLkNvbXBhcmlzb25PcGVyYXRvci5HUkVBVEVSX1RIQU5fVEhSRVNIT0xELFxuICAgICAgZXZhbHVhdGlvblBlcmlvZHM6IDEsXG4gICAgICB0cmVhdE1pc3NpbmdEYXRhOiBjbG91ZHdhdGNoLlRyZWF0TWlzc2luZ0RhdGEuSUdOT1JFLFxuICAgICAgdGhyZXNob2xkOiAocHJvcHM/LmJvdW5jZWRNZXNzYWdlc0FsYXJtPy50aHJlc2hvbGQgPz8gMi41KSAvIDEwMCxcbiAgICB9KVxuXG4gICAgaWYgKHByb3BzPy5ib3VuY2VkTWVzc2FnZXNBbGFybT8uZW5hYmxlZCA/PyB0cnVlKSB7XG4gICAgICBib3VuY2VkTWVzc2FnZXNBbGFybS5hZGRBbGFybUFjdGlvbihcbiAgICAgICAgcHJvcHM/LmJvdW5jZWRNZXNzYWdlc0FsYXJtPy5hY3Rpb24gfHwgdGhpcy5hY3Rpb24sXG4gICAgICApXG4gICAgfVxuXG4gICAgY29uc3QgY29tcGxhaW50TWVzc2FnZXNBbGFybSA9IG5ldyBjbG91ZHdhdGNoLk1ldHJpYyh7XG4gICAgICBtZXRyaWNOYW1lOiBcIlJlcHV0YXRpb24uQ29tcGxhaW50UmF0ZVwiLFxuICAgICAgbmFtZXNwYWNlOiBcIkFXUy9TRVNcIixcbiAgICAgIHN0YXRpc3RpYzogXCJNYXhpbXVtXCIsXG4gICAgICBwZXJpb2Q6IHByb3BzPy5jb21wbGFpbnRSYXRlQWxhcm0/LnBlcmlvZCA/PyBjZGsuRHVyYXRpb24ubWludXRlcygxMCksXG4gICAgfSkuY3JlYXRlQWxhcm0odGhpcywgXCJDb21wbGFpbnRNZXNzYWdlc0FsYXJtXCIsIHtcbiAgICAgIGFsYXJtRGVzY3JpcHRpb246IGBUaGUgU0VTIGNvbXBsYWludCByYXRlIGlzIG92ZXIgJHtcbiAgICAgICAgcHJvcHM/LmNvbXBsYWludFJhdGVBbGFybT8udGhyZXNob2xkID8/IDAuMDVcbiAgICAgIH0lYCxcbiAgICAgIGNvbXBhcmlzb25PcGVyYXRvcjogY2xvdWR3YXRjaC5Db21wYXJpc29uT3BlcmF0b3IuR1JFQVRFUl9USEFOX1RIUkVTSE9MRCxcbiAgICAgIGV2YWx1YXRpb25QZXJpb2RzOiAxLFxuICAgICAgdGhyZXNob2xkOiAocHJvcHM/LmNvbXBsYWludFJhdGVBbGFybT8udGhyZXNob2xkID8/IDAuMDUpIC8gMTAwLFxuICAgICAgdHJlYXRNaXNzaW5nRGF0YTogY2xvdWR3YXRjaC5UcmVhdE1pc3NpbmdEYXRhLklHTk9SRSxcbiAgICB9KVxuXG4gICAgaWYgKHByb3BzPy5jb21wbGFpbnRSYXRlQWxhcm0/LmVuYWJsZWQgPz8gdHJ1ZSkge1xuICAgICAgY29tcGxhaW50TWVzc2FnZXNBbGFybS5hZGRBbGFybUFjdGlvbihcbiAgICAgICAgcHJvcHM/LmNvbXBsYWludFJhdGVBbGFybT8uYWN0aW9uIHx8IHRoaXMuYWN0aW9uLFxuICAgICAgKVxuICAgIH1cbiAgfVxufVxuIl19
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VzLWFsYXJtcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hbGFybXMvc2VzLWFsYXJtcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxtQ0FBa0M7QUFDbEMseURBQXdEO0FBQ3hELHlDQUF3QztBQTJEeEM7OztHQUdHO0FBQ0gsTUFBYSxTQUFVLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFHakQsWUFBWSxLQUEyQixFQUFFLEVBQVUsRUFBRSxLQUFxQjs7UUFDeEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUVoQixJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQyxNQUFNLENBQUE7UUFFMUIsTUFBTSxvQkFBb0IsR0FBRyxJQUFJLFVBQVUsQ0FBQyxNQUFNLENBQUM7WUFDakQsVUFBVSxFQUFFLHVCQUF1QjtZQUNuQyxTQUFTLEVBQUUsU0FBUztZQUNwQixTQUFTLEVBQUUsU0FBUztZQUNwQixNQUFNLEVBQUUsTUFBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxvQkFBb0IsMENBQUUsTUFBTSxtQ0FBSSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7U0FDeEUsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsc0JBQXNCLEVBQUU7WUFDM0MsZ0JBQWdCLEVBQUUsK0JBQ2hCLE1BQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsb0JBQW9CLDBDQUFFLFNBQVMsbUNBQUksR0FDNUMsR0FBRztZQUNILGtCQUFrQixFQUFFLFVBQVUsQ0FBQyxrQkFBa0IsQ0FBQyxzQkFBc0I7WUFDeEUsaUJBQWlCLEVBQUUsQ0FBQztZQUNwQixnQkFBZ0IsRUFBRSxVQUFVLENBQUMsZ0JBQWdCLENBQUMsTUFBTTtZQUNwRCxTQUFTLEVBQUUsQ0FBQyxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLG9CQUFvQiwwQ0FBRSxTQUFTLG1DQUFJLEdBQUcsQ0FBQyxHQUFHLEdBQUc7U0FDakUsQ0FBQyxDQUFBO1FBRUYsSUFBSSxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLG9CQUFvQiwwQ0FBRSxPQUFPLG1DQUFJLElBQUksRUFBRSxDQUFDO1lBQ2pELG9CQUFvQixDQUFDLGNBQWMsQ0FDakMsQ0FBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxvQkFBb0IsMENBQUUsTUFBTSxLQUFJLElBQUksQ0FBQyxNQUFNLENBQ25ELENBQUE7UUFDSCxDQUFDO1FBRUQsTUFBTSxzQkFBc0IsR0FBRyxJQUFJLFVBQVUsQ0FBQyxNQUFNLENBQUM7WUFDbkQsVUFBVSxFQUFFLDBCQUEwQjtZQUN0QyxTQUFTLEVBQUUsU0FBUztZQUNwQixTQUFTLEVBQUUsU0FBUztZQUNwQixNQUFNLEVBQUUsTUFBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxrQkFBa0IsMENBQUUsTUFBTSxtQ0FBSSxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7U0FDdEUsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsd0JBQXdCLEVBQUU7WUFDN0MsZ0JBQWdCLEVBQUUsa0NBQ2hCLE1BQUEsTUFBQSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsa0JBQWtCLDBDQUFFLFNBQVMsbUNBQUksSUFDMUMsR0FBRztZQUNILGtCQUFrQixFQUFFLFVBQVUsQ0FBQyxrQkFBa0IsQ0FBQyxzQkFBc0I7WUFDeEUsaUJBQWlCLEVBQUUsQ0FBQztZQUNwQixTQUFTLEVBQUUsQ0FBQyxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLGtCQUFrQiwwQ0FBRSxTQUFTLG1DQUFJLElBQUksQ0FBQyxHQUFHLEdBQUc7WUFDL0QsZ0JBQWdCLEVBQUUsVUFBVSxDQUFDLGdCQUFnQixDQUFDLE1BQU07U0FDckQsQ0FBQyxDQUFBO1FBRUYsSUFBSSxNQUFBLE1BQUEsS0FBSyxhQUFMLEtBQUssdUJBQUwsS0FBSyxDQUFFLGtCQUFrQiwwQ0FBRSxPQUFPLG1DQUFJLElBQUksRUFBRSxDQUFDO1lBQy9DLHNCQUFzQixDQUFDLGNBQWMsQ0FDbkMsQ0FBQSxNQUFBLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxrQkFBa0IsMENBQUUsTUFBTSxLQUFJLElBQUksQ0FBQyxNQUFNLENBQ2pELENBQUE7UUFDSCxDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBbERELDhCQWtEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGNkayBmcm9tIFwiYXdzLWNkay1saWJcIlxuaW1wb3J0ICogYXMgY2xvdWR3YXRjaCBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2hcIlxuaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5cbmV4cG9ydCBpbnRlcmZhY2UgU2VzQWxhcm1zUHJvcHMgZXh0ZW5kcyBjZGsuU3RhY2tQcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgZGVmYXVsdCBhY3Rpb24gdG8gdXNlIGZvciBDbG91ZFdhdGNoIGFsYXJtIHN0YXRlIGNoYW5nZXNcbiAgICovXG4gIGFjdGlvbjogY2xvdWR3YXRjaC5JQWxhcm1BY3Rpb25cbiAgLyoqXG4gICAqIENvbmZpZ3VyYXRpb24gZm9yIGFuIGFsYXJtIGZvciBoaWdoIHJhdGUgYm91bmNlZCBtZXNzYWdlcy5cbiAgICpcbiAgICogQGRlZmF1bHQgQ29uZmlndXJlZCB3aXRoIHJlYXNvbmFibGUgZGVmYXVsdHMuXG4gICAqL1xuICBib3VuY2VkTWVzc2FnZXNBbGFybT86IHtcbiAgICAvKipcbiAgICAgKiBAZGVmYXVsdCB0cnVlXG4gICAgICovXG4gICAgZW5hYmxlZD86IGJvb2xlYW5cbiAgICAvKipcbiAgICAgKiBBbiBhY3Rpb24gdG8gdXNlIGZvciBDbG91ZFdhdGNoIGFsYXJtIHN0YXRlIGNoYW5nZXMgaW5zdGVhZCBvZiB0aGUgZGVmYXVsdCBhY3Rpb25cbiAgICAgKi9cbiAgICBhY3Rpb24/OiBjbG91ZHdhdGNoLklBbGFybUFjdGlvblxuICAgIC8qKlxuICAgICAqIEBkZWZhdWx0IDEwIG1pbnV0ZXNcbiAgICAgKi9cbiAgICBwZXJpb2Q/OiBjZGsuRHVyYXRpb25cbiAgICAvKipcbiAgICAgKiBUaHJlc2hvbGQgdmFsdWUgZm9yIGFsYXJtIGFzIGEgcGVyY2VudFxuICAgICAqIEBkZWZhdWx0IDIuNSglKVxuICAgICAqIDUlIGlzIHRoZSB0aHJlc2hvbGQgYXQgd2hpY2ggQVdTIGNvbnNpZGVycyBwdXR0aW5nIGFuIGFjY291bnQgdW5kZXIgcmV2aWV3XG4gICAgICovXG4gICAgdGhyZXNob2xkPzogbnVtYmVyXG4gIH1cbiAgLyoqXG4gICAqIENvbmZpZ3VyYXRpb24gZm9yIGFuIGFsYXJtIGZvciBoaWdoIGNvbXBsYWludCByYXRlLlxuICAgKlxuICAgKiBAZGVmYXVsdCBDb25maWd1cmVkIHdpdGggc2FuZSBkZWZhdWx0cy5cbiAgICovXG4gIGNvbXBsYWludFJhdGVBbGFybT86IHtcbiAgICAvKipcbiAgICAgKiBAZGVmYXVsdCB0cnVlXG4gICAgICovXG4gICAgZW5hYmxlZD86IGJvb2xlYW5cbiAgICAvKipcbiAgICAgKiBBbiBhY3Rpb24gdG8gdXNlIGZvciBDbG91ZFdhdGNoIGFsYXJtIHN0YXRlIGNoYW5nZXMgaW5zdGVhZCBvZiB0aGUgZGVmYXVsdCBhY3Rpb25cbiAgICAgKi9cbiAgICBhY3Rpb24/OiBjbG91ZHdhdGNoLklBbGFybUFjdGlvblxuICAgIC8qKlxuICAgICAqIEBkZWZhdWx0IDEwIG1pbnV0ZXNcbiAgICAgKi9cbiAgICBwZXJpb2Q/OiBjZGsuRHVyYXRpb25cbiAgICAvKipcbiAgICAgKiBUaHJlc2hvbGQgdmFsdWUgZm9yIGFsYXJtIGFzIGEgcGVyY2VudFxuICAgICAqIEBkZWZhdWx0IDAuMDUoJSlcbiAgICAgKiAwLjEwJSBpcyB0aGUgdGhyZXNob2xkIGF0IHdoaWNoIEFXUyBjb25zaWRlcnMgcHV0dGluZyBhbiBhY2NvdW50IHVuZGVyIHJldmlld1xuICAgICAqL1xuICAgIHRocmVzaG9sZD86IG51bWJlclxuICB9XG59XG5cbi8qKlxuICpcbiAqIENvbnN0cnVjdCB0aGF0IGNvbmZpZ3VyZXMgdmFyaW91cyBzZW5zaWJsZSBDbG91ZFdhdGNoIGFsYXJtcyBmb3IgQVdTIFNFU1xuICovXG5leHBvcnQgY2xhc3MgU2VzQWxhcm1zIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwcml2YXRlIHJlYWRvbmx5IGFjdGlvbjogY2xvdWR3YXRjaC5JQWxhcm1BY3Rpb25cblxuICBjb25zdHJ1Y3RvcihzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBTZXNBbGFybXNQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuYWN0aW9uID0gcHJvcHMuYWN0aW9uXG5cbiAgICBjb25zdCBib3VuY2VkTWVzc2FnZXNBbGFybSA9IG5ldyBjbG91ZHdhdGNoLk1ldHJpYyh7XG4gICAgICBtZXRyaWNOYW1lOiBcIlJlcHV0YXRpb24uQm91bmNlUmF0ZVwiLFxuICAgICAgbmFtZXNwYWNlOiBcIkFXUy9TRVNcIixcbiAgICAgIHN0YXRpc3RpYzogXCJNYXhpbXVtXCIsXG4gICAgICBwZXJpb2Q6IHByb3BzPy5ib3VuY2VkTWVzc2FnZXNBbGFybT8ucGVyaW9kID8/IGNkay5EdXJhdGlvbi5taW51dGVzKDEwKSxcbiAgICB9KS5jcmVhdGVBbGFybSh0aGlzLCBcIkJvdW5jZWRNZXNzYWdlc0FsYXJtXCIsIHtcbiAgICAgIGFsYXJtRGVzY3JpcHRpb246IGBUaGUgU0VTIGJvdW5jZSByYXRlIGlzIG92ZXIgJHtcbiAgICAgICAgcHJvcHM/LmJvdW5jZWRNZXNzYWdlc0FsYXJtPy50aHJlc2hvbGQgPz8gMi41XG4gICAgICB9JWAsXG4gICAgICBjb21wYXJpc29uT3BlcmF0b3I6IGNsb3Vkd2F0Y2guQ29tcGFyaXNvbk9wZXJhdG9yLkdSRUFURVJfVEhBTl9USFJFU0hPTEQsXG4gICAgICBldmFsdWF0aW9uUGVyaW9kczogMSxcbiAgICAgIHRyZWF0TWlzc2luZ0RhdGE6IGNsb3Vkd2F0Y2guVHJlYXRNaXNzaW5nRGF0YS5JR05PUkUsXG4gICAgICB0aHJlc2hvbGQ6IChwcm9wcz8uYm91bmNlZE1lc3NhZ2VzQWxhcm0/LnRocmVzaG9sZCA/PyAyLjUpIC8gMTAwLFxuICAgIH0pXG5cbiAgICBpZiAocHJvcHM/LmJvdW5jZWRNZXNzYWdlc0FsYXJtPy5lbmFibGVkID8/IHRydWUpIHtcbiAgICAgIGJvdW5jZWRNZXNzYWdlc0FsYXJtLmFkZEFsYXJtQWN0aW9uKFxuICAgICAgICBwcm9wcz8uYm91bmNlZE1lc3NhZ2VzQWxhcm0/LmFjdGlvbiB8fCB0aGlzLmFjdGlvbixcbiAgICAgIClcbiAgICB9XG5cbiAgICBjb25zdCBjb21wbGFpbnRNZXNzYWdlc0FsYXJtID0gbmV3IGNsb3Vkd2F0Y2guTWV0cmljKHtcbiAgICAgIG1ldHJpY05hbWU6IFwiUmVwdXRhdGlvbi5Db21wbGFpbnRSYXRlXCIsXG4gICAgICBuYW1lc3BhY2U6IFwiQVdTL1NFU1wiLFxuICAgICAgc3RhdGlzdGljOiBcIk1heGltdW1cIixcbiAgICAgIHBlcmlvZDogcHJvcHM/LmNvbXBsYWludFJhdGVBbGFybT8ucGVyaW9kID8/IGNkay5EdXJhdGlvbi5taW51dGVzKDEwKSxcbiAgICB9KS5jcmVhdGVBbGFybSh0aGlzLCBcIkNvbXBsYWludE1lc3NhZ2VzQWxhcm1cIiwge1xuICAgICAgYWxhcm1EZXNjcmlwdGlvbjogYFRoZSBTRVMgY29tcGxhaW50IHJhdGUgaXMgb3ZlciAke1xuICAgICAgICBwcm9wcz8uY29tcGxhaW50UmF0ZUFsYXJtPy50aHJlc2hvbGQgPz8gMC4wNVxuICAgICAgfSVgLFxuICAgICAgY29tcGFyaXNvbk9wZXJhdG9yOiBjbG91ZHdhdGNoLkNvbXBhcmlzb25PcGVyYXRvci5HUkVBVEVSX1RIQU5fVEhSRVNIT0xELFxuICAgICAgZXZhbHVhdGlvblBlcmlvZHM6IDEsXG4gICAgICB0aHJlc2hvbGQ6IChwcm9wcz8uY29tcGxhaW50UmF0ZUFsYXJtPy50aHJlc2hvbGQgPz8gMC4wNSkgLyAxMDAsXG4gICAgICB0cmVhdE1pc3NpbmdEYXRhOiBjbG91ZHdhdGNoLlRyZWF0TWlzc2luZ0RhdGEuSUdOT1JFLFxuICAgIH0pXG5cbiAgICBpZiAocHJvcHM/LmNvbXBsYWludFJhdGVBbGFybT8uZW5hYmxlZCA/PyB0cnVlKSB7XG4gICAgICBjb21wbGFpbnRNZXNzYWdlc0FsYXJtLmFkZEFsYXJtQWN0aW9uKFxuICAgICAgICBwcm9wcz8uY29tcGxhaW50UmF0ZUFsYXJtPy5hY3Rpb24gfHwgdGhpcy5hY3Rpb24sXG4gICAgICApXG4gICAgfVxuICB9XG59XG4iXX0=
@@ -1,11 +1,18 @@
1
1
  import * as constructs from "constructs";
2
2
  import * as cloudwatchActions from "aws-cdk-lib/aws-cloudwatch-actions";
3
3
  import * as sns from "aws-cdk-lib/aws-sns";
4
+ import * as secretsmanager from "aws-cdk-lib/aws-secretsmanager";
4
5
  export interface SlackAlarmProps {
5
6
  projectName: string;
6
7
  envName: string;
7
- slackChannel: string;
8
- slackUrl: string;
8
+ /**
9
+ * A plaintext secret containing the URL of a Slack incoming webhook.
10
+ * The webhook should be created through a Slack app, and only allows posting to one specific Slack channel.
11
+ * See Slack's official documentation (e.g., https://api.slack.com/messaging/webhooks) for more details.
12
+ *
13
+ * NOTE: Incoming webhooks created through legacy custom integrations in Slack are not supported.
14
+ */
15
+ slackWebhookUrlSecret: secretsmanager.ISecret;
9
16
  }
10
17
  /**
11
18
  * SNS Topic that can be used to action alarms, with a Lambda
@@ -22,15 +22,15 @@ class SlackAlarm extends constructs.Construct {
22
22
  description: "Receives CloudWatch Alarms through SNS and sends a formatted version to Slack",
23
23
  handler: "index.handler",
24
24
  memorySize: 128,
25
- runtime: lambda.Runtime.PYTHON_3_8,
25
+ runtime: lambda.Runtime.PYTHON_3_11,
26
26
  timeout: aws_cdk_lib_1.Duration.seconds(6),
27
27
  environment: {
28
- SLACK_URL: props.slackUrl,
29
- SLACK_CHANNEL: props.slackChannel,
28
+ SLACK_URL_SECRET_NAME: props.slackWebhookUrlSecret.secretName,
30
29
  PROJECT_NAME: props.projectName,
31
30
  ENVIRONMENT_NAME: props.envName,
32
31
  },
33
32
  });
33
+ props.slackWebhookUrlSecret.grantRead(slackLambda);
34
34
  slackLambda.addPermission("InvokePermission", {
35
35
  action: "lambda:InvokeFunction",
36
36
  principal: new iam.ServicePrincipal("sns.amazonaws.com"),
@@ -44,4 +44,4 @@ class SlackAlarm extends constructs.Construct {
44
44
  }
45
45
  }
46
46
  exports.SlackAlarm = SlackAlarm;
47
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stYWxhcm0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWxhcm1zL3NsYWNrLWFsYXJtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUN4Qyx3RUFBdUU7QUFDdkUsMkNBQTBDO0FBQzFDLGlEQUFnRDtBQUNoRCwyQ0FBMEM7QUFDMUMsNkNBQXNDO0FBQ3RDLDZCQUE0QjtBQVM1Qjs7O0dBR0c7QUFDSCxNQUFhLFVBQVcsU0FBUSxVQUFVLENBQUMsU0FBUztJQUlsRCxZQUFZLEtBQTJCLEVBQUUsRUFBVSxFQUFFLEtBQXNCO1FBQ3pFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxJQUFJLEdBQUcsQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFBO1FBRTlDLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxpQkFBaUIsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFBO1FBRWpFLE1BQU0sV0FBVyxHQUFHLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3hELElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDekIsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsaUNBQWlDLENBQUMsQ0FDeEQ7WUFDRCxXQUFXLEVBQ1QsK0VBQStFO1lBQ2pGLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLFVBQVUsRUFBRSxHQUFHO1lBQ2YsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsVUFBVTtZQUNsQyxPQUFPLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1lBQzVCLFdBQVcsRUFBRTtnQkFDWCxTQUFTLEVBQUUsS0FBSyxDQUFDLFFBQVE7Z0JBQ3pCLGFBQWEsRUFBRSxLQUFLLENBQUMsWUFBWTtnQkFDakMsWUFBWSxFQUFFLEtBQUssQ0FBQyxXQUFXO2dCQUMvQixnQkFBZ0IsRUFBRSxLQUFLLENBQUMsT0FBTzthQUNoQztTQUNGLENBQUMsQ0FBQTtRQUVGLFdBQVcsQ0FBQyxhQUFhLENBQUMsa0JBQWtCLEVBQUU7WUFDNUMsTUFBTSxFQUFFLHVCQUF1QjtZQUMvQixTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsZ0JBQWdCLENBQUMsbUJBQW1CLENBQUM7WUFDeEQsU0FBUyxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUTtTQUNwQyxDQUFDLENBQUE7UUFFRixJQUFJLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxFQUFFLGNBQWMsRUFBRTtZQUN6QyxRQUFRLEVBQUUsV0FBVyxDQUFDLFdBQVc7WUFDakMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxNQUFNO1lBQ3pDLEtBQUssRUFBRSxJQUFJLENBQUMsVUFBVTtTQUN2QixDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0Y7QUF6Q0QsZ0NBeUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjbG91ZHdhdGNoQWN0aW9ucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gtYWN0aW9uc1wiXG5pbXBvcnQgKiBhcyBpYW0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIlxuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtbGFtYmRhXCJcbmltcG9ydCAqIGFzIHNucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNuc1wiXG5pbXBvcnQgeyBEdXJhdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCJcblxuZXhwb3J0IGludGVyZmFjZSBTbGFja0FsYXJtUHJvcHMge1xuICBwcm9qZWN0TmFtZTogc3RyaW5nXG4gIGVudk5hbWU6IHN0cmluZ1xuICBzbGFja0NoYW5uZWw6IHN0cmluZ1xuICBzbGFja1VybDogc3RyaW5nXG59XG5cbi8qKlxuICogU05TIFRvcGljIHRoYXQgY2FuIGJlIHVzZWQgdG8gYWN0aW9uIGFsYXJtcywgd2l0aCBhIExhbWJkYVxuICogdGhhdCB3aWxsIHNlbmQgYSBtZXNzYWdlIHRvIFNsYWNrIGZvciB0aGUgYWxhcm0uXG4gKi9cbmV4cG9ydCBjbGFzcyBTbGFja0FsYXJtIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgYWxhcm1Ub3BpYzogc25zLlRvcGljXG4gIHB1YmxpYyByZWFkb25seSBzbnNBY3Rpb246IGNsb3Vkd2F0Y2hBY3Rpb25zLlNuc0FjdGlvblxuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFNsYWNrQWxhcm1Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuYWxhcm1Ub3BpYyA9IG5ldyBzbnMuVG9waWModGhpcywgXCJUb3BpY1wiKVxuXG4gICAgdGhpcy5zbnNBY3Rpb24gPSBuZXcgY2xvdWR3YXRjaEFjdGlvbnMuU25zQWN0aW9uKHRoaXMuYWxhcm1Ub3BpYylcblxuICAgIGNvbnN0IHNsYWNrTGFtYmRhID0gbmV3IGxhbWJkYS5GdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChcbiAgICAgICAgcGF0aC5qb2luKF9fZGlybmFtZSwgXCIuLi8uLi9hc3NldHMvc2xhY2stYWxhcm0tbGFtYmRhXCIpLFxuICAgICAgKSxcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICBcIlJlY2VpdmVzIENsb3VkV2F0Y2ggQWxhcm1zIHRocm91Z2ggU05TIGFuZCBzZW5kcyBhIGZvcm1hdHRlZCB2ZXJzaW9uIHRvIFNsYWNrXCIsXG4gICAgICBoYW5kbGVyOiBcImluZGV4LmhhbmRsZXJcIixcbiAgICAgIG1lbW9yeVNpemU6IDEyOCxcbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLlBZVEhPTl8zXzgsXG4gICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDYpLFxuICAgICAgZW52aXJvbm1lbnQ6IHtcbiAgICAgICAgU0xBQ0tfVVJMOiBwcm9wcy5zbGFja1VybCxcbiAgICAgICAgU0xBQ0tfQ0hBTk5FTDogcHJvcHMuc2xhY2tDaGFubmVsLFxuICAgICAgICBQUk9KRUNUX05BTUU6IHByb3BzLnByb2plY3ROYW1lLFxuICAgICAgICBFTlZJUk9OTUVOVF9OQU1FOiBwcm9wcy5lbnZOYW1lLFxuICAgICAgfSxcbiAgICB9KVxuXG4gICAgc2xhY2tMYW1iZGEuYWRkUGVybWlzc2lvbihcIkludm9rZVBlcm1pc3Npb25cIiwge1xuICAgICAgYWN0aW9uOiBcImxhbWJkYTpJbnZva2VGdW5jdGlvblwiLFxuICAgICAgcHJpbmNpcGFsOiBuZXcgaWFtLlNlcnZpY2VQcmluY2lwYWwoXCJzbnMuYW1hem9uYXdzLmNvbVwiKSxcbiAgICAgIHNvdXJjZUFybjogdGhpcy5hbGFybVRvcGljLnRvcGljQXJuLFxuICAgIH0pXG5cbiAgICBuZXcgc25zLlN1YnNjcmlwdGlvbih0aGlzLCBcIlN1YnNjcmlwdGlvblwiLCB7XG4gICAgICBlbmRwb2ludDogc2xhY2tMYW1iZGEuZnVuY3Rpb25Bcm4sXG4gICAgICBwcm90b2NvbDogc25zLlN1YnNjcmlwdGlvblByb3RvY29sLkxBTUJEQSxcbiAgICAgIHRvcGljOiB0aGlzLmFsYXJtVG9waWMsXG4gICAgfSlcbiAgfVxufVxuIl19
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stYWxhcm0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWxhcm1zL3NsYWNrLWFsYXJtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUN4Qyx3RUFBdUU7QUFDdkUsMkNBQTBDO0FBQzFDLGlEQUFnRDtBQUNoRCwyQ0FBMEM7QUFDMUMsNkNBQXNDO0FBQ3RDLDZCQUE0QjtBQWdCNUI7OztHQUdHO0FBQ0gsTUFBYSxVQUFXLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFJbEQsWUFBWSxLQUEyQixFQUFFLEVBQVUsRUFBRSxLQUFzQjtRQUN6RSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxHQUFHLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQTtRQUU5QyxJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksaUJBQWlCLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQTtRQUVqRSxNQUFNLFdBQVcsR0FBRyxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUN4RCxJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQ3pCLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLGlDQUFpQyxDQUFDLENBQ3hEO1lBQ0QsV0FBVyxFQUNULCtFQUErRTtZQUNqRixPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsR0FBRztZQUNmLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztZQUM1QixXQUFXLEVBQUU7Z0JBQ1gscUJBQXFCLEVBQUUsS0FBSyxDQUFDLHFCQUFxQixDQUFDLFVBQVU7Z0JBQzdELFlBQVksRUFBRSxLQUFLLENBQUMsV0FBVztnQkFDL0IsZ0JBQWdCLEVBQUUsS0FBSyxDQUFDLE9BQU87YUFDaEM7U0FDRixDQUFDLENBQUE7UUFFRixLQUFLLENBQUMscUJBQXFCLENBQUMsU0FBUyxDQUFDLFdBQVcsQ0FBQyxDQUFBO1FBRWxELFdBQVcsQ0FBQyxhQUFhLENBQUMsa0JBQWtCLEVBQUU7WUFDNUMsTUFBTSxFQUFFLHVCQUF1QjtZQUMvQixTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsZ0JBQWdCLENBQUMsbUJBQW1CLENBQUM7WUFDeEQsU0FBUyxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUTtTQUNwQyxDQUFDLENBQUE7UUFFRixJQUFJLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxFQUFFLGNBQWMsRUFBRTtZQUN6QyxRQUFRLEVBQUUsV0FBVyxDQUFDLFdBQVc7WUFDakMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxNQUFNO1lBQ3pDLEtBQUssRUFBRSxJQUFJLENBQUMsVUFBVTtTQUN2QixDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0Y7QUExQ0QsZ0NBMENDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjbG91ZHdhdGNoQWN0aW9ucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gtYWN0aW9uc1wiXG5pbXBvcnQgKiBhcyBpYW0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIlxuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtbGFtYmRhXCJcbmltcG9ydCAqIGFzIHNucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNuc1wiXG5pbXBvcnQgeyBEdXJhdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCJcbmltcG9ydCAqIGFzIHNlY3JldHNtYW5hZ2VyIGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtc2VjcmV0c21hbmFnZXJcIlxuXG5leHBvcnQgaW50ZXJmYWNlIFNsYWNrQWxhcm1Qcm9wcyB7XG4gIHByb2plY3ROYW1lOiBzdHJpbmdcbiAgZW52TmFtZTogc3RyaW5nXG4gIC8qKlxuICAgKiBBIHBsYWludGV4dCBzZWNyZXQgY29udGFpbmluZyB0aGUgVVJMIG9mIGEgU2xhY2sgaW5jb21pbmcgd2ViaG9vay5cbiAgICogVGhlIHdlYmhvb2sgc2hvdWxkIGJlIGNyZWF0ZWQgdGhyb3VnaCBhIFNsYWNrIGFwcCwgYW5kIG9ubHkgYWxsb3dzIHBvc3RpbmcgdG8gb25lIHNwZWNpZmljIFNsYWNrIGNoYW5uZWwuXG4gICAqIFNlZSBTbGFjaydzIG9mZmljaWFsIGRvY3VtZW50YXRpb24gKGUuZy4sIGh0dHBzOi8vYXBpLnNsYWNrLmNvbS9tZXNzYWdpbmcvd2ViaG9va3MpIGZvciBtb3JlIGRldGFpbHMuXG4gICAqXG4gICAqIE5PVEU6IEluY29taW5nIHdlYmhvb2tzIGNyZWF0ZWQgdGhyb3VnaCBsZWdhY3kgY3VzdG9tIGludGVncmF0aW9ucyBpbiBTbGFjayBhcmUgbm90IHN1cHBvcnRlZC5cbiAgICovXG4gIHNsYWNrV2ViaG9va1VybFNlY3JldDogc2VjcmV0c21hbmFnZXIuSVNlY3JldFxufVxuXG4vKipcbiAqIFNOUyBUb3BpYyB0aGF0IGNhbiBiZSB1c2VkIHRvIGFjdGlvbiBhbGFybXMsIHdpdGggYSBMYW1iZGFcbiAqIHRoYXQgd2lsbCBzZW5kIGEgbWVzc2FnZSB0byBTbGFjayBmb3IgdGhlIGFsYXJtLlxuICovXG5leHBvcnQgY2xhc3MgU2xhY2tBbGFybSBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgcHVibGljIHJlYWRvbmx5IGFsYXJtVG9waWM6IHNucy5Ub3BpY1xuICBwdWJsaWMgcmVhZG9ubHkgc25zQWN0aW9uOiBjbG91ZHdhdGNoQWN0aW9ucy5TbnNBY3Rpb25cblxuICBjb25zdHJ1Y3RvcihzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBTbGFja0FsYXJtUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmFsYXJtVG9waWMgPSBuZXcgc25zLlRvcGljKHRoaXMsIFwiVG9waWNcIilcblxuICAgIHRoaXMuc25zQWN0aW9uID0gbmV3IGNsb3Vkd2F0Y2hBY3Rpb25zLlNuc0FjdGlvbih0aGlzLmFsYXJtVG9waWMpXG5cbiAgICBjb25zdCBzbGFja0xhbWJkYSA9IG5ldyBsYW1iZGEuRnVuY3Rpb24odGhpcywgXCJGdW5jdGlvblwiLCB7XG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQoXG4gICAgICAgIHBhdGguam9pbihfX2Rpcm5hbWUsIFwiLi4vLi4vYXNzZXRzL3NsYWNrLWFsYXJtLWxhbWJkYVwiKSxcbiAgICAgICksXG4gICAgICBkZXNjcmlwdGlvbjpcbiAgICAgICAgXCJSZWNlaXZlcyBDbG91ZFdhdGNoIEFsYXJtcyB0aHJvdWdoIFNOUyBhbmQgc2VuZHMgYSBmb3JtYXR0ZWQgdmVyc2lvbiB0byBTbGFja1wiLFxuICAgICAgaGFuZGxlcjogXCJpbmRleC5oYW5kbGVyXCIsXG4gICAgICBtZW1vcnlTaXplOiAxMjgsXG4gICAgICBydW50aW1lOiBsYW1iZGEuUnVudGltZS5QWVRIT05fM18xMSxcbiAgICAgIHRpbWVvdXQ6IER1cmF0aW9uLnNlY29uZHMoNiksXG4gICAgICBlbnZpcm9ubWVudDoge1xuICAgICAgICBTTEFDS19VUkxfU0VDUkVUX05BTUU6IHByb3BzLnNsYWNrV2ViaG9va1VybFNlY3JldC5zZWNyZXROYW1lLFxuICAgICAgICBQUk9KRUNUX05BTUU6IHByb3BzLnByb2plY3ROYW1lLFxuICAgICAgICBFTlZJUk9OTUVOVF9OQU1FOiBwcm9wcy5lbnZOYW1lLFxuICAgICAgfSxcbiAgICB9KVxuXG4gICAgcHJvcHMuc2xhY2tXZWJob29rVXJsU2VjcmV0LmdyYW50UmVhZChzbGFja0xhbWJkYSlcblxuICAgIHNsYWNrTGFtYmRhLmFkZFBlcm1pc3Npb24oXCJJbnZva2VQZXJtaXNzaW9uXCIsIHtcbiAgICAgIGFjdGlvbjogXCJsYW1iZGE6SW52b2tlRnVuY3Rpb25cIixcbiAgICAgIHByaW5jaXBhbDogbmV3IGlhbS5TZXJ2aWNlUHJpbmNpcGFsKFwic25zLmFtYXpvbmF3cy5jb21cIiksXG4gICAgICBzb3VyY2VBcm46IHRoaXMuYWxhcm1Ub3BpYy50b3BpY0FybixcbiAgICB9KVxuXG4gICAgbmV3IHNucy5TdWJzY3JpcHRpb24odGhpcywgXCJTdWJzY3JpcHRpb25cIiwge1xuICAgICAgZW5kcG9pbnQ6IHNsYWNrTGFtYmRhLmZ1bmN0aW9uQXJuLFxuICAgICAgcHJvdG9jb2w6IHNucy5TdWJzY3JpcHRpb25Qcm90b2NvbC5MQU1CREEsXG4gICAgICB0b3BpYzogdGhpcy5hbGFybVRvcGljLFxuICAgIH0pXG4gIH1cbn1cbiJdfQ==
@@ -19,4 +19,4 @@ else {
19
19
  console.error(e.stack || e.message || e);
20
20
  process.exitCode = 1;
21
21
  });
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2RrLWNyZWF0ZS1zbmFwc2hvdHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYmluL2Nkay1jcmVhdGUtc25hcHNob3RzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLDRDQUEwRDtBQUUxRCxJQUFJLEdBQVcsQ0FBQTtBQUNmLElBQUksR0FBVyxDQUFBO0FBRWYseURBQXlEO0FBQ3pELElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFO0lBQzVCLEdBQUcsR0FBRyxTQUFTLENBQUE7SUFDZixHQUFHLEdBQUcsZUFBZSxDQUFBO0NBQ3RCO0tBQU07SUFDTCxpREFBaUQ7SUFDakQsR0FBRyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDckIsR0FBRyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUE7Q0FDdEI7QUFFRCxJQUFBLHVDQUEyQixFQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRTtJQUNoRCxzRUFBc0U7SUFDdEUsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxJQUFJLENBQUMsQ0FBQyxPQUFPLElBQUksQ0FBQyxDQUFDLENBQUE7SUFDeEMsT0FBTyxDQUFDLFFBQVEsR0FBRyxDQUFDLENBQUE7QUFDdEIsQ0FBQyxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXG5pbXBvcnQgeyBjcmVhdGVDbG91ZEFzc2VtYmx5U25hcHNob3QgfSBmcm9tIFwiLi4vc25hcHNob3RzXCJcblxubGV0IHNyYzogc3RyaW5nXG5sZXQgZHN0OiBzdHJpbmdcblxuLy8gSWYgbm8gYXJndW1lbnRzIGFyZSBnaXZlbiwgdXNlIHNvbWUgc2Vuc2libGUgZGVmYXVsdHMuXG5pZiAocHJvY2Vzcy5hcmd2Lmxlbmd0aCA9PSAyKSB7XG4gIHNyYyA9IFwiY2RrLm91dFwiXG4gIGRzdCA9IFwiX19zbmFwc2hvdHNfX1wiXG59IGVsc2Uge1xuICAvLyBJbiBOb2RlLCBmaXJzdCBwYXJhbWV0ZXIgaXMgdGhlIHRoaXJkIGVsZW1lbnQuXG4gIHNyYyA9IHByb2Nlc3MuYXJndlsyXVxuICBkc3QgPSBwcm9jZXNzLmFyZ3ZbM11cbn1cblxuY3JlYXRlQ2xvdWRBc3NlbWJseVNuYXBzaG90KHNyYywgZHN0KS5jYXRjaCgoZSkgPT4ge1xuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1tZW1iZXItYWNjZXNzXG4gIGNvbnNvbGUuZXJyb3IoZS5zdGFjayB8fCBlLm1lc3NhZ2UgfHwgZSlcbiAgcHJvY2Vzcy5leGl0Q29kZSA9IDFcbn0pXG4iXX0=
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2RrLWNyZWF0ZS1zbmFwc2hvdHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYmluL2Nkay1jcmVhdGUtc25hcHNob3RzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLDRDQUEwRDtBQUUxRCxJQUFJLEdBQVcsQ0FBQTtBQUNmLElBQUksR0FBVyxDQUFBO0FBRWYseURBQXlEO0FBQ3pELElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFLENBQUM7SUFDN0IsR0FBRyxHQUFHLFNBQVMsQ0FBQTtJQUNmLEdBQUcsR0FBRyxlQUFlLENBQUE7QUFDdkIsQ0FBQztLQUFNLENBQUM7SUFDTixpREFBaUQ7SUFDakQsR0FBRyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDckIsR0FBRyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUE7QUFDdkIsQ0FBQztBQUVELElBQUEsdUNBQTJCLEVBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFO0lBQ2hELHNFQUFzRTtJQUN0RSxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLElBQUksQ0FBQyxDQUFDLE9BQU8sSUFBSSxDQUFDLENBQUMsQ0FBQTtJQUN4QyxPQUFPLENBQUMsUUFBUSxHQUFHLENBQUMsQ0FBQTtBQUN0QixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIiMhL3Vzci9iaW4vZW52IG5vZGVcbmltcG9ydCB7IGNyZWF0ZUNsb3VkQXNzZW1ibHlTbmFwc2hvdCB9IGZyb20gXCIuLi9zbmFwc2hvdHNcIlxuXG5sZXQgc3JjOiBzdHJpbmdcbmxldCBkc3Q6IHN0cmluZ1xuXG4vLyBJZiBubyBhcmd1bWVudHMgYXJlIGdpdmVuLCB1c2Ugc29tZSBzZW5zaWJsZSBkZWZhdWx0cy5cbmlmIChwcm9jZXNzLmFyZ3YubGVuZ3RoID09IDIpIHtcbiAgc3JjID0gXCJjZGsub3V0XCJcbiAgZHN0ID0gXCJfX3NuYXBzaG90c19fXCJcbn0gZWxzZSB7XG4gIC8vIEluIE5vZGUsIGZpcnN0IHBhcmFtZXRlciBpcyB0aGUgdGhpcmQgZWxlbWVudC5cbiAgc3JjID0gcHJvY2Vzcy5hcmd2WzJdXG4gIGRzdCA9IHByb2Nlc3MuYXJndlszXVxufVxuXG5jcmVhdGVDbG91ZEFzc2VtYmx5U25hcHNob3Qoc3JjLCBkc3QpLmNhdGNoKChlKSA9PiB7XG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3NcbiAgY29uc29sZS5lcnJvcihlLnN0YWNrIHx8IGUubWVzc2FnZSB8fCBlKVxuICBwcm9jZXNzLmV4aXRDb2RlID0gMVxufSlcbiJdfQ==
@@ -30,4 +30,4 @@ main(namespace).catch((e) => {
30
30
  console.error(e.stack || e.message || e);
31
31
  process.exitCode = 1;
32
32
  });
33
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmV0Y2gtcGlwZWxpbmUtdmFyaWFibGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jpbi9mZXRjaC1waXBlbGluZS12YXJpYWJsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EseUJBQXdCO0FBQ3hCLDBEQUEyRTtBQUUzRSxJQUFJLFNBQWlCLENBQUE7QUFFckIseURBQXlEO0FBQ3pELElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFO0lBQzVCLFNBQVMsR0FBRyxTQUFTLENBQUE7Q0FDdEI7S0FBTTtJQUNMLGlEQUFpRDtJQUNqRCxTQUFTLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQTtDQUM1QjtBQUVELEtBQUssVUFBVSxJQUFJLENBQUMsU0FBaUI7SUFDbkMsTUFBTSxTQUFTLEdBQUc7UUFDaEIsMkRBQTJEO1FBQzNELHdEQUF3RDtRQUN4RCw4Q0FBOEM7UUFDOUMsa0JBQWtCLEVBQUUsSUFBSSxJQUFJLEVBQUUsQ0FBQyxXQUFXLEVBQUU7UUFDNUMsR0FBRyxDQUFDLE1BQU0sSUFBQSwwQ0FBOEIsRUFDdEMsZUFBZSxTQUFTLHNCQUFzQixDQUMvQyxDQUFDO0tBQ0gsQ0FBQTtJQUVELE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsU0FBUyxFQUFFLFNBQVMsRUFBRSxJQUFJLENBQUMsQ0FBQTtJQUV6RCxFQUFFLENBQUMsYUFBYSxDQUFDLGdCQUFnQixFQUFFLE1BQU0sQ0FBQyxDQUFBO0lBQzFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsd0NBQXdDLENBQUMsQ0FBQTtJQUNyRCxPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUE7QUFDMUMsQ0FBQztBQUVELElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRTtJQUMxQixzRUFBc0U7SUFDdEUsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxJQUFJLENBQUMsQ0FBQyxPQUFPLElBQUksQ0FBQyxDQUFDLENBQUE7SUFDeEMsT0FBTyxDQUFDLFFBQVEsR0FBRyxDQUFDLENBQUE7QUFDdEIsQ0FBQyxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIjIS91c3IvYmluL2VudiBub2RlXG5pbXBvcnQgKiBhcyBmcyBmcm9tIFwiZnNcIlxuaW1wb3J0IHsgZ2V0VmFyaWFibGVzRnJvbVBhcmFtZXRlclN0b3JlIH0gZnJvbSBcIi4uL2Nkay1waXBlbGluZXMvdmFyaWFibGVzXCJcblxubGV0IG5hbWVzcGFjZTogc3RyaW5nXG5cbi8vIElmIG5vIGFyZ3VtZW50cyBhcmUgZ2l2ZW4sIHVzZSBzb21lIHNlbnNpYmxlIGRlZmF1bHRzLlxuaWYgKHByb2Nlc3MuYXJndi5sZW5ndGggPT0gMikge1xuICBuYW1lc3BhY2UgPSBcImRlZmF1bHRcIlxufSBlbHNlIHtcbiAgLy8gSW4gTm9kZSwgZmlyc3QgcGFyYW1ldGVyIGlzIHRoZSB0aGlyZCBlbGVtZW50LlxuICBuYW1lc3BhY2UgPSBwcm9jZXNzLmFyZ3ZbMl1cbn1cblxuYXN5bmMgZnVuY3Rpb24gbWFpbihuYW1lc3BhY2U6IHN0cmluZykge1xuICBjb25zdCB2YXJpYWJsZXMgPSB7XG4gICAgLy8gU3BlY2lhbCB2YXJpYWJsZSB0aGF0IGNhbiBiZSB1c2VkIHdoZW4gcmVhZGluZyB2YXJpYWJsZXNcbiAgICAvLyB0byBlbnN1cmUgaXQgaXMgbm90IHN0YWxlLiBJbiB0aGUgcGlwZWxpbmUsIHZhcmlhYmxlc1xuICAgIC8vIHdpbGwgbmV2ZXIgYmUgc3RhbGUsIGJ1dCBsb2NhbGx5IGl0IGNhbiBiZS5cbiAgICB2YXJpYWJsZXNUaW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICAuLi4oYXdhaXQgZ2V0VmFyaWFibGVzRnJvbVBhcmFtZXRlclN0b3JlKFxuICAgICAgYC9saWZsaWctY2RrLyR7bmFtZXNwYWNlfS9waXBlbGluZS12YXJpYWJsZXMvYCxcbiAgICApKSxcbiAgfVxuXG4gIGNvbnN0IHJlc3VsdCA9IEpTT04uc3RyaW5naWZ5KHZhcmlhYmxlcywgdW5kZWZpbmVkLCBcIiAgXCIpXG5cbiAgZnMud3JpdGVGaWxlU3luYyhcInZhcmlhYmxlcy5qc29uXCIsIHJlc3VsdClcbiAgY29uc29sZS5sb2coXCJSZXRyaWV2ZWQgYW5kIHNhdmVkIHRvIHZhcmlhYmxlcy5qc29uOlwiKVxuICBjb25zb2xlLmxvZyhyZXN1bHQucmVwbGFjZSgvXi9nbSwgXCIgIFwiKSlcbn1cblxubWFpbihuYW1lc3BhY2UpLmNhdGNoKChlKSA9PiB7XG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3NcbiAgY29uc29sZS5lcnJvcihlLnN0YWNrIHx8IGUubWVzc2FnZSB8fCBlKVxuICBwcm9jZXNzLmV4aXRDb2RlID0gMVxufSlcbiJdfQ==
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmV0Y2gtcGlwZWxpbmUtdmFyaWFibGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jpbi9mZXRjaC1waXBlbGluZS12YXJpYWJsZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQ0EseUJBQXdCO0FBQ3hCLDBEQUEyRTtBQUUzRSxJQUFJLFNBQWlCLENBQUE7QUFFckIseURBQXlEO0FBQ3pELElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLElBQUksQ0FBQyxFQUFFLENBQUM7SUFDN0IsU0FBUyxHQUFHLFNBQVMsQ0FBQTtBQUN2QixDQUFDO0tBQU0sQ0FBQztJQUNOLGlEQUFpRDtJQUNqRCxTQUFTLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQTtBQUM3QixDQUFDO0FBRUQsS0FBSyxVQUFVLElBQUksQ0FBQyxTQUFpQjtJQUNuQyxNQUFNLFNBQVMsR0FBRztRQUNoQiwyREFBMkQ7UUFDM0Qsd0RBQXdEO1FBQ3hELDhDQUE4QztRQUM5QyxrQkFBa0IsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRTtRQUM1QyxHQUFHLENBQUMsTUFBTSxJQUFBLDBDQUE4QixFQUN0QyxlQUFlLFNBQVMsc0JBQXNCLENBQy9DLENBQUM7S0FDSCxDQUFBO0lBRUQsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxTQUFTLEVBQUUsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFBO0lBRXpELEVBQUUsQ0FBQyxhQUFhLENBQUMsZ0JBQWdCLEVBQUUsTUFBTSxDQUFDLENBQUE7SUFDMUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyx3Q0FBd0MsQ0FBQyxDQUFBO0lBQ3JELE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQTtBQUMxQyxDQUFDO0FBRUQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFO0lBQzFCLHNFQUFzRTtJQUN0RSxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLElBQUksQ0FBQyxDQUFDLE9BQU8sSUFBSSxDQUFDLENBQUMsQ0FBQTtJQUN4QyxPQUFPLENBQUMsUUFBUSxHQUFHLENBQUMsQ0FBQTtBQUN0QixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIiMhL3Vzci9iaW4vZW52IG5vZGVcbmltcG9ydCAqIGFzIGZzIGZyb20gXCJmc1wiXG5pbXBvcnQgeyBnZXRWYXJpYWJsZXNGcm9tUGFyYW1ldGVyU3RvcmUgfSBmcm9tIFwiLi4vY2RrLXBpcGVsaW5lcy92YXJpYWJsZXNcIlxuXG5sZXQgbmFtZXNwYWNlOiBzdHJpbmdcblxuLy8gSWYgbm8gYXJndW1lbnRzIGFyZSBnaXZlbiwgdXNlIHNvbWUgc2Vuc2libGUgZGVmYXVsdHMuXG5pZiAocHJvY2Vzcy5hcmd2Lmxlbmd0aCA9PSAyKSB7XG4gIG5hbWVzcGFjZSA9IFwiZGVmYXVsdFwiXG59IGVsc2Uge1xuICAvLyBJbiBOb2RlLCBmaXJzdCBwYXJhbWV0ZXIgaXMgdGhlIHRoaXJkIGVsZW1lbnQuXG4gIG5hbWVzcGFjZSA9IHByb2Nlc3MuYXJndlsyXVxufVxuXG5hc3luYyBmdW5jdGlvbiBtYWluKG5hbWVzcGFjZTogc3RyaW5nKSB7XG4gIGNvbnN0IHZhcmlhYmxlcyA9IHtcbiAgICAvLyBTcGVjaWFsIHZhcmlhYmxlIHRoYXQgY2FuIGJlIHVzZWQgd2hlbiByZWFkaW5nIHZhcmlhYmxlc1xuICAgIC8vIHRvIGVuc3VyZSBpdCBpcyBub3Qgc3RhbGUuIEluIHRoZSBwaXBlbGluZSwgdmFyaWFibGVzXG4gICAgLy8gd2lsbCBuZXZlciBiZSBzdGFsZSwgYnV0IGxvY2FsbHkgaXQgY2FuIGJlLlxuICAgIHZhcmlhYmxlc1RpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIC4uLihhd2FpdCBnZXRWYXJpYWJsZXNGcm9tUGFyYW1ldGVyU3RvcmUoXG4gICAgICBgL2xpZmxpZy1jZGsvJHtuYW1lc3BhY2V9L3BpcGVsaW5lLXZhcmlhYmxlcy9gLFxuICAgICkpLFxuICB9XG5cbiAgY29uc3QgcmVzdWx0ID0gSlNPTi5zdHJpbmdpZnkodmFyaWFibGVzLCB1bmRlZmluZWQsIFwiICBcIilcblxuICBmcy53cml0ZUZpbGVTeW5jKFwidmFyaWFibGVzLmpzb25cIiwgcmVzdWx0KVxuICBjb25zb2xlLmxvZyhcIlJldHJpZXZlZCBhbmQgc2F2ZWQgdG8gdmFyaWFibGVzLmpzb246XCIpXG4gIGNvbnNvbGUubG9nKHJlc3VsdC5yZXBsYWNlKC9eL2dtLCBcIiAgXCIpKVxufVxuXG5tYWluKG5hbWVzcGFjZSkuY2F0Y2goKGUpID0+IHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtbWVtYmVyLWFjY2Vzc1xuICBjb25zb2xlLmVycm9yKGUuc3RhY2sgfHwgZS5tZXNzYWdlIHx8IGUpXG4gIHByb2Nlc3MuZXhpdENvZGUgPSAxXG59KVxuIl19