low-cost-ecs 0.0.48 → 0.0.49

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 CHANGED
@@ -3023,7 +3023,7 @@
3023
3023
  "stability": "experimental"
3024
3024
  },
3025
3025
  "homepage": "https://github.com/rajyan/low-cost-ecs.git",
3026
- "jsiiVersion": "1.74.0 (build 6d08790)",
3026
+ "jsiiVersion": "1.75.0 (build 63bb957)",
3027
3027
  "keywords": [
3028
3028
  "cdk",
3029
3029
  "certbot",
@@ -3561,6 +3561,6 @@
3561
3561
  "symbolId": "src/low-cost-ecs:LowCostECSTaskDefinitionOptions"
3562
3562
  }
3563
3563
  },
3564
- "version": "0.0.48",
3565
- "fingerprint": "Od4isgU1Ljl+m38UFZl2yTB11BcKme0vQb0HVF21phQ="
3564
+ "version": "0.0.49",
3565
+ "fingerprint": "DMlrTE/ziaz+jQYTWMJjFBXT3n1bPoDa9WTZ2fiKXDA="
3566
3566
  }
package/API.md.md ADDED
@@ -0,0 +1,460 @@
1
+ # API Reference <a name="API Reference" id="api-reference"></a>
2
+
3
+ ## Constructs <a name="Constructs" id="Constructs"></a>
4
+
5
+ ### LowCostECS <a name="LowCostECS" id="low-cost-ecs.LowCostECS"></a>
6
+
7
+ #### Initializers <a name="Initializers" id="low-cost-ecs.LowCostECS.Initializer"></a>
8
+
9
+ ```typescript
10
+ import { LowCostECS } from 'low-cost-ecs'
11
+
12
+ new LowCostECS(scope: Construct, id: string, props: LowCostECSProps)
13
+ ```
14
+
15
+ | **Name** | **Type** | **Description** |
16
+ | --- | --- | --- |
17
+ | <code><a href="#low-cost-ecs.LowCostECS.Initializer.parameter.scope">scope</a></code> | <code>constructs.Construct</code> | *No description.* |
18
+ | <code><a href="#low-cost-ecs.LowCostECS.Initializer.parameter.id">id</a></code> | <code>string</code> | *No description.* |
19
+ | <code><a href="#low-cost-ecs.LowCostECS.Initializer.parameter.props">props</a></code> | <code><a href="#low-cost-ecs.LowCostECSProps">LowCostECSProps</a></code> | *No description.* |
20
+
21
+ ---
22
+
23
+ ##### `scope`<sup>Required</sup> <a name="scope" id="low-cost-ecs.LowCostECS.Initializer.parameter.scope"></a>
24
+
25
+ - *Type:* constructs.Construct
26
+
27
+ ---
28
+
29
+ ##### `id`<sup>Required</sup> <a name="id" id="low-cost-ecs.LowCostECS.Initializer.parameter.id"></a>
30
+
31
+ - *Type:* string
32
+
33
+ ---
34
+
35
+ ##### `props`<sup>Required</sup> <a name="props" id="low-cost-ecs.LowCostECS.Initializer.parameter.props"></a>
36
+
37
+ - *Type:* <a href="#low-cost-ecs.LowCostECSProps">LowCostECSProps</a>
38
+
39
+ ---
40
+
41
+ #### Methods <a name="Methods" id="Methods"></a>
42
+
43
+ | **Name** | **Description** |
44
+ | --- | --- |
45
+ | <code><a href="#low-cost-ecs.LowCostECS.toString">toString</a></code> | Returns a string representation of this construct. |
46
+
47
+ ---
48
+
49
+ ##### `toString` <a name="toString" id="low-cost-ecs.LowCostECS.toString"></a>
50
+
51
+ ```typescript
52
+ public toString(): string
53
+ ```
54
+
55
+ Returns a string representation of this construct.
56
+
57
+ #### Static Functions <a name="Static Functions" id="Static Functions"></a>
58
+
59
+ | **Name** | **Description** |
60
+ | --- | --- |
61
+ | <code><a href="#low-cost-ecs.LowCostECS.isConstruct">isConstruct</a></code> | Checks if `x` is a construct. |
62
+
63
+ ---
64
+
65
+ ##### ~~`isConstruct`~~ <a name="isConstruct" id="low-cost-ecs.LowCostECS.isConstruct"></a>
66
+
67
+ ```typescript
68
+ import { LowCostECS } from 'low-cost-ecs'
69
+
70
+ LowCostECS.isConstruct(x: any)
71
+ ```
72
+
73
+ Checks if `x` is a construct.
74
+
75
+ ###### `x`<sup>Required</sup> <a name="x" id="low-cost-ecs.LowCostECS.isConstruct.parameter.x"></a>
76
+
77
+ - *Type:* any
78
+
79
+ Any object.
80
+
81
+ ---
82
+
83
+ #### Properties <a name="Properties" id="Properties"></a>
84
+
85
+ | **Name** | **Type** | **Description** |
86
+ | --- | --- | --- |
87
+ | <code><a href="#low-cost-ecs.LowCostECS.property.node">node</a></code> | <code>constructs.Node</code> | The tree node. |
88
+ | <code><a href="#low-cost-ecs.LowCostECS.property.certFileSystem">certFileSystem</a></code> | <code>aws-cdk-lib.aws_efs.FileSystem</code> | EFS file system that the SSL/TLS certificates are installed. |
89
+ | <code><a href="#low-cost-ecs.LowCostECS.property.cluster">cluster</a></code> | <code>aws-cdk-lib.aws_ecs.Cluster</code> | ECS cluster created in configured VPC. |
90
+ | <code><a href="#low-cost-ecs.LowCostECS.property.hostAutoScalingGroup">hostAutoScalingGroup</a></code> | <code>aws-cdk-lib.aws_autoscaling.AutoScalingGroup</code> | ECS on EC2 service host instance autoscaling group. |
91
+ | <code><a href="#low-cost-ecs.LowCostECS.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinition</code> | Server task definition generated from LowCostECSTaskDefinitionOptions. |
92
+ | <code><a href="#low-cost-ecs.LowCostECS.property.service">service</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2Service</code> | ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100. |
93
+ | <code><a href="#low-cost-ecs.LowCostECS.property.topic">topic</a></code> | <code>aws-cdk-lib.aws_sns.Topic</code> | SNS topic used to notify certbot renewal failure. |
94
+
95
+ ---
96
+
97
+ ##### `node`<sup>Required</sup> <a name="node" id="low-cost-ecs.LowCostECS.property.node"></a>
98
+
99
+ ```typescript
100
+ public readonly node: Node;
101
+ ```
102
+
103
+ - *Type:* constructs.Node
104
+
105
+ The tree node.
106
+
107
+ ---
108
+
109
+ ##### `certFileSystem`<sup>Required</sup> <a name="certFileSystem" id="low-cost-ecs.LowCostECS.property.certFileSystem"></a>
110
+
111
+ ```typescript
112
+ public readonly certFileSystem: FileSystem;
113
+ ```
114
+
115
+ - *Type:* aws-cdk-lib.aws_efs.FileSystem
116
+
117
+ EFS file system that the SSL/TLS certificates are installed.
118
+
119
+ ---
120
+
121
+ ##### `cluster`<sup>Required</sup> <a name="cluster" id="low-cost-ecs.LowCostECS.property.cluster"></a>
122
+
123
+ ```typescript
124
+ public readonly cluster: Cluster;
125
+ ```
126
+
127
+ - *Type:* aws-cdk-lib.aws_ecs.Cluster
128
+
129
+ ECS cluster created in configured VPC.
130
+
131
+ ---
132
+
133
+ ##### `hostAutoScalingGroup`<sup>Required</sup> <a name="hostAutoScalingGroup" id="low-cost-ecs.LowCostECS.property.hostAutoScalingGroup"></a>
134
+
135
+ ```typescript
136
+ public readonly hostAutoScalingGroup: AutoScalingGroup;
137
+ ```
138
+
139
+ - *Type:* aws-cdk-lib.aws_autoscaling.AutoScalingGroup
140
+
141
+ ECS on EC2 service host instance autoscaling group.
142
+
143
+ ---
144
+
145
+ ##### `serverTaskDefinition`<sup>Required</sup> <a name="serverTaskDefinition" id="low-cost-ecs.LowCostECS.property.serverTaskDefinition"></a>
146
+
147
+ ```typescript
148
+ public readonly serverTaskDefinition: Ec2TaskDefinition;
149
+ ```
150
+
151
+ - *Type:* aws-cdk-lib.aws_ecs.Ec2TaskDefinition
152
+
153
+ Server task definition generated from LowCostECSTaskDefinitionOptions.
154
+
155
+ ---
156
+
157
+ ##### `service`<sup>Required</sup> <a name="service" id="low-cost-ecs.LowCostECS.property.service"></a>
158
+
159
+ ```typescript
160
+ public readonly service: Ec2Service;
161
+ ```
162
+
163
+ - *Type:* aws-cdk-lib.aws_ecs.Ec2Service
164
+
165
+ ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100.
166
+
167
+ > [https://github.com/rajyan/low-cost-ecs#limitations](https://github.com/rajyan/low-cost-ecs#limitations)
168
+
169
+ ---
170
+
171
+ ##### `topic`<sup>Required</sup> <a name="topic" id="low-cost-ecs.LowCostECS.property.topic"></a>
172
+
173
+ ```typescript
174
+ public readonly topic: Topic;
175
+ ```
176
+
177
+ - *Type:* aws-cdk-lib.aws_sns.Topic
178
+
179
+ SNS topic used to notify certbot renewal failure.
180
+
181
+ ---
182
+
183
+
184
+ ## Structs <a name="Structs" id="Structs"></a>
185
+
186
+ ### LowCostECSProps <a name="LowCostECSProps" id="low-cost-ecs.LowCostECSProps"></a>
187
+
188
+ #### Initializer <a name="Initializer" id="low-cost-ecs.LowCostECSProps.Initializer"></a>
189
+
190
+ ```typescript
191
+ import { LowCostECSProps } from 'low-cost-ecs'
192
+
193
+ const lowCostECSProps: LowCostECSProps = { ... }
194
+ ```
195
+
196
+ #### Properties <a name="Properties" id="Properties"></a>
197
+
198
+ | **Name** | **Type** | **Description** |
199
+ | --- | --- | --- |
200
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.email">email</a></code> | <code>string</code> | Email for expiration emails to register to your let's encrypt account. |
201
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.hostedZoneDomain">hostedZoneDomain</a></code> | <code>string</code> | Domain name of the hosted zone. |
202
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.awsCliDockerTag">awsCliDockerTag</a></code> | <code>string</code> | Docker image tag of amazon/aws-cli. |
203
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.certbotDockerTag">certbotDockerTag</a></code> | <code>string</code> | Docker image tag of certbot/dns-route53 to create certificates. |
204
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.certbotScheduleInterval">certbotScheduleInterval</a></code> | <code>number</code> | Certbot task schedule interval in days to renew the certificate. |
205
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.containerInsights">containerInsights</a></code> | <code>boolean</code> | Enable container insights or not. |
206
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.hostInstanceSpotPrice">hostInstanceSpotPrice</a></code> | <code>string</code> | The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request. |
207
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.hostInstanceType">hostInstanceType</a></code> | <code>string</code> | Instance type of the ECS host instance. |
208
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.logGroup">logGroup</a></code> | <code>aws-cdk-lib.aws_logs.ILogGroup</code> | Log group of the certbot task and the aws-cli task. |
209
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.recordDomainNames">recordDomainNames</a></code> | <code>string[]</code> | Domain names for A records to elastic ip of ECS host instance. |
210
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.removalPolicy">removalPolicy</a></code> | <code>aws-cdk-lib.RemovalPolicy</code> | Removal policy for the file system and log group (if using default). |
211
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.securityGroups">securityGroups</a></code> | <code>aws-cdk-lib.aws_ec2.ISecurityGroup[]</code> | Security group of the ECS host instance. |
212
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.serverTaskDefinition">serverTaskDefinition</a></code> | <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions">LowCostECSTaskDefinitionOptions</a></code> | Task definition for the server ecs task. |
213
+ | <code><a href="#low-cost-ecs.LowCostECSProps.property.vpc">vpc</a></code> | <code>aws-cdk-lib.aws_ec2.IVpc</code> | VPC of the ECS cluster and EFS file system. |
214
+
215
+ ---
216
+
217
+ ##### `email`<sup>Required</sup> <a name="email" id="low-cost-ecs.LowCostECSProps.property.email"></a>
218
+
219
+ ```typescript
220
+ public readonly email: string;
221
+ ```
222
+
223
+ - *Type:* string
224
+
225
+ Email for expiration emails to register to your let's encrypt account.
226
+
227
+ > [https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html](https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html)
228
+
229
+ ---
230
+
231
+ ##### `hostedZoneDomain`<sup>Required</sup> <a name="hostedZoneDomain" id="low-cost-ecs.LowCostECSProps.property.hostedZoneDomain"></a>
232
+
233
+ ```typescript
234
+ public readonly hostedZoneDomain: string;
235
+ ```
236
+
237
+ - *Type:* string
238
+
239
+ Domain name of the hosted zone.
240
+
241
+ ---
242
+
243
+ ##### `awsCliDockerTag`<sup>Optional</sup> <a name="awsCliDockerTag" id="low-cost-ecs.LowCostECSProps.property.awsCliDockerTag"></a>
244
+
245
+ ```typescript
246
+ public readonly awsCliDockerTag: string;
247
+ ```
248
+
249
+ - *Type:* string
250
+ - *Default:* latest
251
+
252
+ Docker image tag of amazon/aws-cli.
253
+
254
+ This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.
255
+
256
+ ---
257
+
258
+ ##### `certbotDockerTag`<sup>Optional</sup> <a name="certbotDockerTag" id="low-cost-ecs.LowCostECSProps.property.certbotDockerTag"></a>
259
+
260
+ ```typescript
261
+ public readonly certbotDockerTag: string;
262
+ ```
263
+
264
+ - *Type:* string
265
+ - *Default:* v1.29.0
266
+
267
+ Docker image tag of certbot/dns-route53 to create certificates.
268
+
269
+ > [https://hub.docker.com/r/certbot/dns-route53/tags](https://hub.docker.com/r/certbot/dns-route53/tags)
270
+
271
+ ---
272
+
273
+ ##### `certbotScheduleInterval`<sup>Optional</sup> <a name="certbotScheduleInterval" id="low-cost-ecs.LowCostECSProps.property.certbotScheduleInterval"></a>
274
+
275
+ ```typescript
276
+ public readonly certbotScheduleInterval: number;
277
+ ```
278
+
279
+ - *Type:* number
280
+ - *Default:* 60
281
+
282
+ Certbot task schedule interval in days to renew the certificate.
283
+
284
+ ---
285
+
286
+ ##### `containerInsights`<sup>Optional</sup> <a name="containerInsights" id="low-cost-ecs.LowCostECSProps.property.containerInsights"></a>
287
+
288
+ ```typescript
289
+ public readonly containerInsights: boolean;
290
+ ```
291
+
292
+ - *Type:* boolean
293
+ - *Default:* undefined (container insights disabled)
294
+
295
+ Enable container insights or not.
296
+
297
+ ---
298
+
299
+ ##### `hostInstanceSpotPrice`<sup>Optional</sup> <a name="hostInstanceSpotPrice" id="low-cost-ecs.LowCostECSProps.property.hostInstanceSpotPrice"></a>
300
+
301
+ ```typescript
302
+ public readonly hostInstanceSpotPrice: string;
303
+ ```
304
+
305
+ - *Type:* string
306
+ - *Default:* undefined
307
+
308
+ The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.
309
+
310
+ Host instance asg would use spot instances if hostInstanceSpotPrice is set.
311
+
312
+ > [https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice)
313
+
314
+ ---
315
+
316
+ ##### `hostInstanceType`<sup>Optional</sup> <a name="hostInstanceType" id="low-cost-ecs.LowCostECSProps.property.hostInstanceType"></a>
317
+
318
+ ```typescript
319
+ public readonly hostInstanceType: string;
320
+ ```
321
+
322
+ - *Type:* string
323
+ - *Default:* t2.micro
324
+
325
+ Instance type of the ECS host instance.
326
+
327
+ ---
328
+
329
+ ##### `logGroup`<sup>Optional</sup> <a name="logGroup" id="low-cost-ecs.LowCostECSProps.property.logGroup"></a>
330
+
331
+ ```typescript
332
+ public readonly logGroup: ILogGroup;
333
+ ```
334
+
335
+ - *Type:* aws-cdk-lib.aws_logs.ILogGroup
336
+ - *Default:* Creates default cdk log group
337
+
338
+ Log group of the certbot task and the aws-cli task.
339
+
340
+ ---
341
+
342
+ ##### `recordDomainNames`<sup>Optional</sup> <a name="recordDomainNames" id="low-cost-ecs.LowCostECSProps.property.recordDomainNames"></a>
343
+
344
+ ```typescript
345
+ public readonly recordDomainNames: string[];
346
+ ```
347
+
348
+ - *Type:* string[]
349
+ - *Default:* [ props.hostedZone.zoneName ]
350
+
351
+ Domain names for A records to elastic ip of ECS host instance.
352
+
353
+ ---
354
+
355
+ ##### `removalPolicy`<sup>Optional</sup> <a name="removalPolicy" id="low-cost-ecs.LowCostECSProps.property.removalPolicy"></a>
356
+
357
+ ```typescript
358
+ public readonly removalPolicy: RemovalPolicy;
359
+ ```
360
+
361
+ - *Type:* aws-cdk-lib.RemovalPolicy
362
+ - *Default:* RemovalPolicy.DESTROY
363
+
364
+ Removal policy for the file system and log group (if using default).
365
+
366
+ ---
367
+
368
+ ##### `securityGroups`<sup>Optional</sup> <a name="securityGroups" id="low-cost-ecs.LowCostECSProps.property.securityGroups"></a>
369
+
370
+ ```typescript
371
+ public readonly securityGroups: ISecurityGroup[];
372
+ ```
373
+
374
+ - *Type:* aws-cdk-lib.aws_ec2.ISecurityGroup[]
375
+ - *Default:* Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).
376
+
377
+ Security group of the ECS host instance.
378
+
379
+ ---
380
+
381
+ ##### `serverTaskDefinition`<sup>Optional</sup> <a name="serverTaskDefinition" id="low-cost-ecs.LowCostECSProps.property.serverTaskDefinition"></a>
382
+
383
+ ```typescript
384
+ public readonly serverTaskDefinition: LowCostECSTaskDefinitionOptions;
385
+ ```
386
+
387
+ - *Type:* <a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions">LowCostECSTaskDefinitionOptions</a>
388
+ - *Default:* Nginx server task definition defined in sampleTaskDefinition()
389
+
390
+ Task definition for the server ecs task.
391
+
392
+ > [sampleTaskDefinition](sampleTaskDefinition)
393
+
394
+ ---
395
+
396
+ ##### `vpc`<sup>Optional</sup> <a name="vpc" id="low-cost-ecs.LowCostECSProps.property.vpc"></a>
397
+
398
+ ```typescript
399
+ public readonly vpc: IVpc;
400
+ ```
401
+
402
+ - *Type:* aws-cdk-lib.aws_ec2.IVpc
403
+ - *Default:* Creates vpc with only public subnets and no NAT gateways.
404
+
405
+ VPC of the ECS cluster and EFS file system.
406
+
407
+ ---
408
+
409
+ ### LowCostECSTaskDefinitionOptions <a name="LowCostECSTaskDefinitionOptions" id="low-cost-ecs.LowCostECSTaskDefinitionOptions"></a>
410
+
411
+ #### Initializer <a name="Initializer" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.Initializer"></a>
412
+
413
+ ```typescript
414
+ import { LowCostECSTaskDefinitionOptions } from 'low-cost-ecs'
415
+
416
+ const lowCostECSTaskDefinitionOptions: LowCostECSTaskDefinitionOptions = { ... }
417
+ ```
418
+
419
+ #### Properties <a name="Properties" id="Properties"></a>
420
+
421
+ | **Name** | **Type** | **Description** |
422
+ | --- | --- | --- |
423
+ | <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.containers">containers</a></code> | <code>aws-cdk-lib.aws_ecs.ContainerDefinitionOptions[]</code> | *No description.* |
424
+ | <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.taskDefinition">taskDefinition</a></code> | <code>aws-cdk-lib.aws_ecs.Ec2TaskDefinitionProps</code> | *No description.* |
425
+ | <code><a href="#low-cost-ecs.LowCostECSTaskDefinitionOptions.property.volumes">volumes</a></code> | <code>aws-cdk-lib.aws_ecs.Volume[]</code> | *No description.* |
426
+
427
+ ---
428
+
429
+ ##### `containers`<sup>Required</sup> <a name="containers" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.containers"></a>
430
+
431
+ ```typescript
432
+ public readonly containers: ContainerDefinitionOptions[];
433
+ ```
434
+
435
+ - *Type:* aws-cdk-lib.aws_ecs.ContainerDefinitionOptions[]
436
+
437
+ ---
438
+
439
+ ##### `taskDefinition`<sup>Optional</sup> <a name="taskDefinition" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.taskDefinition"></a>
440
+
441
+ ```typescript
442
+ public readonly taskDefinition: Ec2TaskDefinitionProps;
443
+ ```
444
+
445
+ - *Type:* aws-cdk-lib.aws_ecs.Ec2TaskDefinitionProps
446
+
447
+ ---
448
+
449
+ ##### `volumes`<sup>Optional</sup> <a name="volumes" id="low-cost-ecs.LowCostECSTaskDefinitionOptions.property.volumes"></a>
450
+
451
+ ```typescript
452
+ public readonly volumes: Volume[];
453
+ ```
454
+
455
+ - *Type:* aws-cdk-lib.aws_ecs.Volume[]
456
+
457
+ ---
458
+
459
+
460
+
@@ -290,5 +290,5 @@ class LowCostECS extends constructs_1.Construct {
290
290
  }
291
291
  exports.LowCostECS = LowCostECS;
292
292
  _a = JSII_RTTI_SYMBOL_1;
293
- LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.48" };
293
+ LowCostECS[_a] = { fqn: "low-cost-ecs.LowCostECS", version: "0.0.49" };
294
294
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"low-cost-ecs.js","sourceRoot":"","sources":["../src/low-cost-ecs.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as lib from 'aws-cdk-lib';\nimport { AutoScalingGroup } from 'aws-cdk-lib/aws-autoscaling';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport { FileSystem } from 'aws-cdk-lib/aws-efs';\nimport { Rule, Schedule } from 'aws-cdk-lib/aws-events';\nimport { SfnStateMachine } from 'aws-cdk-lib/aws-events-targets';\nimport { Effect, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ILogGroup, LogGroup, RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport { Subscription, SubscriptionProtocol, Topic } from 'aws-cdk-lib/aws-sns';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as sfn_tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\nexport interface LowCostECSProps {\n  /**\n   * Domain name of the hosted zone.\n   */\n  readonly hostedZoneDomain: string;\n\n  /**\n   * Email for expiration emails to register to your let's encrypt account.\n   *\n   * @link https://letsencrypt.org/docs/expiration-emails/\n   *\n   * Also registered as a subscriber of the sns topic, notified on certbot task failure.\n   * Subscription confirmation email would be sent on stack creation.\n   *\n   * @link https://docs.aws.amazon.com/sns/latest/dg/sns-email-notifications.html\n   */\n  readonly email: string;\n\n  /**\n   * Domain names for A records to elastic ip of ECS host instance.\n   *\n   * @default - [ props.hostedZone.zoneName ]\n   */\n  readonly recordDomainNames?: string[];\n\n  /**\n   * VPC of the ECS cluster and EFS file system.\n   *\n   * @default - Creates vpc with only public subnets and no NAT gateways.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group of the ECS host instance\n   *\n   * @default - Creates security group with allowAllOutbound and ingress rule (ipv4, ipv6) => (tcp 80, 443).\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Instance type of the ECS host instance.\n   *\n   * @default - t2.micro\n   */\n  readonly hostInstanceType?: string;\n\n  /**\n   * The maximum hourly price (in USD) to be paid for any Spot Instance launched to fulfill the request.\n   * Host instance asg would use spot instances if hostInstanceSpotPrice is set.\n   *\n   * @link https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_ecs.AddCapacityOptions.html#spotprice\n   * @default - undefined\n   */\n  readonly hostInstanceSpotPrice?: string;\n\n  /**\n   * Log group of the certbot task and the aws-cli task.\n   *\n   * @default - Creates default cdk log group\n   */\n  readonly logGroup?: ILogGroup;\n\n  /**\n   * Docker image tag of certbot/dns-route53 to create certificates.\n   *\n   * @link https://hub.docker.com/r/certbot/dns-route53/tags\n   * @default - v1.29.0\n   */\n  readonly certbotDockerTag?: string;\n\n  /**\n   * Certbot task schedule interval in days to renew the certificate.\n   *\n   * @default - 60\n   */\n  readonly certbotScheduleInterval?: number;\n\n  /**\n   * Docker image tag of amazon/aws-cli.\n   * This image is used to associate elastic ip on host instance startup, and run certbot cfn on ecs container startup.\n   *\n   * @default - latest\n   */\n  readonly awsCliDockerTag?: string;\n\n  /**\n   * Enable container insights or not.\n   *\n   * @default - undefined (container insights disabled)\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * Removal policy for the file system and log group (if using default).\n   *\n   * @default - RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: lib.RemovalPolicy;\n\n  /**\n   * Task definition for the server ecs task.\n   *\n   * @default - Nginx server task definition defined in sampleTaskDefinition()\n   * @see sampleTaskDefinition\n   */\n  readonly serverTaskDefinition?: LowCostECSTaskDefinitionOptions;\n}\n\nexport interface LowCostECSTaskDefinitionOptions {\n  readonly taskDefinition?: ecs.Ec2TaskDefinitionProps;\n  readonly containers: ecs.ContainerDefinitionOptions[];\n  readonly volumes?: ecs.Volume[];\n}\n\nexport class LowCostECS extends Construct {\n  /**\n   * ECS cluster created in configured VPC.\n   */\n  readonly cluster: ecs.Cluster;\n  /**\n   * ECS on EC2 service host instance autoscaling group.\n   */\n  readonly hostAutoScalingGroup: AutoScalingGroup;\n  /**\n   * EFS file system that the SSL/TLS certificates are installed.\n   */\n  readonly certFileSystem: FileSystem;\n  /**\n   * SNS topic used to notify certbot renewal failure.\n   */\n  readonly topic: Topic;\n  /**\n   * Server task definition generated from LowCostECSTaskDefinitionOptions.\n   */\n  readonly serverTaskDefinition: ecs.Ec2TaskDefinition;\n  /**\n   * ECS service of the server with desiredCount: 1, minHealthyPercent: 0, maxHealthyPercent: 100.\n   *\n   * @link https://github.com/rajyan/low-cost-ecs#limitations\n   */\n  readonly service: ecs.Ec2Service;\n\n  constructor(scope: Construct, id: string, props: LowCostECSProps) {\n    super(scope, id);\n\n    const vpc =\n      props.vpc ??\n      new ec2.Vpc(scope, 'Vpc', {\n        natGateways: 0,\n        subnetConfiguration: [\n          {\n            name: 'PublicSubnet',\n            subnetType: ec2.SubnetType.PUBLIC,\n          },\n        ],\n      });\n\n    this.cluster = new ecs.Cluster(scope, 'Cluster', {\n      vpc: vpc,\n      containerInsights: props.containerInsights,\n    });\n\n    this.hostAutoScalingGroup = this.cluster.addCapacity('HostInstanceCapacity', {\n      machineImage: ecs.EcsOptimizedImage.amazonLinux2(ecs.AmiHardwareType.STANDARD, {\n        cachedInContext: true,\n      }),\n      instanceType: new ec2.InstanceType(props.hostInstanceType ?? 't2.micro'),\n      spotPrice: props.hostInstanceSpotPrice,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      associatePublicIpAddress: true,\n      minCapacity: 1,\n      maxCapacity: 1,\n    });\n\n    if (props.securityGroups) {\n      this.hostAutoScalingGroup.node.tryRemoveChild('InstanceSecurityGroup');\n      props.securityGroups.forEach((sg) => this.hostAutoScalingGroup.addSecurityGroup(sg));\n    } else {\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFromAnyIpv4(ec2.Port.tcp(443));\n      this.hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(80));\n      this.hostAutoScalingGroup.connections.allowFrom(ec2.Peer.anyIpv6(), ec2.Port.tcp(443));\n    }\n\n    /**\n     * Add managed policy to allow ssh through ssm manager\n     */\n    this.hostAutoScalingGroup.role.addManagedPolicy(\n      ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore')\n    );\n    /**\n     * Add policy to associate elastic ip on startup\n     */\n    this.hostAutoScalingGroup.role.addToPrincipalPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['ec2:DescribeAddresses', 'ec2:AssociateAddress'],\n        resources: ['*'],\n      })\n    );\n\n    const hostInstanceIp = new ec2.CfnEIP(this, 'HostInstanceIp');\n    const tagUniqueId = lib.Names.uniqueId(hostInstanceIp);\n    hostInstanceIp.tags.setTag('Name', tagUniqueId);\n\n    const awsCliTag = props.awsCliDockerTag ?? 'latest';\n    this.hostAutoScalingGroup.addUserData(\n      'INSTANCE_ID=$(curl --silent http://169.254.169.254/latest/meta-data/instance-id)',\n      `ALLOCATION_ID=$(docker run --net=host amazon/aws-cli:${awsCliTag} ec2 describe-addresses --region ${this.hostAutoScalingGroup.env.region} --filter Name=tag:Name,Values=${tagUniqueId} --query 'Addresses[].AllocationId' --output text | head)`,\n      `docker run --net=host amazon/aws-cli:${awsCliTag} ec2 associate-address --region ${this.hostAutoScalingGroup.env.region} --instance-id \"$INSTANCE_ID\" --allocation-id \"$ALLOCATION_ID\" --allow-reassociation`\n    );\n\n    this.certFileSystem = new FileSystem(this, 'FileSystem', {\n      vpc,\n      encrypted: true,\n      securityGroup: new ec2.SecurityGroup(this, 'FileSystemSecurityGroup', {\n        vpc: vpc,\n        allowAllOutbound: false,\n      }),\n      removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n    });\n    this.certFileSystem.connections.allowDefaultPortTo(this.hostAutoScalingGroup);\n    this.certFileSystem.connections.allowDefaultPortFrom(this.hostAutoScalingGroup);\n\n    /**\n     * ARecord to Elastic ip\n     */\n    const hostedZone = route53.HostedZone.fromLookup(this, 'HostedZone', {\n      domainName: props.hostedZoneDomain,\n    });\n    const records = props.recordDomainNames ?? [hostedZone.zoneName];\n    records.forEach(\n      (record) =>\n        new route53.ARecord(this, `ARecord${record}`, {\n          zone: hostedZone,\n          recordName: record,\n          target: route53.RecordTarget.fromIpAddresses(hostInstanceIp.ref),\n        })\n    );\n\n    /**\n     * Certbot Task Definition\n     * Mounts generated certificate to EFS\n     */\n    const logGroup =\n      props.logGroup ??\n      new LogGroup(this, 'LogGroup', {\n        retention: RetentionDays.TWO_YEARS,\n        removalPolicy: props.removalPolicy ?? lib.RemovalPolicy.DESTROY,\n      });\n\n    const certbotTaskDefinition = new ecs.Ec2TaskDefinition(this, 'CertbotTaskDefinition');\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ListHostedZones', 'route53:GetChange'],\n        resources: ['*'],\n      })\n    );\n    certbotTaskDefinition.addToTaskRolePolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['route53:ChangeResourceRecordSets'],\n        resources: [hostedZone.hostedZoneArn],\n      })\n    );\n\n    const certbotTag = props.certbotDockerTag ?? 'v1.29.0';\n    const certbotContainer = certbotTaskDefinition.addContainer('CertbotContainer', {\n      image: ecs.ContainerImage.fromRegistry(`certbot/dns-route53:${certbotTag}`),\n      containerName: 'certbot',\n      memoryReservationMiB: 64,\n      command: [\n        'certonly',\n        '--verbose',\n        '--preferred-challenges=dns-01',\n        '--dns-route53',\n        '--dns-route53-propagation-seconds=300',\n        '--non-interactive',\n        '--agree-tos',\n        '--expand',\n        '-m',\n        props.email,\n        '--cert-name',\n        records[0],\n        ...records.flatMap((domain) => ['-d', domain]),\n      ],\n      logging: ecs.LogDriver.awsLogs({\n        logGroup,\n        streamPrefix: certbotTag,\n      }),\n    });\n\n    this.certFileSystem.grant(certbotTaskDefinition.taskRole, 'elasticfilesystem:ClientWrite');\n    certbotTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    certbotContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: false,\n    });\n\n    /**\n     * Schedule Certbot certificate create/renew on Step Functions\n     * Sends email notification on certbot failure\n     */\n    this.topic = new Topic(this, 'Topic');\n    new Subscription(this, 'EmailSubscription', {\n      topic: this.topic,\n      protocol: SubscriptionProtocol.EMAIL,\n      endpoint: props.email,\n    });\n\n    const certbotRunTask = new sfn_tasks.EcsRunTask(this, 'CreateCertificate', {\n      cluster: this.cluster,\n      taskDefinition: certbotTaskDefinition,\n      launchTarget: new sfn_tasks.EcsEc2LaunchTarget(),\n      integrationPattern: sfn.IntegrationPattern.RUN_JOB,\n    });\n    certbotRunTask.addCatch(\n      new sfn_tasks.SnsPublish(this, 'SendEmailOnFailure', {\n        topic: this.topic,\n        message: sfn.TaskInput.fromJsonPathAt('$'),\n      }).next(new sfn.Fail(this, 'Fail'))\n    );\n    certbotRunTask.addRetry({\n      interval: lib.Duration.seconds(20),\n    });\n    const certbotStateMachine = new sfn.StateMachine(this, 'StateMachine', {\n      definition: certbotRunTask,\n    });\n\n    new Rule(this, 'CertbotScheduleRule', {\n      schedule: Schedule.rate(lib.Duration.days(props.certbotScheduleInterval ?? 60)),\n      targets: [new SfnStateMachine(certbotStateMachine)],\n    });\n\n    /**\n     * Server ECS task\n     */\n    this.serverTaskDefinition = this.createTaskDefinition(\n      props.serverTaskDefinition ?? this.sampleTaskDefinition(records, logGroup)\n    );\n\n    if (!this.serverTaskDefinition.defaultContainer) {\n      throw new Error(\n        'defaultContainer is required for serverTaskDefinition. Add at least one essential container.'\n      );\n    }\n\n    this.certFileSystem.grant(this.serverTaskDefinition.taskRole, 'elasticfilesystem:ClientMount');\n    this.serverTaskDefinition.addVolume({\n      name: 'certVolume',\n      efsVolumeConfiguration: {\n        fileSystemId: this.certFileSystem.fileSystemId,\n      },\n    });\n    this.serverTaskDefinition.defaultContainer.addMountPoints({\n      sourceVolume: 'certVolume',\n      containerPath: '/etc/letsencrypt',\n      readOnly: true,\n    });\n\n    /**\n     * AWS cli container to execute certbot sfn before the default container startup.\n     */\n    this.serverTaskDefinition.defaultContainer.addContainerDependencies({\n      container: this.serverTaskDefinition.addContainer('AWSCliContainer', {\n        image: ecs.ContainerImage.fromRegistry(`amazon/aws-cli:${awsCliTag}`),\n        containerName: 'aws-cli',\n        memoryReservationMiB: 64,\n        entryPoint: ['/bin/bash', '-c'],\n        command: [\n          `set -eux\n          aws configure set region ${certbotStateMachine.env.region} && \\\\\n          aws configure set output text && \\\\\n          EXECUTION_ARN=$(aws stepfunctions start-execution --state-machine-arn ${certbotStateMachine.stateMachineArn} --query executionArn) && \\\\\n          until [ $(aws stepfunctions describe-execution --execution-arn \"$EXECUTION_ARN\" --query status) != RUNNING ];\n          do\n            echo \"Waiting for $EXECUTION_ARN\"\n            sleep 10\n          done`,\n        ],\n        essential: false,\n        logging: ecs.LogDriver.awsLogs({\n          logGroup: logGroup,\n          streamPrefix: awsCliTag,\n        }),\n      }),\n      condition: ecs.ContainerDependencyCondition.COMPLETE,\n    });\n    certbotStateMachine.grantExecution(\n      this.serverTaskDefinition.taskRole,\n      'states:DescribeExecution'\n    );\n    certbotStateMachine.grantStartExecution(this.serverTaskDefinition.taskRole);\n\n    this.service = new ecs.Ec2Service(this, 'Service', {\n      cluster: this.cluster,\n      taskDefinition: this.serverTaskDefinition,\n      desiredCount: 1,\n      minHealthyPercent: 0,\n      maxHealthyPercent: 100,\n      circuitBreaker: {\n        rollback: true,\n      },\n      enableExecuteCommand: true,\n    });\n\n    new lib.CfnOutput(this, 'PublicIpAddress', { value: hostInstanceIp.ref });\n    new lib.CfnOutput(this, 'StateMachineName', { value: certbotStateMachine.stateMachineName });\n    new lib.CfnOutput(this, 'ClusterName', { value: this.cluster.clusterName });\n    new lib.CfnOutput(this, 'ServiceName', { value: this.service.serviceName });\n  }\n\n  private createTaskDefinition(\n    taskDefinitionOptions: LowCostECSTaskDefinitionOptions\n  ): ecs.Ec2TaskDefinition {\n    const serverTaskDefinition = new ecs.Ec2TaskDefinition(\n      this,\n      'ServerTaskDefinition',\n      taskDefinitionOptions.taskDefinition\n    );\n    taskDefinitionOptions.containers.forEach((containerDefinition, index) => {\n      serverTaskDefinition.addContainer(\n        containerDefinition.containerName ?? `container${index}`,\n        containerDefinition\n      );\n    });\n    taskDefinitionOptions.volumes?.forEach((volume) => serverTaskDefinition.addVolume(volume));\n    return serverTaskDefinition;\n  }\n\n  private sampleTaskDefinition(\n    records: string[],\n    logGroup: ILogGroup\n  ): LowCostECSTaskDefinitionOptions {\n    return {\n      containers: [\n        {\n          image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../examples/containers/nginx')),\n          containerName: 'nginx',\n          memoryReservationMiB: 64,\n          essential: true,\n          environment: {\n            SERVER_NAME: records.join(' '),\n            CERT_NAME: records[0],\n          },\n          logging: ecs.LogDrivers.awsLogs({\n            logGroup: logGroup,\n            streamPrefix: 'sample',\n          }),\n          portMappings: [\n            {\n              hostPort: 80,\n              containerPort: 80,\n              protocol: ecs.Protocol.TCP,\n            },\n            {\n              hostPort: 443,\n              containerPort: 443,\n              protocol: ecs.Protocol.TCP,\n            },\n          ],\n        },\n      ],\n    };\n  }\n}\n"]}
package/package.json CHANGED
@@ -41,7 +41,7 @@
41
41
  "@types/node": "^14",
42
42
  "@typescript-eslint/eslint-plugin": "^5",
43
43
  "@typescript-eslint/parser": "^5",
44
- "aws-cdk": "^2.64.0",
44
+ "aws-cdk": "^2.65.0",
45
45
  "aws-cdk-lib": "2.37.0",
46
46
  "constructs": "10.0.5",
47
47
  "eslint": "^8",
@@ -52,13 +52,13 @@
52
52
  "eslint-plugin-prettier": "^4.2.1",
53
53
  "jest": "^27",
54
54
  "jest-junit": "^13",
55
- "jsii": "^1.74.0",
56
- "jsii-diff": "^1.74.0",
57
- "jsii-docgen": "^7.0.231",
58
- "jsii-pacmak": "^1.74.0",
55
+ "jsii": "^1.75.0",
56
+ "jsii-diff": "^1.75.0",
57
+ "jsii-docgen": "^7.1.2",
58
+ "jsii-pacmak": "^1.75.0",
59
59
  "npm-check-updates": "^16",
60
60
  "prettier": "^2.8.4",
61
- "projen": "^0.67.50",
61
+ "projen": "^0.67.59",
62
62
  "standard-version": "^9",
63
63
  "ts-jest": "^27",
64
64
  "ts-node": "^10.9.1",
@@ -78,7 +78,7 @@
78
78
  ],
79
79
  "main": "lib/index.js",
80
80
  "license": "MIT",
81
- "version": "0.0.48",
81
+ "version": "0.0.49",
82
82
  "jest": {
83
83
  "testMatch": [
84
84
  "<rootDir>/src/**/__tests__/**/*.ts?(x)",