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 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": "[![NPM version](https://img.shields.io/npm/v/low-cost-ecs?color=brightgreen)](https://www.npmjs.com/package/low-cost-ecs)\n[![PyPI version](https://img.shields.io/pypi/v/low-cost-ecs?color=brightgreen)](https://pypi.org/project/low-cost-ecs)\n[![Release](https://github.com/rajyan/low-cost-ecs/workflows/release/badge.svg)](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\n./node_modules/.bin/cdk 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"
3046
+ "markdown": "[![NPM version](https://img.shields.io/npm/v/low-cost-ecs?color=brightgreen)](https://www.npmjs.com/package/low-cost-ecs)\n[![PyPI version](https://img.shields.io/pypi/v/low-cost-ecs?color=brightgreen)](https://pypi.org/project/low-cost-ecs)\n[![Release](https://github.com/rajyan/low-cost-ecs/workflows/release/badge.svg)](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": 140
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": 134
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": 136
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": 133
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": 137
3159
+ "line": 151
3156
3160
  },
3157
3161
  "name": "serverTaskDefinition",
3158
3162
  "type": {
@@ -3161,12 +3165,16 @@
3161
3165
  },
3162
3166
  {
3163
3167
  "docs": {
3164
- "stability": "experimental"
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": 138
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": 135
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": "securityGroup",
3438
+ "name": "securityGroups",
3444
3439
  "optional": true,
3445
3440
  "type": {
3446
- "fqn": "aws-cdk-lib.aws_ec2.ISecurityGroup"
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": "Vpc of the ECS host instance and cluster."
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.23",
3565
- "fingerprint": "df+yS28a5ob3s6dID5EWnI7AXRvfVQUoyzr+dM99v7w="
3564
+ "version": "0.0.25",
3565
+ "fingerprint": "0lvw2vYzDRgz185s9gv6MkV+NwlVMGWnzDMk5tiZ9Q0="
3566
3566
  }
@@ -0,0 +1 @@
1
+ # ~~ Generated by projen. To modify, edit .projenrc.js and run "npx projen".
@@ -0,0 +1,5 @@
1
+ {
2
+ "printWidth": 100,
3
+ "singleQuote": true,
4
+ "overrides": []
5
+ }
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
- 'cdk',
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> | *No description.* |
89
- | <code><a href="#low-cost-ecs.LowCostECS.property.cluster">cluster</a></code> | <code>aws-cdk-lib.aws_ecs.Cluster</code> | *No description.* |
90
- | <code><a href="#low-cost-ecs.LowCostECS.property.hostAutoScalingGroup">hostAutoScalingGroup</a></code> | <code>aws-cdk-lib.aws_autoscaling.AutoScalingGroup</code> | *No description.* |
91
- | <code><a href="#low-cost-ecs.LowCostECS.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinition</code> | *No description.* |
92
- | <code><a href="#low-cost-ecs.LowCostECS.property.service">service</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2Service</code> | *No description.* |
93
- | <code><a href="#low-cost-ecs.LowCostECS.property.topic">topic</a></code> | <code>aws-cdk-lib.aws_sns.Topic</code> | *No description.* |
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.securityGroup">securityGroup</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup</code> | Security group of the ECS host instance. |
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> | Vpc of the ECS host instance and cluster. |
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
- ##### `securityGroup`<sup>Optional</sup> <a name="securityGroup" id="low-cost-ecs.LowCostECSProps.property.securityGroup"></a>
368
+ ##### `securityGroups`<sup>Optional</sup> <a name="securityGroups" id="low-cost-ecs.LowCostECSProps.property.securityGroups"></a>
366
369
 
367
370
  ```typescript
368
- public readonly securityGroup: ISecurityGroup;
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
- Vpc of the ECS host instance and cluster.
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
- ./node_modules/.bin/cdk deploy
29
+ cdk deploy
30
30
  ```
31
31
 
32
32
  Access the configured `recordDomainNames` and see that the Nginx sample server has been deployed.
@@ -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(stack, 'LogGroup', 'arn:aws:logs:region:account-id:log-group:test'),
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
- securityGroup: SecurityGroup.fromSecurityGroupId(stack, 'SecurityGroup', 'test-sg-id'),
31
+ securityGroups: [SecurityGroup.fromSecurityGroupId(stack, 'SecurityGroup', 'test-sg-id')],
28
32
  serverTaskDefinition: {
29
- containers: [{
30
- containerName: 'test-container',
31
- image: ContainerImage.fromRegistry('test-image'),
32
- memoryLimitMiB: 32,
33
- essential: true,
34
- portMappings: [{
35
- containerPort: 80,
36
- hostPort: 80,
37
- protocol: Protocol.TCP,
38
- }],
39
- }],
40
- volumes: [{
41
- name: 'test-volume',
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
  });
@@ -29,7 +29,7 @@ export interface LowCostECSProps {
29
29
  */
30
30
  readonly recordDomainNames?: string[];
31
31
  /**
32
- * Vpc of the ECS host instance and cluster.
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 securityGroup?: ec2.ISecurityGroup;
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
- readonly vpc: ec2.IVpc;
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
- readonly cluster: ecs.Cluster;
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;
@@ -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
- this.vpc =
24
- props.vpc ??
25
- new ec2.Vpc(scope, 'Vpc', {
26
- natGateways: 0,
27
- subnetConfiguration: [
28
- {
29
- name: 'PublicSubnet',
30
- subnetType: ec2.SubnetType.PUBLIC,
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: this.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.securityGroup) {
48
+ if (props.securityGroups) {
50
49
  this.hostAutoScalingGroup.node.tryRemoveChild('InstanceSecurityGroup');
51
- this.hostAutoScalingGroup.addSecurityGroup(props.securityGroup);
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: this.vpc,
76
+ vpc,
78
77
  encrypted: true,
79
78
  securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {
80
- vpc: this.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, 'CertbotStateMachineName', { value: certbotStateMachine.stateMachineName });
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.23" };
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":"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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 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.23",
82
+ "version": "0.0.25",
80
83
  "jest": {
81
84
  "testMatch": [
82
85
  "<rootDir>/src/**/__tests__/**/*.ts?(x)",