low-cost-ecs 0.0.16 → 0.0.18
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/.jsii +86 -12
- package/.projenrc.ts +12 -4
- package/API.md +57 -7
- package/examples/all-props.ts +49 -0
- package/examples/{scheduled-autoscaling.ts → autoscaling.ts} +3 -3
- package/examples/minimum.ts +1 -1
- package/lib/low-cost-ecs.d.ts +10 -4
- package/lib/low-cost-ecs.js +15 -5
- package/package.json +1 -1
package/.jsii
CHANGED
|
@@ -3073,7 +3073,7 @@
|
|
|
3073
3073
|
},
|
|
3074
3074
|
"locationInModule": {
|
|
3075
3075
|
"filename": "src/low-cost-ecs.ts",
|
|
3076
|
-
"line":
|
|
3076
|
+
"line": 137
|
|
3077
3077
|
},
|
|
3078
3078
|
"parameters": [
|
|
3079
3079
|
{
|
|
@@ -3099,7 +3099,7 @@
|
|
|
3099
3099
|
"kind": "class",
|
|
3100
3100
|
"locationInModule": {
|
|
3101
3101
|
"filename": "src/low-cost-ecs.ts",
|
|
3102
|
-
"line":
|
|
3102
|
+
"line": 130
|
|
3103
3103
|
},
|
|
3104
3104
|
"name": "LowCostECS",
|
|
3105
3105
|
"properties": [
|
|
@@ -3110,7 +3110,7 @@
|
|
|
3110
3110
|
"immutable": true,
|
|
3111
3111
|
"locationInModule": {
|
|
3112
3112
|
"filename": "src/low-cost-ecs.ts",
|
|
3113
|
-
"line":
|
|
3113
|
+
"line": 133
|
|
3114
3114
|
},
|
|
3115
3115
|
"name": "certFileSystem",
|
|
3116
3116
|
"type": {
|
|
@@ -3124,7 +3124,7 @@
|
|
|
3124
3124
|
"immutable": true,
|
|
3125
3125
|
"locationInModule": {
|
|
3126
3126
|
"filename": "src/low-cost-ecs.ts",
|
|
3127
|
-
"line":
|
|
3127
|
+
"line": 134
|
|
3128
3128
|
},
|
|
3129
3129
|
"name": "cluster",
|
|
3130
3130
|
"type": {
|
|
@@ -3138,7 +3138,7 @@
|
|
|
3138
3138
|
"immutable": true,
|
|
3139
3139
|
"locationInModule": {
|
|
3140
3140
|
"filename": "src/low-cost-ecs.ts",
|
|
3141
|
-
"line":
|
|
3141
|
+
"line": 132
|
|
3142
3142
|
},
|
|
3143
3143
|
"name": "hostAutoScalingGroup",
|
|
3144
3144
|
"type": {
|
|
@@ -3152,7 +3152,7 @@
|
|
|
3152
3152
|
"immutable": true,
|
|
3153
3153
|
"locationInModule": {
|
|
3154
3154
|
"filename": "src/low-cost-ecs.ts",
|
|
3155
|
-
"line":
|
|
3155
|
+
"line": 135
|
|
3156
3156
|
},
|
|
3157
3157
|
"name": "service",
|
|
3158
3158
|
"type": {
|
|
@@ -3166,7 +3166,7 @@
|
|
|
3166
3166
|
"immutable": true,
|
|
3167
3167
|
"locationInModule": {
|
|
3168
3168
|
"filename": "src/low-cost-ecs.ts",
|
|
3169
|
-
"line":
|
|
3169
|
+
"line": 131
|
|
3170
3170
|
},
|
|
3171
3171
|
"name": "vpc",
|
|
3172
3172
|
"type": {
|
|
@@ -3418,13 +3418,13 @@
|
|
|
3418
3418
|
"name": "securityGroup",
|
|
3419
3419
|
"optional": true,
|
|
3420
3420
|
"type": {
|
|
3421
|
-
"fqn": "aws-cdk-lib.aws_ec2.
|
|
3421
|
+
"fqn": "aws-cdk-lib.aws_ec2.ISecurityGroup"
|
|
3422
3422
|
}
|
|
3423
3423
|
},
|
|
3424
3424
|
{
|
|
3425
3425
|
"abstract": true,
|
|
3426
3426
|
"docs": {
|
|
3427
|
-
"default": "- Nginx server task definition defined in
|
|
3427
|
+
"default": "- Nginx server task definition defined in createSampleTaskDefinition()",
|
|
3428
3428
|
"stability": "experimental",
|
|
3429
3429
|
"summary": "Task definition for the server ecs task."
|
|
3430
3430
|
},
|
|
@@ -3436,7 +3436,7 @@
|
|
|
3436
3436
|
"name": "serverTaskDefinition",
|
|
3437
3437
|
"optional": true,
|
|
3438
3438
|
"type": {
|
|
3439
|
-
"fqn": "
|
|
3439
|
+
"fqn": "low-cost-ecs.LowCostECSTaskDefinitionOptions"
|
|
3440
3440
|
}
|
|
3441
3441
|
},
|
|
3442
3442
|
{
|
|
@@ -3459,8 +3459,82 @@
|
|
|
3459
3459
|
}
|
|
3460
3460
|
],
|
|
3461
3461
|
"symbolId": "src/low-cost-ecs:LowCostECSProps"
|
|
3462
|
+
},
|
|
3463
|
+
"low-cost-ecs.LowCostECSTaskDefinitionOptions": {
|
|
3464
|
+
"assembly": "low-cost-ecs",
|
|
3465
|
+
"datatype": true,
|
|
3466
|
+
"docs": {
|
|
3467
|
+
"stability": "experimental"
|
|
3468
|
+
},
|
|
3469
|
+
"fqn": "low-cost-ecs.LowCostECSTaskDefinitionOptions",
|
|
3470
|
+
"kind": "interface",
|
|
3471
|
+
"locationInModule": {
|
|
3472
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3473
|
+
"line": 124
|
|
3474
|
+
},
|
|
3475
|
+
"name": "LowCostECSTaskDefinitionOptions",
|
|
3476
|
+
"properties": [
|
|
3477
|
+
{
|
|
3478
|
+
"abstract": true,
|
|
3479
|
+
"docs": {
|
|
3480
|
+
"stability": "experimental"
|
|
3481
|
+
},
|
|
3482
|
+
"immutable": true,
|
|
3483
|
+
"locationInModule": {
|
|
3484
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3485
|
+
"line": 126
|
|
3486
|
+
},
|
|
3487
|
+
"name": "containers",
|
|
3488
|
+
"type": {
|
|
3489
|
+
"collection": {
|
|
3490
|
+
"elementtype": {
|
|
3491
|
+
"fqn": "aws-cdk-lib.aws_ecs.ContainerDefinitionOptions"
|
|
3492
|
+
},
|
|
3493
|
+
"kind": "array"
|
|
3494
|
+
}
|
|
3495
|
+
}
|
|
3496
|
+
},
|
|
3497
|
+
{
|
|
3498
|
+
"abstract": true,
|
|
3499
|
+
"docs": {
|
|
3500
|
+
"stability": "experimental"
|
|
3501
|
+
},
|
|
3502
|
+
"immutable": true,
|
|
3503
|
+
"locationInModule": {
|
|
3504
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3505
|
+
"line": 125
|
|
3506
|
+
},
|
|
3507
|
+
"name": "taskDefinition",
|
|
3508
|
+
"optional": true,
|
|
3509
|
+
"type": {
|
|
3510
|
+
"fqn": "aws-cdk-lib.aws_ecs.Ec2TaskDefinitionProps"
|
|
3511
|
+
}
|
|
3512
|
+
},
|
|
3513
|
+
{
|
|
3514
|
+
"abstract": true,
|
|
3515
|
+
"docs": {
|
|
3516
|
+
"stability": "experimental"
|
|
3517
|
+
},
|
|
3518
|
+
"immutable": true,
|
|
3519
|
+
"locationInModule": {
|
|
3520
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3521
|
+
"line": 127
|
|
3522
|
+
},
|
|
3523
|
+
"name": "volumes",
|
|
3524
|
+
"optional": true,
|
|
3525
|
+
"type": {
|
|
3526
|
+
"collection": {
|
|
3527
|
+
"elementtype": {
|
|
3528
|
+
"fqn": "aws-cdk-lib.aws_ecs.Volume"
|
|
3529
|
+
},
|
|
3530
|
+
"kind": "array"
|
|
3531
|
+
}
|
|
3532
|
+
}
|
|
3533
|
+
}
|
|
3534
|
+
],
|
|
3535
|
+
"symbolId": "src/low-cost-ecs:LowCostECSTaskDefinitionOptions"
|
|
3462
3536
|
}
|
|
3463
3537
|
},
|
|
3464
|
-
"version": "0.0.
|
|
3465
|
-
"fingerprint": "
|
|
3538
|
+
"version": "0.0.18",
|
|
3539
|
+
"fingerprint": "2ZO7vgXaqC8x5IOjBgwW35YTj76EX+ao72JEQgXG9e8="
|
|
3466
3540
|
}
|
package/.projenrc.ts
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import { awscdk } from 'projen';
|
|
2
2
|
import { UpgradeDependenciesSchedule } from 'projen/lib/javascript';
|
|
3
3
|
|
|
4
|
-
const excludes = ['.idea/', 'cdk.out/', 'cdk.context.json', 'yarn-error.log'];
|
|
5
4
|
const project = new awscdk.AwsCdkConstructLibrary({
|
|
6
5
|
author: 'Yohta Kimura',
|
|
7
6
|
authorAddress: 'kitakita7617@gmail.com',
|
|
@@ -29,21 +28,30 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
29
28
|
distName: 'low-cost-ecs',
|
|
30
29
|
module: 'low_cost_ecs',
|
|
31
30
|
},
|
|
32
|
-
|
|
33
|
-
npmignore: excludes,
|
|
34
|
-
gitignore: excludes,
|
|
35
31
|
autoApproveOptions: {
|
|
36
32
|
allowedUsernames: ['rajyan'],
|
|
37
33
|
},
|
|
38
34
|
depsUpgradeOptions: {
|
|
39
35
|
workflowOptions: {
|
|
40
36
|
schedule: UpgradeDependenciesSchedule.WEEKLY,
|
|
37
|
+
labels: ['auto-approve'],
|
|
41
38
|
},
|
|
42
39
|
},
|
|
43
40
|
projenrcTs: true,
|
|
44
41
|
});
|
|
45
42
|
|
|
43
|
+
const excludes = ['.idea/', 'cdk.out/', 'cdk.context.json', 'yarn-error.log'];
|
|
44
|
+
project.npmignore?.exclude(...excludes);
|
|
45
|
+
project.gitignore.exclude(...excludes);
|
|
46
|
+
|
|
46
47
|
project.tsconfigDev.addInclude('examples/**/*.ts');
|
|
48
|
+
|
|
49
|
+
// Remove '--updateSnapshot' from test task
|
|
50
|
+
// Work around until https://github.com/projen/projen/issues/1144 is solved
|
|
51
|
+
const testTask = project.tasks.tryFind('test');
|
|
52
|
+
const newTestCommand = testTask!.steps[0]!.exec!.replace(' --updateSnapshot', '');
|
|
53
|
+
testTask!.reset(newTestCommand);
|
|
54
|
+
|
|
47
55
|
// workaround until fixed https://youtrack.jetbrains.com/issue/WEB-57089/ESLint823-TypeError-thislibOptionsparse-is-not-a-function
|
|
48
56
|
project.addDevDeps('eslint@8.22.0');
|
|
49
57
|
|
package/API.md
CHANGED
|
@@ -887,8 +887,8 @@ const lowCostECSProps: LowCostECSProps = { ... }
|
|
|
887
887
|
| <code><a href="#low-cost-ecs.LowCostECSProps.property.logGroup">logGroup</a></code> | <code>aws-cdk-lib.aws_logs.ILogGroup</code> | Log group of the certbot task and the aws-cli task. |
|
|
888
888
|
| <code><a href="#low-cost-ecs.LowCostECSProps.property.recordDomainNames">recordDomainNames</a></code> | <code>string[]</code> | Domain names for A records to elastic ip of ECS host instance. |
|
|
889
889
|
| <code><a href="#low-cost-ecs.LowCostECSProps.property.removalPolicy">removalPolicy</a></code> | <code>aws-cdk-lib.RemovalPolicy</code> | Removal policy for the file system and log group (if using default). |
|
|
890
|
-
| <code><a href="#low-cost-ecs.LowCostECSProps.property.securityGroup">securityGroup</a></code> | <code>aws-cdk-lib.aws_ec2.
|
|
891
|
-
| <code><a href="#low-cost-ecs.LowCostECSProps.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code
|
|
890
|
+
| <code><a href="#low-cost-ecs.LowCostECSProps.property.securityGroup">securityGroup</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup</code> | Security group of the ECS host instance. |
|
|
891
|
+
| <code><a href="#low-cost-ecs.LowCostECSProps.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions">LowCostECSTaskDefinitionOptions</a></code> | Task definition for the server ecs task. |
|
|
892
892
|
| <code><a href="#low-cost-ecs.LowCostECSProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | Vpc of the ECS host instance and cluster. |
|
|
893
893
|
|
|
894
894
|
---
|
|
@@ -1199,10 +1199,10 @@ Removal policy for the file system and log group (if using default).
|
|
|
1199
1199
|
##### `securityGroup`<sup>Optional</sup> <a name="securityGroup" id="low-cost-ecs.LowCostECSProps.property.securityGroup"></a>
|
|
1200
1200
|
|
|
1201
1201
|
```typescript
|
|
1202
|
-
public readonly securityGroup:
|
|
1202
|
+
public readonly securityGroup: ISecurityGroup;
|
|
1203
1203
|
```
|
|
1204
1204
|
|
|
1205
|
-
- *Type:* aws-cdk-lib.aws_ec2.
|
|
1205
|
+
- *Type:* aws-cdk-lib.aws_ec2.ISecurityGroup
|
|
1206
1206
|
- *Default:* Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).
|
|
1207
1207
|
|
|
1208
1208
|
Security group of the ECS host instance.
|
|
@@ -1212,11 +1212,11 @@ Security group of the ECS host instance.
|
|
|
1212
1212
|
##### `serverTaskDefinition`<sup>Optional</sup> <a name="serverTaskDefinition" id="low-cost-ecs.LowCostECSProps.property.serverTaskDefinition"></a>
|
|
1213
1213
|
|
|
1214
1214
|
```typescript
|
|
1215
|
-
public readonly serverTaskDefinition:
|
|
1215
|
+
public readonly serverTaskDefinition: LowCostECSTaskDefinitionOptions;
|
|
1216
1216
|
```
|
|
1217
1217
|
|
|
1218
|
-
- *Type:*
|
|
1219
|
-
- *Default:* Nginx server task definition defined in
|
|
1218
|
+
- *Type:* <a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions">LowCostECSTaskDefinitionOptions</a>
|
|
1219
|
+
- *Default:* Nginx server task definition defined in createSampleTaskDefinition()
|
|
1220
1220
|
|
|
1221
1221
|
Task definition for the server ecs task.
|
|
1222
1222
|
|
|
@@ -1235,5 +1235,55 @@ Vpc of the ECS host instance and cluster.
|
|
|
1235
1235
|
|
|
1236
1236
|
---
|
|
1237
1237
|
|
|
1238
|
+
### LowCostECSTaskDefinitionOptions <a name="LowCostECSTaskDefinitionOptions" id="low-cost-ecs.LowCostECSTaskDefinitionOptions"></a>
|
|
1239
|
+
|
|
1240
|
+
#### Initializer <a name="Initializer" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.Initializer"></a>
|
|
1241
|
+
|
|
1242
|
+
```typescript
|
|
1243
|
+
import { LowCostECSTaskDefinitionOptions } from 'low-cost-ecs'
|
|
1244
|
+
|
|
1245
|
+
const lowCostECSTaskDefinitionOptions: LowCostECSTaskDefinitionOptions = { ... }
|
|
1246
|
+
```
|
|
1247
|
+
|
|
1248
|
+
#### Properties <a name="Properties" id="Properties"></a>
|
|
1249
|
+
|
|
1250
|
+
| **Name** | **Type** | **Description** |
|
|
1251
|
+
| --- | --- | --- |
|
|
1252
|
+
| <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.containers">containers</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerDefinitionOptions[]</code> | *No description.* |
|
|
1253
|
+
| <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.taskDefinition">taskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinitionProps</code> | *No description.* |
|
|
1254
|
+
| <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.volumes">volumes</a></code> | <code>aws-cdk-lib.aws_ecs.Volume[]</code> | *No description.* |
|
|
1255
|
+
|
|
1256
|
+
---
|
|
1257
|
+
|
|
1258
|
+
##### `containers`<sup>Required</sup> <a name="containers" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.containers"></a>
|
|
1259
|
+
|
|
1260
|
+
```typescript
|
|
1261
|
+
public readonly containers: ContainerDefinitionOptions[];
|
|
1262
|
+
```
|
|
1263
|
+
|
|
1264
|
+
- *Type:* aws-cdk-lib.aws_ecs.ContainerDefinitionOptions[]
|
|
1265
|
+
|
|
1266
|
+
---
|
|
1267
|
+
|
|
1268
|
+
##### `taskDefinition`<sup>Optional</sup> <a name="taskDefinition" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.taskDefinition"></a>
|
|
1269
|
+
|
|
1270
|
+
```typescript
|
|
1271
|
+
public readonly taskDefinition: Ec2TaskDefinitionProps;
|
|
1272
|
+
```
|
|
1273
|
+
|
|
1274
|
+
- *Type:* aws-cdk-lib.aws_ecs.Ec2TaskDefinitionProps
|
|
1275
|
+
|
|
1276
|
+
---
|
|
1277
|
+
|
|
1278
|
+
##### `volumes`<sup>Optional</sup> <a name="volumes" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.volumes"></a>
|
|
1279
|
+
|
|
1280
|
+
```typescript
|
|
1281
|
+
public readonly volumes: Volume[];
|
|
1282
|
+
```
|
|
1283
|
+
|
|
1284
|
+
- *Type:* aws-cdk-lib.aws_ecs.Volume[]
|
|
1285
|
+
|
|
1286
|
+
---
|
|
1287
|
+
|
|
1238
1288
|
|
|
1239
1289
|
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { App, RemovalPolicy, Stack } from 'aws-cdk-lib';
|
|
2
|
+
import { SecurityGroup, Vpc } from 'aws-cdk-lib/aws-ec2';
|
|
3
|
+
import { ContainerImage, Protocol } from 'aws-cdk-lib/aws-ecs';
|
|
4
|
+
import { LogGroup } from 'aws-cdk-lib/aws-logs';
|
|
5
|
+
import { LowCostECS } from '../src';
|
|
6
|
+
|
|
7
|
+
const app = new App();
|
|
8
|
+
const stack = new Stack(app, 'TestStack', {
|
|
9
|
+
env: {
|
|
10
|
+
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
11
|
+
region: process.env.CDK_DEFAULT_REGION,
|
|
12
|
+
},
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
export const allPropsStack = new LowCostECS(app, 'LowCostECSStack', {
|
|
16
|
+
env: {
|
|
17
|
+
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
18
|
+
region: process.env.CDK_DEFAULT_REGION,
|
|
19
|
+
},
|
|
20
|
+
hostedZoneDomain: 'rajyan.net',
|
|
21
|
+
email: 'kitakita7617@gmail.com',
|
|
22
|
+
awsCliDockerTag: 'testTag',
|
|
23
|
+
certbotDockerTag: 'testTag',
|
|
24
|
+
certbotScheduleInterval: 10,
|
|
25
|
+
containerInsights: true,
|
|
26
|
+
hostInstanceSpotPrice: '0.010',
|
|
27
|
+
hostInstanceType: 't3.micro',
|
|
28
|
+
logGroup: LogGroup.fromLogGroupArn(stack, 'LogGroup', 'arn:aws:logs:region:account-id:log-group:test'),
|
|
29
|
+
recordDomainNames: ['test1.rajyan.net', 'test2.rajyan.net'],
|
|
30
|
+
removalPolicy: RemovalPolicy.RETAIN,
|
|
31
|
+
securityGroup: SecurityGroup.fromSecurityGroupId(stack, 'SecurityGroup', 'test-sg-id'),
|
|
32
|
+
serverTaskDefinition: {
|
|
33
|
+
containers: [{
|
|
34
|
+
containerName: 'test-container',
|
|
35
|
+
image: ContainerImage.fromRegistry('test-image'),
|
|
36
|
+
memoryLimitMiB: 32,
|
|
37
|
+
essential: true,
|
|
38
|
+
portMappings: [{
|
|
39
|
+
containerPort: 80,
|
|
40
|
+
hostPort: 80,
|
|
41
|
+
protocol: Protocol.TCP,
|
|
42
|
+
}],
|
|
43
|
+
}],
|
|
44
|
+
volumes: [{
|
|
45
|
+
name: 'test-volume',
|
|
46
|
+
}],
|
|
47
|
+
},
|
|
48
|
+
vpc: new Vpc(stack, 'Vpc'),
|
|
49
|
+
});
|
|
@@ -4,7 +4,7 @@ import { LowCostECS } from '../src';
|
|
|
4
4
|
|
|
5
5
|
const app = new App();
|
|
6
6
|
|
|
7
|
-
const
|
|
7
|
+
export const autoscalingStack = new LowCostECS(app, 'LowCostECSStack', {
|
|
8
8
|
env: {
|
|
9
9
|
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
10
10
|
region: process.env.CDK_DEFAULT_REGION,
|
|
@@ -14,7 +14,7 @@ const stack = new LowCostECS(app, 'LowCostECSStack', {
|
|
|
14
14
|
email: 'kitakita7617@gmail.com',
|
|
15
15
|
hostInstanceSpotPrice: '0.0050',
|
|
16
16
|
});
|
|
17
|
-
|
|
17
|
+
autoscalingStack.hostAutoScalingGroup.scaleOnSchedule('IncreaseAtMorning', {
|
|
18
18
|
timeZone: 'Asia/Tokyo',
|
|
19
19
|
schedule: Schedule.cron({
|
|
20
20
|
minute: '0',
|
|
@@ -22,7 +22,7 @@ stack.hostAutoScalingGroup.scaleOnSchedule('IncreaseAtMorning', {
|
|
|
22
22
|
}),
|
|
23
23
|
desiredCapacity: 1,
|
|
24
24
|
});
|
|
25
|
-
|
|
25
|
+
autoscalingStack.hostAutoScalingGroup.scaleOnSchedule('DecreaseAtNight', {
|
|
26
26
|
timeZone: 'Asia/Tokyo',
|
|
27
27
|
schedule: Schedule.cron({
|
|
28
28
|
minute: '0',
|
package/examples/minimum.ts
CHANGED
|
@@ -3,7 +3,7 @@ import { LowCostECS } from '../src';
|
|
|
3
3
|
|
|
4
4
|
const app = new App();
|
|
5
5
|
|
|
6
|
-
new LowCostECS(app, 'LowCostECSStack', {
|
|
6
|
+
export const minimumStack = new LowCostECS(app, 'LowCostECSStack', {
|
|
7
7
|
env: {
|
|
8
8
|
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
9
9
|
region: process.env.CDK_DEFAULT_REGION,
|
package/lib/low-cost-ecs.d.ts
CHANGED
|
@@ -38,7 +38,7 @@ export interface LowCostECSProps extends lib.StackProps {
|
|
|
38
38
|
*
|
|
39
39
|
* @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).
|
|
40
40
|
*/
|
|
41
|
-
readonly securityGroup?: ec2.
|
|
41
|
+
readonly securityGroup?: ec2.ISecurityGroup;
|
|
42
42
|
/**
|
|
43
43
|
* Instance type of the ECS host instance.
|
|
44
44
|
*
|
|
@@ -94,9 +94,14 @@ export interface LowCostECSProps extends lib.StackProps {
|
|
|
94
94
|
/**
|
|
95
95
|
* Task definition for the server ecs task.
|
|
96
96
|
*
|
|
97
|
-
* @default - Nginx server task definition defined in
|
|
97
|
+
* @default - Nginx server task definition defined in createSampleTaskDefinition()
|
|
98
98
|
*/
|
|
99
|
-
readonly serverTaskDefinition?:
|
|
99
|
+
readonly serverTaskDefinition?: LowCostECSTaskDefinitionOptions;
|
|
100
|
+
}
|
|
101
|
+
export interface LowCostECSTaskDefinitionOptions {
|
|
102
|
+
readonly taskDefinition?: ecs.Ec2TaskDefinitionProps;
|
|
103
|
+
readonly containers: ecs.ContainerDefinitionOptions[];
|
|
104
|
+
readonly volumes?: ecs.Volume[];
|
|
100
105
|
}
|
|
101
106
|
export declare class LowCostECS extends lib.Stack {
|
|
102
107
|
readonly vpc: ec2.IVpc;
|
|
@@ -105,5 +110,6 @@ export declare class LowCostECS extends lib.Stack {
|
|
|
105
110
|
readonly cluster: ecs.Cluster;
|
|
106
111
|
readonly service: ecs.Ec2Service;
|
|
107
112
|
constructor(scope: Construct, id: string, props: LowCostECSProps);
|
|
108
|
-
private
|
|
113
|
+
private createTaskDefinition;
|
|
114
|
+
private createSampleTaskDefinition;
|
|
109
115
|
}
|
package/lib/low-cost-ecs.js
CHANGED
|
@@ -16,7 +16,6 @@ const route53 = require("aws-cdk-lib/aws-route53");
|
|
|
16
16
|
const aws_sns_1 = require("aws-cdk-lib/aws-sns");
|
|
17
17
|
const sfn = require("aws-cdk-lib/aws-stepfunctions");
|
|
18
18
|
const sfn_tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
19
|
-
;
|
|
20
19
|
class LowCostECS extends lib.Stack {
|
|
21
20
|
constructor(scope, id, props) {
|
|
22
21
|
super(scope, id, props);
|
|
@@ -185,7 +184,9 @@ class LowCostECS extends lib.Stack {
|
|
|
185
184
|
/**
|
|
186
185
|
* Server ECS task
|
|
187
186
|
*/
|
|
188
|
-
const serverTaskDefinition = props.serverTaskDefinition
|
|
187
|
+
const serverTaskDefinition = props.serverTaskDefinition
|
|
188
|
+
? this.createTaskDefinition(props.serverTaskDefinition)
|
|
189
|
+
: this.createSampleTaskDefinition(records, logGroup);
|
|
189
190
|
this.certFileSystem.grant(serverTaskDefinition.taskRole, 'elasticfilesystem:ClientMount');
|
|
190
191
|
serverTaskDefinition.addVolume({
|
|
191
192
|
name: 'certVolume',
|
|
@@ -244,7 +245,16 @@ class LowCostECS extends lib.Stack {
|
|
|
244
245
|
new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });
|
|
245
246
|
new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });
|
|
246
247
|
}
|
|
247
|
-
|
|
248
|
+
createTaskDefinition(taskDefinitionOptions) {
|
|
249
|
+
const serverTaskDefinition = new ecs.Ec2TaskDefinition(this, 'ServerTaskDefinition', taskDefinitionOptions.taskDefinition);
|
|
250
|
+
taskDefinitionOptions.containers?.forEach((props, index) => {
|
|
251
|
+
const container = serverTaskDefinition.addContainer(props.containerName ?? `container${index}`, props);
|
|
252
|
+
container.addPortMappings(...(props.portMappings ?? []));
|
|
253
|
+
});
|
|
254
|
+
taskDefinitionOptions.volumes?.forEach((props) => serverTaskDefinition.addVolume(props));
|
|
255
|
+
return serverTaskDefinition;
|
|
256
|
+
}
|
|
257
|
+
createSampleTaskDefinition(records, logGroup) {
|
|
248
258
|
const nginxTaskDefinition = new ecs.Ec2TaskDefinition(this, 'NginxTaskDefinition');
|
|
249
259
|
const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {
|
|
250
260
|
image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),
|
|
@@ -274,5 +284,5 @@ class LowCostECS extends lib.Stack {
|
|
|
274
284
|
}
|
|
275
285
|
exports.LowCostECS = LowCostECS;
|
|
276
286
|
_a = JSII_RTTI_SYMBOL_1;
|
|
277
|
-
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.
|
|
278
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"low-cost-ecs.js","sourceRoot":"","sources":["../src/low-cost-ecs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as lib from 'aws-cdk-lib';\nimport { AutoScalingGroup } from 'aws-cdk-lib/aws-autoscaling';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { FileSystem } from 'aws-cdk-lib/aws-efs';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { SfnStateMachine } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport { Subscription, SubscriptionProtocol, Topic } from 'aws-cdk-lib/aws-sns';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as sfn_tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport interface LowCostECSProps extends lib.StackProps {\n  /**\n   * Domain name of the hosted zone.\n   */\n  readonly hostedZoneDomain: string;\n\n  /**\n   * Email for expiration emails to register to your let's encrypt account.\n   *\n   * @link https://letsencrypt.org/docs/expiration-emails/\n   *\n   * Also registered as a subscriber of the sns topic, notified on certbot task failure.\n   * Subscription confirmation email would be sent on stack creation.\n   *\n   * @link https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html\n   */\n  readonly email: string;\n\n  /**\n   * Domain names for A records to elastic ip of ECS host instance.\n   *\n   * @default - [ props.hostedZone.zoneName ]\n   */\n  readonly recordDomainNames?: string[];\n\n  /**\n   * Vpc of the ECS host instance and cluster.\n   *\n   * @default - Creates vpc with only public subnets and no NAT gateways.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group of the ECS host instance\n   *\n   * @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).\n   */\n  readonly securityGroup?: ec2.SecurityGroup;\n\n  /**\n   * Instance type of the ECS host instance.\n   *\n   * @default - t2.micro\n   */\n  readonly hostInstanceType?: string;\n\n  /**\n   * The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.\n   * Host instance asg would use spot instances if hostInstanceSpotPrice is set.\n   *\n   * @link https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice\n   * @default - undefined\n   */\n  readonly hostInstanceSpotPrice?: string;\n\n  /**\n   * Log group of the certbot task and the aws-cli task.\n   *\n   * @default - Creates default cdk log group\n   */\n  readonly logGroup?: ILogGroup;\n\n  /**\n   * Docker image tag of certbot/dns-route53 to create certificates.\n   *\n   * @link https://hub.docker.com/r/certbot/dns-route53/tags\n   * @default - v1.29.0\n   */\n  readonly certbotDockerTag?: string;\n\n  /**\n   * Certbot task schedule interval in days to renew the certificate.\n   *\n   * @default - 60\n   */\n  readonly certbotScheduleInterval?: number;\n\n  /**\n   * Docker image tag of amazon/aws-cli.\n   * This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.\n   *\n   * @default - latest\n   */\n  readonly awsCliDockerTag?: string;\n\n  /**\n   * Enable container insights or not\n   *\n   * @default - undefined (container insights disabled)\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * Removal policy for the file system and log group (if using default).\n   *\n   * @default - RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: lib.RemovalPolicy;\n\n  /**\n   * Task definition for the server ecs task.\n   *\n   * @default - Nginx server task definition defined in sampleServerTask()\n   */\n  readonly serverTaskDefinition?: ecs.Ec2TaskDefinition;\n};\n\nexport class LowCostECS extends lib.Stack {\n  readonly vpc: ec2.IVpc;\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  readonly certFileSystem: FileSystem;\n  readonly cluster: ecs.Cluster;\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id, props);\n\n    this.vpc =\n      props.vpc ??\n      new ec2.Vpc(this, 'Vpc', {\n        natGateways: 0,\n        subnetConfiguration: [\n          {\n            name: 'PublicSubnet',\n            subnetType: ec2.SubnetType.PUBLIC,\n          },\n        ],\n      });\n\n    this.cluster = new ecs.Cluster(this, 'Cluster', {\n      vpc: this.vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {\n      machineImage: ecs.EcsOptimizedImage.amazonLinux2(\n        ecs.AmiHardwareType.STANDARD,\n        {\n          cachedInContext: true,\n        },\n      ),\n      instanceType: new ec2.InstanceType(props.hostInstanceType ?? 't2.micro'),\n      spotPrice: props.hostInstanceSpotPrice,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      associatePublicIpAddress: true,\n      minCapacity: 1,\n      maxCapacity: 1,\n    });\n\n    if (props.securityGroup) {\n      this.hostAutoScalingGroup.addSecurityGroup(props.securityGroup);\n    } else {\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(80),\n      );\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(443),\n      );\n    }\n\n    /**\n     * Add managed policy to allow ssh through ssm manager\n     */\n    this.hostAutoScalingGroup.role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n    );\n    /**\n     * Add policy to associate elastic ip on startup\n     */\n    this.hostAutoScalingGroup.role.addToPrincipalPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['ec2:DescribeAddresses', 'ec2:AssociateAddress'],\n        resources: ['*'],\n      }),\n    );\n\n    const hostInstanceIp = new ec2.CfnEIP(this, 'HostInstanceIp');\n    const tagUniqueId = lib.Names.uniqueId(hostInstanceIp);\n    hostInstanceIp.tags.setTag('Name', tagUniqueId);\n\n    const awsCliTag = props.awsCliDockerTag ?? 'latest';\n    this.hostAutoScalingGroup.addUserData(\n      'INSTANCE_ID=$(curl --silent http://169.254.169.254/latest/meta-data/instance-id)',\n      `ALLOCATION_ID=$(docker run --net=host amazon/aws-cli:${awsCliTag} ec2 describe-addresses --region ${this.hostAutoScalingGroup.env.region} --filter Name=tag:Name,Values=${tagUniqueId} --query 'Addresses[].AllocationId' --output text | head)`,\n      `docker run --net=host amazon/aws-cli:${awsCliTag} ec2 associate-address --region ${this.hostAutoScalingGroup.env.region} --instance-id \"$INSTANCE_ID\" --allocation-id \"$ALLOCATION_ID\" --allow-reassociation`,\n    );\n\n    this.certFileSystem = new FileSystem(this, 'FileSystem', {\n      vpc: this.vpc,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc: this.vpc,\n        allowAllOutbound: false,\n      }),\n      removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n    });\n    this.certFileSystem.connections.allowDefaultPortTo(this.hostAutoScalingGroup);\n    this.certFileSystem.connections.allowDefaultPortFrom(this.hostAutoScalingGroup);\n\n    /**\n     * ARecord to Elastic ip\n     */\n    const hostedZone = route53.HostedZone.fromLookup(this, 'HostedZone', {\n      domainName: props.hostedZoneDomain,\n    });\n    const records = props.recordDomainNames ?? [hostedZone.zoneName];\n    records.forEach(\n      (record) =>\n        new route53.ARecord(this, `ARecord${record}`, {\n          zone: hostedZone,\n          recordName: record,\n          target: route53.RecordTarget.fromIpAddresses(hostInstanceIp.ref),\n        }),\n    );\n\n    /**\n     * Certbot Task Definition\n     * Mounts generated certificate to EFS\n     */\n    const logGroup =\n      props.logGroup ??\n      new LogGroup(this, 'LogGroup', {\n        retention: RetentionDays.TWO_YEARS,\n        removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n      });\n\n    const certbotTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'CertbotTaskDefinition',\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ListHostedZones', 'route53:GetChange'],\n        resources: ['*'],\n      }),\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ChangeResourceRecordSets'],\n        resources: [hostedZone.hostedZoneArn],\n      }),\n    );\n\n    const certbotTag = props.certbotDockerTag ?? 'v1.29.0';\n    const certbotContainer = certbotTaskDefinition.addContainer(\n      'CertbotContainer',\n      {\n        image: ecs.ContainerImage.fromRegistry(\n          `certbot/dns-route53:${certbotTag}`,\n        ),\n        containerName: 'certbot',\n        memoryReservationMiB: 64,\n        command: [\n          'certonly',\n          '--verbose',\n          '--preferred-challenges=dns-01',\n          '--dns-route53',\n          '--dns-route53-propagation-seconds=300',\n          '--non-interactive',\n          '--agree-tos',\n          '--expand',\n          '-m',\n          props.email,\n          '--cert-name',\n          records[0],\n          ...records.flatMap((domain) => ['-d', domain]),\n        ],\n        logging: ecs.LogDriver.awsLogs({\n          logGroup,\n          streamPrefix: certbotTag,\n        }),\n      },\n    );\n\n    this.certFileSystem.grant(\n      certbotTaskDefinition.taskRole,\n      'elasticfilesystem:ClientWrite',\n    );\n    certbotTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    certbotContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: false,\n    });\n\n    /**\n     * Schedule Certbot certificate create/renew on Step Functions\n     * Sends email notification on certbot failure\n     */\n    const topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: topic,\n      protocol: SubscriptionProtocol.EMAIL,\n      endpoint: props.email,\n    });\n\n    const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {\n      cluster: this.cluster,\n      taskDefinition: certbotTaskDefinition,\n      launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),\n      integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n    });\n    certbotRunTask.addCatch(\n      new sfn_tasks.SnsPublish(this, 'SendEmailOnFailure', {\n        topic: topic,\n        message: sfn.TaskInput.fromJsonPathAt('$'),\n      }).next(new sfn.Fail(this, 'Fail')),\n    );\n    certbotRunTask.addRetry({\n      interval: lib.Duration.seconds(20),\n    });\n    const certbotStateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition: certbotRunTask,\n    });\n\n    new Rule(this, 'CertbotScheduleRule', {\n      schedule: Schedule.rate(\n        lib.Duration.days(props.certbotScheduleInterval ?? 60),\n      ),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    const serverTaskDefinition =\n      props.serverTaskDefinition ?? this.sampleSeverTask(records, logGroup);\n    this.certFileSystem.grant(\n      serverTaskDefinition.taskRole,\n      'elasticfilesystem:ClientMount',\n    );\n    serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    serverTaskDefinition.defaultContainer?.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: true,\n    });\n\n    /**\n     * AWS cli container to execute certbot sfn before the default container startup.\n     */\n    serverTaskDefinition.defaultContainer?.addContainerDependencies({\n      container: serverTaskDefinition.addContainer('AWSCliContainer', {\n        image: ecs.ContainerImage.fromRegistry(`amazon/aws-cli:${awsCliTag}`),\n        containerName: 'aws-cli',\n        memoryReservationMiB: 64,\n        entryPoint: ['/bin/bash', '-c'],\n        command: [\n          `set -eux\n          aws configure set region ${certbotStateMachine.env.region} && \\\\\n          aws configure set output text && \\\\\n          EXECUTION_ARN=$(aws stepfunctions start-execution --state-machine-arn ${certbotStateMachine.stateMachineArn} --query executionArn) && \\\\\n          until [ $(aws stepfunctions describe-execution --execution-arn \"$EXECUTION_ARN\" --query status) != RUNNING ];\n          do\n            echo \"Waiting for $EXECUTION_ARN\"\n            sleep 10\n          done`,\n        ],\n        essential: false,\n        logging: ecs.LogDriver.awsLogs({\n          logGroup: logGroup,\n          streamPrefix: awsCliTag,\n        }),\n      }),\n      condition: ecs.ContainerDependencyCondition.COMPLETE,\n    });\n    certbotStateMachine.grantExecution(\n      serverTaskDefinition.taskRole,\n      'states:DescribeExecution',\n    );\n    certbotStateMachine.grantStartExecution(serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: serverTaskDefinition,\n      desiredCount: 1,\n      minHealthyPercent: 0,\n      maxHealthyPercent: 100,\n      circuitBreaker: {\n        rollback: true,\n      },\n      enableExecuteCommand: true,\n    });\n\n    new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });\n    new lib.CfnOutput(this, 'certbotStateMachineName', { value: certbotStateMachine.stateMachineName });\n    new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });\n    new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });\n  }\n\n  private sampleSeverTask(\n    records: string[],\n    logGroup: ILogGroup,\n  ): ecs.Ec2TaskDefinition {\n    const nginxTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'NginxTaskDefinition',\n    );\n    const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {\n      image: ecs.ContainerImage.fromAsset(\n        path.join(__dirname, '../examples/containers/nginx'),\n      ),\n      containerName: 'nginx',\n      memoryReservationMiB: 64,\n      essential: true,\n      environment: {\n        SERVER_NAME: records.join(' '),\n        CERT_NAME: records[0],\n      },\n      logging: ecs.LogDrivers.awsLogs({\n        logGroup: logGroup,\n        streamPrefix: 'nginx-proxy',\n      }),\n    });\n\n    nginxContainer.addPortMappings(\n      {\n        hostPort: 80,\n        containerPort: 80,\n        protocol: ecs.Protocol.TCP,\n      },\n      {\n        hostPort: 443,\n        containerPort: 443,\n        protocol: ecs.Protocol.TCP,\n      },\n    );\n\n    return nginxTaskDefinition;\n  }\n}\n"]}
|
|
287
|
+
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.18" };
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"low-cost-ecs.js","sourceRoot":"","sources":["../src/low-cost-ecs.ts"],"names":[],"mappings":";;;;;AAAA,6BAA6B;AAC7B,mCAAmC;AAEnC,2CAA2C;AAC3C,2CAA2C;AAC3C,iDAAiD;AACjD,uDAAwD;AACxD,uEAAiE;AACjE,iDAA6E;AAC7E,mDAA0E;AAC1E,mDAAmD;AACnD,iDAAgF;AAChF,qDAAqD;AACrD,iEAAiE;AAoHjE,MAAa,UAAW,SAAQ,GAAG,CAAC,KAAK;IAOvC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QAExB,IAAI,CAAC,GAAG;YACN,KAAK,CAAC,GAAG;gBACT,IAAI,GAAG,CAAC,GAAG,CAAC,IAAI,EAAE,KAAK,EAAE;oBACvB,WAAW,EAAE,CAAC;oBACd,mBAAmB,EAAE;wBACnB;4BACE,IAAI,EAAE,cAAc;4BACpB,UAAU,EAAE,GAAG,CAAC,UAAU,CAAC,MAAM;yBAClC;qBACF;iBACF,CAAC,CAAC;QAEL,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,SAAS,EAAE;YAC9C,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;SAC3C,CAAC,CAAC;QAEH,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,sBAAsB,EAAE;YAC3E,YAAY,EAAE,GAAG,CAAC,iBAAiB,CAAC,YAAY,CAC9C,GAAG,CAAC,eAAe,CAAC,QAAQ,EAC5B;gBACE,eAAe,EAAE,IAAI;aACtB,CACF;YACD,YAAY,EAAE,IAAI,GAAG,CAAC,YAAY,CAAC,KAAK,CAAC,gBAAgB,IAAI,UAAU,CAAC;YACxE,SAAS,EAAE,KAAK,CAAC,qBAAqB;YACtC,UAAU,EAAE,EAAE,UAAU,EAAE,GAAG,CAAC,UAAU,CAAC,MAAM,EAAE;YACjD,wBAAwB,EAAE,IAAI;YAC9B,WAAW,EAAE,CAAC;YACd,WAAW,EAAE,CAAC;SACf,CAAC,CAAC;QAEH,IAAI,KAAK,CAAC,aAAa,EAAE;YACvB,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;SACjE;aAAM;YACL,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,gBAAgB,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YACzE,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,gBAAgB,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;YAC1E,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,SAAS,CAC7C,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,EAClB,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CACjB,CAAC;YACF,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,SAAS,CAC7C,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,EAClB,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAClB,CAAC;SACH;QAED;;WAEG;QACH,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,gBAAgB,CAC7C,uBAAa,CAAC,wBAAwB,CAAC,8BAA8B,CAAC,CACvE,CAAC;QACF;;WAEG;QACH,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,oBAAoB,CACjD,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE,CAAC,uBAAuB,EAAE,sBAAsB,CAAC;YAC1D,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CACH,CAAC;QAEF,MAAM,cAAc,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,gBAAgB,CAAC,CAAC;QAC9D,MAAM,WAAW,GAAG,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;QACvD,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;QAEhD,MAAM,SAAS,GAAG,KAAK,CAAC,eAAe,IAAI,QAAQ,CAAC;QACpD,IAAI,CAAC,oBAAoB,CAAC,WAAW,CACnC,kFAAkF,EAClF,wDAAwD,SAAS,oCAAoC,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,MAAM,kCAAkC,WAAW,2DAA2D,EACjP,wCAAwC,SAAS,mCAAmC,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,MAAM,sFAAsF,CAC/M,CAAC;QAEF,IAAI,CAAC,cAAc,GAAG,IAAI,oBAAU,CAAC,IAAI,EAAE,YAAY,EAAE;YACvD,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,SAAS,EAAE,IAAI;YACf,aAAa,EAAE,IAAI,GAAG,CAAC,aAAa,CAAC,IAAI,EAAE,yBAAyB,EAAE;gBACpE,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,gBAAgB,EAAE,KAAK;aACxB,CAAC;YACF,aAAa,EAAE,KAAK,CAAC,aAAa,IAAI,GAAG,CAAC,aAAa,CAAC,OAAO;SAChE,CAAC,CAAC;QACH,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,kBAAkB,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;QAC9E,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,oBAAoB,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;QAEhF;;WAEG;QACH,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,YAAY,EAAE;YACnE,UAAU,EAAE,KAAK,CAAC,gBAAgB;SACnC,CAAC,CAAC;QACH,MAAM,OAAO,GAAG,KAAK,CAAC,iBAAiB,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACjE,OAAO,CAAC,OAAO,CACb,CAAC,MAAM,EAAE,EAAE,CACT,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE,UAAU,MAAM,EAAE,EAAE;YAC5C,IAAI,EAAE,UAAU;YAChB,UAAU,EAAE,MAAM;YAClB,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,eAAe,CAAC,cAAc,CAAC,GAAG,CAAC;SACjE,CAAC,CACL,CAAC;QAEF;;;WAGG;QACH,MAAM,QAAQ,GACZ,KAAK,CAAC,QAAQ;YACd,IAAI,mBAAQ,CAAC,IAAI,EAAE,UAAU,EAAE;gBAC7B,SAAS,EAAE,wBAAa,CAAC,SAAS;gBAClC,aAAa,EAAE,KAAK,CAAC,aAAa,IAAI,GAAG,CAAC,aAAa,CAAC,OAAO;aAChE,CAAC,CAAC;QAEL,MAAM,qBAAqB,GAAG,IAAI,GAAG,CAAC,iBAAiB,CACrD,IAAI,EACJ,uBAAuB,CACxB,CAAC;QACF,qBAAqB,CAAC,mBAAmB,CACvC,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE,CAAC,yBAAyB,EAAE,mBAAmB,CAAC;YACzD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CACH,CAAC;QACF,qBAAqB,CAAC,mBAAmB,CACvC,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE,CAAC,kCAAkC,CAAC;YAC7C,SAAS,EAAE,CAAC,UAAU,CAAC,aAAa,CAAC;SACtC,CAAC,CACH,CAAC;QAEF,MAAM,UAAU,GAAG,KAAK,CAAC,gBAAgB,IAAI,SAAS,CAAC;QACvD,MAAM,gBAAgB,GAAG,qBAAqB,CAAC,YAAY,CACzD,kBAAkB,EAClB;YACE,KAAK,EAAE,GAAG,CAAC,cAAc,CAAC,YAAY,CACpC,uBAAuB,UAAU,EAAE,CACpC;YACD,aAAa,EAAE,SAAS;YACxB,oBAAoB,EAAE,EAAE;YACxB,OAAO,EAAE;gBACP,UAAU;gBACV,WAAW;gBACX,+BAA+B;gBAC/B,eAAe;gBACf,uCAAuC;gBACvC,mBAAmB;gBACnB,aAAa;gBACb,UAAU;gBACV,IAAI;gBACJ,KAAK,CAAC,KAAK;gBACX,aAAa;gBACb,OAAO,CAAC,CAAC,CAAC;gBACV,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;aAC/C;YACD,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,OAAO,CAAC;gBAC7B,QAAQ;gBACR,YAAY,EAAE,UAAU;aACzB,CAAC;SACH,CACF,CAAC;QAEF,IAAI,CAAC,cAAc,CAAC,KAAK,CACvB,qBAAqB,CAAC,QAAQ,EAC9B,+BAA+B,CAChC,CAAC;QACF,qBAAqB,CAAC,SAAS,CAAC;YAC9B,IAAI,EAAE,YAAY;YAClB,sBAAsB,EAAE;gBACtB,YAAY,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY;aAC/C;SACF,CAAC,CAAC;QACH,gBAAgB,CAAC,cAAc,CAAC;YAC9B,YAAY,EAAE,YAAY;YAC1B,aAAa,EAAE,kBAAkB;YACjC,QAAQ,EAAE,KAAK;SAChB,CAAC,CAAC;QAEH;;;WAGG;QACH,MAAM,KAAK,GAAG,IAAI,eAAK,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QACvC,IAAI,sBAAY,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC1C,KAAK,EAAE,KAAK;YACZ,QAAQ,EAAE,8BAAoB,CAAC,KAAK;YACpC,QAAQ,EAAE,KAAK,CAAC,KAAK;SACtB,CAAC,CAAC;QAEH,MAAM,cAAc,GAAG,IAAI,SAAS,CAAC,UAAU,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACzE,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,cAAc,EAAE,qBAAqB;YACrC,YAAY,EAAE,IAAI,SAAS,CAAC,kBAAkB,EAAE;YAChD,kBAAkB,EAAE,GAAG,CAAC,kBAAkB,CAAC,OAAO;SACnD,CAAC,CAAC;QACH,cAAc,CAAC,QAAQ,CACrB,IAAI,SAAS,CAAC,UAAU,CAAC,IAAI,EAAE,oBAAoB,EAAE;YACnD,KAAK,EAAE,KAAK;YACZ,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,cAAc,CAAC,GAAG,CAAC;SAC3C,CAAC,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CACpC,CAAC;QACF,cAAc,CAAC,QAAQ,CAAC;YACtB,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SACnC,CAAC,CAAC;QACH,MAAM,mBAAmB,GAAG,IAAI,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,cAAc,EAAE;YACrE,UAAU,EAAE,cAAc;SAC3B,CAAC,CAAC;QAEH,IAAI,iBAAI,CAAC,IAAI,EAAE,qBAAqB,EAAE;YACpC,QAAQ,EAAE,qBAAQ,CAAC,IAAI,CACrB,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,IAAI,EAAE,CAAC,CACvD;YACD,OAAO,EAAE,CAAC,IAAI,oCAAe,CAAC,mBAAmB,CAAC,CAAC;SACpD,CAAC,CAAC;QAEH;;WAEG;QACH,MAAM,oBAAoB,GAAG,KAAK,CAAC,oBAAoB;YACrD,CAAC,CAAC,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,oBAAoB,CAAC;YACvD,CAAC,CAAC,IAAI,CAAC,0BAA0B,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;QAEvD,IAAI,CAAC,cAAc,CAAC,KAAK,CACvB,oBAAoB,CAAC,QAAQ,EAC7B,+BAA+B,CAChC,CAAC;QACF,oBAAoB,CAAC,SAAS,CAAC;YAC7B,IAAI,EAAE,YAAY;YAClB,sBAAsB,EAAE;gBACtB,YAAY,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY;aAC/C;SACF,CAAC,CAAC;QACH,oBAAoB,CAAC,gBAAgB,EAAE,cAAc,CAAC;YACpD,YAAY,EAAE,YAAY;YAC1B,aAAa,EAAE,kBAAkB;YACjC,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;QAEH;;WAEG;QACH,oBAAoB,CAAC,gBAAgB,EAAE,wBAAwB,CAAC;YAC9D,SAAS,EAAE,oBAAoB,CAAC,YAAY,CAAC,iBAAiB,EAAE;gBAC9D,KAAK,EAAE,GAAG,CAAC,cAAc,CAAC,YAAY,CAAC,kBAAkB,SAAS,EAAE,CAAC;gBACrE,aAAa,EAAE,SAAS;gBACxB,oBAAoB,EAAE,EAAE;gBACxB,UAAU,EAAE,CAAC,WAAW,EAAE,IAAI,CAAC;gBAC/B,OAAO,EAAE;oBACP;qCAC2B,mBAAmB,CAAC,GAAG,CAAC,MAAM;;kFAEe,mBAAmB,CAAC,eAAe;;;;;eAKtG;iBACN;gBACD,SAAS,EAAE,KAAK;gBAChB,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,OAAO,CAAC;oBAC7B,QAAQ,EAAE,QAAQ;oBAClB,YAAY,EAAE,SAAS;iBACxB,CAAC;aACH,CAAC;YACF,SAAS,EAAE,GAAG,CAAC,4BAA4B,CAAC,QAAQ;SACrD,CAAC,CAAC;QACH,mBAAmB,CAAC,cAAc,CAChC,oBAAoB,CAAC,QAAQ,EAC7B,0BAA0B,CAC3B,CAAC;QACF,mBAAmB,CAAC,mBAAmB,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;QAEvE,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,EAAE;YACjD,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,cAAc,EAAE,oBAAoB;YACpC,YAAY,EAAE,CAAC;YACf,iBAAiB,EAAE,CAAC;YACpB,iBAAiB,EAAE,GAAG;YACtB,cAAc,EAAE;gBACd,QAAQ,EAAE,IAAI;aACf;YACD,oBAAoB,EAAE,IAAI;SAC3B,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,iBAAiB,EAAE,EAAE,KAAK,EAAE,cAAc,CAAC,GAAG,EAAE,CAAC,CAAC;QAC1E,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,yBAAyB,EAAE,EAAE,KAAK,EAAE,mBAAmB,CAAC,gBAAgB,EAAE,CAAC,CAAC;QACpG,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,aAAa,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;QAC5E,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,aAAa,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;IAC9E,CAAC;IAEO,oBAAoB,CAAC,qBAAsD;QACjF,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAC,iBAAiB,CACpD,IAAI,EACJ,sBAAsB,EACtB,qBAAqB,CAAC,cAAc,CACrC,CAAC;QACF,qBAAqB,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;YACzD,MAAM,SAAS,GAAG,oBAAoB,CAAC,YAAY,CAAC,KAAK,CAAC,aAAa,IAAI,YAAY,KAAK,EAAE,EAAE,KAAK,CAAC,CAAC;YACvG,SAAS,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;QACH,qBAAqB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,oBAAoB,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;QACzF,OAAO,oBAAoB,CAAC;IAC9B,CAAC;IAEO,0BAA0B,CAChC,OAAiB,EACjB,QAAmB;QAEnB,MAAM,mBAAmB,GAAG,IAAI,GAAG,CAAC,iBAAiB,CACnD,IAAI,EACJ,qBAAqB,CACtB,CAAC;QACF,MAAM,cAAc,GAAG,mBAAmB,CAAC,YAAY,CAAC,gBAAgB,EAAE;YACxE,KAAK,EAAE,GAAG,CAAC,cAAc,CAAC,SAAS,CACjC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,8BAA8B,CAAC,CACrD;YACD,aAAa,EAAE,OAAO;YACtB,oBAAoB,EAAE,EAAE;YACxB,SAAS,EAAE,IAAI;YACf,WAAW,EAAE;gBACX,WAAW,EAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC;gBAC9B,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;aACtB;YACD,OAAO,EAAE,GAAG,CAAC,UAAU,CAAC,OAAO,CAAC;gBAC9B,QAAQ,EAAE,QAAQ;gBAClB,YAAY,EAAE,aAAa;aAC5B,CAAC;SACH,CAAC,CAAC;QAEH,cAAc,CAAC,eAAe,CAC5B;YACE,QAAQ,EAAE,EAAE;YACZ,aAAa,EAAE,EAAE;YACjB,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,GAAG;SAC3B,EACD;YACE,QAAQ,EAAE,GAAG;YACb,aAAa,EAAE,GAAG;YAClB,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,GAAG;SAC3B,CACF,CAAC;QAEF,OAAO,mBAAmB,CAAC;IAC7B,CAAC;;AAnWH,gCAoWC","sourcesContent":["import * as path from 'path';\nimport * as lib from 'aws-cdk-lib';\nimport { AutoScalingGroup } from 'aws-cdk-lib/aws-autoscaling';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { FileSystem } from 'aws-cdk-lib/aws-efs';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { SfnStateMachine } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport { Subscription, SubscriptionProtocol, Topic } from 'aws-cdk-lib/aws-sns';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as sfn_tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport interface LowCostECSProps extends lib.StackProps {\n  /**\n   * Domain name of the hosted zone.\n   */\n  readonly hostedZoneDomain: string;\n\n  /**\n   * Email for expiration emails to register to your let's encrypt account.\n   *\n   * @link https://letsencrypt.org/docs/expiration-emails/\n   *\n   * Also registered as a subscriber of the sns topic, notified on certbot task failure.\n   * Subscription confirmation email would be sent on stack creation.\n   *\n   * @link https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html\n   */\n  readonly email: string;\n\n  /**\n   * Domain names for A records to elastic ip of ECS host instance.\n   *\n   * @default - [ props.hostedZone.zoneName ]\n   */\n  readonly recordDomainNames?: string[];\n\n  /**\n   * Vpc of the ECS host instance and cluster.\n   *\n   * @default - Creates vpc with only public subnets and no NAT gateways.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group of the ECS host instance\n   *\n   * @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Instance type of the ECS host instance.\n   *\n   * @default - t2.micro\n   */\n  readonly hostInstanceType?: string;\n\n  /**\n   * The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.\n   * Host instance asg would use spot instances if hostInstanceSpotPrice is set.\n   *\n   * @link https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice\n   * @default - undefined\n   */\n  readonly hostInstanceSpotPrice?: string;\n\n  /**\n   * Log group of the certbot task and the aws-cli task.\n   *\n   * @default - Creates default cdk log group\n   */\n  readonly logGroup?: ILogGroup;\n\n  /**\n   * Docker image tag of certbot/dns-route53 to create certificates.\n   *\n   * @link https://hub.docker.com/r/certbot/dns-route53/tags\n   * @default - v1.29.0\n   */\n  readonly certbotDockerTag?: string;\n\n  /**\n   * Certbot task schedule interval in days to renew the certificate.\n   *\n   * @default - 60\n   */\n  readonly certbotScheduleInterval?: number;\n\n  /**\n   * Docker image tag of amazon/aws-cli.\n   * This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.\n   *\n   * @default - latest\n   */\n  readonly awsCliDockerTag?: string;\n\n  /**\n   * Enable container insights or not\n   *\n   * @default - undefined (container insights disabled)\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * Removal policy for the file system and log group (if using default).\n   *\n   * @default - RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: lib.RemovalPolicy;\n\n  /**\n   * Task definition for the server ecs task.\n   *\n   * @default - Nginx server task definition defined in createSampleTaskDefinition()\n   */\n  readonly serverTaskDefinition?: LowCostECSTaskDefinitionOptions;\n}\n\nexport interface LowCostECSTaskDefinitionOptions {\n  readonly taskDefinition?: ecs.Ec2TaskDefinitionProps;\n  readonly containers: ecs.ContainerDefinitionOptions[];\n  readonly volumes?: ecs.Volume[];\n}\n\nexport class LowCostECS extends lib.Stack {\n  readonly vpc: ec2.IVpc;\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  readonly certFileSystem: FileSystem;\n  readonly cluster: ecs.Cluster;\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id, props);\n\n    this.vpc =\n      props.vpc ??\n      new ec2.Vpc(this, 'Vpc', {\n        natGateways: 0,\n        subnetConfiguration: [\n          {\n            name: 'PublicSubnet',\n            subnetType: ec2.SubnetType.PUBLIC,\n          },\n        ],\n      });\n\n    this.cluster = new ecs.Cluster(this, 'Cluster', {\n      vpc: this.vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {\n      machineImage: ecs.EcsOptimizedImage.amazonLinux2(\n        ecs.AmiHardwareType.STANDARD,\n        {\n          cachedInContext: true,\n        },\n      ),\n      instanceType: new ec2.InstanceType(props.hostInstanceType ?? 't2.micro'),\n      spotPrice: props.hostInstanceSpotPrice,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      associatePublicIpAddress: true,\n      minCapacity: 1,\n      maxCapacity: 1,\n    });\n\n    if (props.securityGroup) {\n      this.hostAutoScalingGroup.addSecurityGroup(props.securityGroup);\n    } else {\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(80),\n      );\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(443),\n      );\n    }\n\n    /**\n     * Add managed policy to allow ssh through ssm manager\n     */\n    this.hostAutoScalingGroup.role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n    );\n    /**\n     * Add policy to associate elastic ip on startup\n     */\n    this.hostAutoScalingGroup.role.addToPrincipalPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['ec2:DescribeAddresses', 'ec2:AssociateAddress'],\n        resources: ['*'],\n      }),\n    );\n\n    const hostInstanceIp = new ec2.CfnEIP(this, 'HostInstanceIp');\n    const tagUniqueId = lib.Names.uniqueId(hostInstanceIp);\n    hostInstanceIp.tags.setTag('Name', tagUniqueId);\n\n    const awsCliTag = props.awsCliDockerTag ?? 'latest';\n    this.hostAutoScalingGroup.addUserData(\n      'INSTANCE_ID=$(curl --silent http://169.254.169.254/latest/meta-data/instance-id)',\n      `ALLOCATION_ID=$(docker run --net=host amazon/aws-cli:${awsCliTag} ec2 describe-addresses --region ${this.hostAutoScalingGroup.env.region} --filter Name=tag:Name,Values=${tagUniqueId} --query 'Addresses[].AllocationId' --output text | head)`,\n      `docker run --net=host amazon/aws-cli:${awsCliTag} ec2 associate-address --region ${this.hostAutoScalingGroup.env.region} --instance-id \"$INSTANCE_ID\" --allocation-id \"$ALLOCATION_ID\" --allow-reassociation`,\n    );\n\n    this.certFileSystem = new FileSystem(this, 'FileSystem', {\n      vpc: this.vpc,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc: this.vpc,\n        allowAllOutbound: false,\n      }),\n      removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n    });\n    this.certFileSystem.connections.allowDefaultPortTo(this.hostAutoScalingGroup);\n    this.certFileSystem.connections.allowDefaultPortFrom(this.hostAutoScalingGroup);\n\n    /**\n     * ARecord to Elastic ip\n     */\n    const hostedZone = route53.HostedZone.fromLookup(this, 'HostedZone', {\n      domainName: props.hostedZoneDomain,\n    });\n    const records = props.recordDomainNames ?? [hostedZone.zoneName];\n    records.forEach(\n      (record) =>\n        new route53.ARecord(this, `ARecord${record}`, {\n          zone: hostedZone,\n          recordName: record,\n          target: route53.RecordTarget.fromIpAddresses(hostInstanceIp.ref),\n        }),\n    );\n\n    /**\n     * Certbot Task Definition\n     * Mounts generated certificate to EFS\n     */\n    const logGroup =\n      props.logGroup ??\n      new LogGroup(this, 'LogGroup', {\n        retention: RetentionDays.TWO_YEARS,\n        removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n      });\n\n    const certbotTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'CertbotTaskDefinition',\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ListHostedZones', 'route53:GetChange'],\n        resources: ['*'],\n      }),\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ChangeResourceRecordSets'],\n        resources: [hostedZone.hostedZoneArn],\n      }),\n    );\n\n    const certbotTag = props.certbotDockerTag ?? 'v1.29.0';\n    const certbotContainer = certbotTaskDefinition.addContainer(\n      'CertbotContainer',\n      {\n        image: ecs.ContainerImage.fromRegistry(\n          `certbot/dns-route53:${certbotTag}`,\n        ),\n        containerName: 'certbot',\n        memoryReservationMiB: 64,\n        command: [\n          'certonly',\n          '--verbose',\n          '--preferred-challenges=dns-01',\n          '--dns-route53',\n          '--dns-route53-propagation-seconds=300',\n          '--non-interactive',\n          '--agree-tos',\n          '--expand',\n          '-m',\n          props.email,\n          '--cert-name',\n          records[0],\n          ...records.flatMap((domain) => ['-d', domain]),\n        ],\n        logging: ecs.LogDriver.awsLogs({\n          logGroup,\n          streamPrefix: certbotTag,\n        }),\n      },\n    );\n\n    this.certFileSystem.grant(\n      certbotTaskDefinition.taskRole,\n      'elasticfilesystem:ClientWrite',\n    );\n    certbotTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    certbotContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: false,\n    });\n\n    /**\n     * Schedule Certbot certificate create/renew on Step Functions\n     * Sends email notification on certbot failure\n     */\n    const topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: topic,\n      protocol: SubscriptionProtocol.EMAIL,\n      endpoint: props.email,\n    });\n\n    const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {\n      cluster: this.cluster,\n      taskDefinition: certbotTaskDefinition,\n      launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),\n      integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n    });\n    certbotRunTask.addCatch(\n      new sfn_tasks.SnsPublish(this, 'SendEmailOnFailure', {\n        topic: topic,\n        message: sfn.TaskInput.fromJsonPathAt('$'),\n      }).next(new sfn.Fail(this, 'Fail')),\n    );\n    certbotRunTask.addRetry({\n      interval: lib.Duration.seconds(20),\n    });\n    const certbotStateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition: certbotRunTask,\n    });\n\n    new Rule(this, 'CertbotScheduleRule', {\n      schedule: Schedule.rate(\n        lib.Duration.days(props.certbotScheduleInterval ?? 60),\n      ),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    const serverTaskDefinition = props.serverTaskDefinition\n      ? this.createTaskDefinition(props.serverTaskDefinition)\n      : this.createSampleTaskDefinition(records, logGroup);\n\n    this.certFileSystem.grant(\n      serverTaskDefinition.taskRole,\n      'elasticfilesystem:ClientMount',\n    );\n    serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    serverTaskDefinition.defaultContainer?.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: true,\n    });\n\n    /**\n     * AWS cli container to execute certbot sfn before the default container startup.\n     */\n    serverTaskDefinition.defaultContainer?.addContainerDependencies({\n      container: serverTaskDefinition.addContainer('AWSCliContainer', {\n        image: ecs.ContainerImage.fromRegistry(`amazon/aws-cli:${awsCliTag}`),\n        containerName: 'aws-cli',\n        memoryReservationMiB: 64,\n        entryPoint: ['/bin/bash', '-c'],\n        command: [\n          `set -eux\n          aws configure set region ${certbotStateMachine.env.region} && \\\\\n          aws configure set output text && \\\\\n          EXECUTION_ARN=$(aws stepfunctions start-execution --state-machine-arn ${certbotStateMachine.stateMachineArn} --query executionArn) && \\\\\n          until [ $(aws stepfunctions describe-execution --execution-arn \"$EXECUTION_ARN\" --query status) != RUNNING ];\n          do\n            echo \"Waiting for $EXECUTION_ARN\"\n            sleep 10\n          done`,\n        ],\n        essential: false,\n        logging: ecs.LogDriver.awsLogs({\n          logGroup: logGroup,\n          streamPrefix: awsCliTag,\n        }),\n      }),\n      condition: ecs.ContainerDependencyCondition.COMPLETE,\n    });\n    certbotStateMachine.grantExecution(\n      serverTaskDefinition.taskRole,\n      'states:DescribeExecution',\n    );\n    certbotStateMachine.grantStartExecution(serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: serverTaskDefinition,\n      desiredCount: 1,\n      minHealthyPercent: 0,\n      maxHealthyPercent: 100,\n      circuitBreaker: {\n        rollback: true,\n      },\n      enableExecuteCommand: true,\n    });\n\n    new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });\n    new lib.CfnOutput(this, 'certbotStateMachineName', { value: certbotStateMachine.stateMachineName });\n    new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });\n    new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });\n  }\n\n  private createTaskDefinition(taskDefinitionOptions: LowCostECSTaskDefinitionOptions) : ecs.Ec2TaskDefinition {\n    const serverTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'ServerTaskDefinition',\n      taskDefinitionOptions.taskDefinition,\n    );\n    taskDefinitionOptions.containers?.forEach((props, index) => {\n      const container = serverTaskDefinition.addContainer(props.containerName ?? `container${index}`, props);\n      container.addPortMappings(...(props.portMappings ?? []));\n    });\n    taskDefinitionOptions.volumes?.forEach((props) => serverTaskDefinition.addVolume(props));\n    return serverTaskDefinition;\n  }\n\n  private createSampleTaskDefinition(\n    records: string[],\n    logGroup: ILogGroup,\n  ): ecs.Ec2TaskDefinition {\n    const nginxTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'NginxTaskDefinition',\n    );\n    const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {\n      image: ecs.ContainerImage.fromAsset(\n        path.join(__dirname, '../examples/containers/nginx'),\n      ),\n      containerName: 'nginx',\n      memoryReservationMiB: 64,\n      essential: true,\n      environment: {\n        SERVER_NAME: records.join(' '),\n        CERT_NAME: records[0],\n      },\n      logging: ecs.LogDrivers.awsLogs({\n        logGroup: logGroup,\n        streamPrefix: 'nginx-proxy',\n      }),\n    });\n\n    nginxContainer.addPortMappings(\n      {\n        hostPort: 80,\n        containerPort: 80,\n        protocol: ecs.Protocol.TCP,\n      },\n      {\n        hostPort: 443,\n        containerPort: 443,\n        protocol: ecs.Protocol.TCP,\n      },\n    );\n\n    return nginxTaskDefinition;\n  }\n}\n"]}
|