low-cost-ecs 0.0.8 → 0.0.11
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +92 -20
- package/.projenrc.ts +1 -0
- package/API.md +55 -0
- package/README.md +21 -20
- package/cdk.json +1 -1
- package/examples/minimum.ts +13 -0
- package/examples/scheduled-autoscaling.ts +32 -0
- package/lib/low-cost-ecs.d.ts +7 -0
- package/lib/low-cost-ecs.js +40 -35
- package/package.json +1 -1
- package/todo.md +1 -1
- package/bin/low-cost-ecs.ts +0 -15
package/.jsii
CHANGED
|
@@ -3043,7 +3043,7 @@
|
|
|
3043
3043
|
},
|
|
3044
3044
|
"name": "low-cost-ecs",
|
|
3045
3045
|
"readme": {
|
|
3046
|
-
"markdown": "[](https://www.npmjs.com/package/low-cost-ecs)\n[](https://pypi.org/project/low-cost-ecs)\n[](https://github.com/rajyan/low-cost-ecs/actions/workflows/release.yml)\n[<img src=\"https://constructs.dev/badge?package=low-cost-ecs\" width=\"150\">](https://constructs.dev/packages/low-cost-ecs)\n\n# Low-Cost ECS\n\nA CDK construct that provides easy and [low-cost](#cost) ECS on EC2 server setup without a load balancer.\n\n**This construct is for development purposes only
|
|
3046
|
+
"markdown": "[](https://www.npmjs.com/package/low-cost-ecs)\n[](https://pypi.org/project/low-cost-ecs)\n[](https://github.com/rajyan/low-cost-ecs/actions/workflows/release.yml)\n[<img src=\"https://constructs.dev/badge?package=low-cost-ecs\" width=\"150\">](https://constructs.dev/packages/low-cost-ecs)\n\n# Low-Cost ECS\n\nA CDK construct that provides 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, it scales, and has dynamic port mappings and so on, 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 to make it easy to deploy resources to run a low-cost ECS server.\n\n# Try it out!\n\nThe easiest way to try this construct is to clone this repository and deploy the sample server.\nEdit settings in `bin/low-cost-ecs.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 to 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 ECS-optimized Amazon Linux 2 AMI) in the host instance and `AmazonSSMManagedInstanceCore` is added to the host instance role\nto 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 in 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\nThe ECS service occupies the host port, so only one service can be run 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 service, you may need to manually terminate the task of old the service.\n"
|
|
3047
3047
|
},
|
|
3048
3048
|
"repository": {
|
|
3049
3049
|
"type": "git",
|
|
@@ -3073,7 +3073,7 @@
|
|
|
3073
3073
|
},
|
|
3074
3074
|
"locationInModule": {
|
|
3075
3075
|
"filename": "src/low-cost-ecs.ts",
|
|
3076
|
-
"line":
|
|
3076
|
+
"line": 131
|
|
3077
3077
|
},
|
|
3078
3078
|
"parameters": [
|
|
3079
3079
|
{
|
|
@@ -3099,9 +3099,81 @@
|
|
|
3099
3099
|
"kind": "class",
|
|
3100
3100
|
"locationInModule": {
|
|
3101
3101
|
"filename": "src/low-cost-ecs.ts",
|
|
3102
|
-
"line":
|
|
3102
|
+
"line": 124
|
|
3103
3103
|
},
|
|
3104
3104
|
"name": "LowCostECS",
|
|
3105
|
+
"properties": [
|
|
3106
|
+
{
|
|
3107
|
+
"docs": {
|
|
3108
|
+
"stability": "experimental"
|
|
3109
|
+
},
|
|
3110
|
+
"immutable": true,
|
|
3111
|
+
"locationInModule": {
|
|
3112
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3113
|
+
"line": 127
|
|
3114
|
+
},
|
|
3115
|
+
"name": "certFileSystem",
|
|
3116
|
+
"type": {
|
|
3117
|
+
"fqn": "aws-cdk-lib.aws_efs.FileSystem"
|
|
3118
|
+
}
|
|
3119
|
+
},
|
|
3120
|
+
{
|
|
3121
|
+
"docs": {
|
|
3122
|
+
"stability": "experimental"
|
|
3123
|
+
},
|
|
3124
|
+
"immutable": true,
|
|
3125
|
+
"locationInModule": {
|
|
3126
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3127
|
+
"line": 128
|
|
3128
|
+
},
|
|
3129
|
+
"name": "cluster",
|
|
3130
|
+
"type": {
|
|
3131
|
+
"fqn": "aws-cdk-lib.aws_ecs.Cluster"
|
|
3132
|
+
}
|
|
3133
|
+
},
|
|
3134
|
+
{
|
|
3135
|
+
"docs": {
|
|
3136
|
+
"stability": "experimental"
|
|
3137
|
+
},
|
|
3138
|
+
"immutable": true,
|
|
3139
|
+
"locationInModule": {
|
|
3140
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3141
|
+
"line": 126
|
|
3142
|
+
},
|
|
3143
|
+
"name": "hostAutoScalingGroup",
|
|
3144
|
+
"type": {
|
|
3145
|
+
"fqn": "aws-cdk-lib.aws_autoscaling.AutoScalingGroup"
|
|
3146
|
+
}
|
|
3147
|
+
},
|
|
3148
|
+
{
|
|
3149
|
+
"docs": {
|
|
3150
|
+
"stability": "experimental"
|
|
3151
|
+
},
|
|
3152
|
+
"immutable": true,
|
|
3153
|
+
"locationInModule": {
|
|
3154
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3155
|
+
"line": 129
|
|
3156
|
+
},
|
|
3157
|
+
"name": "service",
|
|
3158
|
+
"type": {
|
|
3159
|
+
"fqn": "aws-cdk-lib.aws_ecs.Ec2Service"
|
|
3160
|
+
}
|
|
3161
|
+
},
|
|
3162
|
+
{
|
|
3163
|
+
"docs": {
|
|
3164
|
+
"stability": "experimental"
|
|
3165
|
+
},
|
|
3166
|
+
"immutable": true,
|
|
3167
|
+
"locationInModule": {
|
|
3168
|
+
"filename": "src/low-cost-ecs.ts",
|
|
3169
|
+
"line": 125
|
|
3170
|
+
},
|
|
3171
|
+
"name": "vpc",
|
|
3172
|
+
"type": {
|
|
3173
|
+
"fqn": "aws-cdk-lib.aws_ec2.IVpc"
|
|
3174
|
+
}
|
|
3175
|
+
}
|
|
3176
|
+
],
|
|
3105
3177
|
"symbolId": "src/low-cost-ecs:LowCostECS"
|
|
3106
3178
|
},
|
|
3107
3179
|
"low-cost-ecs.LowCostECSProps": {
|
|
@@ -3117,7 +3189,7 @@
|
|
|
3117
3189
|
"kind": "interface",
|
|
3118
3190
|
"locationInModule": {
|
|
3119
3191
|
"filename": "src/low-cost-ecs.ts",
|
|
3120
|
-
"line":
|
|
3192
|
+
"line": 17
|
|
3121
3193
|
},
|
|
3122
3194
|
"name": "LowCostECSProps",
|
|
3123
3195
|
"properties": [
|
|
@@ -3133,7 +3205,7 @@
|
|
|
3133
3205
|
"immutable": true,
|
|
3134
3206
|
"locationInModule": {
|
|
3135
3207
|
"filename": "src/low-cost-ecs.ts",
|
|
3136
|
-
"line":
|
|
3208
|
+
"line": 33
|
|
3137
3209
|
},
|
|
3138
3210
|
"name": "email",
|
|
3139
3211
|
"type": {
|
|
@@ -3149,7 +3221,7 @@
|
|
|
3149
3221
|
"immutable": true,
|
|
3150
3222
|
"locationInModule": {
|
|
3151
3223
|
"filename": "src/low-cost-ecs.ts",
|
|
3152
|
-
"line":
|
|
3224
|
+
"line": 21
|
|
3153
3225
|
},
|
|
3154
3226
|
"name": "hostedZoneDomain",
|
|
3155
3227
|
"type": {
|
|
@@ -3167,7 +3239,7 @@
|
|
|
3167
3239
|
"immutable": true,
|
|
3168
3240
|
"locationInModule": {
|
|
3169
3241
|
"filename": "src/low-cost-ecs.ts",
|
|
3170
|
-
"line":
|
|
3242
|
+
"line": 100
|
|
3171
3243
|
},
|
|
3172
3244
|
"name": "awsCliDockerTag",
|
|
3173
3245
|
"optional": true,
|
|
@@ -3188,7 +3260,7 @@
|
|
|
3188
3260
|
"immutable": true,
|
|
3189
3261
|
"locationInModule": {
|
|
3190
3262
|
"filename": "src/low-cost-ecs.ts",
|
|
3191
|
-
"line":
|
|
3263
|
+
"line": 85
|
|
3192
3264
|
},
|
|
3193
3265
|
"name": "certbotDockerTag",
|
|
3194
3266
|
"optional": true,
|
|
@@ -3206,7 +3278,7 @@
|
|
|
3206
3278
|
"immutable": true,
|
|
3207
3279
|
"locationInModule": {
|
|
3208
3280
|
"filename": "src/low-cost-ecs.ts",
|
|
3209
|
-
"line":
|
|
3281
|
+
"line": 92
|
|
3210
3282
|
},
|
|
3211
3283
|
"name": "certbotScheduleInterval",
|
|
3212
3284
|
"optional": true,
|
|
@@ -3224,7 +3296,7 @@
|
|
|
3224
3296
|
"immutable": true,
|
|
3225
3297
|
"locationInModule": {
|
|
3226
3298
|
"filename": "src/low-cost-ecs.ts",
|
|
3227
|
-
"line":
|
|
3299
|
+
"line": 107
|
|
3228
3300
|
},
|
|
3229
3301
|
"name": "containerInsights",
|
|
3230
3302
|
"optional": true,
|
|
@@ -3246,7 +3318,7 @@
|
|
|
3246
3318
|
"immutable": true,
|
|
3247
3319
|
"locationInModule": {
|
|
3248
3320
|
"filename": "src/low-cost-ecs.ts",
|
|
3249
|
-
"line":
|
|
3321
|
+
"line": 70
|
|
3250
3322
|
},
|
|
3251
3323
|
"name": "hostInstanceSpotPrice",
|
|
3252
3324
|
"optional": true,
|
|
@@ -3264,7 +3336,7 @@
|
|
|
3264
3336
|
"immutable": true,
|
|
3265
3337
|
"locationInModule": {
|
|
3266
3338
|
"filename": "src/low-cost-ecs.ts",
|
|
3267
|
-
"line":
|
|
3339
|
+
"line": 61
|
|
3268
3340
|
},
|
|
3269
3341
|
"name": "hostInstanceType",
|
|
3270
3342
|
"optional": true,
|
|
@@ -3282,7 +3354,7 @@
|
|
|
3282
3354
|
"immutable": true,
|
|
3283
3355
|
"locationInModule": {
|
|
3284
3356
|
"filename": "src/low-cost-ecs.ts",
|
|
3285
|
-
"line":
|
|
3357
|
+
"line": 77
|
|
3286
3358
|
},
|
|
3287
3359
|
"name": "logGroup",
|
|
3288
3360
|
"optional": true,
|
|
@@ -3300,7 +3372,7 @@
|
|
|
3300
3372
|
"immutable": true,
|
|
3301
3373
|
"locationInModule": {
|
|
3302
3374
|
"filename": "src/low-cost-ecs.ts",
|
|
3303
|
-
"line":
|
|
3375
|
+
"line": 40
|
|
3304
3376
|
},
|
|
3305
3377
|
"name": "recordDomainNames",
|
|
3306
3378
|
"optional": true,
|
|
@@ -3323,7 +3395,7 @@
|
|
|
3323
3395
|
"immutable": true,
|
|
3324
3396
|
"locationInModule": {
|
|
3325
3397
|
"filename": "src/low-cost-ecs.ts",
|
|
3326
|
-
"line":
|
|
3398
|
+
"line": 114
|
|
3327
3399
|
},
|
|
3328
3400
|
"name": "removalPolicy",
|
|
3329
3401
|
"optional": true,
|
|
@@ -3341,7 +3413,7 @@
|
|
|
3341
3413
|
"immutable": true,
|
|
3342
3414
|
"locationInModule": {
|
|
3343
3415
|
"filename": "src/low-cost-ecs.ts",
|
|
3344
|
-
"line":
|
|
3416
|
+
"line": 54
|
|
3345
3417
|
},
|
|
3346
3418
|
"name": "securityGroup",
|
|
3347
3419
|
"optional": true,
|
|
@@ -3359,7 +3431,7 @@
|
|
|
3359
3431
|
"immutable": true,
|
|
3360
3432
|
"locationInModule": {
|
|
3361
3433
|
"filename": "src/low-cost-ecs.ts",
|
|
3362
|
-
"line":
|
|
3434
|
+
"line": 121
|
|
3363
3435
|
},
|
|
3364
3436
|
"name": "serverTaskDefinition",
|
|
3365
3437
|
"optional": true,
|
|
@@ -3377,7 +3449,7 @@
|
|
|
3377
3449
|
"immutable": true,
|
|
3378
3450
|
"locationInModule": {
|
|
3379
3451
|
"filename": "src/low-cost-ecs.ts",
|
|
3380
|
-
"line":
|
|
3452
|
+
"line": 47
|
|
3381
3453
|
},
|
|
3382
3454
|
"name": "vpc",
|
|
3383
3455
|
"optional": true,
|
|
@@ -3389,6 +3461,6 @@
|
|
|
3389
3461
|
"symbolId": "src/low-cost-ecs:LowCostECSProps"
|
|
3390
3462
|
}
|
|
3391
3463
|
},
|
|
3392
|
-
"version": "0.0.
|
|
3393
|
-
"fingerprint": "
|
|
3464
|
+
"version": "0.0.11",
|
|
3465
|
+
"fingerprint": "F8bDPWAB1W3Il7w415rGPw99MaVy2Tbr8RQvv4+eAEw="
|
|
3394
3466
|
}
|
package/.projenrc.ts
CHANGED
|
@@ -43,6 +43,7 @@ const project = new awscdk.AwsCdkConstructLibrary({
|
|
|
43
43
|
projenrcTs: true,
|
|
44
44
|
});
|
|
45
45
|
|
|
46
|
+
project.tsconfigDev.addInclude('examples/**/*.ts');
|
|
46
47
|
// workaround until fixed https://youtrack.jetbrains.com/issue/WEB-57089/ESLint823-TypeError-thislibOptionsparse-is-not-a-function
|
|
47
48
|
project.addDevDeps('eslint@8.22.0');
|
|
48
49
|
|
package/API.md
CHANGED
|
@@ -464,6 +464,11 @@ The construct to start the search from.
|
|
|
464
464
|
| <code><a href="#low-cost-ecs.LowCostECS.property.nestedStackParent">nestedStackParent</a></code> | <code>aws-cdk-lib.Stack</code> | If this is a nested stack, returns it's parent stack. |
|
|
465
465
|
| <code><a href="#low-cost-ecs.LowCostECS.property.nestedStackResource">nestedStackResource</a></code> | <code>aws-cdk-lib.CfnResource</code> | If this is a nested stack, this represents its `AWS::CloudFormation::Stack` resource. |
|
|
466
466
|
| <code><a href="#low-cost-ecs.LowCostECS.property.terminationProtection">terminationProtection</a></code> | <code>boolean</code> | Whether termination protection is enabled for this stack. |
|
|
467
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.certFileSystem">certFileSystem</a></code> | <code>aws-cdk-lib.aws_efs.FileSystem</code> | *No description.* |
|
|
468
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.cluster">cluster</a></code> | <code>aws-cdk-lib.aws_ecs.Cluster</code> | *No description.* |
|
|
469
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.hostAutoScalingGroup">hostAutoScalingGroup</a></code> | <code>aws-cdk-lib.aws_autoscaling.AutoScalingGroup</code> | *No description.* |
|
|
470
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.service">service</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2Service</code> | *No description.* |
|
|
471
|
+
| <code><a href="#low-cost-ecs.LowCostECS.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | *No description.* |
|
|
467
472
|
|
|
468
473
|
---
|
|
469
474
|
|
|
@@ -797,6 +802,56 @@ Whether termination protection is enabled for this stack.
|
|
|
797
802
|
|
|
798
803
|
---
|
|
799
804
|
|
|
805
|
+
##### `certFileSystem`<sup>Required</sup> <a name="certFileSystem" id="low-cost-ecs.LowCostECS.property.certFileSystem"></a>
|
|
806
|
+
|
|
807
|
+
```typescript
|
|
808
|
+
public readonly certFileSystem: FileSystem;
|
|
809
|
+
```
|
|
810
|
+
|
|
811
|
+
- *Type:* aws-cdk-lib.aws_efs.FileSystem
|
|
812
|
+
|
|
813
|
+
---
|
|
814
|
+
|
|
815
|
+
##### `cluster`<sup>Required</sup> <a name="cluster" id="low-cost-ecs.LowCostECS.property.cluster"></a>
|
|
816
|
+
|
|
817
|
+
```typescript
|
|
818
|
+
public readonly cluster: Cluster;
|
|
819
|
+
```
|
|
820
|
+
|
|
821
|
+
- *Type:* aws-cdk-lib.aws_ecs.Cluster
|
|
822
|
+
|
|
823
|
+
---
|
|
824
|
+
|
|
825
|
+
##### `hostAutoScalingGroup`<sup>Required</sup> <a name="hostAutoScalingGroup" id="low-cost-ecs.LowCostECS.property.hostAutoScalingGroup"></a>
|
|
826
|
+
|
|
827
|
+
```typescript
|
|
828
|
+
public readonly hostAutoScalingGroup: AutoScalingGroup;
|
|
829
|
+
```
|
|
830
|
+
|
|
831
|
+
- *Type:* aws-cdk-lib.aws_autoscaling.AutoScalingGroup
|
|
832
|
+
|
|
833
|
+
---
|
|
834
|
+
|
|
835
|
+
##### `service`<sup>Required</sup> <a name="service" id="low-cost-ecs.LowCostECS.property.service"></a>
|
|
836
|
+
|
|
837
|
+
```typescript
|
|
838
|
+
public readonly service: Ec2Service;
|
|
839
|
+
```
|
|
840
|
+
|
|
841
|
+
- *Type:* aws-cdk-lib.aws_ecs.Ec2Service
|
|
842
|
+
|
|
843
|
+
---
|
|
844
|
+
|
|
845
|
+
##### `vpc`<sup>Required</sup> <a name="vpc" id="low-cost-ecs.LowCostECS.property.vpc"></a>
|
|
846
|
+
|
|
847
|
+
```typescript
|
|
848
|
+
public readonly vpc: IVpc;
|
|
849
|
+
```
|
|
850
|
+
|
|
851
|
+
- *Type:* aws-cdk-lib.aws_ec2.IVpc
|
|
852
|
+
|
|
853
|
+
---
|
|
854
|
+
|
|
800
855
|
|
|
801
856
|
## Structs <a name="Structs" id="Structs"></a>
|
|
802
857
|
|
package/README.md
CHANGED
|
@@ -7,17 +7,26 @@
|
|
|
7
7
|
|
|
8
8
|
A CDK construct that provides easy and [low-cost](#cost) ECS on EC2 server setup without a load balancer.
|
|
9
9
|
|
|
10
|
-
**This construct is for development purposes only
|
|
10
|
+
**This construct is for development purposes only**. See [Limitations](#limitations).
|
|
11
|
+
|
|
12
|
+
# Why
|
|
13
|
+
|
|
14
|
+
ECS may often seem expensive when used for personal development purposes, due to the cost of the load balancer.
|
|
15
|
+
The application load balancer is a great service that is easy to set up managed ACM certificates, it scales, and has dynamic port mappings and so on, but it is over-featured for running 1 ECS task.
|
|
16
|
+
|
|
17
|
+
However, 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.
|
|
18
|
+
This construct aims to automate these works and to make it easy to deploy resources to run a low-cost ECS server.
|
|
11
19
|
|
|
12
20
|
# Try it out!
|
|
13
21
|
|
|
14
|
-
The easiest way to
|
|
15
|
-
Edit settings in `bin/low-cost-ecs.ts` and deploy cdk construct. [Public hosted zone](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) is required.
|
|
22
|
+
The easiest way to try this construct is to clone this repository and deploy the sample server.
|
|
23
|
+
Edit settings in `bin/low-cost-ecs.ts` and deploy the cdk construct. [Public hosted zone](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) is required.
|
|
16
24
|
|
|
17
25
|
```
|
|
18
26
|
git clone https://github.com/rajyan/low-cost-ecs.git
|
|
27
|
+
yarn install
|
|
19
28
|
# edit settings in bin/low-cost-ecs.ts
|
|
20
|
-
|
|
29
|
+
./node_modules/.bin/cdk deploy
|
|
21
30
|
```
|
|
22
31
|
|
|
23
32
|
Access to configured `recordDomainNames` and see that the nginx sample server has been deployed.
|
|
@@ -40,8 +49,8 @@ class SampleStack extends Stack {
|
|
|
40
49
|
super(scope, id, props);
|
|
41
50
|
|
|
42
51
|
const vpc = { /** Your VPC */ };
|
|
43
|
-
const securityGroup = {/** Your security group */ };
|
|
44
|
-
const serverTaskDefinition = {/** Your task definition */ };
|
|
52
|
+
const securityGroup = { /** Your security group */ };
|
|
53
|
+
const serverTaskDefinition = { /** Your task definition */ };
|
|
45
54
|
|
|
46
55
|
new LowCostECS(this, 'LowCostECS', {
|
|
47
56
|
hostedZoneDomain: "rajyan.net",
|
|
@@ -55,16 +64,7 @@ class SampleStack extends Stack {
|
|
|
55
64
|
```
|
|
56
65
|
|
|
57
66
|
The required fields are `hostedZoneDomain` and `email`.
|
|
58
|
-
You can configure your server task definition
|
|
59
|
-
|
|
60
|
-
# Why
|
|
61
|
-
|
|
62
|
-
ECS may often seem expensive when used for personal development purposes, because of the cost of the load balancer.
|
|
63
|
-
The application load balancer is a great service because it is easy to set up managed ACM certificates, it scales, and has dynamic port mapping,
|
|
64
|
-
but it is over-featured for running 1 ECS service.
|
|
65
|
-
|
|
66
|
-
However, to run an ECS sever without a load balancer, you need to associate an Elastic IP to the host instance and install your certificate by yourself.
|
|
67
|
-
This construct aims to automate these works and deploy resources to run a low-cost ECS server.
|
|
67
|
+
You 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.
|
|
68
68
|
|
|
69
69
|
# Overview
|
|
70
70
|
|
|
@@ -77,11 +77,11 @@ Resources generated in this stack
|
|
|
77
77
|
* Scheduled automated renewal every 60 days
|
|
78
78
|
* Email notification on certbot task failure
|
|
79
79
|
* ECS on EC2 host instance
|
|
80
|
-
* ECS-optimized Amazon Linux 2 AMI instance auto
|
|
80
|
+
* ECS-optimized Amazon Linux 2 AMI instance auto-scaling group
|
|
81
81
|
* Automatically associated with Elastic IP on instance initialization
|
|
82
82
|
* ECS Service
|
|
83
|
-
* TLS/SSL certificate installation
|
|
84
|
-
* Certificate EFS mounted on `/etc/letsencrypt`
|
|
83
|
+
* TLS/SSL certificate installation before default container startup
|
|
84
|
+
* Certificate EFS mounted on default container as `/etc/letsencrypt`
|
|
85
85
|
* Others
|
|
86
86
|
* VPC with only public subnets (no NAT Gateways to decrease cost)
|
|
87
87
|
* Security groups with minimum inbounds
|
|
@@ -130,6 +130,7 @@ aws ecs execute-command \
|
|
|
130
130
|
|
|
131
131
|
# Limitations
|
|
132
132
|
|
|
133
|
-
The ECS service occupies the host port, only one service can be run at a time.
|
|
133
|
+
The ECS service occupies the host port, so only one service can be run at a time.
|
|
134
134
|
The old task must be terminated before the new task launches, and this causes downtime on release.
|
|
135
|
+
|
|
135
136
|
Also, if you make changes that require recreating service, you may need to manually terminate the task of old the service.
|
package/cdk.json
CHANGED
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { App } from 'aws-cdk-lib';
|
|
2
|
+
import { LowCostECS } from '../src';
|
|
3
|
+
|
|
4
|
+
const app = new App();
|
|
5
|
+
|
|
6
|
+
new LowCostECS(app, 'LowCostECSStack', {
|
|
7
|
+
env: {
|
|
8
|
+
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
9
|
+
region: process.env.CDK_DEFAULT_REGION,
|
|
10
|
+
},
|
|
11
|
+
hostedZoneDomain: 'rajyan.net',
|
|
12
|
+
email: 'kitakita7617@gmail.com',
|
|
13
|
+
});
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { App } from 'aws-cdk-lib';
|
|
2
|
+
import { Schedule } from 'aws-cdk-lib/aws-autoscaling';
|
|
3
|
+
import { LowCostECS } from '../src';
|
|
4
|
+
|
|
5
|
+
const app = new App();
|
|
6
|
+
|
|
7
|
+
const stack = new LowCostECS(app, 'LowCostECSStack', {
|
|
8
|
+
env: {
|
|
9
|
+
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
10
|
+
region: process.env.CDK_DEFAULT_REGION,
|
|
11
|
+
},
|
|
12
|
+
hostedZoneDomain: 'rajyan.net',
|
|
13
|
+
recordDomainNames: ['test1.rajyan.net', 'test2.rajyan.net'],
|
|
14
|
+
email: 'kitakita7617@gmail.com',
|
|
15
|
+
hostInstanceSpotPrice: '0.0050',
|
|
16
|
+
});
|
|
17
|
+
stack.hostAutoScalingGroup.scaleOnSchedule('IncreaseAtMorning', {
|
|
18
|
+
timeZone: 'Asia/Tokyo',
|
|
19
|
+
schedule: Schedule.cron({
|
|
20
|
+
minute: '0',
|
|
21
|
+
hour: '8',
|
|
22
|
+
}),
|
|
23
|
+
desiredCapacity: 1,
|
|
24
|
+
});
|
|
25
|
+
stack.hostAutoScalingGroup.scaleOnSchedule('DecreaseAtNight', {
|
|
26
|
+
timeZone: 'Asia/Tokyo',
|
|
27
|
+
schedule: Schedule.cron({
|
|
28
|
+
minute: '0',
|
|
29
|
+
hour: '23',
|
|
30
|
+
}),
|
|
31
|
+
desiredCapacity: 0,
|
|
32
|
+
});
|
package/lib/low-cost-ecs.d.ts
CHANGED
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
import * as lib from 'aws-cdk-lib';
|
|
2
|
+
import { AutoScalingGroup } from 'aws-cdk-lib/aws-autoscaling';
|
|
2
3
|
import * as ec2 from 'aws-cdk-lib/aws-ec2';
|
|
3
4
|
import * as ecs from 'aws-cdk-lib/aws-ecs';
|
|
5
|
+
import { FileSystem } from 'aws-cdk-lib/aws-efs';
|
|
4
6
|
import { ILogGroup } from 'aws-cdk-lib/aws-logs';
|
|
5
7
|
import { Construct } from 'constructs';
|
|
6
8
|
export interface LowCostECSProps extends lib.StackProps {
|
|
@@ -97,6 +99,11 @@ export interface LowCostECSProps extends lib.StackProps {
|
|
|
97
99
|
readonly serverTaskDefinition?: ecs.Ec2TaskDefinition;
|
|
98
100
|
}
|
|
99
101
|
export declare class LowCostECS extends lib.Stack {
|
|
102
|
+
readonly vpc: ec2.IVpc;
|
|
103
|
+
readonly hostAutoScalingGroup: AutoScalingGroup;
|
|
104
|
+
readonly certFileSystem: FileSystem;
|
|
105
|
+
readonly cluster: ecs.Cluster;
|
|
106
|
+
readonly service: ecs.Ec2Service;
|
|
100
107
|
constructor(scope: Construct, id: string, props: LowCostECSProps);
|
|
101
108
|
private sampleSeverTask;
|
|
102
109
|
}
|
package/lib/low-cost-ecs.js
CHANGED
|
@@ -20,21 +20,22 @@ const sfn_tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
|
20
20
|
class LowCostECS extends lib.Stack {
|
|
21
21
|
constructor(scope, id, props) {
|
|
22
22
|
super(scope, id, props);
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
23
|
+
this.vpc =
|
|
24
|
+
props.vpc ??
|
|
25
|
+
new ec2.Vpc(this, 'Vpc', {
|
|
26
|
+
natGateways: 0,
|
|
27
|
+
subnetConfiguration: [
|
|
28
|
+
{
|
|
29
|
+
name: 'PublicSubnet',
|
|
30
|
+
subnetType: ec2.SubnetType.PUBLIC,
|
|
31
|
+
},
|
|
32
|
+
],
|
|
33
|
+
});
|
|
34
|
+
this.cluster = new ecs.Cluster(this, 'Cluster', {
|
|
35
|
+
vpc: this.vpc,
|
|
35
36
|
containerInsights: props.containerInsights,
|
|
36
37
|
});
|
|
37
|
-
|
|
38
|
+
this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {
|
|
38
39
|
machineImage: ecs.EcsOptimizedImage.amazonLinux2(ecs.AmiHardwareType.STANDARD, {
|
|
39
40
|
cachedInContext: true,
|
|
40
41
|
}),
|
|
@@ -46,22 +47,22 @@ class LowCostECS extends lib.Stack {
|
|
|
46
47
|
maxCapacity: 1,
|
|
47
48
|
});
|
|
48
49
|
if (props.securityGroup) {
|
|
49
|
-
hostAutoScalingGroup.addSecurityGroup(props.securityGroup);
|
|
50
|
+
this.hostAutoScalingGroup.addSecurityGroup(props.securityGroup);
|
|
50
51
|
}
|
|
51
52
|
else {
|
|
52
|
-
hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));
|
|
53
|
-
hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));
|
|
54
|
-
hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(80));
|
|
55
|
-
hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(443));
|
|
53
|
+
this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));
|
|
54
|
+
this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));
|
|
55
|
+
this.hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(80));
|
|
56
|
+
this.hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(443));
|
|
56
57
|
}
|
|
57
58
|
/**
|
|
58
59
|
* Add managed policy to allow ssh through ssm manager
|
|
59
60
|
*/
|
|
60
|
-
hostAutoScalingGroup.role.addManagedPolicy(aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'));
|
|
61
|
+
this.hostAutoScalingGroup.role.addManagedPolicy(aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'));
|
|
61
62
|
/**
|
|
62
63
|
* Add policy to associate elastic ip on startup
|
|
63
64
|
*/
|
|
64
|
-
hostAutoScalingGroup.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
65
|
+
this.hostAutoScalingGroup.role.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
65
66
|
effect: aws_iam_1.Effect.ALLOW,
|
|
66
67
|
actions: ['ec2:DescribeAddresses', 'ec2:AssociateAddress'],
|
|
67
68
|
resources: ['*'],
|
|
@@ -70,18 +71,18 @@ class LowCostECS extends lib.Stack {
|
|
|
70
71
|
const tagUniqueId = lib.Names.uniqueId(hostInstanceIp);
|
|
71
72
|
hostInstanceIp.tags.setTag('Name', tagUniqueId);
|
|
72
73
|
const awsCliTag = props.awsCliDockerTag ?? 'latest';
|
|
73
|
-
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 ${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 ${hostAutoScalingGroup.env.region} --instance-id "$INSTANCE_ID" --allocation-id "$ALLOCATION_ID" --allow-reassociation`);
|
|
74
|
-
|
|
75
|
-
vpc,
|
|
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`);
|
|
75
|
+
this.certFileSystem = new aws_efs_1.FileSystem(this, 'FileSystem', {
|
|
76
|
+
vpc: this.vpc,
|
|
76
77
|
encrypted: true,
|
|
77
78
|
securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {
|
|
78
|
-
vpc,
|
|
79
|
+
vpc: this.vpc,
|
|
79
80
|
allowAllOutbound: false,
|
|
80
81
|
}),
|
|
81
82
|
removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,
|
|
82
83
|
});
|
|
83
|
-
certFileSystem.connections.allowDefaultPortTo(hostAutoScalingGroup);
|
|
84
|
-
certFileSystem.connections.allowDefaultPortFrom(hostAutoScalingGroup);
|
|
84
|
+
this.certFileSystem.connections.allowDefaultPortTo(this.hostAutoScalingGroup);
|
|
85
|
+
this.certFileSystem.connections.allowDefaultPortFrom(this.hostAutoScalingGroup);
|
|
85
86
|
/**
|
|
86
87
|
* ARecord to Elastic ip
|
|
87
88
|
*/
|
|
@@ -139,11 +140,11 @@ class LowCostECS extends lib.Stack {
|
|
|
139
140
|
streamPrefix: certbotTag,
|
|
140
141
|
}),
|
|
141
142
|
});
|
|
142
|
-
certFileSystem.grant(certbotTaskDefinition.taskRole, 'elasticfilesystem:ClientWrite');
|
|
143
|
+
this.certFileSystem.grant(certbotTaskDefinition.taskRole, 'elasticfilesystem:ClientWrite');
|
|
143
144
|
certbotTaskDefinition.addVolume({
|
|
144
145
|
name: 'certVolume',
|
|
145
146
|
efsVolumeConfiguration: {
|
|
146
|
-
fileSystemId: certFileSystem.fileSystemId,
|
|
147
|
+
fileSystemId: this.certFileSystem.fileSystemId,
|
|
147
148
|
},
|
|
148
149
|
});
|
|
149
150
|
certbotContainer.addMountPoints({
|
|
@@ -162,7 +163,7 @@ class LowCostECS extends lib.Stack {
|
|
|
162
163
|
endpoint: props.email,
|
|
163
164
|
});
|
|
164
165
|
const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {
|
|
165
|
-
cluster: cluster,
|
|
166
|
+
cluster: this.cluster,
|
|
166
167
|
taskDefinition: certbotTaskDefinition,
|
|
167
168
|
launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),
|
|
168
169
|
integrationPattern: sfn.IntegrationPattern.RUN_JOB,
|
|
@@ -185,11 +186,11 @@ class LowCostECS extends lib.Stack {
|
|
|
185
186
|
* Server ECS task
|
|
186
187
|
*/
|
|
187
188
|
const serverTaskDefinition = props.serverTaskDefinition ?? this.sampleSeverTask(records, logGroup);
|
|
188
|
-
certFileSystem.grant(serverTaskDefinition.taskRole, 'elasticfilesystem:ClientMount');
|
|
189
|
+
this.certFileSystem.grant(serverTaskDefinition.taskRole, 'elasticfilesystem:ClientMount');
|
|
189
190
|
serverTaskDefinition.addVolume({
|
|
190
191
|
name: 'certVolume',
|
|
191
192
|
efsVolumeConfiguration: {
|
|
192
|
-
fileSystemId: certFileSystem.fileSystemId,
|
|
193
|
+
fileSystemId: this.certFileSystem.fileSystemId,
|
|
193
194
|
},
|
|
194
195
|
});
|
|
195
196
|
serverTaskDefinition.defaultContainer?.addMountPoints({
|
|
@@ -227,8 +228,8 @@ class LowCostECS extends lib.Stack {
|
|
|
227
228
|
});
|
|
228
229
|
certbotStateMachine.grantExecution(serverTaskDefinition.taskRole, 'states:DescribeExecution');
|
|
229
230
|
certbotStateMachine.grantStartExecution(serverTaskDefinition.taskRole);
|
|
230
|
-
new ecs.Ec2Service(this, 'Service', {
|
|
231
|
-
cluster: cluster,
|
|
231
|
+
this.service = new ecs.Ec2Service(this, 'Service', {
|
|
232
|
+
cluster: this.cluster,
|
|
232
233
|
taskDefinition: serverTaskDefinition,
|
|
233
234
|
desiredCount: 1,
|
|
234
235
|
minHealthyPercent: 0,
|
|
@@ -238,6 +239,10 @@ class LowCostECS extends lib.Stack {
|
|
|
238
239
|
},
|
|
239
240
|
enableExecuteCommand: true,
|
|
240
241
|
});
|
|
242
|
+
new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });
|
|
243
|
+
new lib.CfnOutput(this, 'certbotStateMachineName', { value: certbotStateMachine.stateMachineName });
|
|
244
|
+
new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });
|
|
245
|
+
new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });
|
|
241
246
|
}
|
|
242
247
|
sampleSeverTask(records, logGroup) {
|
|
243
248
|
const nginxTaskDefinition = new ecs.Ec2TaskDefinition(this, 'NginxTaskDefinition');
|
|
@@ -269,5 +274,5 @@ class LowCostECS extends lib.Stack {
|
|
|
269
274
|
}
|
|
270
275
|
exports.LowCostECS = LowCostECS;
|
|
271
276
|
_a = JSII_RTTI_SYMBOL_1;
|
|
272
|
-
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.
|
|
273
|
-
//# 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 * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { FileSystem } from 'aws-cdk-lib/aws-efs';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { SfnStateMachine } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport { Subscription, SubscriptionProtocol, Topic } from 'aws-cdk-lib/aws-sns';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as sfn_tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport interface LowCostECSProps extends lib.StackProps {\n  /**\n   * Domain name of the hosted zone.\n   */\n  readonly hostedZoneDomain: string;\n\n  /**\n   * Email for expiration emails to register to your let's encrypt account.\n   *\n   * @link https://letsencrypt.org/docs/expiration-emails/\n   *\n   * Also registered as a subscriber of the sns topic, notified on certbot task failure.\n   * Subscription confirmation email would be sent on stack creation.\n   *\n   * @link https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html\n   */\n  readonly email: string;\n\n  /**\n   * Domain names for A records to elastic ip of ECS host instance.\n   *\n   * @default - [ props.hostedZone.zoneName ]\n   */\n  readonly recordDomainNames?: string[];\n\n  /**\n   * Vpc of the ECS host instance and cluster.\n   *\n   * @default - Creates vpc with only public subnets and no NAT gateways.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group of the ECS host instance\n   *\n   * @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).\n   */\n  readonly securityGroup?: ec2.SecurityGroup;\n\n  /**\n   * Instance type of the ECS host instance.\n   *\n   * @default - t2.micro\n   */\n  readonly hostInstanceType?: string;\n\n  /**\n   * The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.\n   * Host instance asg would use spot instances if hostInstanceSpotPrice is set.\n   *\n   * @link https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice\n   * @default - undefined\n   */\n  readonly hostInstanceSpotPrice?: string;\n\n  /**\n   * Log group of the certbot task and the aws-cli task.\n   *\n   * @default - Creates default cdk log group\n   */\n  readonly logGroup?: ILogGroup;\n\n  /**\n   * Docker image tag of certbot/dns-route53 to create certificates.\n   *\n   * @link https://hub.docker.com/r/certbot/dns-route53/tags\n   * @default - v1.29.0\n   */\n  readonly certbotDockerTag?: string;\n\n  /**\n   * Certbot task schedule interval in days to renew the certificate.\n   *\n   * @default - 60\n   */\n  readonly certbotScheduleInterval?: number;\n\n  /**\n   * Docker image tag of amazon/aws-cli.\n   * This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.\n   *\n   * @default - latest\n   */\n  readonly awsCliDockerTag?: string;\n\n  /**\n   * Enable container insights or not\n   *\n   * @default - undefined (container insights disabled)\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * Removal policy for the file system and log group (if using default).\n   *\n   * @default - RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: lib.RemovalPolicy;\n\n  /**\n   * Task definition for the server ecs task.\n   *\n   * @default - Nginx server task definition defined in sampleServerTask()\n   */\n  readonly serverTaskDefinition?: ecs.Ec2TaskDefinition;\n};\n\nexport class LowCostECS extends lib.Stack {\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id, props);\n\n    const vpc =\n      props.vpc ??\n      new ec2.Vpc(this, 'Vpc', {\n        natGateways: 0,\n        subnetConfiguration: [\n          {\n            name: 'PublicSubnet',\n            subnetType: ec2.SubnetType.PUBLIC,\n          },\n        ],\n      });\n\n    const cluster = new ecs.Cluster(this, 'Cluster', {\n      vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    const hostAutoScalingGroup = 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      hostAutoScalingGroup.addSecurityGroup(props.securityGroup);\n    } else {\n      hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));\n      hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));\n      hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(80),\n      );\n      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    hostAutoScalingGroup.role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n    );\n    /**\n     * Add policy to associate elastic ip on startup\n     */\n    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    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 ${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 ${hostAutoScalingGroup.env.region} --instance-id \"$INSTANCE_ID\" --allocation-id \"$ALLOCATION_ID\" --allow-reassociation`,\n    );\n\n    const certFileSystem = new FileSystem(this, 'FileSystem', {\n      vpc,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc,\n        allowAllOutbound: false,\n      }),\n      removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n    });\n    certFileSystem.connections.allowDefaultPortTo(hostAutoScalingGroup);\n    certFileSystem.connections.allowDefaultPortFrom(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    certFileSystem.grant(\n      certbotTaskDefinition.taskRole,\n      'elasticfilesystem:ClientWrite',\n    );\n    certbotTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: certFileSystem.fileSystemId,\n      },\n    });\n    certbotContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: false,\n    });\n\n    /**\n     * Schedule Certbot certificate create/renew on Step Functions\n     * Sends email notification on certbot failure\n     */\n    const topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: topic,\n      protocol: SubscriptionProtocol.EMAIL,\n      endpoint: props.email,\n    });\n\n    const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {\n      cluster: cluster,\n      taskDefinition: certbotTaskDefinition,\n      launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),\n      integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n    });\n    certbotRunTask.addCatch(\n      new sfn_tasks.SnsPublish(this, 'SendEmailOnFailure', {\n        topic: topic,\n        message: sfn.TaskInput.fromJsonPathAt('$'),\n      }).next(new sfn.Fail(this, 'Fail')),\n    );\n    certbotRunTask.addRetry({\n      interval: lib.Duration.seconds(20),\n    });\n    const certbotStateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition: certbotRunTask,\n    });\n\n    new Rule(this, 'CertbotScheduleRule', {\n      schedule: Schedule.rate(\n        lib.Duration.days(props.certbotScheduleInterval ?? 60),\n      ),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    const serverTaskDefinition =\n      props.serverTaskDefinition ?? this.sampleSeverTask(records, logGroup);\n    certFileSystem.grant(\n      serverTaskDefinition.taskRole,\n      'elasticfilesystem:ClientMount',\n    );\n    serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: certFileSystem.fileSystemId,\n      },\n    });\n    serverTaskDefinition.defaultContainer?.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: true,\n    });\n\n    /**\n     * AWS cli container to execute certbot sfn before the default container startup.\n     */\n    serverTaskDefinition.defaultContainer?.addContainerDependencies({\n      container: serverTaskDefinition.addContainer('AWSCliContainer', {\n        image: ecs.ContainerImage.fromRegistry(`amazon/aws-cli:${awsCliTag}`),\n        containerName: 'aws-cli',\n        memoryReservationMiB: 64,\n        entryPoint: ['/bin/bash', '-c'],\n        command: [\n          `set -eux\n          aws configure set region ${certbotStateMachine.env.region} && \\\\\n          aws configure set output text && \\\\\n          EXECUTION_ARN=$(aws stepfunctions start-execution --state-machine-arn ${certbotStateMachine.stateMachineArn} --query executionArn) && \\\\\n          until [ $(aws stepfunctions describe-execution --execution-arn \"$EXECUTION_ARN\" --query status) != RUNNING ];\n          do\n            echo \"Waiting for $EXECUTION_ARN\"\n            sleep 10\n          done`,\n        ],\n        essential: false,\n        logging: ecs.LogDriver.awsLogs({\n          logGroup: logGroup,\n          streamPrefix: awsCliTag,\n        }),\n      }),\n      condition: ecs.ContainerDependencyCondition.COMPLETE,\n    });\n    certbotStateMachine.grantExecution(\n      serverTaskDefinition.taskRole,\n      'states:DescribeExecution',\n    );\n    certbotStateMachine.grantStartExecution(serverTaskDefinition.taskRole);\n\n    new ecs.Ec2Service(this, 'Service', {\n      cluster: cluster,\n      taskDefinition: serverTaskDefinition,\n      desiredCount: 1,\n      minHealthyPercent: 0,\n      maxHealthyPercent: 100,\n      circuitBreaker: {\n        rollback: true,\n      },\n      enableExecuteCommand: true,\n    });\n  }\n\n  private sampleSeverTask(\n    records: string[],\n    logGroup: ILogGroup,\n  ): ecs.Ec2TaskDefinition {\n    const nginxTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'NginxTaskDefinition',\n    );\n    const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {\n      image: ecs.ContainerImage.fromAsset(\n        path.join(__dirname, '../containers/nginx-proxy'),\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"]}
|
|
277
|
+
LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.11" };
|
|
278
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"low-cost-ecs.js","sourceRoot":"","sources":["../src/low-cost-ecs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as lib from 'aws-cdk-lib';\nimport { AutoScalingGroup } from 'aws-cdk-lib/aws-autoscaling';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { FileSystem } from 'aws-cdk-lib/aws-efs';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { SfnStateMachine } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport { Subscription, SubscriptionProtocol, Topic } from 'aws-cdk-lib/aws-sns';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as sfn_tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport interface LowCostECSProps extends lib.StackProps {\n  /**\n   * Domain name of the hosted zone.\n   */\n  readonly hostedZoneDomain: string;\n\n  /**\n   * Email for expiration emails to register to your let's encrypt account.\n   *\n   * @link https://letsencrypt.org/docs/expiration-emails/\n   *\n   * Also registered as a subscriber of the sns topic, notified on certbot task failure.\n   * Subscription confirmation email would be sent on stack creation.\n   *\n   * @link https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html\n   */\n  readonly email: string;\n\n  /**\n   * Domain names for A records to elastic ip of ECS host instance.\n   *\n   * @default - [ props.hostedZone.zoneName ]\n   */\n  readonly recordDomainNames?: string[];\n\n  /**\n   * Vpc of the ECS host instance and cluster.\n   *\n   * @default - Creates vpc with only public subnets and no NAT gateways.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group of the ECS host instance\n   *\n   * @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).\n   */\n  readonly securityGroup?: ec2.SecurityGroup;\n\n  /**\n   * Instance type of the ECS host instance.\n   *\n   * @default - t2.micro\n   */\n  readonly hostInstanceType?: string;\n\n  /**\n   * The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.\n   * Host instance asg would use spot instances if hostInstanceSpotPrice is set.\n   *\n   * @link https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice\n   * @default - undefined\n   */\n  readonly hostInstanceSpotPrice?: string;\n\n  /**\n   * Log group of the certbot task and the aws-cli task.\n   *\n   * @default - Creates default cdk log group\n   */\n  readonly logGroup?: ILogGroup;\n\n  /**\n   * Docker image tag of certbot/dns-route53 to create certificates.\n   *\n   * @link https://hub.docker.com/r/certbot/dns-route53/tags\n   * @default - v1.29.0\n   */\n  readonly certbotDockerTag?: string;\n\n  /**\n   * Certbot task schedule interval in days to renew the certificate.\n   *\n   * @default - 60\n   */\n  readonly certbotScheduleInterval?: number;\n\n  /**\n   * Docker image tag of amazon/aws-cli.\n   * This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.\n   *\n   * @default - latest\n   */\n  readonly awsCliDockerTag?: string;\n\n  /**\n   * Enable container insights or not\n   *\n   * @default - undefined (container insights disabled)\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * Removal policy for the file system and log group (if using default).\n   *\n   * @default - RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: lib.RemovalPolicy;\n\n  /**\n   * Task definition for the server ecs task.\n   *\n   * @default - Nginx server task definition defined in sampleServerTask()\n   */\n  readonly serverTaskDefinition?: ecs.Ec2TaskDefinition;\n};\n\nexport class LowCostECS extends lib.Stack {\n  readonly vpc: ec2.IVpc;\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  readonly certFileSystem: FileSystem;\n  readonly cluster: ecs.Cluster;\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id, props);\n\n    this.vpc =\n      props.vpc ??\n      new ec2.Vpc(this, 'Vpc', {\n        natGateways: 0,\n        subnetConfiguration: [\n          {\n            name: 'PublicSubnet',\n            subnetType: ec2.SubnetType.PUBLIC,\n          },\n        ],\n      });\n\n    this.cluster = new ecs.Cluster(this, 'Cluster', {\n      vpc: this.vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {\n      machineImage: ecs.EcsOptimizedImage.amazonLinux2(\n        ecs.AmiHardwareType.STANDARD,\n        {\n          cachedInContext: true,\n        },\n      ),\n      instanceType: new ec2.InstanceType(props.hostInstanceType ?? 't2.micro'),\n      spotPrice: props.hostInstanceSpotPrice,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      associatePublicIpAddress: true,\n      minCapacity: 1,\n      maxCapacity: 1,\n    });\n\n    if (props.securityGroup) {\n      this.hostAutoScalingGroup.addSecurityGroup(props.securityGroup);\n    } else {\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(80),\n      );\n      this.hostAutoScalingGroup.connections.allowFrom(\n        ec2.Peer.anyIpv6(),\n        ec2.Port.tcp(443),\n      );\n    }\n\n    /**\n     * Add managed policy to allow ssh through ssm manager\n     */\n    this.hostAutoScalingGroup.role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n    );\n    /**\n     * Add policy to associate elastic ip on startup\n     */\n    this.hostAutoScalingGroup.role.addToPrincipalPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['ec2:DescribeAddresses', 'ec2:AssociateAddress'],\n        resources: ['*'],\n      }),\n    );\n\n    const hostInstanceIp = new ec2.CfnEIP(this, 'HostInstanceIp');\n    const tagUniqueId = lib.Names.uniqueId(hostInstanceIp);\n    hostInstanceIp.tags.setTag('Name', tagUniqueId);\n\n    const awsCliTag = props.awsCliDockerTag ?? 'latest';\n    this.hostAutoScalingGroup.addUserData(\n      'INSTANCE_ID=$(curl --silent http://169.254.169.254/latest/meta-data/instance-id)',\n      `ALLOCATION_ID=$(docker run --net=host amazon/aws-cli:${awsCliTag} ec2 describe-addresses --region ${this.hostAutoScalingGroup.env.region} --filter Name=tag:Name,Values=${tagUniqueId} --query 'Addresses[].AllocationId' --output text | head)`,\n      `docker run --net=host amazon/aws-cli:${awsCliTag} ec2 associate-address --region ${this.hostAutoScalingGroup.env.region} --instance-id \"$INSTANCE_ID\" --allocation-id \"$ALLOCATION_ID\" --allow-reassociation`,\n    );\n\n    this.certFileSystem = new FileSystem(this, 'FileSystem', {\n      vpc: this.vpc,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc: this.vpc,\n        allowAllOutbound: false,\n      }),\n      removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n    });\n    this.certFileSystem.connections.allowDefaultPortTo(this.hostAutoScalingGroup);\n    this.certFileSystem.connections.allowDefaultPortFrom(this.hostAutoScalingGroup);\n\n    /**\n     * ARecord to Elastic ip\n     */\n    const hostedZone = route53.HostedZone.fromLookup(this, 'HostedZone', {\n      domainName: props.hostedZoneDomain,\n    });\n    const records = props.recordDomainNames ?? [hostedZone.zoneName];\n    records.forEach(\n      (record) =>\n        new route53.ARecord(this, `ARecord${record}`, {\n          zone: hostedZone,\n          recordName: record,\n          target: route53.RecordTarget.fromIpAddresses(hostInstanceIp.ref),\n        }),\n    );\n\n    /**\n     * Certbot Task Definition\n     * Mounts generated certificate to EFS\n     */\n    const logGroup =\n      props.logGroup ??\n      new LogGroup(this, 'LogGroup', {\n        retention: RetentionDays.TWO_YEARS,\n        removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n      });\n\n    const certbotTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'CertbotTaskDefinition',\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ListHostedZones', 'route53:GetChange'],\n        resources: ['*'],\n      }),\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ChangeResourceRecordSets'],\n        resources: [hostedZone.hostedZoneArn],\n      }),\n    );\n\n    const certbotTag = props.certbotDockerTag ?? 'v1.29.0';\n    const certbotContainer = certbotTaskDefinition.addContainer(\n      'CertbotContainer',\n      {\n        image: ecs.ContainerImage.fromRegistry(\n          `certbot/dns-route53:${certbotTag}`,\n        ),\n        containerName: 'certbot',\n        memoryReservationMiB: 64,\n        command: [\n          'certonly',\n          '--verbose',\n          '--preferred-challenges=dns-01',\n          '--dns-route53',\n          '--dns-route53-propagation-seconds=300',\n          '--non-interactive',\n          '--agree-tos',\n          '--expand',\n          '-m',\n          props.email,\n          '--cert-name',\n          records[0],\n          ...records.flatMap((domain) => ['-d', domain]),\n        ],\n        logging: ecs.LogDriver.awsLogs({\n          logGroup,\n          streamPrefix: certbotTag,\n        }),\n      },\n    );\n\n    this.certFileSystem.grant(\n      certbotTaskDefinition.taskRole,\n      'elasticfilesystem:ClientWrite',\n    );\n    certbotTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    certbotContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: false,\n    });\n\n    /**\n     * Schedule Certbot certificate create/renew on Step Functions\n     * Sends email notification on certbot failure\n     */\n    const topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: topic,\n      protocol: SubscriptionProtocol.EMAIL,\n      endpoint: props.email,\n    });\n\n    const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {\n      cluster: this.cluster,\n      taskDefinition: certbotTaskDefinition,\n      launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),\n      integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n    });\n    certbotRunTask.addCatch(\n      new sfn_tasks.SnsPublish(this, 'SendEmailOnFailure', {\n        topic: topic,\n        message: sfn.TaskInput.fromJsonPathAt('$'),\n      }).next(new sfn.Fail(this, 'Fail')),\n    );\n    certbotRunTask.addRetry({\n      interval: lib.Duration.seconds(20),\n    });\n    const certbotStateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition: certbotRunTask,\n    });\n\n    new Rule(this, 'CertbotScheduleRule', {\n      schedule: Schedule.rate(\n        lib.Duration.days(props.certbotScheduleInterval ?? 60),\n      ),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    const serverTaskDefinition =\n      props.serverTaskDefinition ?? this.sampleSeverTask(records, logGroup);\n    this.certFileSystem.grant(\n      serverTaskDefinition.taskRole,\n      'elasticfilesystem:ClientMount',\n    );\n    serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    serverTaskDefinition.defaultContainer?.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: true,\n    });\n\n    /**\n     * AWS cli container to execute certbot sfn before the default container startup.\n     */\n    serverTaskDefinition.defaultContainer?.addContainerDependencies({\n      container: serverTaskDefinition.addContainer('AWSCliContainer', {\n        image: ecs.ContainerImage.fromRegistry(`amazon/aws-cli:${awsCliTag}`),\n        containerName: 'aws-cli',\n        memoryReservationMiB: 64,\n        entryPoint: ['/bin/bash', '-c'],\n        command: [\n          `set -eux\n          aws configure set region ${certbotStateMachine.env.region} && \\\\\n          aws configure set output text && \\\\\n          EXECUTION_ARN=$(aws stepfunctions start-execution --state-machine-arn ${certbotStateMachine.stateMachineArn} --query executionArn) && \\\\\n          until [ $(aws stepfunctions describe-execution --execution-arn \"$EXECUTION_ARN\" --query status) != RUNNING ];\n          do\n            echo \"Waiting for $EXECUTION_ARN\"\n            sleep 10\n          done`,\n        ],\n        essential: false,\n        logging: ecs.LogDriver.awsLogs({\n          logGroup: logGroup,\n          streamPrefix: awsCliTag,\n        }),\n      }),\n      condition: ecs.ContainerDependencyCondition.COMPLETE,\n    });\n    certbotStateMachine.grantExecution(\n      serverTaskDefinition.taskRole,\n      'states:DescribeExecution',\n    );\n    certbotStateMachine.grantStartExecution(serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: serverTaskDefinition,\n      desiredCount: 1,\n      minHealthyPercent: 0,\n      maxHealthyPercent: 100,\n      circuitBreaker: {\n        rollback: true,\n      },\n      enableExecuteCommand: true,\n    });\n\n    new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });\n    new lib.CfnOutput(this, 'certbotStateMachineName', { value: certbotStateMachine.stateMachineName });\n    new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });\n    new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });\n  }\n\n  private sampleSeverTask(\n    records: string[],\n    logGroup: ILogGroup,\n  ): ecs.Ec2TaskDefinition {\n    const nginxTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'NginxTaskDefinition',\n    );\n    const nginxContainer = nginxTaskDefinition.addContainer('NginxContainer', {\n      image: ecs.ContainerImage.fromAsset(\n        path.join(__dirname, '../containers/nginx-proxy'),\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"]}
|
package/package.json
CHANGED
package/todo.md
CHANGED
package/bin/low-cost-ecs.ts
DELETED
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
import { App } from "aws-cdk-lib";
|
|
2
|
-
import { LowCostECS } from '../src';
|
|
3
|
-
|
|
4
|
-
const app = new App();
|
|
5
|
-
|
|
6
|
-
new LowCostECS(app, "LowCostECSStack", {
|
|
7
|
-
env: {
|
|
8
|
-
account: process.env.CDK_DEFAULT_ACCOUNT,
|
|
9
|
-
region: process.env.CDK_DEFAULT_REGION,
|
|
10
|
-
},
|
|
11
|
-
hostedZoneDomain: "rajyan.net",
|
|
12
|
-
recordDomainNames: ["test1.rajyan.net", "test2.rajyan.net"],
|
|
13
|
-
email: "kitakita7617@gmail.com",
|
|
14
|
-
hostInstanceSpotPrice: "0.0050",
|
|
15
|
-
});
|