@liflig/cdk 1.48.1 → 1.51.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.
@@ -0,0 +1,249 @@
1
+ #!/usr/bin/env python
2
+
3
+ """
4
+ Transform CloudTrail events to payloads formatted for Slack's API, and send them
5
+ directly to Slack or through an SQS FIFO queue for deduplication.
6
+
7
+ The code below contains entrypoints for two Lambda functions (prefixed with `handler_`).
8
+ """
9
+
10
+ import os
11
+ import logging
12
+ import json
13
+ import urllib.request
14
+ import boto3
15
+
16
+ logger = logging.getLogger()
17
+ logger.setLevel(logging.INFO)
18
+
19
+
20
+ def get_slack_payload_for_assume_role_event(event, account_friendly_names):
21
+ """Parse a CloudTrail event related to the API call sts:AssumeRole,
22
+ and return a Slack-formatted attachment"""
23
+ event_account_id = event["account"]
24
+ event_detail = event["detail"]
25
+ request_parameters = event_detail.get("requestParameters", {}) or {}
26
+
27
+ timestamp = event_detail["eventTime"]
28
+ user_identity = event_detail["userIdentity"]
29
+ principal_id = user_identity["principalId"]
30
+ principal_account_id = user_identity["accountId"]
31
+ source_identity = request_parameters.get("sourceIdentity", "")
32
+ source_ip = event_detail.get("sourceIPAddress", "")
33
+ role_arn = request_parameters.get("roleArn", "")
34
+
35
+ fallback = f"Sensitive role accessed in '{event_account_id}'"
36
+ pretext_messages = [f":warning: Sensitive role in `{event_account_id}` assumed by"]
37
+ if principal_id.startswith("AIDA"):
38
+ pretext_messages.append("IAM user")
39
+ elif principal_id.startswith("AROA"):
40
+ pretext_messages.append("IAM role")
41
+ else:
42
+ pretext_messages.append("principal")
43
+ pretext_messages.append(f"in `{principal_account_id}`")
44
+ pretext = " ".join(pretext_messages)
45
+
46
+ text = [
47
+ f"*Role ARN:* `{role_arn}`",
48
+ f"*Principal Account ID:* `{principal_account_id}`",
49
+ f"*Principal ID:* `{principal_id}`",
50
+ f"*Source IP:* `{source_ip}`",
51
+ f"*Source Identity:* `{source_identity}`" if source_identity else "",
52
+ f"*Timestamp:* `{timestamp}`",
53
+ ]
54
+ text = "\n".join(line for line in text if line)
55
+ for account_id, friendly_name in account_friendly_names.items():
56
+ pretext = pretext.replace(account_id, friendly_name)
57
+ fallback = fallback.replace(account_id, friendly_name)
58
+ return {
59
+ "attachments": [
60
+ {
61
+ "pretext": pretext,
62
+ "color": "warning",
63
+ "text": text,
64
+ "fallback": fallback,
65
+ "mrkdwn_in": ["pretext", "text"],
66
+ }
67
+ ]
68
+ }
69
+
70
+
71
+ def get_fallback_slack_payload_for_event(
72
+ event, account_friendly_names, fallback_parse_behavior=""
73
+ ):
74
+ """Parse a generic CloudTrail event related to an API call
75
+ and return a Slack-formatted attachment"""
76
+ event_account_id = event["account"]
77
+ event_detail = event["detail"]
78
+ event_name = event_detail["eventName"]
79
+ event_type = event_detail["eventType"]
80
+ event_time = event_detail["eventTime"]
81
+ pretext = f":warning: CloudTrail event in account `{event_account_id}`"
82
+ fallback = f"CloudTrail event in account '{event_account_id}'"
83
+ if fallback_parse_behavior == "DUMP_EVENT":
84
+ text = "\n".join(
85
+ ["*Event:*", "```", json.dumps(event, sort_keys=True, indent=2), "```"]
86
+ )
87
+ else:
88
+ error_message = event_detail.get("errorMessage", "")
89
+ # This may be None, in which case we force it to an empty dict instead
90
+ response_element = (event_detail.get("responseElements", {}) or {}).get(
91
+ event_name, ""
92
+ )
93
+ user_identity = event_detail["userIdentity"]
94
+ principal_id = user_identity.get("principalId", "")
95
+ principal_type = user_identity.get("type", "")
96
+ principal_account_id = user_identity.get("accountId", "")
97
+ principal_arn = user_identity.get("arn", "")
98
+ source_ip = event_detail.get("sourceIPAddress", "")
99
+ resources = event_detail.get("resources", []) or []
100
+ text = [
101
+ f"*Event Type:* `{event_type}`",
102
+ f"*Event Name:* `{event_name}`",
103
+ f"*Event Time:* `{event_time}`",
104
+ f"*Error Message:* `{error_message}`" if error_message else "",
105
+ f"*Response Code:* `{response_element}`" if response_element else "",
106
+ f"*Principal Type:* `{principal_type}`" if principal_type else "",
107
+ f"*Principal Account ID:* `{principal_account_id}`"
108
+ if principal_account_id
109
+ else "",
110
+ f"*Principal ARN:* `{principal_arn}`" if principal_arn else "",
111
+ f"*Principal ID:* `{principal_id}`" if principal_id else "",
112
+ f"*Source IP:* `{source_ip}`" if source_ip else "",
113
+ f"*Resources:*\n```{json.dumps(resources, indent=2, sort_keys=True)}\n```"
114
+ if len(resources)
115
+ else "",
116
+ ]
117
+ # Filter out empty strings
118
+ text = "\n".join(line for line in text if line)
119
+
120
+ for account_id, friendly_name in account_friendly_names.items():
121
+ pretext = pretext.replace(account_id, friendly_name)
122
+ fallback = fallback.replace(account_id, friendly_name)
123
+ return {
124
+ "attachments": [
125
+ {
126
+ "pretext": pretext,
127
+ "color": "warning",
128
+ "text": text,
129
+ "fallback": fallback,
130
+ "mrkdwn_in": ["pretext", "text"],
131
+ }
132
+ ]
133
+ }
134
+
135
+
136
+ def get_augmented_account_friendly_names(event, account_friendly_names):
137
+ """Return an augmented dictionary containing the alias of the current
138
+ AWS account if relevant"""
139
+ augmented_account_friendly_names = {**account_friendly_names}
140
+ try:
141
+ event_account_id = event["account"]
142
+ event_detail = event["detail"]
143
+ recipient_account_id = event_detail["recipientAccountId"]
144
+ if (
145
+ not augmented_account_friendly_names.get(event_account_id, "")
146
+ and event_account_id == recipient_account_id
147
+ ):
148
+ logger.info(
149
+ "No friendly name was supplied for current account '%s', so looking up account alias",
150
+ event_account_id,
151
+ )
152
+ iam = boto3.client("iam")
153
+ aliases = iam.list_account_aliases()["AccountAliases"]
154
+ if len(aliases):
155
+ augmented_account_friendly_names[event_account_id] = aliases[0]
156
+ except:
157
+ logger.exception("Failed to look up alias of current AWS account")
158
+
159
+ return augmented_account_friendly_names
160
+
161
+
162
+ def post_to_slack(slack_payload, slack_webhook_url):
163
+ """Post a payload to Slack's webhook API"""
164
+ encoded_slack_payload = json.dumps(slack_payload).encode("utf-8")
165
+ try:
166
+ slack_request = urllib.request.Request(
167
+ slack_webhook_url,
168
+ data=encoded_slack_payload,
169
+ headers={"Content-Type": "application/json"},
170
+ )
171
+ urllib.request.urlopen(slack_request)
172
+ except:
173
+ logger.exception("Failed to post to Slack")
174
+ raise
175
+
176
+
177
+ def handler_event_transformer(event, context):
178
+ """Lambda handler for the event transformer Lambda"""
179
+ logger.info("Triggered with event: %s", json.dumps(event, indent=2))
180
+
181
+ account_friendly_names = json.loads(os.environ["ACCOUNT_FRIENDLY_NAMES"])
182
+ slack_webhook_url = os.environ["SLACK_WEBHOOK_URL"]
183
+ slack_channel = os.environ["SLACK_CHANNEL"]
184
+ sqs_queue_url = os.environ.get("SQS_QUEUE_URL", "")
185
+ fallback_parse_behavior = os.environ.get("FALLBACK_PARSE_BEHAVIOR", "")
186
+ deduplicate_events = os.environ.get("DEDUPLICATE_EVENTS", "false") == "true"
187
+
188
+ account_friendly_names = get_augmented_account_friendly_names(
189
+ event, account_friendly_names
190
+ )
191
+
192
+ if not event["detail-type"].endswith("via CloudTrail"):
193
+ logger.warn("Invalid event received")
194
+ return
195
+
196
+ slack_payload = {}
197
+ try:
198
+ if event["detail"]["eventName"] == "AssumeRole":
199
+ slack_payload = get_slack_payload_for_assume_role_event(
200
+ event, account_friendly_names
201
+ )
202
+ except:
203
+ logger.exception("Failed to parse event using predefined schema")
204
+ if not slack_payload:
205
+ logger.warn("Using a fallback schema to parse event")
206
+ slack_payload = get_fallback_slack_payload_for_event(
207
+ event,
208
+ account_friendly_names,
209
+ fallback_parse_behavior=fallback_parse_behavior,
210
+ )
211
+ slack_payload = {**slack_payload, "channel": slack_channel}
212
+
213
+ if deduplicate_events and sqs_queue_url:
214
+ logger.info("Sending message to SQS for deduplication")
215
+ deduplication_id = (
216
+ event["detail"].get("requestID", "")
217
+ or event["detail"].get("eventID", "")
218
+ or event["id"]
219
+ )
220
+ body = {
221
+ "slackWebhookUrl": slack_webhook_url,
222
+ "slackPayload": slack_payload,
223
+ }
224
+
225
+ sqs = boto3.client("sqs")
226
+ sqs.send_message(
227
+ QueueUrl=sqs_queue_url,
228
+ MessageBody=json.dumps(body),
229
+ MessageDeduplicationId=deduplication_id,
230
+ MessageGroupId=deduplication_id,
231
+ )
232
+ else:
233
+ logger.info("Sending message directly to Slack")
234
+ post_to_slack(slack_payload, slack_webhook_url)
235
+
236
+
237
+ def handler_slack_forwarder(event, context):
238
+ """Lambda handler for the Slack forwarder Lambda"""
239
+ logger.info("Triggered with event: %s", json.dumps(event, indent=2))
240
+ records = event["Records"]
241
+ for record in records:
242
+ body = json.loads(record["body"])
243
+ slack_channel = body.get("slackChannel", "")
244
+ slack_webhook_url = body.get("slackWebhookUrl", "")
245
+ slack_payload = {
246
+ **body["slackPayload"],
247
+ **({"channel": slack_channel} if slack_channel else {}),
248
+ }
249
+ post_to_slack(slack_payload, slack_webhook_url)
@@ -69,4 +69,4 @@ const startDeployHandler = async (event, context) => {
69
69
  };
70
70
  };
71
71
  exports.startDeployHandler = startDeployHandler;
72
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhcnQtZGVwbG95LWhhbmRsZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2RrLWRlcGxveS9zdGFydC1kZXBsb3ktaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFhQSxtREFBbUQ7QUFDbkQsNkJBQTZCO0FBQ3RCLE1BQU0sa0JBQWtCLEdBQzdCLEtBQUssRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLEVBQUU7O0lBQ3ZCLE1BQU0sR0FBRyxHQUFHLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQTtJQUU5QixNQUFNLFNBQVMsR0FBRyxJQUFJLEdBQUcsQ0FBQyxTQUFTLEVBQW9CLENBQUE7SUFDdkQsTUFBTSxFQUFFLEdBQUcsSUFBSSxHQUFHLENBQUMsRUFBRSxFQUFhLENBQUE7SUFFbEMsU0FBUyxVQUFVLENBQUMsSUFBWTtRQUM5QixNQUFNLEtBQUssR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFBO1FBQy9CLElBQUksS0FBSyxLQUFLLFNBQVMsRUFBRTtZQUN2QixNQUFNLElBQUksS0FBSyxDQUFDLFdBQVcsSUFBSSxFQUFFLENBQUMsQ0FBQTtTQUNuQztRQUNELE9BQU8sS0FBSyxDQUFBO0lBQ2QsQ0FBQztJQUVELE1BQU0sV0FBVyxHQUFHLFVBQVUsQ0FBQyxjQUFjLENBQUMsQ0FBQTtJQUM5QyxNQUFNLFVBQVUsR0FBRyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUE7SUFDNUMsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQTtJQUV4RCx5REFBeUQ7SUFDekQsdURBQXVEO0lBQ3ZELE1BQU0sY0FBYyxHQUFHLDBCQUEwQixDQUFBO0lBRWpELE1BQU0sV0FBVyxHQUFHLEdBQUcsT0FBTyxDQUFDLFlBQVksR0FBRyxDQUFBO0lBRTlDLHNCQUFzQjtJQUN0QixJQUNFLE9BQU8sS0FBSyxDQUFDLFVBQVUsS0FBSyxRQUFRO1FBQ3BDLE9BQU8sS0FBSyxDQUFDLFNBQVMsS0FBSyxRQUFRO1FBQ25DLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDO1FBQ2hDLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQ3JCLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxPQUFPLEVBQUUsS0FBSyxRQUFRLElBQUksY0FBYyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FDMUQsRUFDRDtRQUNBLE1BQU0sSUFBSSxLQUFLLENBQ2IsaUJBQWlCLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLEVBQUUsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUMzRCxDQUFBO0tBQ0Y7SUFFRCxLQUFLLFVBQVUsR0FBRyxDQUFDLElBQVksRUFBRSxJQUFpQjtRQUNoRCxNQUFNLEVBQUU7YUFDTCxTQUFTLENBQUM7WUFDVCxNQUFNLEVBQUUsVUFBVTtZQUNsQixHQUFHLEVBQUUsR0FBRyxXQUFXLEdBQUcsSUFBSSxFQUFFO1lBQzVCLElBQUksRUFBRSxJQUFJO1NBQ1gsQ0FBQzthQUNELE9BQU8sRUFBRSxDQUFBO0lBQ2QsQ0FBQztJQUVELE1BQU0sR0FBRyxDQUFDLGlCQUFpQixFQUFFLEtBQUssQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUE7SUFDeEQsMERBQTBEO0lBQzFELE1BQU0sR0FBRyxDQUNQLFVBQVUsRUFDVixJQUFJLENBQUMsU0FBUyxDQUFDO1FBQ2IsT0FBTyxFQUFFLFVBQVU7S0FDcEIsQ0FBQyxDQUNILENBQUE7SUFFRCxNQUFNLEtBQUssR0FBRyxNQUFNLFNBQVM7U0FDMUIsVUFBVSxDQUFDO1FBQ1YsV0FBVztRQUNYLGtCQUFrQixFQUFFLElBQUk7UUFDeEIsc0JBQXNCLEVBQUUsR0FBRyxVQUFVLElBQUksV0FBVyxFQUFFO1FBQ3RELHdCQUF3QixFQUFFO1lBQ3hCO2dCQUNFLElBQUksRUFBRSxJQUFJO2dCQUNWLFFBQVEsRUFBRSxHQUFHLEtBQUssQ0FBQyxVQUFVLElBQUksS0FBSyxDQUFDLFNBQVMsRUFBRTtnQkFDbEQsZ0JBQWdCLEVBQUUsZUFBZTthQUNsQztTQUNGO0tBQ0YsQ0FBQztTQUNELE9BQU8sRUFBRSxDQUFBO0lBRVosTUFBTSxPQUFPLEdBQUcsTUFBQSxLQUFLLENBQUMsS0FBSywwQ0FBRSxFQUFFLENBQUE7SUFDL0IsSUFBSSxPQUFPLElBQUksSUFBSSxFQUFFO1FBQ25CLE1BQU0sSUFBSSxLQUFLLENBQUMsa0JBQWtCLENBQUMsQ0FBQTtLQUNwQztJQUVELE9BQU87UUFDTCxpRUFBaUU7UUFDakUsb0VBQW9FO1FBQ3BFLG9DQUFvQztRQUNwQyxLQUFLLEVBQUUsT0FBTztLQUNmLENBQUE7QUFDSCxDQUFDLENBQUE7QUFwRlUsUUFBQSxrQkFBa0Isc0JBb0Y1QiIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtYXNzaWdubWVudCAqL1xuLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1jYWxsICovXG4vKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3MgKi9cbi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXMgKi9cbmltcG9ydCB7IEhhbmRsZXIgfSBmcm9tIFwiYXdzLWxhbWJkYVwiXG5pbXBvcnQgdHlwZSAqIGFzIF9BV1MgZnJvbSBcImF3cy1zZGtcIlxuXG5pbnRlcmZhY2UgU3RhcnREZXBsb3lFeHBlY3RlZElucHV0IHtcbiAgYnVja2V0TmFtZTogc3RyaW5nXG4gIGJ1Y2tldEtleTogc3RyaW5nXG4gIHN0YWNrTmFtZXM6IHN0cmluZ1tdXG59XG5cbi8vIFRoaXMgZnVuY3Rpb24gaXMgaW5saW5lLWNvbXBpbGVkIGZvciB0aGUgbGFtYmRhLlxuLy8gSXQgbXVzdCBiZSBzZWxmLWNvbnRhaW5lZC5cbmV4cG9ydCBjb25zdCBzdGFydERlcGxveUhhbmRsZXI6IEhhbmRsZXI8UGFydGlhbDxTdGFydERlcGxveUV4cGVjdGVkSW5wdXQ+PiA9XG4gIGFzeW5jIChldmVudCwgY29udGV4dCkgPT4ge1xuICAgIGNvbnN0IEFXUyA9IHJlcXVpcmUoXCJhd3Mtc2RrXCIpXG5cbiAgICBjb25zdCBjb2RlYnVpbGQgPSBuZXcgQVdTLkNvZGVCdWlsZCgpIGFzIF9BV1MuQ29kZUJ1aWxkXG4gICAgY29uc3QgczMgPSBuZXcgQVdTLlMzKCkgYXMgX0FXUy5TM1xuXG4gICAgZnVuY3Rpb24gcmVxdWlyZUVudihuYW1lOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgICAgY29uc3QgdmFsdWUgPSBwcm9jZXNzLmVudltuYW1lXVxuICAgICAgaWYgKHZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBNaXNzaW5nICR7bmFtZX1gKVxuICAgICAgfVxuICAgICAgcmV0dXJuIHZhbHVlXG4gICAgfVxuXG4gICAgY29uc3QgcHJvamVjdE5hbWUgPSByZXF1aXJlRW52KFwiUFJPSkVDVF9OQU1FXCIpXG4gICAgY29uc3QgYnVja2V0TmFtZSA9IHJlcXVpcmVFbnYoXCJCVUNLRVRfTkFNRVwiKVxuICAgIGNvbnN0IGNka0NvbnRleHQgPSBKU09OLnBhcnNlKHJlcXVpcmVFbnYoXCJDREtfQ09OVEVYVFwiKSlcblxuICAgIC8vIFNpbmNlIHdlIHBhc3MgdGhlIHN0YWNrIG5hbWVzIGFzIHN0cmluZ3MgdG8gdGhlIHNoZWxsLFxuICAgIC8vIGJlIGEgYml0IHJlc3RyaWN0aXZlIG9mIHRoZSB2YWxpZCB2YWx1ZXMgd2UgY2FuIHVzZS5cbiAgICBjb25zdCB2YWxpZFN0YWNrTmFtZSA9IC9eW2EtejAtOV9dW2EtejAtOVxcLV9dKiQvaVxuXG4gICAgY29uc3QgczNLZXlQcmVmaXggPSBgJHtjb250ZXh0LmF3c1JlcXVlc3RJZH0vYFxuXG4gICAgLy8gVmFsaWRhdGUgdGhlIGlucHV0LlxuICAgIGlmIChcbiAgICAgIHR5cGVvZiBldmVudC5idWNrZXROYW1lICE9PSBcInN0cmluZ1wiIHx8XG4gICAgICB0eXBlb2YgZXZlbnQuYnVja2V0S2V5ICE9PSBcInN0cmluZ1wiIHx8XG4gICAgICAhQXJyYXkuaXNBcnJheShldmVudC5zdGFja05hbWVzKSB8fFxuICAgICAgIWV2ZW50LnN0YWNrTmFtZXMuZXZlcnkoXG4gICAgICAgIChpdCkgPT4gdHlwZW9mIGl0ID09PSBcInN0cmluZ1wiICYmIHZhbGlkU3RhY2tOYW1lLnRlc3QoaXQpLFxuICAgICAgKVxuICAgICkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBcIklucHV0IGludmFsaWQ6IFwiICsgSlNPTi5zdHJpbmdpZnkoZXZlbnQsIHVuZGVmaW5lZCwgXCIgIFwiKSxcbiAgICAgIClcbiAgICB9XG5cbiAgICBhc3luYyBmdW5jdGlvbiBwdXQobmFtZTogc3RyaW5nLCBkYXRhOiBBV1MuUzMuQm9keSkge1xuICAgICAgYXdhaXQgczNcbiAgICAgICAgLnB1dE9iamVjdCh7XG4gICAgICAgICAgQnVja2V0OiBidWNrZXROYW1lLFxuICAgICAgICAgIEtleTogYCR7czNLZXlQcmVmaXh9JHtuYW1lfWAsXG4gICAgICAgICAgQm9keTogZGF0YSxcbiAgICAgICAgfSlcbiAgICAgICAgLnByb21pc2UoKVxuICAgIH1cblxuICAgIGF3YWl0IHB1dChcInN0YWNrLW5hbWVzLnR4dFwiLCBldmVudC5zdGFja05hbWVzLmpvaW4oXCIgXCIpKVxuICAgIC8vIEVuc3VyZSB0aGF0IHdlIHJ1biB0aGUgc2NyaXB0IHVzaW5nIHNhbWUgZmVhdHVyZSBmbGFncy5cbiAgICBhd2FpdCBwdXQoXG4gICAgICBcImNkay5qc29uXCIsXG4gICAgICBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgIGNvbnRleHQ6IGNka0NvbnRleHQsXG4gICAgICB9KSxcbiAgICApXG5cbiAgICBjb25zdCBidWlsZCA9IGF3YWl0IGNvZGVidWlsZFxuICAgICAgLnN0YXJ0QnVpbGQoe1xuICAgICAgICBwcm9qZWN0TmFtZSxcbiAgICAgICAgc291cmNlVHlwZU92ZXJyaWRlOiBcIlMzXCIsXG4gICAgICAgIHNvdXJjZUxvY2F0aW9uT3ZlcnJpZGU6IGAke2J1Y2tldE5hbWV9LyR7czNLZXlQcmVmaXh9YCxcbiAgICAgICAgc2Vjb25kYXJ5U291cmNlc092ZXJyaWRlOiBbXG4gICAgICAgICAge1xuICAgICAgICAgICAgdHlwZTogXCJTM1wiLFxuICAgICAgICAgICAgbG9jYXRpb246IGAke2V2ZW50LmJ1Y2tldE5hbWV9LyR7ZXZlbnQuYnVja2V0S2V5fWAsXG4gICAgICAgICAgICBzb3VyY2VJZGVudGlmaWVyOiBcIkNMT1VEQVNTRU1CTFlcIixcbiAgICAgICAgICB9LFxuICAgICAgICBdLFxuICAgICAgfSlcbiAgICAgIC5wcm9taXNlKClcblxuICAgIGNvbnN0IGJ1aWxkSWQgPSBidWlsZC5idWlsZD8uaWRcbiAgICBpZiAoYnVpbGRJZCA9PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJVbmtub3duIGJ1aWxkIElEXCIpXG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIC8vIFRoaXMgaXMgdGhlIHZhbHVlIHRoZSBjYWxsZXIgd2lsbCB1c2UgdG8gZmV0Y2ggdXBkYXRlZCBzdGF0dXMuXG4gICAgICAvLyBBdm9pZCBleHBvc2luZyB3aGF0IGtpbmQgb2YgSUQgdGhpcyBpcywgYmVjYXVzZSB3ZSBzaG91bGQgYmUgZnJlZVxuICAgICAgLy8gdG8gY2hhbmdlIGltcGxlbWVudGF0aW9uIGRldGFpbHMuXG4gICAgICBqb2JJZDogYnVpbGRJZCxcbiAgICB9XG4gIH1cbiJdfQ==
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhcnQtZGVwbG95LWhhbmRsZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2RrLWRlcGxveS9zdGFydC1kZXBsb3ktaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFhQSxtREFBbUQ7QUFDbkQsNkJBQTZCO0FBQ3RCLE1BQU0sa0JBQWtCLEdBRTNCLEtBQUssRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLEVBQUU7O0lBQzNCLE1BQU0sR0FBRyxHQUFHLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQTtJQUU5QixNQUFNLFNBQVMsR0FBRyxJQUFJLEdBQUcsQ0FBQyxTQUFTLEVBQW9CLENBQUE7SUFDdkQsTUFBTSxFQUFFLEdBQUcsSUFBSSxHQUFHLENBQUMsRUFBRSxFQUFhLENBQUE7SUFFbEMsU0FBUyxVQUFVLENBQUMsSUFBWTtRQUM5QixNQUFNLEtBQUssR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFBO1FBQy9CLElBQUksS0FBSyxLQUFLLFNBQVMsRUFBRTtZQUN2QixNQUFNLElBQUksS0FBSyxDQUFDLFdBQVcsSUFBSSxFQUFFLENBQUMsQ0FBQTtTQUNuQztRQUNELE9BQU8sS0FBSyxDQUFBO0lBQ2QsQ0FBQztJQUVELE1BQU0sV0FBVyxHQUFHLFVBQVUsQ0FBQyxjQUFjLENBQUMsQ0FBQTtJQUM5QyxNQUFNLFVBQVUsR0FBRyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUE7SUFDNUMsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQTtJQUV4RCx5REFBeUQ7SUFDekQsdURBQXVEO0lBQ3ZELE1BQU0sY0FBYyxHQUFHLDBCQUEwQixDQUFBO0lBRWpELE1BQU0sV0FBVyxHQUFHLEdBQUcsT0FBTyxDQUFDLFlBQVksR0FBRyxDQUFBO0lBRTlDLHNCQUFzQjtJQUN0QixJQUNFLE9BQU8sS0FBSyxDQUFDLFVBQVUsS0FBSyxRQUFRO1FBQ3BDLE9BQU8sS0FBSyxDQUFDLFNBQVMsS0FBSyxRQUFRO1FBQ25DLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDO1FBQ2hDLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQ3JCLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxPQUFPLEVBQUUsS0FBSyxRQUFRLElBQUksY0FBYyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FDMUQsRUFDRDtRQUNBLE1BQU0sSUFBSSxLQUFLLENBQUMsaUJBQWlCLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLEVBQUUsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUE7S0FDNUU7SUFFRCxLQUFLLFVBQVUsR0FBRyxDQUFDLElBQVksRUFBRSxJQUFpQjtRQUNoRCxNQUFNLEVBQUU7YUFDTCxTQUFTLENBQUM7WUFDVCxNQUFNLEVBQUUsVUFBVTtZQUNsQixHQUFHLEVBQUUsR0FBRyxXQUFXLEdBQUcsSUFBSSxFQUFFO1lBQzVCLElBQUksRUFBRSxJQUFJO1NBQ1gsQ0FBQzthQUNELE9BQU8sRUFBRSxDQUFBO0lBQ2QsQ0FBQztJQUVELE1BQU0sR0FBRyxDQUFDLGlCQUFpQixFQUFFLEtBQUssQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUE7SUFDeEQsMERBQTBEO0lBQzFELE1BQU0sR0FBRyxDQUNQLFVBQVUsRUFDVixJQUFJLENBQUMsU0FBUyxDQUFDO1FBQ2IsT0FBTyxFQUFFLFVBQVU7S0FDcEIsQ0FBQyxDQUNILENBQUE7SUFFRCxNQUFNLEtBQUssR0FBRyxNQUFNLFNBQVM7U0FDMUIsVUFBVSxDQUFDO1FBQ1YsV0FBVztRQUNYLGtCQUFrQixFQUFFLElBQUk7UUFDeEIsc0JBQXNCLEVBQUUsR0FBRyxVQUFVLElBQUksV0FBVyxFQUFFO1FBQ3RELHdCQUF3QixFQUFFO1lBQ3hCO2dCQUNFLElBQUksRUFBRSxJQUFJO2dCQUNWLFFBQVEsRUFBRSxHQUFHLEtBQUssQ0FBQyxVQUFVLElBQUksS0FBSyxDQUFDLFNBQVMsRUFBRTtnQkFDbEQsZ0JBQWdCLEVBQUUsZUFBZTthQUNsQztTQUNGO0tBQ0YsQ0FBQztTQUNELE9BQU8sRUFBRSxDQUFBO0lBRVosTUFBTSxPQUFPLEdBQUcsTUFBQSxLQUFLLENBQUMsS0FBSywwQ0FBRSxFQUFFLENBQUE7SUFDL0IsSUFBSSxPQUFPLElBQUksSUFBSSxFQUFFO1FBQ25CLE1BQU0sSUFBSSxLQUFLLENBQUMsa0JBQWtCLENBQUMsQ0FBQTtLQUNwQztJQUVELE9BQU87UUFDTCxpRUFBaUU7UUFDakUsb0VBQW9FO1FBQ3BFLG9DQUFvQztRQUNwQyxLQUFLLEVBQUUsT0FBTztLQUNmLENBQUE7QUFDSCxDQUFDLENBQUE7QUFuRlksUUFBQSxrQkFBa0Isc0JBbUY5QiIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtYXNzaWdubWVudCAqL1xuLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1jYWxsICovXG4vKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3MgKi9cbi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby12YXItcmVxdWlyZXMgKi9cbmltcG9ydCB7IEhhbmRsZXIgfSBmcm9tIFwiYXdzLWxhbWJkYVwiXG5pbXBvcnQgdHlwZSAqIGFzIF9BV1MgZnJvbSBcImF3cy1zZGtcIlxuXG5pbnRlcmZhY2UgU3RhcnREZXBsb3lFeHBlY3RlZElucHV0IHtcbiAgYnVja2V0TmFtZTogc3RyaW5nXG4gIGJ1Y2tldEtleTogc3RyaW5nXG4gIHN0YWNrTmFtZXM6IHN0cmluZ1tdXG59XG5cbi8vIFRoaXMgZnVuY3Rpb24gaXMgaW5saW5lLWNvbXBpbGVkIGZvciB0aGUgbGFtYmRhLlxuLy8gSXQgbXVzdCBiZSBzZWxmLWNvbnRhaW5lZC5cbmV4cG9ydCBjb25zdCBzdGFydERlcGxveUhhbmRsZXI6IEhhbmRsZXI8XG4gIFBhcnRpYWw8U3RhcnREZXBsb3lFeHBlY3RlZElucHV0PlxuPiA9IGFzeW5jIChldmVudCwgY29udGV4dCkgPT4ge1xuICBjb25zdCBBV1MgPSByZXF1aXJlKFwiYXdzLXNka1wiKVxuXG4gIGNvbnN0IGNvZGVidWlsZCA9IG5ldyBBV1MuQ29kZUJ1aWxkKCkgYXMgX0FXUy5Db2RlQnVpbGRcbiAgY29uc3QgczMgPSBuZXcgQVdTLlMzKCkgYXMgX0FXUy5TM1xuXG4gIGZ1bmN0aW9uIHJlcXVpcmVFbnYobmFtZTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCB2YWx1ZSA9IHByb2Nlc3MuZW52W25hbWVdXG4gICAgaWYgKHZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgTWlzc2luZyAke25hbWV9YClcbiAgICB9XG4gICAgcmV0dXJuIHZhbHVlXG4gIH1cblxuICBjb25zdCBwcm9qZWN0TmFtZSA9IHJlcXVpcmVFbnYoXCJQUk9KRUNUX05BTUVcIilcbiAgY29uc3QgYnVja2V0TmFtZSA9IHJlcXVpcmVFbnYoXCJCVUNLRVRfTkFNRVwiKVxuICBjb25zdCBjZGtDb250ZXh0ID0gSlNPTi5wYXJzZShyZXF1aXJlRW52KFwiQ0RLX0NPTlRFWFRcIikpXG5cbiAgLy8gU2luY2Ugd2UgcGFzcyB0aGUgc3RhY2sgbmFtZXMgYXMgc3RyaW5ncyB0byB0aGUgc2hlbGwsXG4gIC8vIGJlIGEgYml0IHJlc3RyaWN0aXZlIG9mIHRoZSB2YWxpZCB2YWx1ZXMgd2UgY2FuIHVzZS5cbiAgY29uc3QgdmFsaWRTdGFja05hbWUgPSAvXlthLXowLTlfXVthLXowLTlcXC1fXSokL2lcblxuICBjb25zdCBzM0tleVByZWZpeCA9IGAke2NvbnRleHQuYXdzUmVxdWVzdElkfS9gXG5cbiAgLy8gVmFsaWRhdGUgdGhlIGlucHV0LlxuICBpZiAoXG4gICAgdHlwZW9mIGV2ZW50LmJ1Y2tldE5hbWUgIT09IFwic3RyaW5nXCIgfHxcbiAgICB0eXBlb2YgZXZlbnQuYnVja2V0S2V5ICE9PSBcInN0cmluZ1wiIHx8XG4gICAgIUFycmF5LmlzQXJyYXkoZXZlbnQuc3RhY2tOYW1lcykgfHxcbiAgICAhZXZlbnQuc3RhY2tOYW1lcy5ldmVyeShcbiAgICAgIChpdCkgPT4gdHlwZW9mIGl0ID09PSBcInN0cmluZ1wiICYmIHZhbGlkU3RhY2tOYW1lLnRlc3QoaXQpLFxuICAgIClcbiAgKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFwiSW5wdXQgaW52YWxpZDogXCIgKyBKU09OLnN0cmluZ2lmeShldmVudCwgdW5kZWZpbmVkLCBcIiAgXCIpKVxuICB9XG5cbiAgYXN5bmMgZnVuY3Rpb24gcHV0KG5hbWU6IHN0cmluZywgZGF0YTogQVdTLlMzLkJvZHkpIHtcbiAgICBhd2FpdCBzM1xuICAgICAgLnB1dE9iamVjdCh7XG4gICAgICAgIEJ1Y2tldDogYnVja2V0TmFtZSxcbiAgICAgICAgS2V5OiBgJHtzM0tleVByZWZpeH0ke25hbWV9YCxcbiAgICAgICAgQm9keTogZGF0YSxcbiAgICAgIH0pXG4gICAgICAucHJvbWlzZSgpXG4gIH1cblxuICBhd2FpdCBwdXQoXCJzdGFjay1uYW1lcy50eHRcIiwgZXZlbnQuc3RhY2tOYW1lcy5qb2luKFwiIFwiKSlcbiAgLy8gRW5zdXJlIHRoYXQgd2UgcnVuIHRoZSBzY3JpcHQgdXNpbmcgc2FtZSBmZWF0dXJlIGZsYWdzLlxuICBhd2FpdCBwdXQoXG4gICAgXCJjZGsuanNvblwiLFxuICAgIEpTT04uc3RyaW5naWZ5KHtcbiAgICAgIGNvbnRleHQ6IGNka0NvbnRleHQsXG4gICAgfSksXG4gIClcblxuICBjb25zdCBidWlsZCA9IGF3YWl0IGNvZGVidWlsZFxuICAgIC5zdGFydEJ1aWxkKHtcbiAgICAgIHByb2plY3ROYW1lLFxuICAgICAgc291cmNlVHlwZU92ZXJyaWRlOiBcIlMzXCIsXG4gICAgICBzb3VyY2VMb2NhdGlvbk92ZXJyaWRlOiBgJHtidWNrZXROYW1lfS8ke3MzS2V5UHJlZml4fWAsXG4gICAgICBzZWNvbmRhcnlTb3VyY2VzT3ZlcnJpZGU6IFtcbiAgICAgICAge1xuICAgICAgICAgIHR5cGU6IFwiUzNcIixcbiAgICAgICAgICBsb2NhdGlvbjogYCR7ZXZlbnQuYnVja2V0TmFtZX0vJHtldmVudC5idWNrZXRLZXl9YCxcbiAgICAgICAgICBzb3VyY2VJZGVudGlmaWVyOiBcIkNMT1VEQVNTRU1CTFlcIixcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSlcbiAgICAucHJvbWlzZSgpXG5cbiAgY29uc3QgYnVpbGRJZCA9IGJ1aWxkLmJ1aWxkPy5pZFxuICBpZiAoYnVpbGRJZCA9PSBudWxsKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFwiVW5rbm93biBidWlsZCBJRFwiKVxuICB9XG5cbiAgcmV0dXJuIHtcbiAgICAvLyBUaGlzIGlzIHRoZSB2YWx1ZSB0aGUgY2FsbGVyIHdpbGwgdXNlIHRvIGZldGNoIHVwZGF0ZWQgc3RhdHVzLlxuICAgIC8vIEF2b2lkIGV4cG9zaW5nIHdoYXQga2luZCBvZiBJRCB0aGlzIGlzLCBiZWNhdXNlIHdlIHNob3VsZCBiZSBmcmVlXG4gICAgLy8gdG8gY2hhbmdlIGltcGxlbWVudGF0aW9uIGRldGFpbHMuXG4gICAgam9iSWQ6IGJ1aWxkSWQsXG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,43 @@
1
+ import * as cdk from "@aws-cdk/core";
2
+ import * as cloudwatch from "@aws-cdk/aws-cloudwatch";
3
+ export interface CloudTrailSlackIntegrationProps extends cdk.StackProps {
4
+ /**
5
+ * A key-value pair of AWS account IDs and friendly names of these accounts
6
+ * to use when sending messages to Slack.
7
+ */
8
+ accountFriendlyNames?: {
9
+ [key: string]: string;
10
+ };
11
+ slackWebhookUrl: string;
12
+ slackChannel: string;
13
+ /**
14
+ * A list of ARNs of roles in the current account to monitor usage of.
15
+ */
16
+ rolesToMonitor?: string[];
17
+ /**
18
+ * Whether to monitor various IAM API calls associated with the current account's root user (e.g., console login, password reset, etc.)
19
+ *
20
+ * @default true
21
+ */
22
+ monitorRootUserActions?: boolean;
23
+ /**
24
+ * Whether to set up additional AWS infrastructure to deduplicate CloudTrail events in order to avoid duplicate Slack messages. May be used to decrease noise.
25
+ *
26
+ * @default false
27
+ */
28
+ deduplicateEvents?: boolean;
29
+ /**
30
+ * If supplied, CloudWatch alarms will be created for the construct's underlying infrastructure (e.g., Lambda functions) and the action will be used to notify on OK and ALARM actions.
31
+ */
32
+ infrastructureAlarmAction?: cloudwatch.IAlarmAction;
33
+ }
34
+ /**
35
+ * Forward a predefined set of CloudTrail API events to Slack using EventBridge, Lambda
36
+ * and an optional SQS FIFO queue for deduplicating events.
37
+ * The API events are limited to monitoring access to the current account's root user and/or specific IAM roles.
38
+ *
39
+ * NOTE: The construct needs to be provisioned in us-east-1, and requires an existing CloudTrail set up in that region.
40
+ */
41
+ export declare class CloudTrailSlackIntegration extends cdk.Construct {
42
+ constructor(scope: cdk.Construct, id: string, props: CloudTrailSlackIntegrationProps);
43
+ }
@@ -0,0 +1,210 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CloudTrailSlackIntegration = void 0;
4
+ const cdk = require("@aws-cdk/core");
5
+ const iam = require("@aws-cdk/aws-iam");
6
+ const logs = require("@aws-cdk/aws-logs");
7
+ const cloudwatch = require("@aws-cdk/aws-cloudwatch");
8
+ const lambda = require("@aws-cdk/aws-lambda");
9
+ const events = require("@aws-cdk/aws-events");
10
+ const sources = require("@aws-cdk/aws-lambda-event-sources");
11
+ const sqs = require("@aws-cdk/aws-sqs");
12
+ const targets = require("@aws-cdk/aws-events-targets");
13
+ const path = require("path");
14
+ /**
15
+ * Forward a predefined set of CloudTrail API events to Slack using EventBridge, Lambda
16
+ * and an optional SQS FIFO queue for deduplicating events.
17
+ * The API events are limited to monitoring access to the current account's root user and/or specific IAM roles.
18
+ *
19
+ * NOTE: The construct needs to be provisioned in us-east-1, and requires an existing CloudTrail set up in that region.
20
+ */
21
+ class CloudTrailSlackIntegration extends cdk.Construct {
22
+ constructor(scope, id, props) {
23
+ super(scope, id);
24
+ const eventTransformer = new lambda.Function(this, "EventTransformerLambda", {
25
+ code: lambda.Code.fromAsset(path.join(__dirname, "../../assets/cloudtrail-slack-integration-lambda")),
26
+ description: "Formats CloudTrail API calls sent through EventBridge, and posts them directly to Slack or first to an SQS FIFO queue for deduplication",
27
+ handler: "main.handler_event_transformer",
28
+ runtime: lambda.Runtime.PYTHON_3_9,
29
+ timeout: cdk.Duration.seconds(15),
30
+ logRetention: logs.RetentionDays.SIX_MONTHS,
31
+ environment: {
32
+ SLACK_CHANNEL: props.slackChannel,
33
+ DEDUPLICATE_EVENTS: JSON.stringify(!!props.deduplicateEvents),
34
+ ACCOUNT_FRIENDLY_NAMES: JSON.stringify(props.accountFriendlyNames || {}),
35
+ SLACK_WEBHOOK_URL: props.slackWebhookUrl,
36
+ },
37
+ });
38
+ eventTransformer.addToRolePolicy(new iam.PolicyStatement({
39
+ actions: ["iam:ListAccountAliases"],
40
+ resources: ["*"],
41
+ }));
42
+ if (props.infrastructureAlarmAction) {
43
+ const eventTransformerAlarm = eventTransformer
44
+ .metricErrors({
45
+ period: cdk.Duration.minutes(5),
46
+ statistic: cloudwatch.Statistic.SUM,
47
+ })
48
+ .createAlarm(this, "EventTransformerErrorAlarm", {
49
+ threshold: 1,
50
+ evaluationPeriods: 1,
51
+ alarmDescription: "Triggers if the Lambda function that transforms CloudTrail API calls received through EventBridge fails (e.g., it fails to process the event)",
52
+ datapointsToAlarm: 1,
53
+ treatMissingData: cloudwatch.TreatMissingData.IGNORE,
54
+ });
55
+ eventTransformerAlarm.addOkAction(props.infrastructureAlarmAction);
56
+ eventTransformerAlarm.addAlarmAction(props.infrastructureAlarmAction);
57
+ }
58
+ if (props.deduplicateEvents) {
59
+ const deduplicationQueue = new sqs.Queue(this, "Queue", {
60
+ // We explicitly give the queue a name due to bug https://github.com/aws/aws-cdk/issues/5860
61
+ queueName: `${this.node.id.substring(0, 33)}${this.node.addr}`.substring(0, 75) +
62
+ ".fifo",
63
+ fifo: true,
64
+ });
65
+ eventTransformer.addEnvironment("SQS_QUEUE_URL", deduplicationQueue.queueUrl);
66
+ deduplicationQueue.grantSendMessages(eventTransformer);
67
+ const slackForwarder = new lambda.Function(this, "SlackForwarderLambda", {
68
+ code: lambda.Code.fromAsset(path.join(__dirname, "../../assets/cloudtrail-slack-integration-lambda")),
69
+ description: "Polls from an SQS FIFO queue containing formatted CloudTrail API calls and sends them to Slack.",
70
+ handler: "main.handler_slack_forwarder",
71
+ runtime: lambda.Runtime.PYTHON_3_9,
72
+ timeout: cdk.Duration.seconds(15),
73
+ logRetention: logs.RetentionDays.TWO_WEEKS,
74
+ });
75
+ if (props.infrastructureAlarmAction) {
76
+ const slackForwarderAlarm = slackForwarder
77
+ .metricErrors({
78
+ period: cdk.Duration.minutes(5),
79
+ statistic: cloudwatch.Statistic.SUM,
80
+ })
81
+ .createAlarm(this, "SlackForwarderErrorAlarm", {
82
+ threshold: 1,
83
+ alarmDescription: "Triggers if the Lambda function that polls from SQS and posts deduplicated CloudTrail API calls received through EventBridge to Slack fails (e.g., invalid Slack webhook URL)",
84
+ evaluationPeriods: 1,
85
+ datapointsToAlarm: 1,
86
+ treatMissingData: cloudwatch.TreatMissingData.IGNORE,
87
+ });
88
+ slackForwarderAlarm.addOkAction(props.infrastructureAlarmAction);
89
+ slackForwarderAlarm.addAlarmAction(props.infrastructureAlarmAction);
90
+ }
91
+ slackForwarder.addEventSource(new sources.SqsEventSource(deduplicationQueue));
92
+ }
93
+ if (props.rolesToMonitor && props.rolesToMonitor.length > 0) {
94
+ new events.Rule(this, "RuleForAssumeRole", {
95
+ enabled: true,
96
+ targets: [new targets.LambdaFunction(eventTransformer)],
97
+ eventPattern: {
98
+ detail: {
99
+ eventName: ["AssumeRole"],
100
+ requestParameters: {
101
+ roleArn: props.rolesToMonitor,
102
+ },
103
+ },
104
+ },
105
+ });
106
+ }
107
+ if (props.monitorRootUserActions !== false) {
108
+ // Triggers when the root password has been changed
109
+ new events.Rule(this, "RuleForRootUserPasswordChange", {
110
+ enabled: true,
111
+ targets: [new targets.LambdaFunction(eventTransformer)],
112
+ eventPattern: {
113
+ detail: {
114
+ userIdentity: {
115
+ type: ["Root"],
116
+ },
117
+ eventName: ["PasswordUpdated"],
118
+ eventType: ["AwsConsoleSignIn"],
119
+ },
120
+ },
121
+ });
122
+ // Triggers when MFA for the root user has been set up
123
+ new events.Rule(this, "RuleForRootUserMfaChange", {
124
+ enabled: true,
125
+ targets: [new targets.LambdaFunction(eventTransformer)],
126
+ eventPattern: {
127
+ detail: {
128
+ userIdentity: {
129
+ type: ["Root"],
130
+ },
131
+ eventName: ["EnableMFADevice"],
132
+ requestParameters: {
133
+ userName: ["AWS ROOT USER"],
134
+ },
135
+ },
136
+ },
137
+ });
138
+ // Triggers when a root user succesfully logs in to the console
139
+ new events.Rule(this, "RuleForRootUserSuccessfulLogin", {
140
+ enabled: true,
141
+ targets: [new targets.LambdaFunction(eventTransformer)],
142
+ eventPattern: {
143
+ detail: {
144
+ userIdentity: {
145
+ type: ["Root"],
146
+ },
147
+ eventName: ["ConsoleLogin"],
148
+ eventType: ["AwsConsoleSignIn"],
149
+ responseElements: {
150
+ ConsoleLogin: ["Success"],
151
+ },
152
+ },
153
+ },
154
+ });
155
+ // Triggers for bad login attemps for root user (e.g., wrong password)
156
+ new events.Rule(this, "RuleForRootUserUnsuccessfulLogin", {
157
+ enabled: true,
158
+ targets: [new targets.LambdaFunction(eventTransformer)],
159
+ eventPattern: {
160
+ detail: {
161
+ userIdentity: {
162
+ type: ["Root"],
163
+ },
164
+ eventName: ["ConsoleLogin"],
165
+ eventType: ["AwsConsoleSignIn"],
166
+ responseElements: {
167
+ ConsoleLogin: ["Failure"],
168
+ },
169
+ },
170
+ },
171
+ });
172
+ // Triggered when password reset has been requested
173
+ new events.Rule(this, "RuleForRootUserPasswordRecoveryRequest", {
174
+ enabled: true,
175
+ targets: [new targets.LambdaFunction(eventTransformer)],
176
+ eventPattern: {
177
+ detail: {
178
+ userIdentity: {
179
+ type: ["Root"],
180
+ },
181
+ eventName: ["PasswordRecoveryRequested"],
182
+ eventType: ["AwsConsoleSignIn"],
183
+ responseElements: {
184
+ PasswordRecoveryRequested: ["Success"],
185
+ },
186
+ },
187
+ },
188
+ });
189
+ // Triggered when password has been successfully reset
190
+ new events.Rule(this, "RuleForRootUserPasswordRecoveryComplete", {
191
+ enabled: true,
192
+ targets: [new targets.LambdaFunction(eventTransformer)],
193
+ eventPattern: {
194
+ detail: {
195
+ userIdentity: {
196
+ type: ["Root"],
197
+ },
198
+ eventName: ["PasswordRecoveryCompleted"],
199
+ eventType: ["AwsConsoleSignIn"],
200
+ responseElements: {
201
+ PasswordRecoveryCompleted: ["Success"],
202
+ },
203
+ },
204
+ },
205
+ });
206
+ }
207
+ }
208
+ }
209
+ exports.CloudTrailSlackIntegration = CloudTrailSlackIntegration;
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudtrail-slack-integration.js","sourceRoot":"","sources":["../../src/cloudtrail-slack-integration/cloudtrail-slack-integration.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"@aws-cdk/core\"\nimport * as iam from \"@aws-cdk/aws-iam\"\nimport * as logs from \"@aws-cdk/aws-logs\"\nimport * as cloudwatch from \"@aws-cdk/aws-cloudwatch\"\nimport * as lambda from \"@aws-cdk/aws-lambda\"\nimport * as events from \"@aws-cdk/aws-events\"\nimport * as sources from \"@aws-cdk/aws-lambda-event-sources\"\nimport * as sqs from \"@aws-cdk/aws-sqs\"\nimport * as targets from \"@aws-cdk/aws-events-targets\"\nimport * as path from \"path\"\n\nexport interface CloudTrailSlackIntegrationProps extends cdk.StackProps {\n  /**\n   * A key-value pair of AWS account IDs and friendly names of these accounts\n   * to use when sending messages to Slack.\n   */\n  accountFriendlyNames?: {\n    [key: string]: string\n  }\n  slackWebhookUrl: string\n  slackChannel: string\n  /**\n   * A list of ARNs of roles in the current account to monitor usage of.\n   */\n  rolesToMonitor?: string[]\n  /**\n   * Whether to monitor various IAM API calls associated with the current account's root user (e.g., console login, password reset, etc.)\n   *\n   * @default true\n   */\n  monitorRootUserActions?: boolean\n  /**\n   * Whether to set up additional AWS infrastructure to deduplicate CloudTrail events in order to avoid duplicate Slack messages. May be used to decrease noise.\n   *\n   * @default false\n   */\n  deduplicateEvents?: boolean\n  /**\n   * If supplied, CloudWatch alarms will be created for the construct's underlying infrastructure (e.g., Lambda functions) and the action will be used to notify on OK and ALARM actions.\n   */\n  infrastructureAlarmAction?: cloudwatch.IAlarmAction\n}\n\n/**\n * Forward a predefined set of CloudTrail API events to Slack using EventBridge, Lambda\n * and an optional SQS FIFO queue for deduplicating events.\n * The API events are limited to monitoring access to the current account's root user and/or specific IAM roles.\n *\n * NOTE: The construct needs to be provisioned in us-east-1, and requires an existing CloudTrail set up in that region.\n */\nexport class CloudTrailSlackIntegration extends cdk.Construct {\n  constructor(\n    scope: cdk.Construct,\n    id: string,\n    props: CloudTrailSlackIntegrationProps,\n  ) {\n    super(scope, id)\n\n    const eventTransformer = new lambda.Function(\n      this,\n      \"EventTransformerLambda\",\n      {\n        code: lambda.Code.fromAsset(\n          path.join(\n            __dirname,\n            \"../../assets/cloudtrail-slack-integration-lambda\",\n          ),\n        ),\n        description:\n          \"Formats CloudTrail API calls sent through EventBridge, and posts them directly to Slack or first to an SQS FIFO queue for deduplication\",\n        handler: \"main.handler_event_transformer\",\n        runtime: lambda.Runtime.PYTHON_3_9,\n        timeout: cdk.Duration.seconds(15),\n        logRetention: logs.RetentionDays.SIX_MONTHS,\n        environment: {\n          SLACK_CHANNEL: props.slackChannel,\n          DEDUPLICATE_EVENTS: JSON.stringify(!!props.deduplicateEvents),\n          ACCOUNT_FRIENDLY_NAMES: JSON.stringify(\n            props.accountFriendlyNames || {},\n          ),\n          SLACK_WEBHOOK_URL: props.slackWebhookUrl,\n        },\n      },\n    )\n    eventTransformer.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\"iam:ListAccountAliases\"],\n        resources: [\"*\"],\n      }),\n    )\n    if (props.infrastructureAlarmAction) {\n      const eventTransformerAlarm = eventTransformer\n        .metricErrors({\n          period: cdk.Duration.minutes(5),\n          statistic: cloudwatch.Statistic.SUM,\n        })\n        .createAlarm(this, \"EventTransformerErrorAlarm\", {\n          threshold: 1,\n          evaluationPeriods: 1,\n          alarmDescription:\n            \"Triggers if the Lambda function that transforms CloudTrail API calls received through EventBridge fails (e.g., it fails to process the event)\",\n          datapointsToAlarm: 1,\n          treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n        })\n      eventTransformerAlarm.addOkAction(props.infrastructureAlarmAction)\n      eventTransformerAlarm.addAlarmAction(props.infrastructureAlarmAction)\n    }\n    if (props.deduplicateEvents) {\n      const deduplicationQueue = new sqs.Queue(this, \"Queue\", {\n        // We explicitly give the queue a name due to bug https://github.com/aws/aws-cdk/issues/5860\n        queueName:\n          `${this.node.id.substring(0, 33)}${this.node.addr}`.substring(0, 75) +\n          \".fifo\",\n        fifo: true,\n      })\n      eventTransformer.addEnvironment(\n        \"SQS_QUEUE_URL\",\n        deduplicationQueue.queueUrl,\n      )\n      deduplicationQueue.grantSendMessages(eventTransformer)\n      const slackForwarder = new lambda.Function(this, \"SlackForwarderLambda\", {\n        code: lambda.Code.fromAsset(\n          path.join(\n            __dirname,\n            \"../../assets/cloudtrail-slack-integration-lambda\",\n          ),\n        ),\n        description:\n          \"Polls from an SQS FIFO queue containing formatted CloudTrail API calls and sends them to Slack.\",\n        handler: \"main.handler_slack_forwarder\",\n        runtime: lambda.Runtime.PYTHON_3_9,\n        timeout: cdk.Duration.seconds(15),\n        logRetention: logs.RetentionDays.TWO_WEEKS,\n      })\n\n      if (props.infrastructureAlarmAction) {\n        const slackForwarderAlarm = slackForwarder\n          .metricErrors({\n            period: cdk.Duration.minutes(5),\n            statistic: cloudwatch.Statistic.SUM,\n          })\n          .createAlarm(this, \"SlackForwarderErrorAlarm\", {\n            threshold: 1,\n            alarmDescription:\n              \"Triggers if the Lambda function that polls from SQS and posts deduplicated CloudTrail API calls received through EventBridge to Slack fails (e.g., invalid Slack webhook URL)\",\n            evaluationPeriods: 1,\n            datapointsToAlarm: 1,\n            treatMissingData: cloudwatch.TreatMissingData.IGNORE,\n          })\n        slackForwarderAlarm.addOkAction(props.infrastructureAlarmAction)\n        slackForwarderAlarm.addAlarmAction(props.infrastructureAlarmAction)\n      }\n      slackForwarder.addEventSource(\n        new sources.SqsEventSource(deduplicationQueue),\n      )\n    }\n\n    if (props.rolesToMonitor && props.rolesToMonitor.length > 0) {\n      new events.Rule(this, \"RuleForAssumeRole\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            eventName: [\"AssumeRole\"],\n            requestParameters: {\n              roleArn: props.rolesToMonitor,\n            },\n          },\n        },\n      })\n    }\n\n    if (props.monitorRootUserActions !== false) {\n      // Triggers when the root password has been changed\n      new events.Rule(this, \"RuleForRootUserPasswordChange\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"PasswordUpdated\"],\n            eventType: [\"AwsConsoleSignIn\"],\n          },\n        },\n      })\n\n      // Triggers when MFA for the root user has been set up\n      new events.Rule(this, \"RuleForRootUserMfaChange\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"EnableMFADevice\"],\n            requestParameters: {\n              userName: [\"AWS ROOT USER\"],\n            },\n          },\n        },\n      })\n\n      // Triggers when a root user succesfully logs in to the console\n      new events.Rule(this, \"RuleForRootUserSuccessfulLogin\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"ConsoleLogin\"],\n            eventType: [\"AwsConsoleSignIn\"],\n            responseElements: {\n              ConsoleLogin: [\"Success\"],\n            },\n          },\n        },\n      })\n\n      // Triggers for bad login attemps for root user (e.g., wrong password)\n      new events.Rule(this, \"RuleForRootUserUnsuccessfulLogin\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"ConsoleLogin\"],\n            eventType: [\"AwsConsoleSignIn\"],\n            responseElements: {\n              ConsoleLogin: [\"Failure\"],\n            },\n          },\n        },\n      })\n\n      // Triggered when password reset has been requested\n      new events.Rule(this, \"RuleForRootUserPasswordRecoveryRequest\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"PasswordRecoveryRequested\"],\n            eventType: [\"AwsConsoleSignIn\"],\n            responseElements: {\n              PasswordRecoveryRequested: [\"Success\"],\n            },\n          },\n        },\n      })\n\n      // Triggered when password has been successfully reset\n      new events.Rule(this, \"RuleForRootUserPasswordRecoveryComplete\", {\n        enabled: true,\n        targets: [new targets.LambdaFunction(eventTransformer)],\n        eventPattern: {\n          detail: {\n            userIdentity: {\n              type: [\"Root\"],\n            },\n            eventName: [\"PasswordRecoveryCompleted\"],\n            eventType: [\"AwsConsoleSignIn\"],\n            responseElements: {\n              PasswordRecoveryCompleted: [\"Success\"],\n            },\n          },\n        },\n      })\n    }\n  }\n}\n"]}
@@ -0,0 +1 @@
1
+ export { CloudTrailSlackIntegration, CloudTrailSlackIntegrationProps, } from "./cloudtrail-slack-integration";
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CloudTrailSlackIntegration = void 0;
4
+ var cloudtrail_slack_integration_1 = require("./cloudtrail-slack-integration");
5
+ Object.defineProperty(exports, "CloudTrailSlackIntegration", { enumerable: true, get: function () { return cloudtrail_slack_integration_1.CloudTrailSlackIntegration; } });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2xvdWR0cmFpbC1zbGFjay1pbnRlZ3JhdGlvbi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSwrRUFHdUM7QUFGckMsMElBQUEsMEJBQTBCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQge1xuICBDbG91ZFRyYWlsU2xhY2tJbnRlZ3JhdGlvbixcbiAgQ2xvdWRUcmFpbFNsYWNrSW50ZWdyYXRpb25Qcm9wcyxcbn0gZnJvbSBcIi4vY2xvdWR0cmFpbC1zbGFjay1pbnRlZ3JhdGlvblwiXG4iXX0=
@@ -54,7 +54,6 @@ const startDeployHandler = async (event) => {
54
54
  "status",
55
55
  "taskDefinitionArn",
56
56
  ];
57
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
58
57
  const updatedSpec = {
59
58
  ...Object.fromEntries(Object.entries(prevTaskDefinition).filter(([key]) => !exclude.includes(key))),
60
59
  containerDefinitions: [
@@ -102,4 +101,4 @@ const startDeployHandler = async (event) => {
102
101
  return {};
103
102
  };
104
103
  exports.startDeployHandler = startDeployHandler;
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"start-deploy-handler.js","sourceRoot":"","sources":["../../src/ecs-update-image/start-deploy-handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type { Handler } from \"aws-lambda\"\nimport type * as _AWS from \"aws-sdk\"\n\ninterface ExpectedInput {\n  tag: string\n}\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const startDeployHandler: Handler<Partial<ExpectedInput>> = async (\n  event,\n) => {\n  const AWS = require(\"aws-sdk\")\n  const ecs = new AWS.ECS() as _AWS.ECS\n  const sm = new AWS.SecretsManager() as _AWS.SecretsManager\n\n  function requireEnv(name: string): string {\n    const value = process.env[name]\n    if (value === undefined) {\n      throw new Error(`Missing ${name}`)\n    }\n    return value\n  }\n\n  async function getService(\n    clusterName: string,\n    serviceName: string,\n  ): Promise<AWS.ECS.Service> {\n    const services = await ecs\n      .describeServices({\n        cluster: clusterName,\n        services: [serviceName],\n      })\n      .promise()\n\n    if (services.services?.length !== 1) {\n      throw new Error(`Service not found: ${clusterName}/${serviceName}`)\n    }\n\n    return services.services[0]\n  }\n\n  async function getTaskDefinition(\n    taskDefinition: string,\n  ): Promise<AWS.ECS.TaskDefinition> {\n    return (\n      await ecs\n        .describeTaskDefinition({\n          taskDefinition: taskDefinition,\n        })\n        .promise()\n    ).taskDefinition!\n  }\n\n  async function updateServiceToImage(\n    clusterName: string,\n    serviceName: string,\n    image: string,\n  ) {\n    console.log(`Cluster name: ${clusterName}`)\n    console.log(`Service name: ${serviceName}`)\n\n    const service = await getService(clusterName, serviceName)\n    const prevTaskDefinition = await getTaskDefinition(service.taskDefinition!)\n\n    // Don't bother updating the service if the image is already the latest.\n    const prevImage = prevTaskDefinition.containerDefinitions![0].image!\n    if (prevImage === image) {\n      return\n    }\n\n    console.log(\n      `Updating image for service '${serviceName}' from '${prevImage}' to '${image}'`,\n    )\n\n    const exclude = [\n      \"registeredAt\",\n      \"registeredBy\",\n      \"compatibilities\",\n      \"requiresAttributes\",\n      \"revision\",\n      \"status\",\n      \"taskDefinitionArn\",\n    ]\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const updatedSpec: any = {\n      ...Object.fromEntries(\n        Object.entries(prevTaskDefinition).filter(\n          ([key]) => !exclude.includes(key),\n        ),\n      ),\n      containerDefinitions: [\n        {\n          ...prevTaskDefinition.containerDefinitions![0],\n          image,\n        },\n      ],\n    }\n\n    const updatedTaskDefinition = (\n      await ecs.registerTaskDefinition(updatedSpec).promise()\n    ).taskDefinition!\n\n    await ecs\n      .updateService({\n        cluster: clusterName,\n        service: serviceName,\n        taskDefinition: updatedTaskDefinition.taskDefinitionArn,\n      })\n      .promise()\n\n    console.log(\"Service is updated\")\n  }\n\n  const clusterName = requireEnv(\"CLUSTER_NAME\")\n  const serviceName = requireEnv(\"SERVICE_NAME\")\n  const repositoryUrl = requireEnv(\"REPOSITORY_URL\")\n  const ecrTagSecretArn = requireEnv(\"ECR_TAG_SECRET_ARN\")\n\n  // Validate the input.\n  if (typeof event.tag !== \"string\") {\n    throw new Error(\"Input invalid: \" + JSON.stringify(event, undefined, \"  \"))\n  }\n\n  // Register tag as current target.\n  // This is needed so that CloudFormation deployments, e.g.\n  // updates to the Task Definition, will use the same image.\n  await sm\n    .updateSecret({\n      SecretId: ecrTagSecretArn,\n      SecretString: JSON.stringify({\n        tag: event.tag,\n      }),\n    })\n    .promise()\n\n  // Update the service if we know the service name. This is unknown\n  // during initial deployment of the stack.\n  if (serviceName !== \"\") {\n    const image = `${repositoryUrl}:${event.tag}`\n    await updateServiceToImage(clusterName, serviceName, image)\n  }\n\n  return {}\n}\n"]}
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"start-deploy-handler.js","sourceRoot":"","sources":["../../src/ecs-update-image/start-deploy-handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type { Handler } from \"aws-lambda\"\nimport type * as _AWS from \"aws-sdk\"\nimport { RegisterTaskDefinitionRequest } from \"aws-sdk/clients/ecs\"\n\ninterface ExpectedInput {\n  tag: string\n}\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const startDeployHandler: Handler<Partial<ExpectedInput>> = async (\n  event,\n) => {\n  const AWS = require(\"aws-sdk\")\n  const ecs = new AWS.ECS() as _AWS.ECS\n  const sm = new AWS.SecretsManager() as _AWS.SecretsManager\n\n  function requireEnv(name: string): string {\n    const value = process.env[name]\n    if (value === undefined) {\n      throw new Error(`Missing ${name}`)\n    }\n    return value\n  }\n\n  async function getService(\n    clusterName: string,\n    serviceName: string,\n  ): Promise<AWS.ECS.Service> {\n    const services = await ecs\n      .describeServices({\n        cluster: clusterName,\n        services: [serviceName],\n      })\n      .promise()\n\n    if (services.services?.length !== 1) {\n      throw new Error(`Service not found: ${clusterName}/${serviceName}`)\n    }\n\n    return services.services[0]\n  }\n\n  async function getTaskDefinition(\n    taskDefinition: string,\n  ): Promise<AWS.ECS.TaskDefinition> {\n    return (\n      await ecs\n        .describeTaskDefinition({\n          taskDefinition: taskDefinition,\n        })\n        .promise()\n    ).taskDefinition!\n  }\n\n  async function updateServiceToImage(\n    clusterName: string,\n    serviceName: string,\n    image: string,\n  ) {\n    console.log(`Cluster name: ${clusterName}`)\n    console.log(`Service name: ${serviceName}`)\n\n    const service = await getService(clusterName, serviceName)\n    const prevTaskDefinition = await getTaskDefinition(service.taskDefinition!)\n\n    // Don't bother updating the service if the image is already the latest.\n    const prevImage = prevTaskDefinition.containerDefinitions![0].image!\n    if (prevImage === image) {\n      return\n    }\n\n    console.log(\n      `Updating image for service '${serviceName}' from '${prevImage}' to '${image}'`,\n    )\n\n    const exclude = [\n      \"registeredAt\",\n      \"registeredBy\",\n      \"compatibilities\",\n      \"requiresAttributes\",\n      \"revision\",\n      \"status\",\n      \"taskDefinitionArn\",\n    ]\n\n    const updatedSpec: RegisterTaskDefinitionRequest = {\n      ...Object.fromEntries(\n        Object.entries(prevTaskDefinition).filter(\n          ([key]) => !exclude.includes(key),\n        ),\n      ),\n      containerDefinitions: [\n        {\n          ...prevTaskDefinition.containerDefinitions![0],\n          image,\n        },\n      ],\n    } as RegisterTaskDefinitionRequest\n\n    const updatedTaskDefinition = (\n      await ecs.registerTaskDefinition(updatedSpec).promise()\n    ).taskDefinition!\n\n    await ecs\n      .updateService({\n        cluster: clusterName,\n        service: serviceName,\n        taskDefinition: updatedTaskDefinition.taskDefinitionArn,\n      })\n      .promise()\n\n    console.log(\"Service is updated\")\n  }\n\n  const clusterName = requireEnv(\"CLUSTER_NAME\")\n  const serviceName = requireEnv(\"SERVICE_NAME\")\n  const repositoryUrl = requireEnv(\"REPOSITORY_URL\")\n  const ecrTagSecretArn = requireEnv(\"ECR_TAG_SECRET_ARN\")\n\n  // Validate the input.\n  if (typeof event.tag !== \"string\") {\n    throw new Error(\"Input invalid: \" + JSON.stringify(event, undefined, \"  \"))\n  }\n\n  // Register tag as current target.\n  // This is needed so that CloudFormation deployments, e.g.\n  // updates to the Task Definition, will use the same image.\n  await sm\n    .updateSecret({\n      SecretId: ecrTagSecretArn,\n      SecretString: JSON.stringify({\n        tag: event.tag,\n      }),\n    })\n    .promise()\n\n  // Update the service if we know the service name. This is unknown\n  // during initial deployment of the stack.\n  if (serviceName !== \"\") {\n    const image = `${repositoryUrl}:${event.tag}`\n    await updateServiceToImage(clusterName, serviceName, image)\n  }\n\n  return {}\n}\n"]}