low-cost-ecs 0.0.22 → 0.0.24

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": 139
3076
+ "line": 140
3077
3077
  },
3078
3078
  "parameters": [
3079
3079
  {
@@ -3099,7 +3099,7 @@
3099
3099
  "kind": "class",
3100
3100
  "locationInModule": {
3101
3101
  "filename": "src/low-cost-ecs.ts",
3102
- "line": 130
3102
+ "line": 131
3103
3103
  },
3104
3104
  "name": "LowCostECS",
3105
3105
  "properties": [
@@ -3110,7 +3110,7 @@
3110
3110
  "immutable": true,
3111
3111
  "locationInModule": {
3112
3112
  "filename": "src/low-cost-ecs.ts",
3113
- "line": 133
3113
+ "line": 134
3114
3114
  },
3115
3115
  "name": "certFileSystem",
3116
3116
  "type": {
@@ -3124,7 +3124,7 @@
3124
3124
  "immutable": true,
3125
3125
  "locationInModule": {
3126
3126
  "filename": "src/low-cost-ecs.ts",
3127
- "line": 135
3127
+ "line": 136
3128
3128
  },
3129
3129
  "name": "cluster",
3130
3130
  "type": {
@@ -3138,7 +3138,7 @@
3138
3138
  "immutable": true,
3139
3139
  "locationInModule": {
3140
3140
  "filename": "src/low-cost-ecs.ts",
3141
- "line": 132
3141
+ "line": 133
3142
3142
  },
3143
3143
  "name": "hostAutoScalingGroup",
3144
3144
  "type": {
@@ -3152,7 +3152,7 @@
3152
3152
  "immutable": true,
3153
3153
  "locationInModule": {
3154
3154
  "filename": "src/low-cost-ecs.ts",
3155
- "line": 136
3155
+ "line": 137
3156
3156
  },
3157
3157
  "name": "serverTaskDefinition",
3158
3158
  "type": {
@@ -3166,7 +3166,7 @@
3166
3166
  "immutable": true,
3167
3167
  "locationInModule": {
3168
3168
  "filename": "src/low-cost-ecs.ts",
3169
- "line": 137
3169
+ "line": 138
3170
3170
  },
3171
3171
  "name": "service",
3172
3172
  "type": {
@@ -3180,7 +3180,7 @@
3180
3180
  "immutable": true,
3181
3181
  "locationInModule": {
3182
3182
  "filename": "src/low-cost-ecs.ts",
3183
- "line": 134
3183
+ "line": 135
3184
3184
  },
3185
3185
  "name": "topic",
3186
3186
  "type": {
@@ -3194,7 +3194,7 @@
3194
3194
  "immutable": true,
3195
3195
  "locationInModule": {
3196
3196
  "filename": "src/low-cost-ecs.ts",
3197
- "line": 131
3197
+ "line": 132
3198
3198
  },
3199
3199
  "name": "vpc",
3200
3200
  "type": {
@@ -3449,14 +3449,15 @@
3449
3449
  {
3450
3450
  "abstract": true,
3451
3451
  "docs": {
3452
- "default": "- Nginx server task definition defined in createSampleTaskDefinition()",
3452
+ "default": "- Nginx server task definition defined in sampleTaskDefinition()",
3453
+ "see": "sampleTaskDefinition",
3453
3454
  "stability": "experimental",
3454
3455
  "summary": "Task definition for the server ecs task."
3455
3456
  },
3456
3457
  "immutable": true,
3457
3458
  "locationInModule": {
3458
3459
  "filename": "src/low-cost-ecs.ts",
3459
- "line": 121
3460
+ "line": 122
3460
3461
  },
3461
3462
  "name": "serverTaskDefinition",
3462
3463
  "optional": true,
@@ -3495,7 +3496,7 @@
3495
3496
  "kind": "interface",
3496
3497
  "locationInModule": {
3497
3498
  "filename": "src/low-cost-ecs.ts",
3498
- "line": 124
3499
+ "line": 125
3499
3500
  },
3500
3501
  "name": "LowCostECSTaskDefinitionOptions",
3501
3502
  "properties": [
@@ -3507,7 +3508,7 @@
3507
3508
  "immutable": true,
3508
3509
  "locationInModule": {
3509
3510
  "filename": "src/low-cost-ecs.ts",
3510
- "line": 126
3511
+ "line": 127
3511
3512
  },
3512
3513
  "name": "containers",
3513
3514
  "type": {
@@ -3527,7 +3528,7 @@
3527
3528
  "immutable": true,
3528
3529
  "locationInModule": {
3529
3530
  "filename": "src/low-cost-ecs.ts",
3530
- "line": 125
3531
+ "line": 126
3531
3532
  },
3532
3533
  "name": "taskDefinition",
3533
3534
  "optional": true,
@@ -3543,7 +3544,7 @@
3543
3544
  "immutable": true,
3544
3545
  "locationInModule": {
3545
3546
  "filename": "src/low-cost-ecs.ts",
3546
- "line": 127
3547
+ "line": 128
3547
3548
  },
3548
3549
  "name": "volumes",
3549
3550
  "optional": true,
@@ -3560,6 +3561,6 @@
3560
3561
  "symbolId": "src/low-cost-ecs:LowCostECSTaskDefinitionOptions"
3561
3562
  }
3562
3563
  },
3563
- "version": "0.0.22",
3564
- "fingerprint": "SEDtANmtj6kiupr0Ty6oKWlkmTg84dhrcXWKnOFF+oM="
3564
+ "version": "0.0.24",
3565
+ "fingerprint": "mY2OnK2X4isP1cFA/MBGNyiJUlg1lvtZRciahXTbM/4="
3565
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
@@ -382,10 +382,12 @@ public readonly serverTaskDefinition: LowCostECSTaskDefinitionOptions;
382
382
  ```
383
383
 
384
384
  - *Type:* <a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions">LowCostECSTaskDefinitionOptions</a>
385
- - *Default:* Nginx server task definition defined in createSampleTaskDefinition()
385
+ - *Default:* Nginx server task definition defined in sampleTaskDefinition()
386
386
 
387
387
  Task definition for the server ecs task.
388
388
 
389
+ > [sampleTaskDefinition](sampleTaskDefinition)
390
+
389
391
  ---
390
392
 
391
393
  ##### `vpc`<sup>Optional</sup> <a name="vpc" id="low-cost-ecs.LowCostECSProps.property.vpc"></a>
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
31
  securityGroup: 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
  });
@@ -95,7 +95,8 @@ export interface LowCostECSProps {
95
95
  /**
96
96
  * Task definition for the server ecs task.
97
97
  *
98
- * @default - Nginx server task definition defined in createSampleTaskDefinition()
98
+ * @default - Nginx server task definition defined in sampleTaskDefinition()
99
+ * @see sampleTaskDefinition
99
100
  */
100
101
  readonly serverTaskDefinition?: LowCostECSTaskDefinitionOptions;
101
102
  }
@@ -114,5 +115,5 @@ export declare class LowCostECS extends Construct {
114
115
  readonly service: ecs.Ec2Service;
115
116
  constructor(scope: Construct, id: string, props: LowCostECSProps);
116
117
  private createTaskDefinition;
117
- private createSampleTaskDefinition;
118
+ private sampleTaskDefinition;
118
119
  }
@@ -186,9 +186,7 @@ class LowCostECS extends constructs_1.Construct {
186
186
  /**
187
187
  * Server ECS task
188
188
  */
189
- this.serverTaskDefinition = props.serverTaskDefinition
190
- ? this.createTaskDefinition(props.serverTaskDefinition)
191
- : this.createSampleTaskDefinition(records, logGroup);
189
+ this.serverTaskDefinition = this.createTaskDefinition(props.serverTaskDefinition ?? this.sampleTaskDefinition(records, logGroup));
192
190
  if (!this.serverTaskDefinition.defaultContainer) {
193
191
  throw new Error('defaultContainer is required for serverTaskDefinition. Add at least one essential container.');
194
192
  }
@@ -246,7 +244,9 @@ class LowCostECS extends constructs_1.Construct {
246
244
  enableExecuteCommand: true,
247
245
  });
248
246
  new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });
249
- new lib.CfnOutput(this, 'certbotStateMachineName', { value: certbotStateMachine.stateMachineName });
247
+ new lib.CfnOutput(this, 'CertbotStateMachineName', {
248
+ value: certbotStateMachine.stateMachineName,
249
+ });
250
250
  new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });
251
251
  new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });
252
252
  }
@@ -258,35 +258,40 @@ class LowCostECS extends constructs_1.Construct {
258
258
  taskDefinitionOptions.volumes?.forEach((volume) => serverTaskDefinition.addVolume(volume));
259
259
  return serverTaskDefinition;
260
260
  }
261
- createSampleTaskDefinition(records, logGroup) {
262
- const nginxTaskDefinition = new ecs.Ec2TaskDefinition(this, 'NginxTaskDefinition');
263
- const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {
264
- image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),
265
- containerName: 'nginx',
266
- memoryReservationMiB: 64,
267
- essential: true,
268
- environment: {
269
- SERVER_NAME: records.join(' '),
270
- CERT_NAME: records[0],
271
- },
272
- logging: ecs.LogDrivers.awsLogs({
273
- logGroup: logGroup,
274
- streamPrefix: 'nginx-proxy',
275
- }),
276
- });
277
- nginxContainer.addPortMappings({
278
- hostPort: 80,
279
- containerPort: 80,
280
- protocol: ecs.Protocol.TCP,
281
- }, {
282
- hostPort: 443,
283
- containerPort: 443,
284
- protocol: ecs.Protocol.TCP,
285
- });
286
- return nginxTaskDefinition;
261
+ sampleTaskDefinition(records, logGroup) {
262
+ return {
263
+ containers: [
264
+ {
265
+ image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),
266
+ containerName: 'nginx',
267
+ memoryReservationMiB: 64,
268
+ essential: true,
269
+ environment: {
270
+ SERVER_NAME: records.join(' '),
271
+ CERT_NAME: records[0],
272
+ },
273
+ logging: ecs.LogDrivers.awsLogs({
274
+ logGroup: logGroup,
275
+ streamPrefix: 'sample',
276
+ }),
277
+ portMappings: [
278
+ {
279
+ hostPort: 80,
280
+ containerPort: 80,
281
+ protocol: ecs.Protocol.TCP,
282
+ },
283
+ {
284
+ hostPort: 443,
285
+ containerPort: 443,
286
+ protocol: ecs.Protocol.TCP,
287
+ },
288
+ ],
289
+ },
290
+ ],
291
+ };
287
292
  }
288
293
  }
289
294
  exports.LowCostECS = LowCostECS;
290
295
  _a = JSII_RTTI_SYMBOL_1;
291
- LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.22" };
292
- //# 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 createSampleTaskDefinition()\n   */\n  readonly serverTaskDefinition?: LowCostECSTaskDefinitionOptions;\n}\n\nexport interface LowCostECSTaskDefinitionOptions {\n  readonly taskDefinition?: ecs.Ec2TaskDefinitionProps;\n  readonly containers: ecs.ContainerDefinitionOptions[];\n  readonly volumes?: ecs.Volume[];\n}\n\nexport class LowCostECS extends 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 = props.serverTaskDefinition\n      ? this.createTaskDefinition(props.serverTaskDefinition)\n      : this.createSampleTaskDefinition(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 createSampleTaskDefinition(\n    records: string[],\n    logGroup: ILogGroup,\n  ): ecs.Ec2TaskDefinition {\n    const nginxTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'NginxTaskDefinition',\n    );\n    const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {\n      image: ecs.ContainerImage.fromAsset(\n        path.join(__dirname, '../examples/containers/nginx'),\n      ),\n      containerName: 'nginx',\n      memoryReservationMiB: 64,\n      essential: true,\n      environment: {\n        SERVER_NAME: records.join(' '),\n        CERT_NAME: records[0],\n      },\n      logging: ecs.LogDrivers.awsLogs({\n        logGroup: logGroup,\n        streamPrefix: 'nginx-proxy',\n      }),\n    });\n\n    nginxContainer.addPortMappings(\n      {\n        hostPort: 80,\n        containerPort: 80,\n        protocol: ecs.Protocol.TCP,\n      },\n      {\n        hostPort: 443,\n        containerPort: 443,\n        protocol: ecs.Protocol.TCP,\n      },\n    );\n\n    return nginxTaskDefinition;\n  }\n}\n"]}
296
+ LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.24" };
297
+ //# 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(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.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(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: 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(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, 'CertbotStateMachineName', {\n      value: certbotStateMachine.stateMachineName,\n    });\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.22",
82
+ "version": "0.0.24",
80
83
  "jest": {
81
84
  "testMatch": [
82
85
  "<rootDir>/src/**/__tests__/**/*.ts?(x)",