low-cost-ecs 0.0.23 → 0.0.25
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/.gitattributes +2 -0
- package/.jsii +33 -33
- package/.prettierignore +1 -0
- package/.prettierrc.json +5 -0
- package/.projenrc.ts +10 -13
- package/API.md +25 -22
- package/README.md +1 -1
- package/examples/all-props.ts +26 -16
- package/lib/low-cost-ecs.d.ts +24 -5
- package/lib/low-cost-ecs.js +26 -23
- package/package.json +4 -1
package/.gitattributes
CHANGED
|
@@ -12,6 +12,8 @@
|
|
|
12
12
|
/.gitignore linguist-generated
|
|
13
13
|
/.mergify.yml linguist-generated
|
|
14
14
|
/.npmignore linguist-generated
|
|
15
|
+
/.prettierignore linguist-generated
|
|
16
|
+
/.prettierrc.json linguist-generated
|
|
15
17
|
/.projen/** linguist-generated
|
|
16
18
|
/.projen/deps.json linguist-generated
|
|
17
19
|
/.projen/files.json linguist-generated
|
package/.jsii
CHANGED
|
@@ -3043,7 +3043,7 @@
|
|
|
3043
3043
|
},
|
|
3044
3044
|
"name": "low-cost-ecs",
|
|
3045
3045
|
"readme": {
|
|
3046
|
-
"markdown": "[](https://www.npmjs.com/package/low-cost-ecs)\n[](https://pypi.org/project/low-cost-ecs)\n[](https://github.com/rajyan/low-cost-ecs/actions/workflows/release.yml)\n[<img src=\"https://constructs.dev/badge?package=low-cost-ecs\" width=\"150\">](https://constructs.dev/packages/low-cost-ecs)\n\n# Low-Cost ECS\n\nA CDK construct that provides an easy and [low-cost](#cost) ECS on EC2 server setup without a load balancer.\n\n**This construct is for development purposes only**. See [Limitations](#limitations).\n\n# Why\n\nECS may often seem expensive when used for personal development purposes, due to the cost of the load balancer.\nThe application load balancer is a great service that is easy to set up managed ACM certificates, easy scaling, and has dynamic port mappings..., but it is over-featured for running 1 ECS task.\n\nHowever, to run an ECS server without a load balancer, you need to associate an Elastic IP to the host instance and install your certificate to your service every time you start up the server.\nThis construct aims to automate these works and make it easy to deploy resources to run a low-cost ECS server.\n\n# Try it out!\n\nThe easiest way to try the construct is to clone this repository and deploy the sample server.\nEdit settings in `examples/minimum.ts` and deploy the cdk construct. [Public hosted zone](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) is required.\n\n```\ngit clone https://github.com/rajyan/low-cost-ecs.git\nyarn install\n# edit settings in bin/low-cost-ecs.ts\
|
|
3046
|
+
"markdown": "[](https://www.npmjs.com/package/low-cost-ecs)\n[](https://pypi.org/project/low-cost-ecs)\n[](https://github.com/rajyan/low-cost-ecs/actions/workflows/release.yml)\n[<img src=\"https://constructs.dev/badge?package=low-cost-ecs\" width=\"150\">](https://constructs.dev/packages/low-cost-ecs)\n\n# Low-Cost ECS\n\nA CDK construct that provides an easy and [low-cost](#cost) ECS on EC2 server setup without a load balancer.\n\n**This construct is for development purposes only**. See [Limitations](#limitations).\n\n# Why\n\nECS may often seem expensive when used for personal development purposes, due to the cost of the load balancer.\nThe application load balancer is a great service that is easy to set up managed ACM certificates, easy scaling, and has dynamic port mappings..., but it is over-featured for running 1 ECS task.\n\nHowever, to run an ECS server without a load balancer, you need to associate an Elastic IP to the host instance and install your certificate to your service every time you start up the server.\nThis construct aims to automate these works and make it easy to deploy resources to run a low-cost ECS server.\n\n# Try it out!\n\nThe easiest way to try the construct is to clone this repository and deploy the sample server.\nEdit settings in `examples/minimum.ts` and deploy the cdk construct. [Public hosted zone](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) is required.\n\n```\ngit clone https://github.com/rajyan/low-cost-ecs.git\nyarn install\n# edit settings in bin/low-cost-ecs.ts\ncdk deploy\n```\n\nAccess the configured `recordDomainNames` and see that the Nginx sample server has been deployed.\n\n# Installation\n\nTo use this construct in your cdk stack as a library,\n\n```\nnpm install low-cost-ecs\n```\n\n```ts\nimport { Stack, StackProps } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { LowCostECS } from 'low-cost-ecs';\n\nclass SampleStack extends Stack {\n constructor(scope: Construct, id: string, props?: StackProps) {\n super(scope, id, props);\n\n const vpc = { /** Your VPC */ };\n const securityGroup = { /** Your security group */ };\n const serverTaskDefinition = { /** Your task definition */ };\n\n new LowCostECS(this, 'LowCostECS', {\n hostedZoneDomain: \"rajyan.net\",\n email: \"kitakita7617@gmail.com\",\n vpc: vpc,\n securityGroup: securityGroup,\n serverTaskDefinition: serverTaskDefinition\n });\n }\n}\n```\n\nThe required fields are `hostedZoneDomain` and `email`.\nYou can configure your server task definition and other props. Read [`LowCostECSProps` documentation](https://github.com/rajyan/low-cost-ecs/blob/main/API.md#low-cost-ecs.LowCostECSProps) for details.\n\n# Overview\n\nResources generated in this stack\n\n* Route53 A record\n * Forwarding to host instance Elastic IP\n* Certificate State Machine\n * Install and renew certificates to EFS using [certbot-dns-route53](https://certbot-dns-route53.readthedocs.io/en/stable/)\n * Scheduled automated renewal every 60 days\n * Email notification on certbot task failure\n* ECS on EC2 host instance\n * ECS-optimized Amazon Linux 2 AMI instance auto-scaling group\n * Automatically associated with Elastic IP on instance initialization\n* ECS Service\n * TLS/SSL certificate installation before default container startup\n * Certificate EFS mounted on default container as `/etc/letsencrypt`\n* Others\n * VPC with only public subnets (no NAT Gateways to decrease cost)\n * Security groups with minimum inbounds\n * IAM roles with minimum privileges\n\n# Cost\n\nAll resources except Route53 HostedZone should be included in [AWS Free Tier](https://docs.aws.amazon.com/whitepapers/latest/how-aws-pricing-works/get-started-with-the-aws-free-tier.html)\n***if you are in the 12 Months Free period***.\nAfter your 12 Months Free period, setting [`hostInstanceSpotPrice`](https://github.com/rajyan/low-cost-ecs/blob/main/API.md#low-cost-ecs.LowCostECSProps.property.hostInstanceSpotPrice) to use spot instances is recommended.\n\n* EC2\n * t2.micro 750 instance hours (12 Months Free Tier)\n * 30GB EBS volume (12 Months Free Tier)\n* ECS\n * No additional charge because using ECS on EC2\n* EFS\n * Usage is very small, it should be free\n* Cloud Watch\n * Usage is very small, and it should be included in the free tier\n * Enabling [`containerInsights`](https://github.com/rajyan/low-cost-ecs/blob/main/API.md#low-cost-ecs.LowCostECSProps.property.containerInsights) will charge for custom metrics\n\n# Debugging\n\n* SSM Session Manager\n\nSSM manager is pre-installed in the host instance (by ECS-optimized Amazon Linux 2 AMI) and `AmazonSSMManagedInstanceCore` is added to the host instance role to access and debug in your host instance.\n\n```\naws ssm start-session --target $INSTANCE_ID\n```\n\n* ECS Exec\n\nService ECS Exec is enabled, so execute commands can be used to debug your server task container.\n\n```\naws ecs execute-command \\\n--cluster $CLUSTER_ID \\\n--task $TASK_ID \\\n--container nginx \\\n--command bash \\\n--interactive\n```\n\n# Limitations\n\nBecause the ECS service occupies a host port, only one task can be executed at a time.\nThe old task must be terminated before the new task launches, and this causes downtime on release.\n\nAlso, if you make changes that require recreating the service, you may need to manually terminate the task of the old service.\n"
|
|
3047
3047
|
},
|
|
3048
3048
|
"repository": {
|
|
3049
3049
|
"type": "git",
|
|
@@ -3073,7 +3073,7 @@
|
|
|
3073
3073
|
},
|
|
3074
3074
|
"locationInModule": {
|
|
3075
3075
|
"filename": "src/low-cost-ecs.ts",
|
|
3076
|
-
"line":
|
|
3076
|
+
"line": 159
|
|
3077
3077
|
},
|
|
3078
3078
|
"parameters": [
|
|
3079
3079
|
{
|
|
@@ -3105,12 +3105,13 @@
|
|
|
3105
3105
|
"properties": [
|
|
3106
3106
|
{
|
|
3107
3107
|
"docs": {
|
|
3108
|
-
"stability": "experimental"
|
|
3108
|
+
"stability": "experimental",
|
|
3109
|
+
"summary": "EFS file system that the SSL/TLS certificates are installed."
|
|
3109
3110
|
},
|
|
3110
3111
|
"immutable": true,
|
|
3111
3112
|
"locationInModule": {
|
|
3112
3113
|
"filename": "src/low-cost-ecs.ts",
|
|
3113
|
-
"line":
|
|
3114
|
+
"line": 143
|
|
3114
3115
|
},
|
|
3115
3116
|
"name": "certFileSystem",
|
|
3116
3117
|
"type": {
|
|
@@ -3119,12 +3120,13 @@
|
|
|
3119
3120
|
},
|
|
3120
3121
|
{
|
|
3121
3122
|
"docs": {
|
|
3122
|
-
"stability": "experimental"
|
|
3123
|
+
"stability": "experimental",
|
|
3124
|
+
"summary": "ECS cluster created in configured VPC."
|
|
3123
3125
|
},
|
|
3124
3126
|
"immutable": true,
|
|
3125
3127
|
"locationInModule": {
|
|
3126
3128
|
"filename": "src/low-cost-ecs.ts",
|
|
3127
|
-
"line":
|
|
3129
|
+
"line": 135
|
|
3128
3130
|
},
|
|
3129
3131
|
"name": "cluster",
|
|
3130
3132
|
"type": {
|
|
@@ -3133,12 +3135,13 @@
|
|
|
3133
3135
|
},
|
|
3134
3136
|
{
|
|
3135
3137
|
"docs": {
|
|
3136
|
-
"stability": "experimental"
|
|
3138
|
+
"stability": "experimental",
|
|
3139
|
+
"summary": "ECS on EC2 service host instance autoscaling group."
|
|
3137
3140
|
},
|
|
3138
3141
|
"immutable": true,
|
|
3139
3142
|
"locationInModule": {
|
|
3140
3143
|
"filename": "src/low-cost-ecs.ts",
|
|
3141
|
-
"line":
|
|
3144
|
+
"line": 139
|
|
3142
3145
|
},
|
|
3143
3146
|
"name": "hostAutoScalingGroup",
|
|
3144
3147
|
"type": {
|
|
@@ -3147,12 +3150,13 @@
|
|
|
3147
3150
|
},
|
|
3148
3151
|
{
|
|
3149
3152
|
"docs": {
|
|
3150
|
-
"stability": "experimental"
|
|
3153
|
+
"stability": "experimental",
|
|
3154
|
+
"summary": "Server task definition generated from LowCostECSTaskDefinitionOptions."
|
|
3151
3155
|
},
|
|
3152
3156
|
"immutable": true,
|
|
3153
3157
|
"locationInModule": {
|
|
3154
3158
|
"filename": "src/low-cost-ecs.ts",
|
|
3155
|
-
"line":
|
|
3159
|
+
"line": 151
|
|
3156
3160
|
},
|
|
3157
3161
|
"name": "serverTaskDefinition",
|
|
3158
3162
|
"type": {
|
|
@@ -3161,12 +3165,16 @@
|
|
|
3161
3165
|
},
|
|
3162
3166
|
{
|
|
3163
3167
|
"docs": {
|
|
3164
|
-
"
|
|
3168
|
+
"custom": {
|
|
3169
|
+
"link": "https://github.com/rajyan/low-cost-ecs#limitations"
|
|
3170
|
+
},
|
|
3171
|
+
"stability": "experimental",
|
|
3172
|
+
"summary": "ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100."
|
|
3165
3173
|
},
|
|
3166
3174
|
"immutable": true,
|
|
3167
3175
|
"locationInModule": {
|
|
3168
3176
|
"filename": "src/low-cost-ecs.ts",
|
|
3169
|
-
"line":
|
|
3177
|
+
"line": 157
|
|
3170
3178
|
},
|
|
3171
3179
|
"name": "service",
|
|
3172
3180
|
"type": {
|
|
@@ -3175,31 +3183,18 @@
|
|
|
3175
3183
|
},
|
|
3176
3184
|
{
|
|
3177
3185
|
"docs": {
|
|
3178
|
-
"stability": "experimental"
|
|
3186
|
+
"stability": "experimental",
|
|
3187
|
+
"summary": "SNS topic used to notify certbot renewal failure."
|
|
3179
3188
|
},
|
|
3180
3189
|
"immutable": true,
|
|
3181
3190
|
"locationInModule": {
|
|
3182
3191
|
"filename": "src/low-cost-ecs.ts",
|
|
3183
|
-
"line":
|
|
3192
|
+
"line": 147
|
|
3184
3193
|
},
|
|
3185
3194
|
"name": "topic",
|
|
3186
3195
|
"type": {
|
|
3187
3196
|
"fqn": "aws-cdk-lib.aws_sns.Topic"
|
|
3188
3197
|
}
|
|
3189
|
-
},
|
|
3190
|
-
{
|
|
3191
|
-
"docs": {
|
|
3192
|
-
"stability": "experimental"
|
|
3193
|
-
},
|
|
3194
|
-
"immutable": true,
|
|
3195
|
-
"locationInModule": {
|
|
3196
|
-
"filename": "src/low-cost-ecs.ts",
|
|
3197
|
-
"line": 132
|
|
3198
|
-
},
|
|
3199
|
-
"name": "vpc",
|
|
3200
|
-
"type": {
|
|
3201
|
-
"fqn": "aws-cdk-lib.aws_ec2.IVpc"
|
|
3202
|
-
}
|
|
3203
3198
|
}
|
|
3204
3199
|
],
|
|
3205
3200
|
"symbolId": "src/low-cost-ecs:LowCostECS"
|
|
@@ -3440,10 +3435,15 @@
|
|
|
3440
3435
|
"filename": "src/low-cost-ecs.ts",
|
|
3441
3436
|
"line": 54
|
|
3442
3437
|
},
|
|
3443
|
-
"name": "
|
|
3438
|
+
"name": "securityGroups",
|
|
3444
3439
|
"optional": true,
|
|
3445
3440
|
"type": {
|
|
3446
|
-
"
|
|
3441
|
+
"collection": {
|
|
3442
|
+
"elementtype": {
|
|
3443
|
+
"fqn": "aws-cdk-lib.aws_ec2.ISecurityGroup"
|
|
3444
|
+
},
|
|
3445
|
+
"kind": "array"
|
|
3446
|
+
}
|
|
3447
3447
|
}
|
|
3448
3448
|
},
|
|
3449
3449
|
{
|
|
@@ -3470,7 +3470,7 @@
|
|
|
3470
3470
|
"docs": {
|
|
3471
3471
|
"default": "- Creates vpc with only public subnets and no NAT gateways.",
|
|
3472
3472
|
"stability": "experimental",
|
|
3473
|
-
"summary": "
|
|
3473
|
+
"summary": "VPC of the ECS cluster and EFS file system."
|
|
3474
3474
|
},
|
|
3475
3475
|
"immutable": true,
|
|
3476
3476
|
"locationInModule": {
|
|
@@ -3561,6 +3561,6 @@
|
|
|
3561
3561
|
"symbolId": "src/low-cost-ecs:LowCostECSTaskDefinitionOptions"
|
|
3562
3562
|
}
|
|
3563
3563
|
},
|
|
3564
|
-
"version": "0.0.
|
|
3565
|
-
"fingerprint": "
|
|
3564
|
+
"version": "0.0.25",
|
|
3565
|
+
"fingerprint": "0lvw2vYzDRgz185s9gv6MkV+NwlVMGWnzDMk5tiZ9Q0="
|
|
3566
3566
|
}
|
package/.prettierignore
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
# ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
|
package/.prettierrc.json
ADDED
package/.projenrc.ts
CHANGED
|
@@ -10,18 +10,8 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
10
10
|
license: 'MIT',
|
|
11
11
|
cdkVersion: '2.37.0',
|
|
12
12
|
defaultReleaseBranch: 'main',
|
|
13
|
-
keywords: [
|
|
14
|
-
|
|
15
|
-
'ecs',
|
|
16
|
-
'stepfunctions',
|
|
17
|
-
'route53',
|
|
18
|
-
'certbot',
|
|
19
|
-
'loadbalancer',
|
|
20
|
-
],
|
|
21
|
-
devDeps: [
|
|
22
|
-
'aws-cdk',
|
|
23
|
-
'ts-node',
|
|
24
|
-
],
|
|
13
|
+
keywords: ['cdk', 'ecs', 'stepfunctions', 'route53', 'certbot', 'loadbalancer'],
|
|
14
|
+
devDeps: ['aws-cdk', 'ts-node'],
|
|
25
15
|
stability: 'experimental',
|
|
26
16
|
|
|
27
17
|
python: {
|
|
@@ -37,6 +27,13 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
37
27
|
labels: ['auto-approve'],
|
|
38
28
|
},
|
|
39
29
|
},
|
|
30
|
+
prettier: true,
|
|
31
|
+
prettierOptions: {
|
|
32
|
+
settings: {
|
|
33
|
+
printWidth: 100,
|
|
34
|
+
singleQuote: true,
|
|
35
|
+
},
|
|
36
|
+
},
|
|
40
37
|
projenrcTs: true,
|
|
41
38
|
});
|
|
42
39
|
|
|
@@ -52,4 +49,4 @@ const testTask = project.tasks.tryFind('test');
|
|
|
52
49
|
const newTestCommand = testTask!.steps[0]!.exec!.replace(' --updateSnapshot', '');
|
|
53
50
|
testTask!.reset(newTestCommand);
|
|
54
51
|
|
|
55
|
-
project.synth();
|
|
52
|
+
project.synth();
|
package/API.md
CHANGED
|
@@ -85,13 +85,12 @@ Any object.
|
|
|
85
85
|
| **Name** | **Type** | **Description** |
|
|
86
86
|
| --- | --- | --- |
|
|
87
87
|
| <code><a href="#low-cost-ecs.LowCostECS.property.node">node</a></code> | <code>constructs.Node</code> | The tree node. |
|
|
88
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.certFileSystem">certFileSystem</a></code> | <code>aws-cdk-lib.aws_efs.FileSystem</code> |
|
|
89
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.cluster">cluster</a></code> | <code>aws-cdk-lib.aws_ecs.Cluster</code> |
|
|
90
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.hostAutoScalingGroup">hostAutoScalingGroup</a></code> | <code>aws-cdk-lib.aws_autoscaling.AutoScalingGroup</code> |
|
|
91
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinition</code> |
|
|
92
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.service">service</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2Service</code> |
|
|
93
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.topic">topic</a></code> | <code>aws-cdk-lib.aws_sns.Topic</code> |
|
|
94
|
-
| <code><a href="#low-cost-ecs.LowCostECS.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | *No description.* |
|
|
88
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.certFileSystem">certFileSystem</a></code> | <code>aws-cdk-lib.aws_efs.FileSystem</code> | EFS file system that the SSL/TLS certificates are installed. |
|
|
89
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.cluster">cluster</a></code> | <code>aws-cdk-lib.aws_ecs.Cluster</code> | ECS cluster created in configured VPC. |
|
|
90
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.hostAutoScalingGroup">hostAutoScalingGroup</a></code> | <code>aws-cdk-lib.aws_autoscaling.AutoScalingGroup</code> | ECS on EC2 service host instance autoscaling group. |
|
|
91
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinition</code> | Server task definition generated from LowCostECSTaskDefinitionOptions. |
|
|
92
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.service">service</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2Service</code> | ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100. |
|
|
93
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.topic">topic</a></code> | <code>aws-cdk-lib.aws_sns.Topic</code> | SNS topic used to notify certbot renewal failure. |
|
|
95
94
|
|
|
96
95
|
---
|
|
97
96
|
|
|
@@ -115,6 +114,8 @@ public readonly certFileSystem: FileSystem;
|
|
|
115
114
|
|
|
116
115
|
- *Type:* aws-cdk-lib.aws_efs.FileSystem
|
|
117
116
|
|
|
117
|
+
EFS file system that the SSL/TLS certificates are installed.
|
|
118
|
+
|
|
118
119
|
---
|
|
119
120
|
|
|
120
121
|
##### `cluster`<sup>Required</sup> <a name="cluster" id="low-cost-ecs.LowCostECS.property.cluster"></a>
|
|
@@ -125,6 +126,8 @@ public readonly cluster: Cluster;
|
|
|
125
126
|
|
|
126
127
|
- *Type:* aws-cdk-lib.aws_ecs.Cluster
|
|
127
128
|
|
|
129
|
+
ECS cluster created in configured VPC.
|
|
130
|
+
|
|
128
131
|
---
|
|
129
132
|
|
|
130
133
|
##### `hostAutoScalingGroup`<sup>Required</sup> <a name="hostAutoScalingGroup" id="low-cost-ecs.LowCostECS.property.hostAutoScalingGroup"></a>
|
|
@@ -135,6 +138,8 @@ public readonly hostAutoScalingGroup: AutoScalingGroup;
|
|
|
135
138
|
|
|
136
139
|
- *Type:* aws-cdk-lib.aws_autoscaling.AutoScalingGroup
|
|
137
140
|
|
|
141
|
+
ECS on EC2 service host instance autoscaling group.
|
|
142
|
+
|
|
138
143
|
---
|
|
139
144
|
|
|
140
145
|
##### `serverTaskDefinition`<sup>Required</sup> <a name="serverTaskDefinition" id="low-cost-ecs.LowCostECS.property.serverTaskDefinition"></a>
|
|
@@ -145,6 +150,8 @@ public readonly serverTaskDefinition: Ec2TaskDefinition;
|
|
|
145
150
|
|
|
146
151
|
- *Type:* aws-cdk-lib.aws_ecs.Ec2TaskDefinition
|
|
147
152
|
|
|
153
|
+
Server task definition generated from LowCostECSTaskDefinitionOptions.
|
|
154
|
+
|
|
148
155
|
---
|
|
149
156
|
|
|
150
157
|
##### `service`<sup>Required</sup> <a name="service" id="low-cost-ecs.LowCostECS.property.service"></a>
|
|
@@ -155,6 +162,10 @@ public readonly service: Ec2Service;
|
|
|
155
162
|
|
|
156
163
|
- *Type:* aws-cdk-lib.aws_ecs.Ec2Service
|
|
157
164
|
|
|
165
|
+
ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100.
|
|
166
|
+
|
|
167
|
+
> [https://github.com/rajyan/low-cost-ecs#limitations](https://github.com/rajyan/low-cost-ecs#limitations)
|
|
168
|
+
|
|
158
169
|
---
|
|
159
170
|
|
|
160
171
|
##### `topic`<sup>Required</sup> <a name="topic" id="low-cost-ecs.LowCostECS.property.topic"></a>
|
|
@@ -165,15 +176,7 @@ public readonly topic: Topic;
|
|
|
165
176
|
|
|
166
177
|
- *Type:* aws-cdk-lib.aws_sns.Topic
|
|
167
178
|
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
##### `vpc`<sup>Required</sup> <a name="vpc" id="low-cost-ecs.LowCostECS.property.vpc"></a>
|
|
171
|
-
|
|
172
|
-
```typescript
|
|
173
|
-
public readonly vpc: IVpc;
|
|
174
|
-
```
|
|
175
|
-
|
|
176
|
-
- *Type:* aws-cdk-lib.aws_ec2.IVpc
|
|
179
|
+
SNS topic used to notify certbot renewal failure.
|
|
177
180
|
|
|
178
181
|
---
|
|
179
182
|
|
|
@@ -205,9 +208,9 @@ const lowCostECSProps: LowCostECSProps = { ... }
|
|
|
205
208
|
| <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. |
|
|
206
209
|
| <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. |
|
|
207
210
|
| <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). |
|
|
208
|
-
| <code><a href="#low-cost-ecs.LowCostECSProps.property.
|
|
211
|
+
| <code><a href="#low-cost-ecs.LowCostECSProps.property.securityGroups">securityGroups</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup[]</code> | Security group of the ECS host instance. |
|
|
209
212
|
| <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. |
|
|
210
|
-
| <code><a href="#low-cost-ecs.LowCostECSProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> |
|
|
213
|
+
| <code><a href="#low-cost-ecs.LowCostECSProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | VPC of the ECS cluster and EFS file system. |
|
|
211
214
|
|
|
212
215
|
---
|
|
213
216
|
|
|
@@ -362,13 +365,13 @@ Removal policy for the file system and log group (if using default).
|
|
|
362
365
|
|
|
363
366
|
---
|
|
364
367
|
|
|
365
|
-
##### `
|
|
368
|
+
##### `securityGroups`<sup>Optional</sup> <a name="securityGroups" id="low-cost-ecs.LowCostECSProps.property.securityGroups"></a>
|
|
366
369
|
|
|
367
370
|
```typescript
|
|
368
|
-
public readonly
|
|
371
|
+
public readonly securityGroups: ISecurityGroup[];
|
|
369
372
|
```
|
|
370
373
|
|
|
371
|
-
- *Type:* aws-cdk-lib.aws_ec2.ISecurityGroup
|
|
374
|
+
- *Type:* aws-cdk-lib.aws_ec2.ISecurityGroup[]
|
|
372
375
|
- *Default:* Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).
|
|
373
376
|
|
|
374
377
|
Security group of the ECS host instance.
|
|
@@ -399,7 +402,7 @@ public readonly vpc: IVpc;
|
|
|
399
402
|
- *Type:* aws-cdk-lib.aws_ec2.IVpc
|
|
400
403
|
- *Default:* Creates vpc with only public subnets and no NAT gateways.
|
|
401
404
|
|
|
402
|
-
|
|
405
|
+
VPC of the ECS cluster and EFS file system.
|
|
403
406
|
|
|
404
407
|
---
|
|
405
408
|
|
package/README.md
CHANGED
|
@@ -26,7 +26,7 @@ Edit settings in `examples/minimum.ts` and deploy the cdk construct. [Public hos
|
|
|
26
26
|
git clone https://github.com/rajyan/low-cost-ecs.git
|
|
27
27
|
yarn install
|
|
28
28
|
# edit settings in bin/low-cost-ecs.ts
|
|
29
|
-
|
|
29
|
+
cdk deploy
|
|
30
30
|
```
|
|
31
31
|
|
|
32
32
|
Access the configured `recordDomainNames` and see that the Nginx sample server has been deployed.
|
package/examples/all-props.ts
CHANGED
|
@@ -21,25 +21,35 @@ export const allProps = new LowCostECS(stack, 'LowCostECS', {
|
|
|
21
21
|
containerInsights: true,
|
|
22
22
|
hostInstanceSpotPrice: '0.010',
|
|
23
23
|
hostInstanceType: 't3.micro',
|
|
24
|
-
logGroup: LogGroup.fromLogGroupArn(
|
|
24
|
+
logGroup: LogGroup.fromLogGroupArn(
|
|
25
|
+
stack,
|
|
26
|
+
'LogGroup',
|
|
27
|
+
'arn:aws:logs:region:account-id:log-group:test'
|
|
28
|
+
),
|
|
25
29
|
recordDomainNames: ['test1.rajyan.net', 'test2.rajyan.net'],
|
|
26
30
|
removalPolicy: RemovalPolicy.RETAIN,
|
|
27
|
-
|
|
31
|
+
securityGroups: [SecurityGroup.fromSecurityGroupId(stack, 'SecurityGroup', 'test-sg-id')],
|
|
28
32
|
serverTaskDefinition: {
|
|
29
|
-
containers: [
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
33
|
+
containers: [
|
|
34
|
+
{
|
|
35
|
+
containerName: 'test-container',
|
|
36
|
+
image: ContainerImage.fromRegistry('test-image'),
|
|
37
|
+
memoryLimitMiB: 32,
|
|
38
|
+
essential: true,
|
|
39
|
+
portMappings: [
|
|
40
|
+
{
|
|
41
|
+
containerPort: 80,
|
|
42
|
+
hostPort: 80,
|
|
43
|
+
protocol: Protocol.TCP,
|
|
44
|
+
},
|
|
45
|
+
],
|
|
46
|
+
},
|
|
47
|
+
],
|
|
48
|
+
volumes: [
|
|
49
|
+
{
|
|
50
|
+
name: 'test-volume',
|
|
51
|
+
},
|
|
52
|
+
],
|
|
43
53
|
},
|
|
44
54
|
vpc: new Vpc(stack, 'Vpc'),
|
|
45
55
|
});
|
package/lib/low-cost-ecs.d.ts
CHANGED
|
@@ -29,7 +29,7 @@ export interface LowCostECSProps {
|
|
|
29
29
|
*/
|
|
30
30
|
readonly recordDomainNames?: string[];
|
|
31
31
|
/**
|
|
32
|
-
*
|
|
32
|
+
* VPC of the ECS cluster and EFS file system.
|
|
33
33
|
*
|
|
34
34
|
* @default - Creates vpc with only public subnets and no NAT gateways.
|
|
35
35
|
*/
|
|
@@ -39,7 +39,7 @@ export interface LowCostECSProps {
|
|
|
39
39
|
*
|
|
40
40
|
* @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).
|
|
41
41
|
*/
|
|
42
|
-
readonly
|
|
42
|
+
readonly securityGroups?: ec2.ISecurityGroup[];
|
|
43
43
|
/**
|
|
44
44
|
* Instance type of the ECS host instance.
|
|
45
45
|
*
|
|
@@ -81,7 +81,7 @@ export interface LowCostECSProps {
|
|
|
81
81
|
*/
|
|
82
82
|
readonly awsCliDockerTag?: string;
|
|
83
83
|
/**
|
|
84
|
-
* Enable container insights or not
|
|
84
|
+
* Enable container insights or not.
|
|
85
85
|
*
|
|
86
86
|
* @default - undefined (container insights disabled)
|
|
87
87
|
*/
|
|
@@ -106,12 +106,31 @@ export interface LowCostECSTaskDefinitionOptions {
|
|
|
106
106
|
readonly volumes?: ecs.Volume[];
|
|
107
107
|
}
|
|
108
108
|
export declare class LowCostECS extends Construct {
|
|
109
|
-
|
|
109
|
+
/**
|
|
110
|
+
* ECS cluster created in configured VPC.
|
|
111
|
+
*/
|
|
112
|
+
readonly cluster: ecs.Cluster;
|
|
113
|
+
/**
|
|
114
|
+
* ECS on EC2 service host instance autoscaling group.
|
|
115
|
+
*/
|
|
110
116
|
readonly hostAutoScalingGroup: AutoScalingGroup;
|
|
117
|
+
/**
|
|
118
|
+
* EFS file system that the SSL/TLS certificates are installed.
|
|
119
|
+
*/
|
|
111
120
|
readonly certFileSystem: FileSystem;
|
|
121
|
+
/**
|
|
122
|
+
* SNS topic used to notify certbot renewal failure.
|
|
123
|
+
*/
|
|
112
124
|
readonly topic: Topic;
|
|
113
|
-
|
|
125
|
+
/**
|
|
126
|
+
* Server task definition generated from LowCostECSTaskDefinitionOptions.
|
|
127
|
+
*/
|
|
114
128
|
readonly serverTaskDefinition: ecs.Ec2TaskDefinition;
|
|
129
|
+
/**
|
|
130
|
+
* ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100.
|
|
131
|
+
*
|
|
132
|
+
* @link https://github.com/rajyan/low-cost-ecs#limitations
|
|
133
|
+
*/
|
|
115
134
|
readonly service: ecs.Ec2Service;
|
|
116
135
|
constructor(scope: Construct, id: string, props: LowCostECSProps);
|
|
117
136
|
private createTaskDefinition;
|
package/lib/low-cost-ecs.js
CHANGED
|
@@ -20,19 +20,18 @@ const constructs_1 = require("constructs");
|
|
|
20
20
|
class LowCostECS extends constructs_1.Construct {
|
|
21
21
|
constructor(scope, id, props) {
|
|
22
22
|
super(scope, id);
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
});
|
|
23
|
+
const vpc = props.vpc ??
|
|
24
|
+
new ec2.Vpc(scope, 'Vpc', {
|
|
25
|
+
natGateways: 0,
|
|
26
|
+
subnetConfiguration: [
|
|
27
|
+
{
|
|
28
|
+
name: 'PublicSubnet',
|
|
29
|
+
subnetType: ec2.SubnetType.PUBLIC,
|
|
30
|
+
},
|
|
31
|
+
],
|
|
32
|
+
});
|
|
34
33
|
this.cluster = new ecs.Cluster(scope, 'Cluster', {
|
|
35
|
-
vpc:
|
|
34
|
+
vpc: vpc,
|
|
36
35
|
containerInsights: props.containerInsights,
|
|
37
36
|
});
|
|
38
37
|
this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {
|
|
@@ -46,9 +45,9 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
46
45
|
minCapacity: 1,
|
|
47
46
|
maxCapacity: 1,
|
|
48
47
|
});
|
|
49
|
-
if (props.
|
|
48
|
+
if (props.securityGroups) {
|
|
50
49
|
this.hostAutoScalingGroup.node.tryRemoveChild('InstanceSecurityGroup');
|
|
51
|
-
this.hostAutoScalingGroup.addSecurityGroup(
|
|
50
|
+
props.securityGroups.forEach((sg) => this.hostAutoScalingGroup.addSecurityGroup(sg));
|
|
52
51
|
}
|
|
53
52
|
else {
|
|
54
53
|
this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));
|
|
@@ -74,10 +73,10 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
74
73
|
const awsCliTag = props.awsCliDockerTag ?? 'latest';
|
|
75
74
|
this.hostAutoScalingGroup.addUserData('INSTANCE_ID=$(curl --silent http://169.254.169.254/latest/meta-data/instance-id)', `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)`, `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`);
|
|
76
75
|
this.certFileSystem = new aws_efs_1.FileSystem(this, 'FileSystem', {
|
|
77
|
-
vpc
|
|
76
|
+
vpc,
|
|
78
77
|
encrypted: true,
|
|
79
78
|
securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {
|
|
80
|
-
vpc:
|
|
79
|
+
vpc: vpc,
|
|
81
80
|
allowAllOutbound: false,
|
|
82
81
|
}),
|
|
83
82
|
removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,
|
|
@@ -244,7 +243,7 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
244
243
|
enableExecuteCommand: true,
|
|
245
244
|
});
|
|
246
245
|
new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });
|
|
247
|
-
new lib.CfnOutput(this, '
|
|
246
|
+
new lib.CfnOutput(this, 'StateMachineName', { value: certbotStateMachine.stateMachineName });
|
|
248
247
|
new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });
|
|
249
248
|
new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });
|
|
250
249
|
}
|
|
@@ -258,7 +257,8 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
258
257
|
}
|
|
259
258
|
sampleTaskDefinition(records, logGroup) {
|
|
260
259
|
return {
|
|
261
|
-
containers: [
|
|
260
|
+
containers: [
|
|
261
|
+
{
|
|
262
262
|
image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),
|
|
263
263
|
containerName: 'nginx',
|
|
264
264
|
memoryReservationMiB: 64,
|
|
@@ -271,7 +271,8 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
271
271
|
logGroup: logGroup,
|
|
272
272
|
streamPrefix: 'sample',
|
|
273
273
|
}),
|
|
274
|
-
portMappings: [
|
|
274
|
+
portMappings: [
|
|
275
|
+
{
|
|
275
276
|
hostPort: 80,
|
|
276
277
|
containerPort: 80,
|
|
277
278
|
protocol: ecs.Protocol.TCP,
|
|
@@ -280,12 +281,14 @@ class LowCostECS extends constructs_1.Construct {
|
|
|
280
281
|
hostPort: 443,
|
|
281
282
|
containerPort: 443,
|
|
282
283
|
protocol: ecs.Protocol.TCP,
|
|
283
|
-
}
|
|
284
|
-
|
|
284
|
+
},
|
|
285
|
+
],
|
|
286
|
+
},
|
|
287
|
+
],
|
|
285
288
|
};
|
|
286
289
|
}
|
|
287
290
|
}
|
|
288
291
|
exports.LowCostECS = LowCostECS;
|
|
289
292
|
_a = JSII_RTTI_SYMBOL_1;
|
|
290
|
-
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.
|
|
291
|
-
//# 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 {\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 sampleTaskDefinition()\n   * @see sampleTaskDefinition\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 Construct {\n  readonly vpc: ec2.IVpc;\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  readonly certFileSystem: FileSystem;\n  readonly topic: Topic;\n  readonly cluster: ecs.Cluster;\n  readonly serverTaskDefinition: ecs.Ec2TaskDefinition;\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id);\n\n    this.vpc =\n      props.vpc ??\n      new ec2.Vpc(scope, '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(scope, '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.node.tryRemoveChild('InstanceSecurityGroup');\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    this.topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: this.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: this.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    this.serverTaskDefinition = this.createTaskDefinition(props.serverTaskDefinition ?? this.sampleTaskDefinition(records, logGroup));\n\n    if (!this.serverTaskDefinition.defaultContainer) {\n      throw new Error('defaultContainer is required for serverTaskDefinition. Add at least one essential container.');\n    }\n\n    this.certFileSystem.grant(\n      this.serverTaskDefinition.taskRole,\n      'elasticfilesystem:ClientMount',\n    );\n    this.serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    this.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    this.serverTaskDefinition.defaultContainer.addContainerDependencies({\n      container: this.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      this.serverTaskDefinition.taskRole,\n      'states:DescribeExecution',\n    );\n    certbotStateMachine.grantStartExecution(this.serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: this.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((containerDefinition, index) => {\n      serverTaskDefinition.addContainer(containerDefinition.containerName ?? `container${index}`, containerDefinition);\n    });\n    taskDefinitionOptions.volumes?.forEach((volume) => serverTaskDefinition.addVolume(volume));\n    return serverTaskDefinition;\n  }\n\n  private sampleTaskDefinition(\n    records: string[],\n    logGroup: ILogGroup,\n  ): LowCostECSTaskDefinitionOptions {\n    return {\n      containers: [{\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: 'sample',\n        }),\n        portMappings: [{\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  }\n}\n"]}
|
|
293
|
+
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.25" };
|
|
294
|
+
//# 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;AACjE,2CAAuC;AAoHvC,MAAa,UAAW,SAAQ,sBAAS;IA4BvC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,GAAG,GACP,KAAK,CAAC,GAAG;YACT,IAAI,GAAG,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE;gBACxB,WAAW,EAAE,CAAC;gBACd,mBAAmB,EAAE;oBACnB;wBACE,IAAI,EAAE,cAAc;wBACpB,UAAU,EAAE,GAAG,CAAC,UAAU,CAAC,MAAM;qBAClC;iBACF;aACF,CAAC,CAAC;QAEL,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,SAAS,EAAE;YAC/C,GAAG,EAAE,GAAG;YACR,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,CAAC,GAAG,CAAC,eAAe,CAAC,QAAQ,EAAE;gBAC7E,eAAe,EAAE,IAAI;aACtB,CAAC;YACF,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,cAAc,EAAE;YACxB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,cAAc,CAAC,uBAAuB,CAAC,CAAC;YACvE,KAAK,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC,CAAC;SACtF;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,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;YACtF,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;SACxF;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;YACH,SAAS,EAAE,IAAI;YACf,aAAa,EAAE,IAAI,GAAG,CAAC,aAAa,CAAC,IAAI,EAAE,yBAAyB,EAAE;gBACpE,GAAG,EAAE,GAAG;gBACR,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,CAAC,IAAI,EAAE,uBAAuB,CAAC,CAAC;QACvF,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,CAAC,kBAAkB,EAAE;YAC9E,KAAK,EAAE,GAAG,CAAC,cAAc,CAAC,YAAY,CAAC,uBAAuB,UAAU,EAAE,CAAC;YAC3E,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,CAAC,CAAC;QAEH,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,qBAAqB,CAAC,QAAQ,EAAE,+BAA+B,CAAC,CAAC;QAC3F,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,IAAI,CAAC,KAAK,GAAG,IAAI,eAAK,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QACtC,IAAI,sBAAY,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC1C,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,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,IAAI,CAAC,KAAK;YACjB,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,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC;YAC/E,OAAO,EAAE,CAAC,IAAI,oCAAe,CAAC,mBAAmB,CAAC,CAAC;SACpD,CAAC,CAAC;QAEH;;WAEG;QACH,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,oBAAoB,CACnD,KAAK,CAAC,oBAAoB,IAAI,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,QAAQ,CAAC,CAC3E,CAAC;QAEF,IAAI,CAAC,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE;YAC/C,MAAM,IAAI,KAAK,CACb,8FAA8F,CAC/F,CAAC;SACH;QAED,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAAE,+BAA+B,CAAC,CAAC;QAC/F,IAAI,CAAC,oBAAoB,CAAC,SAAS,CAAC;YAClC,IAAI,EAAE,YAAY;YAClB,sBAAsB,EAAE;gBACtB,YAAY,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY;aAC/C;SACF,CAAC,CAAC;QACH,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,cAAc,CAAC;YACxD,YAAY,EAAE,YAAY;YAC1B,aAAa,EAAE,kBAAkB;YACjC,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;QAEH;;WAEG;QACH,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,wBAAwB,CAAC;YAClE,SAAS,EAAE,IAAI,CAAC,oBAAoB,CAAC,YAAY,CAAC,iBAAiB,EAAE;gBACnE,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,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAClC,0BAA0B,CAC3B,CAAC;QACF,mBAAmB,CAAC,mBAAmB,CAAC,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;QAE5E,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,EAAE;YACjD,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,cAAc,EAAE,IAAI,CAAC,oBAAoB;YACzC,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,kBAAkB,EAAE,EAAE,KAAK,EAAE,mBAAmB,CAAC,gBAAgB,EAAE,CAAC,CAAC;QAC7F,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,CAC1B,qBAAsD;QAEtD,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAC,iBAAiB,CACpD,IAAI,EACJ,sBAAsB,EACtB,qBAAqB,CAAC,cAAc,CACrC,CAAC;QACF,qBAAqB,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,mBAAmB,EAAE,KAAK,EAAE,EAAE;YACtE,oBAAoB,CAAC,YAAY,CAC/B,mBAAmB,CAAC,aAAa,IAAI,YAAY,KAAK,EAAE,EACxD,mBAAmB,CACpB,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,qBAAqB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,oBAAoB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;QAC3F,OAAO,oBAAoB,CAAC;IAC9B,CAAC;IAEO,oBAAoB,CAC1B,OAAiB,EACjB,QAAmB;QAEnB,OAAO;YACL,UAAU,EAAE;gBACV;oBACE,KAAK,EAAE,GAAG,CAAC,cAAc,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,8BAA8B,CAAC,CAAC;oBACzF,aAAa,EAAE,OAAO;oBACtB,oBAAoB,EAAE,EAAE;oBACxB,SAAS,EAAE,IAAI;oBACf,WAAW,EAAE;wBACX,WAAW,EAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC;wBAC9B,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC;qBACtB;oBACD,OAAO,EAAE,GAAG,CAAC,UAAU,CAAC,OAAO,CAAC;wBAC9B,QAAQ,EAAE,QAAQ;wBAClB,YAAY,EAAE,QAAQ;qBACvB,CAAC;oBACF,YAAY,EAAE;wBACZ;4BACE,QAAQ,EAAE,EAAE;4BACZ,aAAa,EAAE,EAAE;4BACjB,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,GAAG;yBAC3B;wBACD;4BACE,QAAQ,EAAE,GAAG;4BACb,aAAa,EAAE,GAAG;4BAClB,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,GAAG;yBAC3B;qBACF;iBACF;aACF;SACF,CAAC;IACJ,CAAC;;AArWH,gCAsWC","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 {\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 cluster and EFS file system.\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 securityGroups?: 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 sampleTaskDefinition()\n   * @see sampleTaskDefinition\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 Construct {\n  /**\n   * ECS cluster created in configured VPC.\n   */\n  readonly cluster: ecs.Cluster;\n  /**\n   * ECS on EC2 service host instance autoscaling group.\n   */\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  /**\n   * EFS file system that the SSL/TLS certificates are installed.\n   */\n  readonly certFileSystem: FileSystem;\n  /**\n   * SNS topic used to notify certbot renewal failure.\n   */\n  readonly topic: Topic;\n  /**\n   * Server task definition generated from LowCostECSTaskDefinitionOptions.\n   */\n  readonly serverTaskDefinition: ecs.Ec2TaskDefinition;\n  /**\n   * ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100.\n   *\n   * @link https://github.com/rajyan/low-cost-ecs#limitations\n   */\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id);\n\n    const vpc =\n      props.vpc ??\n      new ec2.Vpc(scope, '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(scope, 'Cluster', {\n      vpc: vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {\n      machineImage: ecs.EcsOptimizedImage.amazonLinux2(ecs.AmiHardwareType.STANDARD, {\n        cachedInContext: true,\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.securityGroups) {\n      this.hostAutoScalingGroup.node.tryRemoveChild('InstanceSecurityGroup');\n      props.securityGroups.forEach((sg) => this.hostAutoScalingGroup.addSecurityGroup(sg));\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(ec2.Peer.anyIpv6(), ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(443));\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,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc: 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(this, 'CertbotTaskDefinition');\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('CertbotContainer', {\n      image: ecs.ContainerImage.fromRegistry(`certbot/dns-route53:${certbotTag}`),\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    this.certFileSystem.grant(certbotTaskDefinition.taskRole, 'elasticfilesystem:ClientWrite');\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    this.topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: this.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: this.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(lib.Duration.days(props.certbotScheduleInterval ?? 60)),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    this.serverTaskDefinition = this.createTaskDefinition(\n      props.serverTaskDefinition ?? this.sampleTaskDefinition(records, logGroup)\n    );\n\n    if (!this.serverTaskDefinition.defaultContainer) {\n      throw new Error(\n        'defaultContainer is required for serverTaskDefinition. Add at least one essential container.'\n      );\n    }\n\n    this.certFileSystem.grant(this.serverTaskDefinition.taskRole, 'elasticfilesystem:ClientMount');\n    this.serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    this.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    this.serverTaskDefinition.defaultContainer.addContainerDependencies({\n      container: this.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      this.serverTaskDefinition.taskRole,\n      'states:DescribeExecution'\n    );\n    certbotStateMachine.grantStartExecution(this.serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: this.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, 'StateMachineName', { 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(\n    taskDefinitionOptions: LowCostECSTaskDefinitionOptions\n  ): ecs.Ec2TaskDefinition {\n    const serverTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'ServerTaskDefinition',\n      taskDefinitionOptions.taskDefinition\n    );\n    taskDefinitionOptions.containers.forEach((containerDefinition, index) => {\n      serverTaskDefinition.addContainer(\n        containerDefinition.containerName ?? `container${index}`,\n        containerDefinition\n      );\n    });\n    taskDefinitionOptions.volumes?.forEach((volume) => serverTaskDefinition.addVolume(volume));\n    return serverTaskDefinition;\n  }\n\n  private sampleTaskDefinition(\n    records: string[],\n    logGroup: ILogGroup\n  ): LowCostECSTaskDefinitionOptions {\n    return {\n      containers: [\n        {\n          image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),\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: 'sample',\n          }),\n          portMappings: [\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      ],\n    };\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -45,9 +45,11 @@
|
|
|
45
45
|
"aws-cdk-lib": "2.37.0",
|
|
46
46
|
"constructs": "10.0.5",
|
|
47
47
|
"eslint": "^8",
|
|
48
|
+
"eslint-config-prettier": "^8.5.0",
|
|
48
49
|
"eslint-import-resolver-node": "^0.3.6",
|
|
49
50
|
"eslint-import-resolver-typescript": "^3.5.1",
|
|
50
51
|
"eslint-plugin-import": "^2.26.0",
|
|
52
|
+
"eslint-plugin-prettier": "^4.2.1",
|
|
51
53
|
"jest": "^27",
|
|
52
54
|
"jest-junit": "^13",
|
|
53
55
|
"jsii": "^1.67.0",
|
|
@@ -56,6 +58,7 @@
|
|
|
56
58
|
"jsii-pacmak": "^1.67.0",
|
|
57
59
|
"json-schema": "^0.4.0",
|
|
58
60
|
"npm-check-updates": "^15",
|
|
61
|
+
"prettier": "^2.7.1",
|
|
59
62
|
"projen": "^0.62.12",
|
|
60
63
|
"standard-version": "^9",
|
|
61
64
|
"ts-jest": "^27",
|
|
@@ -76,7 +79,7 @@
|
|
|
76
79
|
],
|
|
77
80
|
"main": "lib/index.js",
|
|
78
81
|
"license": "MIT",
|
|
79
|
-
"version": "0.0.
|
|
82
|
+
"version": "0.0.25",
|
|
80
83
|
"jest": {
|
|
81
84
|
"testMatch": [
|
|
82
85
|
"<rootDir>/src/**/__tests__/**/*.ts?(x)",
|