@liflig/cdk 1.52.1 → 2.0.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.
- package/README.md +5 -7
- package/assets/cloudtrail-slack-integration-lambda/main.py +44 -26
- package/lib/alarms/service-alarms.d.ts +6 -5
- package/lib/alarms/service-alarms.js +8 -7
- package/lib/alarms/slack-alarm.d.ts +5 -5
- package/lib/alarms/slack-alarm.js +9 -9
- package/lib/bastion-host.d.ts +4 -4
- package/lib/bastion-host.js +6 -5
- package/lib/build-artifacts/index.d.ts +3 -3
- package/lib/build-artifacts/index.js +7 -6
- package/lib/cdk-deploy/cdk-deploy.d.ts +4 -3
- package/lib/cdk-deploy/cdk-deploy.js +9 -8
- package/lib/cdk-pipelines/liflig-cdk-pipeline.d.ts +7 -7
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +28 -23
- package/lib/cdk-pipelines/slack-notification.d.ts +4 -4
- package/lib/cdk-pipelines/slack-notification.js +7 -6
- package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.d.ts +10 -6
- package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.js +13 -12
- package/lib/configure-parameters/configure-parameters.d.ts +6 -6
- package/lib/configure-parameters/configure-parameters.js +5 -5
- package/lib/cross-region-ssm-parameter.d.ts +3 -3
- package/lib/cross-region-ssm-parameter.js +4 -4
- package/lib/ecs/cluster.d.ts +7 -7
- package/lib/ecs/cluster.js +9 -8
- package/lib/ecs/fargate-service.d.ts +8 -7
- package/lib/ecs/fargate-service.js +11 -10
- package/lib/ecs/listener-rule.d.ts +12 -5
- package/lib/ecs/listener-rule.js +8 -8
- package/lib/ecs-update-image/ecs-update-image.d.ts +6 -6
- package/lib/ecs-update-image/ecs-update-image.js +6 -5
- package/lib/ecs-update-image/tag.d.ts +4 -4
- package/lib/ecs-update-image/tag.js +5 -5
- package/lib/griid/artefact-bucket.d.ts +3 -3
- package/lib/griid/artefact-bucket.js +5 -5
- package/lib/griid/index.d.ts +3 -3
- package/lib/griid/index.js +3 -3
- package/lib/hosted-zone-with-param.d.ts +5 -5
- package/lib/hosted-zone-with-param.js +6 -5
- package/lib/kinesis/kinesis-to-datadog-stream.d.ts +4 -4
- package/lib/kinesis/kinesis-to-datadog-stream.js +10 -9
- package/lib/load-balancer/load-balancer.d.ts +7 -7
- package/lib/load-balancer/load-balancer.js +8 -7
- package/lib/pipelines/deploy-env.d.ts +6 -6
- package/lib/pipelines/deploy-env.js +11 -10
- package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +3 -3
- package/lib/pipelines/liflig-cdk-deployer-deps.js +5 -4
- package/lib/pipelines/pipeline.d.ts +6 -6
- package/lib/pipelines/pipeline.js +11 -10
- package/lib/platform/platform.d.ts +6 -6
- package/lib/platform/platform.js +5 -5
- package/lib/rds/database.d.ts +7 -6
- package/lib/rds/database.js +6 -5
- package/lib/ses/configurationsetsnsdestination/index.d.ts +4 -4
- package/lib/ses/configurationsetsnsdestination/index.js +8 -7
- package/lib/ses/sesdomain/index.d.ts +5 -4
- package/lib/ses/sesdomain/index.js +9 -8
- package/lib/ses/sesverifyemail/index.d.ts +4 -3
- package/lib/ses/sesverifyemail/index.js +8 -7
- package/lib/ssm-parameter-backed-resource.d.ts +5 -5
- package/lib/ssm-parameter-backed-resource.js +5 -4
- package/lib/ssm-parameter-reader.d.ts +3 -3
- package/lib/ssm-parameter-reader.js +3 -3
- package/lib/tags.d.ts +3 -2
- package/lib/tags.js +3 -3
- package/lib/utils.d.ts +2 -2
- package/lib/utils.js +2 -2
- package/lib/webapp/security-headers.d.ts +4 -4
- package/lib/webapp/security-headers.js +4 -4
- package/lib/webapp/webapp.d.ts +8 -8
- package/lib/webapp/webapp.js +9 -9
- package/lib/webapp-deploy-via-role.d.ts +3 -3
- package/lib/webapp-deploy-via-role.js +5 -4
- package/package.json +24 -75
package/README.md
CHANGED
|
@@ -1,14 +1,12 @@
|
|
|
1
1
|
# Liflig CDK
|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
This is a collection of reusable constructs and patterns for
|
|
4
|
+
CDK setups, for use within Liflig.
|
|
4
5
|
|
|
5
6
|
## State of repository and package
|
|
6
7
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
and as such will not be following semantic versioning strictly. There will be
|
|
10
|
-
breaking changes across both minor and patch releases, as we will be
|
|
11
|
-
coordinating changes internally.
|
|
8
|
+
We do not expect others to depend on this, and as such will not be following semantic versioning strictly.
|
|
9
|
+
There will be breaking changes across both minor and patch releases, as we will be coordinating changes internally.
|
|
12
10
|
|
|
13
11
|
CDK has some major issues for 3rd party library authors which
|
|
14
12
|
are not yet resolved. Some relevant information:
|
|
@@ -30,7 +28,7 @@ are not yet resolved. Some relevant information:
|
|
|
30
28
|
npm run test -- -u
|
|
31
29
|
```
|
|
32
30
|
|
|
33
|
-
Investigate any changes before
|
|
31
|
+
Investigate any changes before committing.
|
|
34
32
|
|
|
35
33
|
## Testing library changes before releasing
|
|
36
34
|
|
|
@@ -11,17 +11,29 @@ import os
|
|
|
11
11
|
import logging
|
|
12
12
|
import json
|
|
13
13
|
import urllib.request
|
|
14
|
+
import re
|
|
14
15
|
import boto3
|
|
15
16
|
|
|
16
17
|
logger = logging.getLogger()
|
|
17
18
|
logger.setLevel(logging.INFO)
|
|
18
19
|
|
|
20
|
+
def augment_strings_with_friendly_names(strings, friendly_names):
|
|
21
|
+
"""A helper method for augmenting various values (e.g., AWS account ID) in
|
|
22
|
+
a list of strings with a more friendly name"""
|
|
23
|
+
# We avoid replacing values that are directly prefixed and/or suffixed with ':'
|
|
24
|
+
# as it is most likely an ARN or similiar. We don't want to replace account IDs
|
|
25
|
+
# inside ARNs as this would look messy.This is a quite basic heuristic, but it should allow
|
|
26
|
+
# us to easily replace most relevant values (e.g., principal ID, account ID, etc.) with
|
|
27
|
+
# friendly names without a complicated regex.
|
|
28
|
+
pattern = re.compile("|".join([f"(?<!:)({re.escape(key)})(?!:)" for key in friendly_names]))
|
|
29
|
+
return [pattern.sub(lambda m: m[0] + f" ({friendly_names[m.string[m.start():m.end()]]})", s) for s in strings]
|
|
19
30
|
|
|
20
|
-
|
|
31
|
+
|
|
32
|
+
def get_slack_payload_for_assume_role_event(event, friendly_names):
|
|
21
33
|
"""Parse a CloudTrail event related to the API call sts:AssumeRole,
|
|
22
34
|
and return a Slack-formatted attachment"""
|
|
23
|
-
event_account_id = event["account"]
|
|
24
35
|
event_detail = event["detail"]
|
|
36
|
+
recipient_account_id = event_detail["recipientAccountId"]
|
|
25
37
|
request_parameters = event_detail.get("requestParameters", {}) or {}
|
|
26
38
|
|
|
27
39
|
timestamp = event_detail["eventTime"]
|
|
@@ -32,12 +44,14 @@ def get_slack_payload_for_assume_role_event(event, account_friendly_names):
|
|
|
32
44
|
source_ip = event_detail.get("sourceIPAddress", "")
|
|
33
45
|
role_arn = request_parameters.get("roleArn", "")
|
|
34
46
|
|
|
35
|
-
fallback = f"Sensitive role accessed in '{
|
|
36
|
-
pretext_messages = [f":warning: Sensitive role in `{
|
|
47
|
+
fallback = f"Sensitive role accessed in '{recipient_account_id}'"
|
|
48
|
+
pretext_messages = [f":warning: Sensitive role in `{recipient_account_id}` assumed by"]
|
|
37
49
|
if principal_id.startswith("AIDA"):
|
|
38
50
|
pretext_messages.append("IAM user")
|
|
39
51
|
elif principal_id.startswith("AROA"):
|
|
40
|
-
|
|
52
|
+
# The other part of the principal ID for a role is the name of the session
|
|
53
|
+
principal_id = principal_id.split(":")[0]
|
|
54
|
+
pretext_messages.append(f"IAM role")
|
|
41
55
|
else:
|
|
42
56
|
pretext_messages.append("principal")
|
|
43
57
|
pretext_messages.append(f"in `{principal_account_id}`")
|
|
@@ -52,9 +66,11 @@ def get_slack_payload_for_assume_role_event(event, account_friendly_names):
|
|
|
52
66
|
f"*Timestamp:* `{timestamp}`",
|
|
53
67
|
]
|
|
54
68
|
text = "\n".join(line for line in text if line)
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
fallback =
|
|
69
|
+
|
|
70
|
+
try:
|
|
71
|
+
pretext, fallback, text = augment_strings_with_friendly_names([pretext, fallback, text], friendly_names)
|
|
72
|
+
except:
|
|
73
|
+
logger.exception("Failed to augment strings with friendly names")
|
|
58
74
|
return {
|
|
59
75
|
"attachments": [
|
|
60
76
|
{
|
|
@@ -69,17 +85,17 @@ def get_slack_payload_for_assume_role_event(event, account_friendly_names):
|
|
|
69
85
|
|
|
70
86
|
|
|
71
87
|
def get_fallback_slack_payload_for_event(
|
|
72
|
-
event,
|
|
88
|
+
event, friendly_names, fallback_parse_behavior=""
|
|
73
89
|
):
|
|
74
90
|
"""Parse a generic CloudTrail event related to an API call
|
|
75
91
|
and return a Slack-formatted attachment"""
|
|
76
|
-
event_account_id = event["account"]
|
|
77
92
|
event_detail = event["detail"]
|
|
78
93
|
event_name = event_detail["eventName"]
|
|
79
94
|
event_type = event_detail["eventType"]
|
|
80
95
|
event_time = event_detail["eventTime"]
|
|
81
|
-
|
|
82
|
-
|
|
96
|
+
recipient_account_id = event_detail["recipientAccountId"]
|
|
97
|
+
pretext = f":warning: CloudTrail event in account `{recipient_account_id}`"
|
|
98
|
+
fallback = f"CloudTrail event in account '{recipient_account_id}'"
|
|
83
99
|
if fallback_parse_behavior == "DUMP_EVENT":
|
|
84
100
|
text = "\n".join(
|
|
85
101
|
["*Event:*", "```", json.dumps(event, sort_keys=True, indent=2), "```"]
|
|
@@ -117,9 +133,11 @@ def get_fallback_slack_payload_for_event(
|
|
|
117
133
|
# Filter out empty strings
|
|
118
134
|
text = "\n".join(line for line in text if line)
|
|
119
135
|
|
|
120
|
-
|
|
121
|
-
pretext = pretext
|
|
122
|
-
|
|
136
|
+
try:
|
|
137
|
+
pretext, fallback, text = augment_strings_with_friendly_names([pretext, fallback, text], friendly_names)
|
|
138
|
+
except:
|
|
139
|
+
logger.exception("Failed to augment strings with friendly names")
|
|
140
|
+
|
|
123
141
|
return {
|
|
124
142
|
"attachments": [
|
|
125
143
|
{
|
|
@@ -133,16 +151,16 @@ def get_fallback_slack_payload_for_event(
|
|
|
133
151
|
}
|
|
134
152
|
|
|
135
153
|
|
|
136
|
-
def
|
|
154
|
+
def get_augmented_friendly_names(event, friendly_names):
|
|
137
155
|
"""Return an augmented dictionary containing the alias of the current
|
|
138
|
-
AWS account if relevant"""
|
|
139
|
-
|
|
156
|
+
AWS account as a friendly name for the current account ID if relevant"""
|
|
157
|
+
augmented_friendly_names = {**friendly_names}
|
|
140
158
|
try:
|
|
141
159
|
event_account_id = event["account"]
|
|
142
160
|
event_detail = event["detail"]
|
|
143
161
|
recipient_account_id = event_detail["recipientAccountId"]
|
|
144
162
|
if (
|
|
145
|
-
not
|
|
163
|
+
not friendly_names.get(event_account_id, "")
|
|
146
164
|
and event_account_id == recipient_account_id
|
|
147
165
|
):
|
|
148
166
|
logger.info(
|
|
@@ -152,11 +170,11 @@ def get_augmented_account_friendly_names(event, account_friendly_names):
|
|
|
152
170
|
iam = boto3.client("iam")
|
|
153
171
|
aliases = iam.list_account_aliases()["AccountAliases"]
|
|
154
172
|
if len(aliases):
|
|
155
|
-
|
|
173
|
+
augmented_friendly_names[event_account_id] = aliases[0]
|
|
156
174
|
except:
|
|
157
175
|
logger.exception("Failed to look up alias of current AWS account")
|
|
158
176
|
|
|
159
|
-
return
|
|
177
|
+
return augmented_friendly_names
|
|
160
178
|
|
|
161
179
|
|
|
162
180
|
def post_to_slack(slack_payload, slack_webhook_url):
|
|
@@ -178,15 +196,15 @@ def handler_event_transformer(event, context):
|
|
|
178
196
|
"""Lambda handler for the event transformer Lambda"""
|
|
179
197
|
logger.info("Triggered with event: %s", json.dumps(event, indent=2))
|
|
180
198
|
|
|
181
|
-
|
|
199
|
+
friendly_names = json.loads(os.environ["FRIENDLY_NAMES"])
|
|
182
200
|
slack_webhook_url = os.environ["SLACK_WEBHOOK_URL"]
|
|
183
201
|
slack_channel = os.environ["SLACK_CHANNEL"]
|
|
184
202
|
sqs_queue_url = os.environ.get("SQS_QUEUE_URL", "")
|
|
185
203
|
fallback_parse_behavior = os.environ.get("FALLBACK_PARSE_BEHAVIOR", "")
|
|
186
204
|
deduplicate_events = os.environ.get("DEDUPLICATE_EVENTS", "false") == "true"
|
|
187
205
|
|
|
188
|
-
|
|
189
|
-
event,
|
|
206
|
+
friendly_names = get_augmented_friendly_names(
|
|
207
|
+
event, friendly_names
|
|
190
208
|
)
|
|
191
209
|
|
|
192
210
|
if not event["detail-type"].endswith("via CloudTrail"):
|
|
@@ -197,7 +215,7 @@ def handler_event_transformer(event, context):
|
|
|
197
215
|
try:
|
|
198
216
|
if event["detail"]["eventName"] == "AssumeRole":
|
|
199
217
|
slack_payload = get_slack_payload_for_assume_role_event(
|
|
200
|
-
event,
|
|
218
|
+
event, friendly_names
|
|
201
219
|
)
|
|
202
220
|
except:
|
|
203
221
|
logger.exception("Failed to parse event using predefined schema")
|
|
@@ -205,7 +223,7 @@ def handler_event_transformer(event, context):
|
|
|
205
223
|
logger.warn("Using a fallback schema to parse event")
|
|
206
224
|
slack_payload = get_fallback_slack_payload_for_event(
|
|
207
225
|
event,
|
|
208
|
-
|
|
226
|
+
friendly_names,
|
|
209
227
|
fallback_parse_behavior=fallback_parse_behavior,
|
|
210
228
|
)
|
|
211
229
|
slack_payload = {**slack_payload, "channel": slack_channel}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
|
-
import * as
|
|
2
|
-
import * as
|
|
3
|
-
import * as
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
|
|
3
|
+
import * as logs from "aws-cdk-lib/aws-logs";
|
|
4
|
+
import * as cdk from "aws-cdk-lib";
|
|
4
5
|
export interface ServiceAlarmsProps extends cdk.StackProps {
|
|
5
6
|
action: cloudwatch.IAlarmAction;
|
|
6
7
|
serviceName: string;
|
|
@@ -13,10 +14,10 @@ export interface ServiceAlarmsProps extends cdk.StackProps {
|
|
|
13
14
|
*
|
|
14
15
|
* See SlackAlarm construct for SNS Action.
|
|
15
16
|
*/
|
|
16
|
-
export declare class ServiceAlarms extends
|
|
17
|
+
export declare class ServiceAlarms extends constructs.Construct {
|
|
17
18
|
private readonly action;
|
|
18
19
|
private readonly serviceName;
|
|
19
|
-
constructor(scope:
|
|
20
|
+
constructor(scope: constructs.Construct, id: string, props: ServiceAlarmsProps);
|
|
20
21
|
/**
|
|
21
22
|
* For logs stored as JSON, monitor log entries logged
|
|
22
23
|
* with level ERROR or higher, as well as any requests
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.ServiceAlarms = void 0;
|
|
4
|
-
const
|
|
5
|
-
const
|
|
6
|
-
const
|
|
4
|
+
const constructs = require("constructs");
|
|
5
|
+
const cloudwatch = require("aws-cdk-lib/aws-cloudwatch");
|
|
6
|
+
const logs = require("aws-cdk-lib/aws-logs");
|
|
7
|
+
const cdk = require("aws-cdk-lib");
|
|
7
8
|
/**
|
|
8
9
|
* Various alarms and monitoring.
|
|
9
10
|
*
|
|
@@ -12,7 +13,7 @@ const cdk = require("@aws-cdk/core");
|
|
|
12
13
|
*
|
|
13
14
|
* See SlackAlarm construct for SNS Action.
|
|
14
15
|
*/
|
|
15
|
-
class ServiceAlarms extends
|
|
16
|
+
class ServiceAlarms extends constructs.Construct {
|
|
16
17
|
constructor(scope, id, props) {
|
|
17
18
|
super(scope, id);
|
|
18
19
|
this.action = props.action;
|
|
@@ -59,7 +60,7 @@ class ServiceAlarms extends cdk.Construct {
|
|
|
59
60
|
namespace: "AWS/ApplicationELB",
|
|
60
61
|
statistic: "Average",
|
|
61
62
|
period: cdk.Duration.seconds(60),
|
|
62
|
-
|
|
63
|
+
dimensionsMap: {
|
|
63
64
|
TargetGroup: props.targetGroupFullName,
|
|
64
65
|
LoadBalancer: props.loadBalancerFullName,
|
|
65
66
|
},
|
|
@@ -77,7 +78,7 @@ class ServiceAlarms extends cdk.Construct {
|
|
|
77
78
|
namespace: "AWS/ApplicationELB",
|
|
78
79
|
statistic: "Sum",
|
|
79
80
|
period: cdk.Duration.seconds(60),
|
|
80
|
-
|
|
81
|
+
dimensionsMap: {
|
|
81
82
|
TargetGroup: props.targetGroupFullName,
|
|
82
83
|
LoadBalancer: props.loadBalancerFullName,
|
|
83
84
|
},
|
|
@@ -93,4 +94,4 @@ class ServiceAlarms extends cdk.Construct {
|
|
|
93
94
|
}
|
|
94
95
|
}
|
|
95
96
|
exports.ServiceAlarms = ServiceAlarms;
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
97
|
+
//# 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 constructs from \"constructs\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as cdk from \"aws-cdk-lib\"\n\nexport interface ServiceAlarmsProps extends cdk.StackProps {\n  action: cloudwatch.IAlarmAction\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  }): 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(this.action)\n    errorAlarm.addOkAction(this.action)\n  }\n\n  /**\n   * Monitor healthy host count and connection errors from load balancer.\n   */\n  addTargetGroupAlarm(props: {\n    targetGroupFullName: string\n    loadBalancerFullName: string\n  }): void {\n    const healthAlarm = new cloudwatch.Metric({\n      metricName: \"HealthyHostCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Average\",\n      period: cdk.Duration.seconds(60),\n      dimensionsMap: {\n        TargetGroup: props.targetGroupFullName,\n        LoadBalancer: props.loadBalancerFullName,\n      },\n    }).createAlarm(this, \"HealthAlarm\", {\n      alarmDescription:\n        \"Service might be unavailable! It is not responding to health checks.\",\n      comparisonOperator: cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD,\n      evaluationPeriods: 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.BREACHING,\n    })\n\n    healthAlarm.addAlarmAction(this.action)\n    healthAlarm.addOkAction(this.action)\n\n    const connectionAlarm = new cloudwatch.Metric({\n      metricName: \"TargetConnectionErrorCount\",\n      namespace: \"AWS/ApplicationELB\",\n      statistic: \"Sum\",\n      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 could not connect to target\",\n      evaluationPeriods: 1,\n      threshold: 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n\n    connectionAlarm.addAlarmAction(this.action)\n    connectionAlarm.addOkAction(this.action)\n  }\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import * as
|
|
2
|
-
import * as
|
|
3
|
-
import * as
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as cloudwatchActions from "aws-cdk-lib/aws-cloudwatch-actions";
|
|
3
|
+
import * as sns from "aws-cdk-lib/aws-sns";
|
|
4
4
|
export interface SlackAlarmProps {
|
|
5
5
|
projectName: string;
|
|
6
6
|
envName: string;
|
|
@@ -11,8 +11,8 @@ export interface SlackAlarmProps {
|
|
|
11
11
|
* SNS Topic that can be used to action alarms, with a Lambda
|
|
12
12
|
* that will send a message to Slack for the alarm.
|
|
13
13
|
*/
|
|
14
|
-
export declare class SlackAlarm extends
|
|
14
|
+
export declare class SlackAlarm extends constructs.Construct {
|
|
15
15
|
readonly alarmTopic: sns.Topic;
|
|
16
16
|
readonly snsAction: cloudwatchActions.SnsAction;
|
|
17
|
-
constructor(scope:
|
|
17
|
+
constructor(scope: constructs.Construct, id: string, props: SlackAlarmProps);
|
|
18
18
|
}
|
|
@@ -1,18 +1,18 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.SlackAlarm = void 0;
|
|
4
|
-
const
|
|
5
|
-
const
|
|
6
|
-
const
|
|
7
|
-
const
|
|
8
|
-
const
|
|
9
|
-
const
|
|
4
|
+
const constructs = require("constructs");
|
|
5
|
+
const cloudwatchActions = require("aws-cdk-lib/aws-cloudwatch-actions");
|
|
6
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
7
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
8
|
+
const sns = require("aws-cdk-lib/aws-sns");
|
|
9
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
10
10
|
const path = require("path");
|
|
11
11
|
/**
|
|
12
12
|
* SNS Topic that can be used to action alarms, with a Lambda
|
|
13
13
|
* that will send a message to Slack for the alarm.
|
|
14
14
|
*/
|
|
15
|
-
class SlackAlarm extends
|
|
15
|
+
class SlackAlarm extends constructs.Construct {
|
|
16
16
|
constructor(scope, id, props) {
|
|
17
17
|
super(scope, id);
|
|
18
18
|
this.alarmTopic = new sns.Topic(this, "Topic");
|
|
@@ -23,7 +23,7 @@ class SlackAlarm extends cdk.Construct {
|
|
|
23
23
|
handler: "index.handler",
|
|
24
24
|
memorySize: 128,
|
|
25
25
|
runtime: lambda.Runtime.PYTHON_3_8,
|
|
26
|
-
timeout:
|
|
26
|
+
timeout: aws_cdk_lib_1.Duration.seconds(6),
|
|
27
27
|
environment: {
|
|
28
28
|
SLACK_URL: props.slackUrl,
|
|
29
29
|
SLACK_CHANNEL: props.slackChannel,
|
|
@@ -44,4 +44,4 @@ class SlackAlarm extends cdk.Construct {
|
|
|
44
44
|
}
|
|
45
45
|
}
|
|
46
46
|
exports.SlackAlarm = SlackAlarm;
|
|
47
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
47
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stYWxhcm0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWxhcm1zL3NsYWNrLWFsYXJtLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUN4Qyx3RUFBdUU7QUFDdkUsMkNBQTBDO0FBQzFDLGlEQUFnRDtBQUNoRCwyQ0FBMEM7QUFDMUMsNkNBQXNDO0FBQ3RDLDZCQUE0QjtBQVM1Qjs7O0dBR0c7QUFDSCxNQUFhLFVBQVcsU0FBUSxVQUFVLENBQUMsU0FBUztJQUlsRCxZQUFZLEtBQTJCLEVBQUUsRUFBVSxFQUFFLEtBQXNCO1FBQ3pFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxJQUFJLEdBQUcsQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFBO1FBRTlDLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxpQkFBaUIsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFBO1FBRWpFLE1BQU0sV0FBVyxHQUFHLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3hELElBQUksRUFBRSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDekIsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUsaUNBQWlDLENBQUMsQ0FDeEQ7WUFDRCxXQUFXLEVBQ1QsK0VBQStFO1lBQ2pGLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLFVBQVUsRUFBRSxHQUFHO1lBQ2YsT0FBTyxFQUFFLE1BQU0sQ0FBQyxPQUFPLENBQUMsVUFBVTtZQUNsQyxPQUFPLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1lBQzVCLFdBQVcsRUFBRTtnQkFDWCxTQUFTLEVBQUUsS0FBSyxDQUFDLFFBQVE7Z0JBQ3pCLGFBQWEsRUFBRSxLQUFLLENBQUMsWUFBWTtnQkFDakMsWUFBWSxFQUFFLEtBQUssQ0FBQyxXQUFXO2dCQUMvQixnQkFBZ0IsRUFBRSxLQUFLLENBQUMsT0FBTzthQUNoQztTQUNGLENBQUMsQ0FBQTtRQUVGLFdBQVcsQ0FBQyxhQUFhLENBQUMsa0JBQWtCLEVBQUU7WUFDNUMsTUFBTSxFQUFFLHVCQUF1QjtZQUMvQixTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsZ0JBQWdCLENBQUMsbUJBQW1CLENBQUM7WUFDeEQsU0FBUyxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUTtTQUNwQyxDQUFDLENBQUE7UUFFRixJQUFJLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxFQUFFLGNBQWMsRUFBRTtZQUN6QyxRQUFRLEVBQUUsV0FBVyxDQUFDLFdBQVc7WUFDakMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxNQUFNO1lBQ3pDLEtBQUssRUFBRSxJQUFJLENBQUMsVUFBVTtTQUN2QixDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0Y7QUF6Q0QsZ0NBeUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjbG91ZHdhdGNoQWN0aW9ucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gtYWN0aW9uc1wiXG5pbXBvcnQgKiBhcyBpYW0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIlxuaW1wb3J0ICogYXMgbGFtYmRhIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtbGFtYmRhXCJcbmltcG9ydCAqIGFzIHNucyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNuc1wiXG5pbXBvcnQgeyBEdXJhdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCJcblxuZXhwb3J0IGludGVyZmFjZSBTbGFja0FsYXJtUHJvcHMge1xuICBwcm9qZWN0TmFtZTogc3RyaW5nXG4gIGVudk5hbWU6IHN0cmluZ1xuICBzbGFja0NoYW5uZWw6IHN0cmluZ1xuICBzbGFja1VybDogc3RyaW5nXG59XG5cbi8qKlxuICogU05TIFRvcGljIHRoYXQgY2FuIGJlIHVzZWQgdG8gYWN0aW9uIGFsYXJtcywgd2l0aCBhIExhbWJkYVxuICogdGhhdCB3aWxsIHNlbmQgYSBtZXNzYWdlIHRvIFNsYWNrIGZvciB0aGUgYWxhcm0uXG4gKi9cbmV4cG9ydCBjbGFzcyBTbGFja0FsYXJtIGV4dGVuZHMgY29uc3RydWN0cy5Db25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgYWxhcm1Ub3BpYzogc25zLlRvcGljXG4gIHB1YmxpYyByZWFkb25seSBzbnNBY3Rpb246IGNsb3Vkd2F0Y2hBY3Rpb25zLlNuc0FjdGlvblxuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFNsYWNrQWxhcm1Qcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIHRoaXMuYWxhcm1Ub3BpYyA9IG5ldyBzbnMuVG9waWModGhpcywgXCJUb3BpY1wiKVxuXG4gICAgdGhpcy5zbnNBY3Rpb24gPSBuZXcgY2xvdWR3YXRjaEFjdGlvbnMuU25zQWN0aW9uKHRoaXMuYWxhcm1Ub3BpYylcblxuICAgIGNvbnN0IHNsYWNrTGFtYmRhID0gbmV3IGxhbWJkYS5GdW5jdGlvbih0aGlzLCBcIkZ1bmN0aW9uXCIsIHtcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChcbiAgICAgICAgcGF0aC5qb2luKF9fZGlybmFtZSwgXCIuLi8uLi9hc3NldHMvc2xhY2stYWxhcm0tbGFtYmRhXCIpLFxuICAgICAgKSxcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICBcIlJlY2VpdmVzIENsb3VkV2F0Y2ggQWxhcm1zIHRocm91Z2ggU05TIGFuZCBzZW5kcyBhIGZvcm1hdHRlZCB2ZXJzaW9uIHRvIFNsYWNrXCIsXG4gICAgICBoYW5kbGVyOiBcImluZGV4LmhhbmRsZXJcIixcbiAgICAgIG1lbW9yeVNpemU6IDEyOCxcbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLlBZVEhPTl8zXzgsXG4gICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDYpLFxuICAgICAgZW52aXJvbm1lbnQ6IHtcbiAgICAgICAgU0xBQ0tfVVJMOiBwcm9wcy5zbGFja1VybCxcbiAgICAgICAgU0xBQ0tfQ0hBTk5FTDogcHJvcHMuc2xhY2tDaGFubmVsLFxuICAgICAgICBQUk9KRUNUX05BTUU6IHByb3BzLnByb2plY3ROYW1lLFxuICAgICAgICBFTlZJUk9OTUVOVF9OQU1FOiBwcm9wcy5lbnZOYW1lLFxuICAgICAgfSxcbiAgICB9KVxuXG4gICAgc2xhY2tMYW1iZGEuYWRkUGVybWlzc2lvbihcIkludm9rZVBlcm1pc3Npb25cIiwge1xuICAgICAgYWN0aW9uOiBcImxhbWJkYTpJbnZva2VGdW5jdGlvblwiLFxuICAgICAgcHJpbmNpcGFsOiBuZXcgaWFtLlNlcnZpY2VQcmluY2lwYWwoXCJzbnMuYW1hem9uYXdzLmNvbVwiKSxcbiAgICAgIHNvdXJjZUFybjogdGhpcy5hbGFybVRvcGljLnRvcGljQXJuLFxuICAgIH0pXG5cbiAgICBuZXcgc25zLlN1YnNjcmlwdGlvbih0aGlzLCBcIlN1YnNjcmlwdGlvblwiLCB7XG4gICAgICBlbmRwb2ludDogc2xhY2tMYW1iZGEuZnVuY3Rpb25Bcm4sXG4gICAgICBwcm90b2NvbDogc25zLlN1YnNjcmlwdGlvblByb3RvY29sLkxBTUJEQSxcbiAgICAgIHRvcGljOiB0aGlzLmFsYXJtVG9waWMsXG4gICAgfSlcbiAgfVxufVxuIl19
|
package/lib/bastion-host.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import * as
|
|
2
|
-
import * as
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as ec2 from "aws-cdk-lib/aws-ec2";
|
|
3
3
|
interface Props {
|
|
4
4
|
vpc: ec2.IVpc;
|
|
5
5
|
/**
|
|
@@ -34,8 +34,8 @@ interface Props {
|
|
|
34
34
|
* For more internal details, see
|
|
35
35
|
* https://confluence.capraconsulting.no/x/q8UBC
|
|
36
36
|
*/
|
|
37
|
-
export declare class BastionHost extends
|
|
37
|
+
export declare class BastionHost extends constructs.Construct {
|
|
38
38
|
readonly securityGroup: ec2.ISecurityGroup;
|
|
39
|
-
constructor(scope:
|
|
39
|
+
constructor(scope: constructs.Construct, id: string, props: Props);
|
|
40
40
|
}
|
|
41
41
|
export {};
|
package/lib/bastion-host.js
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.BastionHost = void 0;
|
|
4
|
-
const
|
|
5
|
-
const
|
|
6
|
-
const
|
|
4
|
+
const constructs = require("constructs");
|
|
5
|
+
const ec2 = require("aws-cdk-lib/aws-ec2");
|
|
6
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
7
|
+
const cdk = require("aws-cdk-lib");
|
|
7
8
|
/**
|
|
8
9
|
* This creates a EC2 bastion host that can be used to connect
|
|
9
10
|
* to database instances and other internal resources.
|
|
@@ -17,7 +18,7 @@ const cdk = require("@aws-cdk/core");
|
|
|
17
18
|
* For more internal details, see
|
|
18
19
|
* https://confluence.capraconsulting.no/x/q8UBC
|
|
19
20
|
*/
|
|
20
|
-
class BastionHost extends
|
|
21
|
+
class BastionHost extends constructs.Construct {
|
|
21
22
|
constructor(scope, id, props) {
|
|
22
23
|
var _a, _b;
|
|
23
24
|
super(scope, id);
|
|
@@ -82,4 +83,4 @@ class BastionHost extends cdk.Construct {
|
|
|
82
83
|
}
|
|
83
84
|
}
|
|
84
85
|
exports.BastionHost = BastionHost;
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
86
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bastion-host.js","sourceRoot":"","sources":["../src/bastion-host.ts"],"names":[],"mappings":";;;AAAA,yCAAwC;AACxC,2CAA0C;AAC1C,2CAA0C;AAC1C,mCAAkC;AAwBlC;;;;;;;;;;;;GAYG;AACH,MAAa,WAAY,SAAQ,UAAU,CAAC,SAAS;IAGnD,YAAY,KAA2B,EAAE,EAAU,EAAE,KAAY;;QAC/D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,MAAM,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,CAAA;QAExC,IAAI,CAAC,aAAa;YAChB,MAAA,KAAK,CAAC,aAAa,mCACnB,IAAI,GAAG,CAAC,aAAa,CAAC,IAAI,EAAE,eAAe,EAAE;gBAC3C,GAAG,EAAE,KAAK,CAAC,GAAG;aACf,CAAC,CAAA;QAEJ,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,UAAU,EAAE;YAClD,GAAG,EAAE,KAAK,CAAC,GAAG;YACd,UAAU,EAAE,MAAA,KAAK,CAAC,eAAe,mCAAI;gBACnC,UAAU,EAAE,GAAG,CAAC,UAAU,CAAC,MAAM;aAClC;YACD,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,YAAY,EAAE,SAAS;YACvB,YAAY,EAAE,GAAG,CAAC,YAAY,CAAC,EAAE,CAC/B,GAAG,CAAC,aAAa,CAAC,EAAE,EACpB,GAAG,CAAC,YAAY,CAAC,IAAI,CACtB;YACD,YAAY,EAAE,GAAG,CAAC,YAAY,CAAC,iBAAiB,CAAC;gBAC/C,UAAU,EAAE,GAAG,CAAC,qBAAqB,CAAC,cAAc;aACrD,CAAC;SACH,CAAC,CAAA;QAEF,QAAQ,CAAC,WAAW,CAClB,qCAAqC,MAAM,oFAAoF,CAChI,CAAA;QAED,eAAe;QACf,QAAQ,CAAC,eAAe;QACtB,6CAA6C;QAC7C,0CAA0C;QAC1C,EAAE;QACF,iEAAiE;QACjE,kEAAkE;QAClE,mEAAmE;QACnE,2DAA2D;QAC3D,EAAE;QACF,8DAA8D;QAC9D,oGAAoG;QACpG,IAAI,GAAG,CAAC,eAAe,CAAC;YACtB,OAAO,EAAE;gBACP,2GAA2G;gBAC3G,+EAA+E;gBAC/E,8BAA8B;gBAC9B,+BAA+B;gBAC/B,iBAAiB;gBACjB,kBAAkB;gBAClB,qCAAqC;gBACrC,2GAA2G;gBAC3G,kCAAkC;gBAClC,+BAA+B;gBAC/B,gCAAgC;gBAChC,6BAA6B;gBAC7B,0FAA0F;gBAC1F,gCAAgC;gBAChC,2BAA2B;gBAC3B,yBAAyB;gBACzB,yBAAyB;gBACzB,yBAAyB;gBACzB,uBAAuB;aACxB;YACD,8CAA8C;YAC9C,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CACH,CAAA;QAED,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC3C,KAAK,EAAE,QAAQ,CAAC,UAAU;SAC3B,CAAC,CAAA;IACJ,CAAC;CACF;AA7ED,kCA6EC","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as cdk from \"aws-cdk-lib\"\n\ninterface Props {\n  vpc: ec2.IVpc\n  /**\n   * The security group used for the EC2 instance.\n   *\n   * @default - a security group will be created\n   */\n  securityGroup?: ec2.ISecurityGroup\n  /**\n   * The subnets to place the bastion host.\n   *\n   * Note that if placed inside private subnet, the VPC must have\n   * VPC endpoints to access relevant AWS services for Systems Manager\n   * to work in order to be able to connect to the instance.\n   *\n   * See https://aws.amazon.com/premiumsupport/knowledge-center/ec2-systems-manager-vpc-endpoints/\n   *\n   * @default - public subnets\n   */\n  subnetSelection?: ec2.SubnetSelection\n}\n\n/**\n * This creates a EC2 bastion host that can be used to connect\n * to database instances and other internal resources.\n *\n * The instance is supposed to have no open ingress ports, and users\n * are supposed to connect only through SSM Session Manager.\n *\n * The resources that the bastion host should be allowed to access\n * must have the bastion host security group as allowed ingress.\n *\n * For more internal details, see\n * https://confluence.capraconsulting.no/x/q8UBC\n */\nexport class BastionHost extends constructs.Construct {\n  public readonly securityGroup: ec2.ISecurityGroup\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    const region = cdk.Stack.of(this).region\n\n    this.securityGroup =\n      props.securityGroup ??\n      new ec2.SecurityGroup(this, \"SecurityGroup\", {\n        vpc: props.vpc,\n      })\n\n    const instance = new ec2.Instance(this, \"Instance\", {\n      vpc: props.vpc,\n      vpcSubnets: props.subnetSelection ?? {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      securityGroup: this.securityGroup,\n      instanceName: \"Bastion\",\n      instanceType: ec2.InstanceType.of(\n        ec2.InstanceClass.T3,\n        ec2.InstanceSize.NANO,\n      ),\n      machineImage: ec2.MachineImage.latestAmazonLinux({\n        generation: ec2.AmazonLinuxGeneration.AMAZON_LINUX_2,\n      }),\n    })\n\n    instance.addUserData(\n      `yum install -y https://amazon-ssm-${region}.s3.amazonaws.com/latest/linux_amd64/amazon-ssm-agent.rpm socat postgresql mariadb`,\n    )\n\n    // SSM support.\n    instance.addToRolePolicy(\n      // This mimics the AmazonEC2RoleforSSM policy\n      // while granting least privileges needed.\n      //\n      // The default AmazonEC2RoleforSSM policy gives read/write access\n      // to all objects in S3, all parameters in Parameter Store, amoung\n      // more. We primarily use the SSM agent for limited remote control,\n      // and the policy here covers that as the primary use case.\n      //\n      // See https://www.cflee.com/posts/aws-ssm-iam-policy-caveats/\n      // See also https://docs.aws.amazon.com/systems-manager/latest/userguide/setup-instance-profile.html\n      new iam.PolicyStatement({\n        actions: [\n          // https://docs.aws.amazon.com/systems-manager/latest/userguide/systems-manager-setting-up-messageAPIs.html\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/list_awssystemsmanager.html\n          \"ssm:ListInstanceAssociations\",\n          \"ssm:UpdateInstanceInformation\",\n          \"ssm:GetDocument\",\n          \"ssm:PutInventory\",\n          \"ssm:UpdateInstanceAssociationStatus\",\n          // https://docs.aws.amazon.com/systems-manager/latest/userguide/systems-manager-setting-up-messageAPIs.html\n          \"ssmmessages:CreateControlChannel\",\n          \"ssmmessages:CreateDataChannel\",\n          \"ssmmessages:OpenControlChannel\",\n          \"ssmmessages:OpenDataChannel\",\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/list_amazonmessagedeliveryservice.html\n          \"ec2messages:AcknowledgeMessage\",\n          \"ec2messages:DeleteMessage\",\n          \"ec2messages:FailMessage\",\n          \"ec2messages:GetEndpoint\",\n          \"ec2messages:GetMessages\",\n          \"ec2messages:SendReply\",\n        ],\n        // Seems this is needed for the given actions.\n        resources: [\"*\"],\n      }),\n    )\n\n    new cdk.CfnOutput(this, \"BastionInstanceId\", {\n      value: instance.instanceId,\n    })\n  }\n}\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
2
|
interface Props {
|
|
3
3
|
/**
|
|
4
4
|
* The name to use for the S3 Bucket. Should include both account and region
|
|
@@ -52,10 +52,10 @@ interface Props {
|
|
|
52
52
|
*
|
|
53
53
|
* @experimental
|
|
54
54
|
*/
|
|
55
|
-
export declare class BuildArtifacts extends
|
|
55
|
+
export declare class BuildArtifacts extends constructs.Construct {
|
|
56
56
|
readonly bucketName: string | undefined;
|
|
57
57
|
readonly ecrRepositoryArn: string;
|
|
58
58
|
readonly ecrRepositoryName: string;
|
|
59
|
-
constructor(scope:
|
|
59
|
+
constructor(scope: constructs.Construct, id: string, props: Props);
|
|
60
60
|
}
|
|
61
61
|
export {};
|
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.BuildArtifacts = void 0;
|
|
4
|
-
const
|
|
5
|
-
const
|
|
6
|
-
const
|
|
7
|
-
const
|
|
4
|
+
const constructs = require("constructs");
|
|
5
|
+
const ecr = require("aws-cdk-lib/aws-ecr");
|
|
6
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
7
|
+
const s3 = require("aws-cdk-lib/aws-s3");
|
|
8
|
+
const cdk = require("aws-cdk-lib");
|
|
8
9
|
const griid_1 = require("../griid");
|
|
9
10
|
/**
|
|
10
11
|
* Build artifacts.
|
|
@@ -18,7 +19,7 @@ const griid_1 = require("../griid");
|
|
|
18
19
|
*
|
|
19
20
|
* @experimental
|
|
20
21
|
*/
|
|
21
|
-
class BuildArtifacts extends
|
|
22
|
+
class BuildArtifacts extends constructs.Construct {
|
|
22
23
|
constructor(scope, id, props) {
|
|
23
24
|
var _a;
|
|
24
25
|
super(scope, id);
|
|
@@ -107,4 +108,4 @@ class BuildArtifacts extends cdk.Construct {
|
|
|
107
108
|
}
|
|
108
109
|
}
|
|
109
110
|
exports.BuildArtifacts = BuildArtifacts;
|
|
110
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/build-artifacts/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ecr from \"@aws-cdk/aws-ecr\"\nimport * as iam from \"@aws-cdk/aws-iam\"\nimport * as s3 from \"@aws-cdk/aws-s3\"\nimport * as cdk from \"@aws-cdk/core\"\nimport { getGriidCiRole } from \"../griid\"\n\ninterface Props {\n  /**\n   * The name to use for the S3 Bucket. Should include both account and region\n   * so that it will not conflict with other accounts/regions.\n   *\n   * @default - no bucket will be created\n   */\n  bucketName?: string\n  /**\n   * The name to use for the ECR Repository.\n   */\n  ecrRepositoryName: string\n  /**\n   * Reference to the IAM Role that will be granted permission to\n   * assume the CI role. This role must have permission to assume\n   * the CI role.\n   *\n   * @default - use Liflig Jenkins role\n   */\n  externalRoleArn?: string\n  /**\n   * The name of the role that will be created that will be assumed\n   * from the CI system.\n   *\n   * @default - no role will be created\n   */\n  ciRoleName?: string\n  /**\n   * The AWS Accounts that will be granted permission to read from\n   * the artifact repos.\n   */\n  targetAccountIds: string[]\n  /**\n   * Flag if Griid is bootstrapped and the account this construct is\n   * deployed to is the build account. Will attach policies and\n   * reference existing artifacts and roles.\n   *\n   * @default false\n   */\n  griid?: boolean\n}\n\n/**\n * Build artifacts.\n *\n * This holds a S3 Bucket, a ECR Repository and roles to be used\n * from CI system for uploading.\n *\n * TODO: How can we cleanup stuff that goes into this S3 Bucket and\n *  ECR Repository? Can we ever reliably cleanup? We probably need\n *  some strategy for how we put stuff here to be able to do it.\n *\n * @experimental\n */\nexport class BuildArtifacts extends cdk.Construct {\n  readonly bucketName: string | undefined\n  readonly ecrRepositoryArn: string\n  readonly ecrRepositoryName: string\n\n  constructor(scope: cdk.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    this.bucketName = props.bucketName\n    this.ecrRepositoryName = props.ecrRepositoryName\n    this.ecrRepositoryArn = cdk.Arn.format(\n      {\n        service: \"ecr\",\n        resource: \"repository\",\n        resourceName: this.ecrRepositoryName,\n      },\n      cdk.Stack.of(this),\n    )\n\n    const externalRoleArn =\n      props.externalRoleArn ??\n      \"arn:aws:iam::923402097046:role/buildtools-jenkins-RoleJenkinsSlave-JQGYHR5WE6C5\"\n\n    const ecrRepositoryName = props.ecrRepositoryName\n\n    let bucket: s3.Bucket | undefined = undefined\n\n    if (props.bucketName) {\n      bucket = new s3.Bucket(this, \"S3Bucket\", {\n        bucketName: props.bucketName,\n        encryption: s3.BucketEncryption.S3_MANAGED,\n        blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n        versioned: true,\n        lifecycleRules: [\n          {\n            noncurrentVersionExpiration: cdk.Duration.days(10),\n          },\n        ],\n      })\n    }\n\n    const ciRole: iam.Role | undefined = props.ciRoleName\n      ? new iam.Role(this, \"CiRole\", {\n          roleName: props.ciRoleName,\n          assumedBy: new iam.ArnPrincipal(externalRoleArn),\n        })\n      : undefined\n\n    const griidCiRole: iam.IRole | undefined = props.griid\n      ? getGriidCiRole(this)\n      : undefined\n\n    if (bucket && ciRole) {\n      bucket.grantReadWrite(ciRole)\n    }\n\n    if (bucket && griidCiRole) {\n      bucket.grantReadWrite(griidCiRole)\n    }\n\n    const ecrRepo = new ecr.Repository(this, \"EcrRepository\", {\n      repositoryName: ecrRepositoryName,\n    })\n\n    if (ciRole) {\n      ecrRepo.grantPullPush(ciRole)\n    }\n\n    if (griidCiRole) {\n      ecrRepo.grantPullPush(griidCiRole)\n    }\n\n    // Allow a target to read from the repos. As any specific roles need\n    // to exist before we can grant access, we delegate that responsibility\n    // to the target account.\n    for (const targetAccountId of props.targetAccountIds) {\n      if (bucket) {\n        bucket.grantRead(new iam.AccountPrincipal(targetAccountId))\n      }\n      ecrRepo.grantPull(new iam.AccountPrincipal(targetAccountId))\n    }\n\n    // Grant permissions to write pipeline variables.\n    if (ciRole || griidCiRole) {\n      const account = cdk.Stack.of(this).account\n      const region = cdk.Stack.of(this).region\n      const statement = new iam.PolicyStatement({\n        actions: [\"ssm:PutParameter\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      })\n\n      ciRole?.grantPrincipal.addToPrincipalPolicy(statement)\n      griidCiRole?.grantPrincipal.addToPrincipalPolicy(statement)\n    }\n\n    new cdk.CfnOutput(this, \"EcrRepoUri\", {\n      value: ecrRepo.repositoryUri,\n    })\n\n    if (bucket) {\n      new cdk.CfnOutput(this, \"BucketName\", {\n        value: bucket.bucketName,\n      })\n    }\n\n    if (ciRole) {\n      new cdk.CfnOutput(this, \"CiRoleArn\", {\n        value: ciRole.roleArn,\n      })\n    }\n  }\n}\n"]}
|
|
111
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/build-artifacts/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ecr from \"aws-cdk-lib/aws-ecr\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport { getGriidCiRole } from \"../griid\"\n\ninterface Props {\n  /**\n   * The name to use for the S3 Bucket. Should include both account and region\n   * so that it will not conflict with other accounts/regions.\n   *\n   * @default - no bucket will be created\n   */\n  bucketName?: string\n  /**\n   * The name to use for the ECR Repository.\n   */\n  ecrRepositoryName: string\n  /**\n   * Reference to the IAM Role that will be granted permission to\n   * assume the CI role. This role must have permission to assume\n   * the CI role.\n   *\n   * @default - use Liflig Jenkins role\n   */\n  externalRoleArn?: string\n  /**\n   * The name of the role that will be created that will be assumed\n   * from the CI system.\n   *\n   * @default - no role will be created\n   */\n  ciRoleName?: string\n  /**\n   * The AWS Accounts that will be granted permission to read from\n   * the artifact repos.\n   */\n  targetAccountIds: string[]\n  /**\n   * Flag if Griid is bootstrapped and the account this construct is\n   * deployed to is the build account. Will attach policies and\n   * reference existing artifacts and roles.\n   *\n   * @default false\n   */\n  griid?: boolean\n}\n\n/**\n * Build artifacts.\n *\n * This holds a S3 Bucket, a ECR Repository and roles to be used\n * from CI system for uploading.\n *\n * TODO: How can we cleanup stuff that goes into this S3 Bucket and\n *  ECR Repository? Can we ever reliably cleanup? We probably need\n *  some strategy for how we put stuff here to be able to do it.\n *\n * @experimental\n */\nexport class BuildArtifacts extends constructs.Construct {\n  readonly bucketName: string | undefined\n  readonly ecrRepositoryArn: string\n  readonly ecrRepositoryName: string\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    this.bucketName = props.bucketName\n    this.ecrRepositoryName = props.ecrRepositoryName\n    this.ecrRepositoryArn = cdk.Arn.format(\n      {\n        service: \"ecr\",\n        resource: \"repository\",\n        resourceName: this.ecrRepositoryName,\n      },\n      cdk.Stack.of(this),\n    )\n\n    const externalRoleArn =\n      props.externalRoleArn ??\n      \"arn:aws:iam::923402097046:role/buildtools-jenkins-RoleJenkinsSlave-JQGYHR5WE6C5\"\n\n    const ecrRepositoryName = props.ecrRepositoryName\n\n    let bucket: s3.Bucket | undefined = undefined\n\n    if (props.bucketName) {\n      bucket = new s3.Bucket(this, \"S3Bucket\", {\n        bucketName: props.bucketName,\n        encryption: s3.BucketEncryption.S3_MANAGED,\n        blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n        versioned: true,\n        lifecycleRules: [\n          {\n            noncurrentVersionExpiration: cdk.Duration.days(10),\n          },\n        ],\n      })\n    }\n\n    const ciRole: iam.Role | undefined = props.ciRoleName\n      ? new iam.Role(this, \"CiRole\", {\n          roleName: props.ciRoleName,\n          assumedBy: new iam.ArnPrincipal(externalRoleArn),\n        })\n      : undefined\n\n    const griidCiRole: iam.IRole | undefined = props.griid\n      ? getGriidCiRole(this)\n      : undefined\n\n    if (bucket && ciRole) {\n      bucket.grantReadWrite(ciRole)\n    }\n\n    if (bucket && griidCiRole) {\n      bucket.grantReadWrite(griidCiRole)\n    }\n\n    const ecrRepo = new ecr.Repository(this, \"EcrRepository\", {\n      repositoryName: ecrRepositoryName,\n    })\n\n    if (ciRole) {\n      ecrRepo.grantPullPush(ciRole)\n    }\n\n    if (griidCiRole) {\n      ecrRepo.grantPullPush(griidCiRole)\n    }\n\n    // Allow a target to read from the repos. As any specific roles need\n    // to exist before we can grant access, we delegate that responsibility\n    // to the target account.\n    for (const targetAccountId of props.targetAccountIds) {\n      if (bucket) {\n        bucket.grantRead(new iam.AccountPrincipal(targetAccountId))\n      }\n      ecrRepo.grantPull(new iam.AccountPrincipal(targetAccountId))\n    }\n\n    // Grant permissions to write pipeline variables.\n    if (ciRole || griidCiRole) {\n      const account = cdk.Stack.of(this).account\n      const region = cdk.Stack.of(this).region\n      const statement = new iam.PolicyStatement({\n        actions: [\"ssm:PutParameter\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      })\n\n      ciRole?.grantPrincipal.addToPrincipalPolicy(statement)\n      griidCiRole?.grantPrincipal.addToPrincipalPolicy(statement)\n    }\n\n    new cdk.CfnOutput(this, \"EcrRepoUri\", {\n      value: ecrRepo.repositoryUri,\n    })\n\n    if (bucket) {\n      new cdk.CfnOutput(this, \"BucketName\", {\n        value: bucket.bucketName,\n      })\n    }\n\n    if (ciRole) {\n      new cdk.CfnOutput(this, \"CiRoleArn\", {\n        value: ciRole.roleArn,\n      })\n    }\n  }\n}\n"]}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
import * as
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as cdk from "aws-cdk-lib";
|
|
2
3
|
interface Props extends cdk.StackProps {
|
|
3
4
|
/**
|
|
4
5
|
* The role that will be granted permission to assume the deploy
|
|
@@ -56,7 +57,7 @@ interface Props extends cdk.StackProps {
|
|
|
56
57
|
* The "status" lambda can be used to poll for completion, and will
|
|
57
58
|
* also return logs from the job upon completion.
|
|
58
59
|
*/
|
|
59
|
-
export declare class CdkDeploy extends
|
|
60
|
-
constructor(scope:
|
|
60
|
+
export declare class CdkDeploy extends constructs.Construct {
|
|
61
|
+
constructor(scope: constructs.Construct, id: string, props: Props);
|
|
61
62
|
}
|
|
62
63
|
export {};
|