@liflig/cdk 3.12.0 → 3.13.0
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/assets/open-telemetry/otel-collector-task-metrics-config.yaml +310 -0
- package/lib/build-artifacts/index.d.ts +1 -1
- package/lib/build-artifacts/index.js +5 -1
- package/lib/ecs/index.d.ts +2 -0
- package/lib/ecs/index.js +2 -1
- package/lib/ecs/open-telemetry.d.ts +98 -0
- package/lib/ecs/open-telemetry.js +211 -0
- package/package.json +4 -4
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
## Validate using https://www.otelbin.io/#distro=adot%7E&distroVersion=v0.43.1%7E .
|
|
2
|
+
##
|
|
3
|
+
## Original copyright is Apache 2.0 to AWS (aws-observability on GitHub).
|
|
4
|
+
## https://github.com/aws-observability/aws-otel-collector/blob/14833d4af543da709c77cf9dc6827351dbd529b1/config/ecs/container-insights/otel-task-metrics-config.yaml
|
|
5
|
+
##
|
|
6
|
+
## This is a modified version of `./etc/ecs/container-insights/otel-task-metrics-config.yaml`
|
|
7
|
+
## where we add `resource_to_telemetry_conversion` so the otel resource `service.name`
|
|
8
|
+
## can be added to the metric. This is useful because you can filter in metrics based on service,
|
|
9
|
+
## instead of grouping e.g. all services' memory usage under the same metric.
|
|
10
|
+
##
|
|
11
|
+
## Search for `# THIS IS WHAT LIFLIG ADDED` to find our modifications.
|
|
12
|
+
##
|
|
13
|
+
##
|
|
14
|
+
## @see https://aws-otel.github.io/docs/setup/ecs/config-through-ssm
|
|
15
|
+
extensions:
|
|
16
|
+
health_check:
|
|
17
|
+
|
|
18
|
+
## These ports are also set in `taskDefinition.defaultContainer` as
|
|
19
|
+
## `OTEL_EXPORTER_OTLP_ENDPOINT` and so on.
|
|
20
|
+
## Changes must happen in both places.
|
|
21
|
+
receivers:
|
|
22
|
+
otlp:
|
|
23
|
+
protocols:
|
|
24
|
+
grpc:
|
|
25
|
+
endpoint: 0.0.0.0:4317
|
|
26
|
+
http:
|
|
27
|
+
endpoint: 0.0.0.0:4318
|
|
28
|
+
awsxray:
|
|
29
|
+
endpoint: 0.0.0.0:2000
|
|
30
|
+
transport: udp
|
|
31
|
+
awsecscontainermetrics:
|
|
32
|
+
|
|
33
|
+
processors:
|
|
34
|
+
batch/traces:
|
|
35
|
+
timeout: 1s
|
|
36
|
+
send_batch_size: 50
|
|
37
|
+
batch/metrics:
|
|
38
|
+
timeout: 60s
|
|
39
|
+
filter:
|
|
40
|
+
metrics:
|
|
41
|
+
include:
|
|
42
|
+
match_type: strict
|
|
43
|
+
metric_names:
|
|
44
|
+
- ecs.task.memory.reserved
|
|
45
|
+
- ecs.task.memory.utilized
|
|
46
|
+
- ecs.task.cpu.reserved
|
|
47
|
+
- ecs.task.cpu.utilized
|
|
48
|
+
- ecs.task.network.rate.rx
|
|
49
|
+
- ecs.task.network.rate.tx
|
|
50
|
+
- ecs.task.storage.read_bytes
|
|
51
|
+
- ecs.task.storage.write_bytes
|
|
52
|
+
- container.duration
|
|
53
|
+
metricstransform:
|
|
54
|
+
transforms:
|
|
55
|
+
- include: ecs.task.memory.utilized
|
|
56
|
+
action: update
|
|
57
|
+
new_name: MemoryUtilized
|
|
58
|
+
- include: ecs.task.memory.reserved
|
|
59
|
+
action: update
|
|
60
|
+
new_name: MemoryReserved
|
|
61
|
+
- include: ecs.task.cpu.utilized
|
|
62
|
+
action: update
|
|
63
|
+
new_name: CpuUtilized
|
|
64
|
+
- include: ecs.task.cpu.reserved
|
|
65
|
+
action: update
|
|
66
|
+
new_name: CpuReserved
|
|
67
|
+
- include: ecs.task.network.rate.rx
|
|
68
|
+
action: update
|
|
69
|
+
new_name: NetworkRxBytes
|
|
70
|
+
- include: ecs.task.network.rate.tx
|
|
71
|
+
action: update
|
|
72
|
+
new_name: NetworkTxBytes
|
|
73
|
+
- include: ecs.task.storage.read_bytes
|
|
74
|
+
action: update
|
|
75
|
+
new_name: StorageReadBytes
|
|
76
|
+
- include: ecs.task.storage.write_bytes
|
|
77
|
+
action: update
|
|
78
|
+
new_name: StorageWriteBytes
|
|
79
|
+
## THIS IS WHAT LIFLIG ADDED
|
|
80
|
+
filter/application:
|
|
81
|
+
error_mode: ignore
|
|
82
|
+
metrics:
|
|
83
|
+
metric:
|
|
84
|
+
- 'resource.attributes["http.target"] == "/health"'
|
|
85
|
+
- 'resource.attributes["net.peer.name"] == "169.254.170.2"' # EC2 Metadata Endpoint
|
|
86
|
+
filter/traces:
|
|
87
|
+
error_mode: ignore
|
|
88
|
+
traces:
|
|
89
|
+
span:
|
|
90
|
+
- 'attributes["http.route"] == "health"'
|
|
91
|
+
- 'attributes["http.url"] == "/health"'
|
|
92
|
+
- 'name == "GET health"'
|
|
93
|
+
attributes/application:
|
|
94
|
+
actions:
|
|
95
|
+
# Http adds lots of high cardinality, which xray creates 1 metric per. Expensive.
|
|
96
|
+
- key: http.url
|
|
97
|
+
action: delete
|
|
98
|
+
- key: http.target
|
|
99
|
+
action: delete
|
|
100
|
+
- key: http.client_ip
|
|
101
|
+
action: delete
|
|
102
|
+
- key: http.response_content_length
|
|
103
|
+
action: delete
|
|
104
|
+
- key: http.request_content_length
|
|
105
|
+
action: delete
|
|
106
|
+
- key: net.host.name
|
|
107
|
+
action: delete
|
|
108
|
+
- key: net.sock.peer.addr
|
|
109
|
+
action: delete
|
|
110
|
+
- key: net.sock.peer.port
|
|
111
|
+
action: delete
|
|
112
|
+
- key: net.sock.host.addr
|
|
113
|
+
action: delete
|
|
114
|
+
- key: net.sock.host.port
|
|
115
|
+
action: delete
|
|
116
|
+
- key: user_agent.original
|
|
117
|
+
action: delete
|
|
118
|
+
resource/application:
|
|
119
|
+
attributes:
|
|
120
|
+
- key: cloud.provider
|
|
121
|
+
action: delete
|
|
122
|
+
- key: host.arch
|
|
123
|
+
action: delete
|
|
124
|
+
- key: aws.ecs.container.image.id
|
|
125
|
+
action: delete
|
|
126
|
+
- key: aws.ecs.task.arn
|
|
127
|
+
action: delete
|
|
128
|
+
- key: aws.log.stream.names
|
|
129
|
+
action: delete
|
|
130
|
+
- key: aws.log.stream.arns
|
|
131
|
+
action: delete
|
|
132
|
+
- key: cloud.platform
|
|
133
|
+
action: delete
|
|
134
|
+
- key: container.name
|
|
135
|
+
action: delete
|
|
136
|
+
- key: process.executable.path
|
|
137
|
+
action: delete
|
|
138
|
+
- key: process.runtime.version
|
|
139
|
+
action: delete
|
|
140
|
+
- key: telemetry.auto.version
|
|
141
|
+
action: delete
|
|
142
|
+
- key: telemetry.sdk.name
|
|
143
|
+
action: delete
|
|
144
|
+
- key: container.id
|
|
145
|
+
action: delete
|
|
146
|
+
- key: container.image.tag
|
|
147
|
+
action: delete
|
|
148
|
+
- key: process.runtime.name
|
|
149
|
+
action: delete
|
|
150
|
+
- key: service.namespace
|
|
151
|
+
action: delete
|
|
152
|
+
- key: telemetry.sdk.version
|
|
153
|
+
action: delete
|
|
154
|
+
- key: aws.ecs.task.family
|
|
155
|
+
action: delete
|
|
156
|
+
- key: aws.ecs.task.revision
|
|
157
|
+
action: delete
|
|
158
|
+
- key: aws.log.group.arns
|
|
159
|
+
action: delete
|
|
160
|
+
- key: container.image.name
|
|
161
|
+
action: delete
|
|
162
|
+
- key: os.description
|
|
163
|
+
action: delete
|
|
164
|
+
- key: os.type
|
|
165
|
+
action: delete
|
|
166
|
+
- key: process.runtime.description
|
|
167
|
+
action: delete
|
|
168
|
+
- key: version
|
|
169
|
+
action: delete
|
|
170
|
+
- key: aws.ecs.launchtype
|
|
171
|
+
action: delete
|
|
172
|
+
- key: aws.log.group.names
|
|
173
|
+
action: delete
|
|
174
|
+
- key: host.name
|
|
175
|
+
action: delete
|
|
176
|
+
- key: process.pid
|
|
177
|
+
action: delete
|
|
178
|
+
- key: telemetry.sdk.language
|
|
179
|
+
action: delete
|
|
180
|
+
- key: aws.ecs.container.arn
|
|
181
|
+
action: delete
|
|
182
|
+
- key: service.version
|
|
183
|
+
action: delete
|
|
184
|
+
- key: process.command_args
|
|
185
|
+
action: delete
|
|
186
|
+
- key: service
|
|
187
|
+
action: delete
|
|
188
|
+
## END OF LIFLIG CHANGES
|
|
189
|
+
resource:
|
|
190
|
+
attributes:
|
|
191
|
+
- key: ClusterName
|
|
192
|
+
from_attribute: aws.ecs.cluster.name
|
|
193
|
+
action: insert
|
|
194
|
+
- key: aws.ecs.cluster.name
|
|
195
|
+
action: delete
|
|
196
|
+
- key: ServiceName
|
|
197
|
+
from_attribute: aws.ecs.service.name
|
|
198
|
+
action: insert
|
|
199
|
+
- key: aws.ecs.service.name
|
|
200
|
+
action: delete
|
|
201
|
+
## THIS IS WHAT LIFLIG ADDED (bunch of sporadic removals)
|
|
202
|
+
- key: aws.ecs.task.id
|
|
203
|
+
action: delete
|
|
204
|
+
- key: TaskDefinitionFamily
|
|
205
|
+
from_attribute: aws.ecs.task.family
|
|
206
|
+
action: insert
|
|
207
|
+
- key: aws.ecs.task.family
|
|
208
|
+
action: delete
|
|
209
|
+
- key: aws.ecs.task.arn
|
|
210
|
+
action: delete
|
|
211
|
+
- key: aws.ecs.docker.name
|
|
212
|
+
action: delete
|
|
213
|
+
- key: aws.ecs.task.version
|
|
214
|
+
action: delete
|
|
215
|
+
- key: aws.ecs.task.pull_started_at
|
|
216
|
+
action: delete
|
|
217
|
+
- key: aws.ecs.task.pull_stopped_at
|
|
218
|
+
action: delete
|
|
219
|
+
- key: AvailabilityZone
|
|
220
|
+
from_attribute: cloud.zone
|
|
221
|
+
action: insert
|
|
222
|
+
- key: cloud.zone
|
|
223
|
+
action: delete
|
|
224
|
+
- key: aws.ecs.task.launch_type
|
|
225
|
+
action: delete
|
|
226
|
+
- key: Region
|
|
227
|
+
from_attribute: cloud.region
|
|
228
|
+
action: insert
|
|
229
|
+
- key: cloud.region
|
|
230
|
+
action: delete
|
|
231
|
+
- key: AccountId
|
|
232
|
+
from_attribute: cloud.account.id
|
|
233
|
+
action: insert
|
|
234
|
+
- key: cloud.account.id
|
|
235
|
+
action: delete
|
|
236
|
+
- key: container.id
|
|
237
|
+
action: delete
|
|
238
|
+
- key: container.name
|
|
239
|
+
action: delete
|
|
240
|
+
- key: container.image.name
|
|
241
|
+
action: delete
|
|
242
|
+
- key: aws.ecs.container.image.id
|
|
243
|
+
action: delete
|
|
244
|
+
- key: aws.ecs.container.exit_code
|
|
245
|
+
action: delete
|
|
246
|
+
- key: aws.ecs.container.created_at
|
|
247
|
+
action: delete
|
|
248
|
+
- key: aws.ecs.container.started_at
|
|
249
|
+
action: delete
|
|
250
|
+
- key: aws.ecs.container.finished_at
|
|
251
|
+
action: delete
|
|
252
|
+
- key: container.image.tag
|
|
253
|
+
action: delete
|
|
254
|
+
## END OF LIFLIG CHANGES
|
|
255
|
+
|
|
256
|
+
exporters:
|
|
257
|
+
awsxray:
|
|
258
|
+
## THIS IS WHAT LIFLIG ADDED
|
|
259
|
+
indexed_attributes: ["otel.resource.service.name"] # Max 50 attributes
|
|
260
|
+
## END OF LIFLIG CHANGES
|
|
261
|
+
awsemf/application:
|
|
262
|
+
namespace: ECS/AWSOTel/Application
|
|
263
|
+
log_group_name: '/aws/ecs/application/metrics'
|
|
264
|
+
## THIS IS WHAT LIFLIG ADDED
|
|
265
|
+
## Config docs at https://github.com/open-telemetry/opentelemetry-collector-contrib/blob/main/exporter/awsemfexporter/README.md
|
|
266
|
+
## log retention is days to keep the emf metric logs.
|
|
267
|
+
log_retention: 365
|
|
268
|
+
resource_to_telemetry_conversion:
|
|
269
|
+
enabled: true
|
|
270
|
+
dimension_rollup_option: NoDimensionRollup
|
|
271
|
+
## No rollup means we keep all the dimensions on the metric, instead of merging similar metrics to one in CloudWatch
|
|
272
|
+
## END OF LIFLIG CHANGES
|
|
273
|
+
awsemf/performance:
|
|
274
|
+
namespace: ECS/ContainerInsights
|
|
275
|
+
log_group_name: '/aws/ecs/containerinsights/{ClusterName}/performance'
|
|
276
|
+
log_stream_name: '{TaskId}'
|
|
277
|
+
resource_to_telemetry_conversion:
|
|
278
|
+
enabled: true
|
|
279
|
+
dimension_rollup_option: NoDimensionRollup
|
|
280
|
+
metric_declarations:
|
|
281
|
+
- dimensions: [ [ ClusterName ], [ ClusterName, TaskDefinitionFamily ] ]
|
|
282
|
+
metric_name_selectors:
|
|
283
|
+
- MemoryUtilized
|
|
284
|
+
- MemoryReserved
|
|
285
|
+
- CpuUtilized
|
|
286
|
+
- CpuReserved
|
|
287
|
+
- NetworkRxBytes
|
|
288
|
+
- NetworkTxBytes
|
|
289
|
+
- StorageReadBytes
|
|
290
|
+
- StorageWriteBytes
|
|
291
|
+
- metric_name_selectors: [container.*]
|
|
292
|
+
|
|
293
|
+
service:
|
|
294
|
+
pipelines:
|
|
295
|
+
traces:
|
|
296
|
+
receivers: [otlp,awsxray]
|
|
297
|
+
## THIS IS WHAT LIFLIG ADDED
|
|
298
|
+
processors: [filter/traces, batch/traces]
|
|
299
|
+
exporters: [awsxray]
|
|
300
|
+
metrics/application:
|
|
301
|
+
receivers: [otlp]
|
|
302
|
+
processors: [filter/application, resource/application, attributes/application, batch/metrics]
|
|
303
|
+
exporters: [awsemf/application]
|
|
304
|
+
## END OF LIFLIG CHANGES
|
|
305
|
+
metrics/performance:
|
|
306
|
+
receivers: [awsecscontainermetrics ]
|
|
307
|
+
processors: [filter, metricstransform, resource]
|
|
308
|
+
exporters: [ awsemf/performance ]
|
|
309
|
+
|
|
310
|
+
extensions: [health_check]
|
|
@@ -24,7 +24,7 @@ interface Props {
|
|
|
24
24
|
/**
|
|
25
25
|
* The lifecycle rules to apply to images stored in the ECR repository.
|
|
26
26
|
*
|
|
27
|
-
* @default - Expire images after 180 days
|
|
27
|
+
* @default - Expire untagged images after 10 days and any image older than 180 days
|
|
28
28
|
*/
|
|
29
29
|
ecrRepositoryLifecycleRules?: ecr.LifecycleRule[];
|
|
30
30
|
/**
|
|
@@ -90,6 +90,10 @@ export class BuildArtifacts extends constructs.Construct {
|
|
|
90
90
|
const ecrRepo = new ecr.Repository(this, "EcrRepository", {
|
|
91
91
|
repositoryName: ecrRepositoryName,
|
|
92
92
|
lifecycleRules: props.ecrRepositoryLifecycleRules || [
|
|
93
|
+
{
|
|
94
|
+
maxImageAge: cdk.Duration.days(10),
|
|
95
|
+
tagStatus: ecr.TagStatus.UNTAGGED,
|
|
96
|
+
},
|
|
93
97
|
{
|
|
94
98
|
maxImageAge: cdk.Duration.days(180),
|
|
95
99
|
tagStatus: ecr.TagStatus.ANY,
|
|
@@ -181,4 +185,4 @@ export class BuildArtifacts extends constructs.Construct {
|
|
|
181
185
|
}
|
|
182
186
|
}
|
|
183
187
|
}
|
|
184
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/build-artifacts/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ecr from \"aws-cdk-lib/aws-ecr\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport {\n  GithubActionsRole,\n  Props as GithubActionsRoleProps,\n} from \"./github-actions-role\"\n\ninterface Props {\n  /**\n   * The name to use for the S3 Bucket.\n   *\n   * NOTE: Should include both account and region so that it will\n   * not conflict with other accounts/regions.\n   */\n  bucketName: string\n  /**\n   * The name to use for the ECR Repository.\n   */\n  ecrRepositoryName: string\n  /**\n   * Create a role that can be assumed by Liflig Jenkins.\n   *\n   * @deprecated\n   * @default - no role will be created\n   */\n  ciRoleName?: string\n  /**\n   * The lifecycle rules to apply to images stored in the ECR repository.\n   *\n   * @default - Expire images after 180 days\n   */\n  ecrRepositoryLifecycleRules?: ecr.LifecycleRule[]\n  /**\n   * ID of AWS accounts that will be granted permission to read from\n   * the artifact repos.\n   */\n  targetAccountIds?: string[]\n  /**\n   * Configuration for creating IAM roles that can be assumed\n   * by GitHub Actions in specific GitHub repositories.\n   *\n   * @default - no roles are created.\n   */\n  githubActions?: Omit<\n    GithubActionsRoleProps,\n    \"trustedOwners\" | \"trustedBranch\" | \"oidcProvider\" | \"roleName\"\n  > & {\n    /**\n     * A list of trusted GitHub repository owners.\n     *\n     * @default [\"capralifecycle\"]\n     */\n    trustedOwners?: string[]\n    /**\n     * The name of the role to create.\n     *\n     * @default \"github-actions-role\"\n     */\n    roleName?: string\n    /**\n     * The name of the default branch in all repositories.\n     *\n     * @default \"master\"\n     */\n    defaultBranch?: string\n    /**\n     * Configuration for a limited role that can be used on non-default\n     * branches with less IAM permissions than the main role.\n     *\n     * @default - a limited role is created.\n     */\n    limitedRoleConfiguration?: {\n      /**\n       * Whether to create the role or not.\n       *\n       * @default true\n       */\n      enabled?: boolean\n      /**\n       * The name of the role to create.\n       *\n       * @default \"github-actions-limited-role\"\n       */\n      roleName?: string\n    }\n  }\n}\n\n/**\n * Utility functions for generating IAM statements.\n */\nconst policyStatements = {\n  allowPipelineVariables: (scope: constructs.Construct, prefix?: string) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: [\"ssm:PutParameter\"],\n      resources: [\n        `arn:aws:ssm:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:parameter/liflig-cdk/*/pipeline-variables/${prefix ?? \"*\"}`,\n      ],\n    }),\n  denyProdPipelines: (scope: constructs.Construct, bucket: s3.IBucket) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.DENY,\n      actions: [\"s3:*\"],\n      resources: [\n        bucket.arnForObjects(\"pipelines/*-prod/*\"),\n        bucket.arnForObjects(\"pipelines/*-prod-*/*\"),\n      ],\n    }),\n  denyProdPipelineVariables: (scope: constructs.Construct) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.DENY,\n      actions: [\"ssm:PutParameter\"],\n      resources: [\n        `arn:aws:ssm:${cdk.Stack.of(scope).region}:${\n          cdk.Stack.of(scope).account\n        }:parameter/liflig-cdk/*/pipeline-variables/prod*`,\n      ],\n    }),\n}\n\n/**\n * Utility function for validating the construct properties.\n */\nexport const validateProps = (props: Props) => {\n  let valid = true\n  if (\n    props.githubActions?.defaultBranch &&\n    !props.githubActions.defaultBranch.match(/^[a-zA-Z0-9-]+$/)\n  ) {\n    console.error(\n      `Default branch ${props.githubActions.defaultBranch} contains invalid characters`,\n    )\n    valid = false\n  }\n  return valid\n}\n\n/**\n * Create various artifacts used as part of Continuous Integration (CI).\n *\n * This includes artifact repositories such as S3 bucket and ECR\n * repository, as well as IAM role(s) that grant the CI server write\n * access to these repositories.\n *\n * TODO: How can we cleanup stuff that goes into this S3 Bucket and\n *  ECR Repository? Can we ever reliably cleanup? We probably need\n *  some strategy for how we put stuff here to be able to do it.\n *\n * @experimental\n */\nexport class BuildArtifacts extends constructs.Construct {\n  public readonly bucketName: string | undefined\n  public readonly ecrRepositoryArn: string\n  public readonly ecrRepositoryName: string\n  public readonly role?: iam.Role\n  public readonly limitedRole?: iam.Role\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    if (!validateProps(props)) {\n      throw new Error(\"Invalid props were supplied\")\n    }\n\n    this.bucketName = props.bucketName\n    this.ecrRepositoryName = props.ecrRepositoryName\n    this.ecrRepositoryArn = cdk.Arn.format(\n      {\n        service: \"ecr\",\n        resource: \"repository\",\n        resourceName: this.ecrRepositoryName,\n      },\n      cdk.Stack.of(this),\n    )\n\n    const ecrRepositoryName = props.ecrRepositoryName\n\n    const bucket = new s3.Bucket(this, \"S3Bucket\", {\n      bucketName: props.bucketName,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      eventBridgeEnabled: true,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      versioned: true,\n      lifecycleRules: [\n        {\n          noncurrentVersionExpiration: cdk.Duration.days(10),\n        },\n      ],\n    })\n\n    const ecrRepo = new ecr.Repository(this, \"EcrRepository\", {\n      repositoryName: ecrRepositoryName,\n      lifecycleRules: props.ecrRepositoryLifecycleRules || [\n        {\n          maxImageAge: cdk.Duration.days(180),\n          tagStatus: ecr.TagStatus.ANY,\n        },\n      ],\n    })\n\n    // Allow a target to read from the repos. As any specific roles need\n    // to exist before we can grant access, we delegate that responsibility\n    // to the target account.\n    for (const targetAccountId of props.targetAccountIds || []) {\n      bucket.grantRead(new iam.AccountPrincipal(targetAccountId))\n      ecrRepo.grantPull(new iam.AccountPrincipal(targetAccountId))\n    }\n\n    if (props.githubActions) {\n      const trustedOwners = props.githubActions.trustedOwners ?? [\n        \"capralifecycle\",\n      ]\n      // NOTE: There's an L2 construct that predates the official CloudFormation resource,\n      // and it is therefore implemented as a custom resource.\n      // We use the L1 CloudFormation resource instead because:\n      // 1) it's simpler\n      // 2) there's a chance the L2 construct will be deprecated in the future in favor\n      // of the official CloudFormation resource.\n      const cfnOidcProvider = new iam.CfnOIDCProvider(\n        this,\n        \"OpenIdConnectProvider\",\n        {\n          url: \"https://token.actions.githubusercontent.com\",\n          clientIdList: [\"sts.amazonaws.com\"],\n          // NOTE: The thumbprint isn't actually used, but the AWS API still requires us to supply it.\n          // More details here: https://web.archive.org/web/20240122155758/https://github.com/aws-actions/configure-aws-credentials/issues/357#issuecomment-1626357333\n          thumbprintList: [\"1c58a3a8518e8759bf075b76b750d4f2df264fcd\"],\n        },\n      )\n      const oidcProvider =\n        iam.OpenIdConnectProvider.fromOpenIdConnectProviderArn(\n          this,\n          \"Provider\",\n          cfnOidcProvider.ref,\n        )\n      const createLimitedRole =\n        props.githubActions.limitedRoleConfiguration?.enabled ?? true\n      const role = new GithubActionsRole(this, \"GithubActionsRole\", {\n        oidcProvider,\n        trustedOwners,\n        roleName: props.githubActions.roleName ?? \"github-actions-role\",\n        trustedBranch: createLimitedRole\n          ? (props.githubActions.defaultBranch ?? \"master\")\n          : // NOTE: If the limited role is disabled, only one role will be created\n            // and that role then needs to be assumable from all branches.\n            \"*\",\n        repositories: props.githubActions.repositories,\n      })\n      this.role = role.role\n      this.role.addToPolicy(policyStatements.allowPipelineVariables(this))\n      bucket.grantPut(this.role)\n      ecrRepo.grantPullPush(this.role)\n      if (createLimitedRole) {\n        const limitedRole = new GithubActionsRole(\n          this,\n          \"GithubActionsLimitedRole\",\n          {\n            oidcProvider,\n            trustedOwners,\n            roleName:\n              props.githubActions.limitedRoleConfiguration?.roleName ??\n              \"github-actions-limited-role\",\n            // NOTE: The limited role can be assumed from all branches.\n            trustedBranch: \"*\",\n            repositories: props.githubActions.repositories,\n          },\n        )\n        this.limitedRole = limitedRole.role\n        this.limitedRole.addToPolicy(\n          policyStatements.allowPipelineVariables(this, \"dev*\"),\n        )\n        this.limitedRole.addToPolicy(\n          policyStatements.denyProdPipelines(scope, bucket),\n        )\n        bucket.grantPut(this.limitedRole)\n        ecrRepo.grantPullPush(this.limitedRole)\n      }\n    }\n    if (props.ciRoleName) {\n      const legacyRoleForJenkins = new iam.Role(this, \"CiRole\", {\n        roleName: props.ciRoleName,\n        assumedBy: new iam.ArnPrincipal(\n          \"arn:aws:iam::923402097046:role/buildtools-jenkins-RoleJenkinsSlave-JQGYHR5WE6C5\",\n        ),\n      })\n      legacyRoleForJenkins.addToPolicy(\n        policyStatements.allowPipelineVariables(this),\n      )\n      bucket.grantPut(legacyRoleForJenkins)\n      ecrRepo.grantPullPush(legacyRoleForJenkins)\n    }\n\n    new cdk.CfnOutput(this, \"EcrRepoUri\", {\n      value: ecrRepo.repositoryUri,\n    })\n\n    new cdk.CfnOutput(this, \"BucketName\", {\n      value: bucket.bucketName,\n    })\n\n    if (this.role) {\n      new cdk.CfnOutput(this, \"RoleArn\", {\n        value: this.role.roleArn,\n      })\n    }\n    if (this.limitedRole) {\n      new cdk.CfnOutput(this, \"LimitedRoleArn\", {\n        value: this.limitedRole.roleArn,\n      })\n    }\n  }\n}\n"]}
|
|
188
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/build-artifacts/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ecr from \"aws-cdk-lib/aws-ecr\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport {\n  GithubActionsRole,\n  Props as GithubActionsRoleProps,\n} from \"./github-actions-role\"\n\ninterface Props {\n  /**\n   * The name to use for the S3 Bucket.\n   *\n   * NOTE: Should include both account and region so that it will\n   * not conflict with other accounts/regions.\n   */\n  bucketName: string\n  /**\n   * The name to use for the ECR Repository.\n   */\n  ecrRepositoryName: string\n  /**\n   * Create a role that can be assumed by Liflig Jenkins.\n   *\n   * @deprecated\n   * @default - no role will be created\n   */\n  ciRoleName?: string\n  /**\n   * The lifecycle rules to apply to images stored in the ECR repository.\n   *\n   * @default - Expire untagged images after 10 days and any image older than 180 days\n   */\n  ecrRepositoryLifecycleRules?: ecr.LifecycleRule[]\n  /**\n   * ID of AWS accounts that will be granted permission to read from\n   * the artifact repos.\n   */\n  targetAccountIds?: string[]\n  /**\n   * Configuration for creating IAM roles that can be assumed\n   * by GitHub Actions in specific GitHub repositories.\n   *\n   * @default - no roles are created.\n   */\n  githubActions?: Omit<\n    GithubActionsRoleProps,\n    \"trustedOwners\" | \"trustedBranch\" | \"oidcProvider\" | \"roleName\"\n  > & {\n    /**\n     * A list of trusted GitHub repository owners.\n     *\n     * @default [\"capralifecycle\"]\n     */\n    trustedOwners?: string[]\n    /**\n     * The name of the role to create.\n     *\n     * @default \"github-actions-role\"\n     */\n    roleName?: string\n    /**\n     * The name of the default branch in all repositories.\n     *\n     * @default \"master\"\n     */\n    defaultBranch?: string\n    /**\n     * Configuration for a limited role that can be used on non-default\n     * branches with less IAM permissions than the main role.\n     *\n     * @default - a limited role is created.\n     */\n    limitedRoleConfiguration?: {\n      /**\n       * Whether to create the role or not.\n       *\n       * @default true\n       */\n      enabled?: boolean\n      /**\n       * The name of the role to create.\n       *\n       * @default \"github-actions-limited-role\"\n       */\n      roleName?: string\n    }\n  }\n}\n\n/**\n * Utility functions for generating IAM statements.\n */\nconst policyStatements = {\n  allowPipelineVariables: (scope: constructs.Construct, prefix?: string) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      actions: [\"ssm:PutParameter\"],\n      resources: [\n        `arn:aws:ssm:${cdk.Stack.of(scope).region}:${cdk.Stack.of(scope).account}:parameter/liflig-cdk/*/pipeline-variables/${prefix ?? \"*\"}`,\n      ],\n    }),\n  denyProdPipelines: (scope: constructs.Construct, bucket: s3.IBucket) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.DENY,\n      actions: [\"s3:*\"],\n      resources: [\n        bucket.arnForObjects(\"pipelines/*-prod/*\"),\n        bucket.arnForObjects(\"pipelines/*-prod-*/*\"),\n      ],\n    }),\n  denyProdPipelineVariables: (scope: constructs.Construct) =>\n    new iam.PolicyStatement({\n      effect: iam.Effect.DENY,\n      actions: [\"ssm:PutParameter\"],\n      resources: [\n        `arn:aws:ssm:${cdk.Stack.of(scope).region}:${\n          cdk.Stack.of(scope).account\n        }:parameter/liflig-cdk/*/pipeline-variables/prod*`,\n      ],\n    }),\n}\n\n/**\n * Utility function for validating the construct properties.\n */\nexport const validateProps = (props: Props) => {\n  let valid = true\n  if (\n    props.githubActions?.defaultBranch &&\n    !props.githubActions.defaultBranch.match(/^[a-zA-Z0-9-]+$/)\n  ) {\n    console.error(\n      `Default branch ${props.githubActions.defaultBranch} contains invalid characters`,\n    )\n    valid = false\n  }\n  return valid\n}\n\n/**\n * Create various artifacts used as part of Continuous Integration (CI).\n *\n * This includes artifact repositories such as S3 bucket and ECR\n * repository, as well as IAM role(s) that grant the CI server write\n * access to these repositories.\n *\n * TODO: How can we cleanup stuff that goes into this S3 Bucket and\n *  ECR Repository? Can we ever reliably cleanup? We probably need\n *  some strategy for how we put stuff here to be able to do it.\n *\n * @experimental\n */\nexport class BuildArtifacts extends constructs.Construct {\n  public readonly bucketName: string | undefined\n  public readonly ecrRepositoryArn: string\n  public readonly ecrRepositoryName: string\n  public readonly role?: iam.Role\n  public readonly limitedRole?: iam.Role\n\n  constructor(scope: constructs.Construct, id: string, props: Props) {\n    super(scope, id)\n\n    if (!validateProps(props)) {\n      throw new Error(\"Invalid props were supplied\")\n    }\n\n    this.bucketName = props.bucketName\n    this.ecrRepositoryName = props.ecrRepositoryName\n    this.ecrRepositoryArn = cdk.Arn.format(\n      {\n        service: \"ecr\",\n        resource: \"repository\",\n        resourceName: this.ecrRepositoryName,\n      },\n      cdk.Stack.of(this),\n    )\n\n    const ecrRepositoryName = props.ecrRepositoryName\n\n    const bucket = new s3.Bucket(this, \"S3Bucket\", {\n      bucketName: props.bucketName,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      eventBridgeEnabled: true,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      versioned: true,\n      lifecycleRules: [\n        {\n          noncurrentVersionExpiration: cdk.Duration.days(10),\n        },\n      ],\n    })\n\n    const ecrRepo = new ecr.Repository(this, \"EcrRepository\", {\n      repositoryName: ecrRepositoryName,\n      lifecycleRules: props.ecrRepositoryLifecycleRules || [\n        {\n          maxImageAge: cdk.Duration.days(10),\n          tagStatus: ecr.TagStatus.UNTAGGED,\n        },\n        {\n          maxImageAge: cdk.Duration.days(180),\n          tagStatus: ecr.TagStatus.ANY,\n        },\n      ],\n    })\n\n    // Allow a target to read from the repos. As any specific roles need\n    // to exist before we can grant access, we delegate that responsibility\n    // to the target account.\n    for (const targetAccountId of props.targetAccountIds || []) {\n      bucket.grantRead(new iam.AccountPrincipal(targetAccountId))\n      ecrRepo.grantPull(new iam.AccountPrincipal(targetAccountId))\n    }\n\n    if (props.githubActions) {\n      const trustedOwners = props.githubActions.trustedOwners ?? [\n        \"capralifecycle\",\n      ]\n      // NOTE: There's an L2 construct that predates the official CloudFormation resource,\n      // and it is therefore implemented as a custom resource.\n      // We use the L1 CloudFormation resource instead because:\n      // 1) it's simpler\n      // 2) there's a chance the L2 construct will be deprecated in the future in favor\n      // of the official CloudFormation resource.\n      const cfnOidcProvider = new iam.CfnOIDCProvider(\n        this,\n        \"OpenIdConnectProvider\",\n        {\n          url: \"https://token.actions.githubusercontent.com\",\n          clientIdList: [\"sts.amazonaws.com\"],\n          // NOTE: The thumbprint isn't actually used, but the AWS API still requires us to supply it.\n          // More details here: https://web.archive.org/web/20240122155758/https://github.com/aws-actions/configure-aws-credentials/issues/357#issuecomment-1626357333\n          thumbprintList: [\"1c58a3a8518e8759bf075b76b750d4f2df264fcd\"],\n        },\n      )\n      const oidcProvider =\n        iam.OpenIdConnectProvider.fromOpenIdConnectProviderArn(\n          this,\n          \"Provider\",\n          cfnOidcProvider.ref,\n        )\n      const createLimitedRole =\n        props.githubActions.limitedRoleConfiguration?.enabled ?? true\n      const role = new GithubActionsRole(this, \"GithubActionsRole\", {\n        oidcProvider,\n        trustedOwners,\n        roleName: props.githubActions.roleName ?? \"github-actions-role\",\n        trustedBranch: createLimitedRole\n          ? (props.githubActions.defaultBranch ?? \"master\")\n          : // NOTE: If the limited role is disabled, only one role will be created\n            // and that role then needs to be assumable from all branches.\n            \"*\",\n        repositories: props.githubActions.repositories,\n      })\n      this.role = role.role\n      this.role.addToPolicy(policyStatements.allowPipelineVariables(this))\n      bucket.grantPut(this.role)\n      ecrRepo.grantPullPush(this.role)\n      if (createLimitedRole) {\n        const limitedRole = new GithubActionsRole(\n          this,\n          \"GithubActionsLimitedRole\",\n          {\n            oidcProvider,\n            trustedOwners,\n            roleName:\n              props.githubActions.limitedRoleConfiguration?.roleName ??\n              \"github-actions-limited-role\",\n            // NOTE: The limited role can be assumed from all branches.\n            trustedBranch: \"*\",\n            repositories: props.githubActions.repositories,\n          },\n        )\n        this.limitedRole = limitedRole.role\n        this.limitedRole.addToPolicy(\n          policyStatements.allowPipelineVariables(this, \"dev*\"),\n        )\n        this.limitedRole.addToPolicy(\n          policyStatements.denyProdPipelines(scope, bucket),\n        )\n        bucket.grantPut(this.limitedRole)\n        ecrRepo.grantPullPush(this.limitedRole)\n      }\n    }\n    if (props.ciRoleName) {\n      const legacyRoleForJenkins = new iam.Role(this, \"CiRole\", {\n        roleName: props.ciRoleName,\n        assumedBy: new iam.ArnPrincipal(\n          \"arn:aws:iam::923402097046:role/buildtools-jenkins-RoleJenkinsSlave-JQGYHR5WE6C5\",\n        ),\n      })\n      legacyRoleForJenkins.addToPolicy(\n        policyStatements.allowPipelineVariables(this),\n      )\n      bucket.grantPut(legacyRoleForJenkins)\n      ecrRepo.grantPullPush(legacyRoleForJenkins)\n    }\n\n    new cdk.CfnOutput(this, \"EcrRepoUri\", {\n      value: ecrRepo.repositoryUri,\n    })\n\n    new cdk.CfnOutput(this, \"BucketName\", {\n      value: bucket.bucketName,\n    })\n\n    if (this.role) {\n      new cdk.CfnOutput(this, \"RoleArn\", {\n        value: this.role.roleArn,\n      })\n    }\n    if (this.limitedRole) {\n      new cdk.CfnOutput(this, \"LimitedRoleArn\", {\n        value: this.limitedRole.roleArn,\n      })\n    }\n  }\n}\n"]}
|
package/lib/ecs/index.d.ts
CHANGED
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
export { Cluster } from "./cluster";
|
|
2
2
|
export { FargateService } from "./fargate-service";
|
|
3
3
|
export { ListenerRule } from "./listener-rule";
|
|
4
|
+
export { OpenTelemetryCollectors } from "./open-telemetry";
|
|
4
5
|
export type { ClusterProps } from "./cluster";
|
|
5
6
|
export type { FargateServiceProps } from "./fargate-service";
|
|
6
7
|
export type { ListenerRuleProps } from "./listener-rule";
|
|
8
|
+
export type { OpenTelemetryCollectorsProps } from "./open-telemetry";
|
package/lib/ecs/index.js
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
export { Cluster } from "./cluster";
|
|
2
2
|
export { FargateService } from "./fargate-service";
|
|
3
3
|
export { ListenerRule } from "./listener-rule";
|
|
4
|
-
|
|
4
|
+
export { OpenTelemetryCollectors } from "./open-telemetry";
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZWNzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxXQUFXLENBQUE7QUFDbkMsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1CQUFtQixDQUFBO0FBQ2xELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQTtBQUM5QyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IENsdXN0ZXIgfSBmcm9tIFwiLi9jbHVzdGVyXCJcbmV4cG9ydCB7IEZhcmdhdGVTZXJ2aWNlIH0gZnJvbSBcIi4vZmFyZ2F0ZS1zZXJ2aWNlXCJcbmV4cG9ydCB7IExpc3RlbmVyUnVsZSB9IGZyb20gXCIuL2xpc3RlbmVyLXJ1bGVcIlxuZXhwb3J0IHsgT3BlblRlbGVtZXRyeUNvbGxlY3RvcnMgfSBmcm9tIFwiLi9vcGVuLXRlbGVtZXRyeVwiXG5leHBvcnQgdHlwZSB7IENsdXN0ZXJQcm9wcyB9IGZyb20gXCIuL2NsdXN0ZXJcIlxuZXhwb3J0IHR5cGUgeyBGYXJnYXRlU2VydmljZVByb3BzIH0gZnJvbSBcIi4vZmFyZ2F0ZS1zZXJ2aWNlXCJcbmV4cG9ydCB0eXBlIHsgTGlzdGVuZXJSdWxlUHJvcHMgfSBmcm9tIFwiLi9saXN0ZW5lci1ydWxlXCJcbmV4cG9ydCB0eXBlIHsgT3BlblRlbGVtZXRyeUNvbGxlY3RvcnNQcm9wcyB9IGZyb20gXCIuL29wZW4tdGVsZW1ldHJ5XCJcbiJdfQ==
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as ecs from "aws-cdk-lib/aws-ecs";
|
|
3
|
+
import { RetentionDays } from "aws-cdk-lib/aws-logs";
|
|
4
|
+
import { FargateService } from "./fargate-service";
|
|
5
|
+
export interface OpenTelemetryCollectorsProps {
|
|
6
|
+
service: FargateService;
|
|
7
|
+
/** @default 6 months **/
|
|
8
|
+
logRetention?: RetentionDays;
|
|
9
|
+
/** @default "amazon/aws-otel-collector:v0.43.1" */
|
|
10
|
+
dockerImage?: string;
|
|
11
|
+
/** Should be kept as `undefined` unless you know what you are doing.
|
|
12
|
+
* This is the YAML config for the OpenTelemetry collector sidecar.
|
|
13
|
+
*
|
|
14
|
+
* An example of a config can be found at https://github.com/aws-observability/aws-otel-collector/blob/0ae198c7e7b8c43bcc8715f54e52c879c04407b6/config/ecs/container-insights/otel-task-metrics-config.yaml
|
|
15
|
+
*
|
|
16
|
+
* @default a file in `assets` tuned to work for aws and strips known high-cardinality metrics (like those containing IP addresses and ports)
|
|
17
|
+
*/
|
|
18
|
+
awsOtelConfig?: string;
|
|
19
|
+
/** Overrides for the sidecar container.
|
|
20
|
+
* You do not need to specify this.
|
|
21
|
+
*
|
|
22
|
+
* Defaults:
|
|
23
|
+
* - cpu: 32 units
|
|
24
|
+
* - memory reservation: 24 MiB
|
|
25
|
+
* - memory limit: 256 MiB
|
|
26
|
+
*/
|
|
27
|
+
containerProps?: SidecarContainerProps;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Methods to enable collection of Open Telemetry (otel) data of a {@link FargateService}
|
|
31
|
+
* using a docker container with an otel agent.
|
|
32
|
+
*
|
|
33
|
+
*
|
|
34
|
+
* An example of a Java auto-instrumentation agent in Docker can be found
|
|
35
|
+
* [in liflig-rest-baseline Dockerfile](https://github.com/capralifecycle/liflig-rest-service-baseline/blob/a29b5a472c982aa7ce04d09d0e7cfdc92a6cc977/docker/Dockerfile#L9-L29).
|
|
36
|
+
*
|
|
37
|
+
* The agent must be configured to output metrics to a collector.
|
|
38
|
+
* That collector is what this construct provides.
|
|
39
|
+
* Usually, the agent is specified in the Dockerfile or as a dependency/library,
|
|
40
|
+
* and configured in the Dockerfile or in the application source code.
|
|
41
|
+
*
|
|
42
|
+
* Use this construct on a {@link FargateService} by constructing a new instance of {@link OpenTelemetryCollectors}
|
|
43
|
+
* and calling the {@link addOpenTelemetryCollectorSidecar} method on it.
|
|
44
|
+
*
|
|
45
|
+
* ```ts
|
|
46
|
+
* const service = FargateService(...);
|
|
47
|
+
*
|
|
48
|
+
* new OpenTelemetryCollectors(this, "OtelSidecar").addOpenTelemetryCollectorSidecar(service)
|
|
49
|
+
* ```
|
|
50
|
+
*
|
|
51
|
+
* The sidecar exposes these ports to your service:
|
|
52
|
+
* - udp 2000 : AWS XRay
|
|
53
|
+
* - tcp 4317 : OpenTelemetry collection GRPC
|
|
54
|
+
* - tcp 4318 : OpenTelemetry collection HTTP
|
|
55
|
+
*
|
|
56
|
+
* ---
|
|
57
|
+
*
|
|
58
|
+
* You can also disable the OpenTelemetry instrumentation agent
|
|
59
|
+
* for Java-based services,
|
|
60
|
+
* by setting the appropriate environment variable with {@link disableOpenTelemetryJavaAgent}:
|
|
61
|
+
* ```ts
|
|
62
|
+
* const service = FargateService(...);
|
|
63
|
+
*
|
|
64
|
+
* OpenTelemetryCollectors.disableOpenTelemetryJavaAgent(service)
|
|
65
|
+
* ```
|
|
66
|
+
*
|
|
67
|
+
* @see OpenTelemetryCollectors.addOpenTelemetryCollectorSidecar
|
|
68
|
+
*/
|
|
69
|
+
export declare class OpenTelemetryCollectors extends constructs.Construct {
|
|
70
|
+
private readonly props;
|
|
71
|
+
constructor(scope: constructs.Construct, id: string, props: OpenTelemetryCollectorsProps);
|
|
72
|
+
/**
|
|
73
|
+
* The OpenTelemetry Java agent may run by default in the Docker image.
|
|
74
|
+
* This method will tell the agent to disable itself.
|
|
75
|
+
*
|
|
76
|
+
* You might want to do this to avoid overhead or error logs from failed
|
|
77
|
+
* connection attempts to the otel collector.
|
|
78
|
+
*/
|
|
79
|
+
disableOpenTelemetryJavaAgent(): void;
|
|
80
|
+
/**
|
|
81
|
+
* The OpenTelemetry Java agent may run by default in the Docker image.
|
|
82
|
+
* This method will tell the agent to disable itself.
|
|
83
|
+
*
|
|
84
|
+
* You might want to do this to avoid overhead or error logs from failed
|
|
85
|
+
* connection attempts to the otel collector.
|
|
86
|
+
* @param service
|
|
87
|
+
*/
|
|
88
|
+
static disableOpenTelemetryJavaAgent(service: FargateService): void;
|
|
89
|
+
/**
|
|
90
|
+
* Adds a sidecar with an AWS Distro OpenTelemetry Collector.
|
|
91
|
+
* https://aws-otel.github.io/docs/setup/ecs
|
|
92
|
+
*
|
|
93
|
+
* You also need to add either the Java SDK for OTel or a Java agent,
|
|
94
|
+
* to capture telemetry and send to this collector.
|
|
95
|
+
*/
|
|
96
|
+
addOpenTelemetryCollectorSidecar(): void;
|
|
97
|
+
}
|
|
98
|
+
export type SidecarContainerProps = Pick<ecs.ContainerDefinitionProps, "cpu" | "memoryReservationMiB" | "memoryLimitMiB">;
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as ecs from "aws-cdk-lib/aws-ecs";
|
|
3
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
4
|
+
import * as logs from "aws-cdk-lib/aws-logs";
|
|
5
|
+
import { RetentionDays } from "aws-cdk-lib/aws-logs";
|
|
6
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
7
|
+
import { readFileSync } from "fs";
|
|
8
|
+
import { fileURLToPath } from "url";
|
|
9
|
+
import * as path from "path";
|
|
10
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
11
|
+
const __dirname = path.dirname(__filename);
|
|
12
|
+
/**
|
|
13
|
+
* Methods to enable collection of Open Telemetry (otel) data of a {@link FargateService}
|
|
14
|
+
* using a docker container with an otel agent.
|
|
15
|
+
*
|
|
16
|
+
*
|
|
17
|
+
* An example of a Java auto-instrumentation agent in Docker can be found
|
|
18
|
+
* [in liflig-rest-baseline Dockerfile](https://github.com/capralifecycle/liflig-rest-service-baseline/blob/a29b5a472c982aa7ce04d09d0e7cfdc92a6cc977/docker/Dockerfile#L9-L29).
|
|
19
|
+
*
|
|
20
|
+
* The agent must be configured to output metrics to a collector.
|
|
21
|
+
* That collector is what this construct provides.
|
|
22
|
+
* Usually, the agent is specified in the Dockerfile or as a dependency/library,
|
|
23
|
+
* and configured in the Dockerfile or in the application source code.
|
|
24
|
+
*
|
|
25
|
+
* Use this construct on a {@link FargateService} by constructing a new instance of {@link OpenTelemetryCollectors}
|
|
26
|
+
* and calling the {@link addOpenTelemetryCollectorSidecar} method on it.
|
|
27
|
+
*
|
|
28
|
+
* ```ts
|
|
29
|
+
* const service = FargateService(...);
|
|
30
|
+
*
|
|
31
|
+
* new OpenTelemetryCollectors(this, "OtelSidecar").addOpenTelemetryCollectorSidecar(service)
|
|
32
|
+
* ```
|
|
33
|
+
*
|
|
34
|
+
* The sidecar exposes these ports to your service:
|
|
35
|
+
* - udp 2000 : AWS XRay
|
|
36
|
+
* - tcp 4317 : OpenTelemetry collection GRPC
|
|
37
|
+
* - tcp 4318 : OpenTelemetry collection HTTP
|
|
38
|
+
*
|
|
39
|
+
* ---
|
|
40
|
+
*
|
|
41
|
+
* You can also disable the OpenTelemetry instrumentation agent
|
|
42
|
+
* for Java-based services,
|
|
43
|
+
* by setting the appropriate environment variable with {@link disableOpenTelemetryJavaAgent}:
|
|
44
|
+
* ```ts
|
|
45
|
+
* const service = FargateService(...);
|
|
46
|
+
*
|
|
47
|
+
* OpenTelemetryCollectors.disableOpenTelemetryJavaAgent(service)
|
|
48
|
+
* ```
|
|
49
|
+
*
|
|
50
|
+
* @see OpenTelemetryCollectors.addOpenTelemetryCollectorSidecar
|
|
51
|
+
*/
|
|
52
|
+
export class OpenTelemetryCollectors extends constructs.Construct {
|
|
53
|
+
props;
|
|
54
|
+
constructor(scope, id, props) {
|
|
55
|
+
super(scope, id);
|
|
56
|
+
this.props = props;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* The OpenTelemetry Java agent may run by default in the Docker image.
|
|
60
|
+
* This method will tell the agent to disable itself.
|
|
61
|
+
*
|
|
62
|
+
* You might want to do this to avoid overhead or error logs from failed
|
|
63
|
+
* connection attempts to the otel collector.
|
|
64
|
+
*/
|
|
65
|
+
disableOpenTelemetryJavaAgent() {
|
|
66
|
+
OpenTelemetryCollectors.disableOpenTelemetryJavaAgent(this.props.service);
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* The OpenTelemetry Java agent may run by default in the Docker image.
|
|
70
|
+
* This method will tell the agent to disable itself.
|
|
71
|
+
*
|
|
72
|
+
* You might want to do this to avoid overhead or error logs from failed
|
|
73
|
+
* connection attempts to the otel collector.
|
|
74
|
+
* @param service
|
|
75
|
+
*/
|
|
76
|
+
static disableOpenTelemetryJavaAgent(service) {
|
|
77
|
+
service.taskDefinition.defaultContainer?.addEnvironment("OTEL_JAVAAGENT_ENABLED", "false");
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Adds a sidecar with an AWS Distro OpenTelemetry Collector.
|
|
81
|
+
* https://aws-otel.github.io/docs/setup/ecs
|
|
82
|
+
*
|
|
83
|
+
* You also need to add either the Java SDK for OTel or a Java agent,
|
|
84
|
+
* to capture telemetry and send to this collector.
|
|
85
|
+
*/
|
|
86
|
+
addOpenTelemetryCollectorSidecar() {
|
|
87
|
+
new OpenTelemetryPolicies(this, "OpenTelemetryPolicies", {
|
|
88
|
+
taskDefinition: this.props.service.taskDefinition,
|
|
89
|
+
});
|
|
90
|
+
this.props.service.taskDefinition.addExtension(new OpenTelemetryCollectorSidecar(this, this.props.logRetention, this.props.dockerImage, this.props.awsOtelConfig, this.props.containerProps));
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Adds a sidecar with an AWS Distro OpenTelemetry Collector.
|
|
95
|
+
* https://aws-otel.github.io/docs/setup/ecs
|
|
96
|
+
*
|
|
97
|
+
* You also need to add either the Java SDK for OTel or a Java agent,
|
|
98
|
+
* to capture telemetry and send to this collector.
|
|
99
|
+
*/
|
|
100
|
+
class OpenTelemetryCollectorSidecar {
|
|
101
|
+
construct;
|
|
102
|
+
logRetention;
|
|
103
|
+
dockerImage;
|
|
104
|
+
awsOtelConfig;
|
|
105
|
+
containerProps;
|
|
106
|
+
constructor(construct, logRetention, dockerImage, awsOtelConfig, containerProps) {
|
|
107
|
+
this.construct = construct;
|
|
108
|
+
this.logRetention = logRetention;
|
|
109
|
+
this.dockerImage = dockerImage;
|
|
110
|
+
this.awsOtelConfig = awsOtelConfig;
|
|
111
|
+
this.containerProps = containerProps;
|
|
112
|
+
}
|
|
113
|
+
extend(taskDefinition) {
|
|
114
|
+
if (taskDefinition.networkMode !== ecs.NetworkMode.AWS_VPC) {
|
|
115
|
+
throw new Error("Task NetworkMode must be AWS_VPC: " + taskDefinition.networkMode);
|
|
116
|
+
}
|
|
117
|
+
const commands = {
|
|
118
|
+
metricsAndTraces: "--config=/etc/ecs/ecs-default-config.yaml",
|
|
119
|
+
metricsAndTracesAndContainerResources: "--config=/etc/ecs/container-insights/otel-task-metrics-config.yaml",
|
|
120
|
+
};
|
|
121
|
+
const logGroup = new logs.LogGroup(this.construct, "CollectorLogGroup", {
|
|
122
|
+
retention: this.logRetention ?? RetentionDays.SIX_MONTHS,
|
|
123
|
+
removalPolicy: RemovalPolicy.DESTROY,
|
|
124
|
+
});
|
|
125
|
+
const sidecarImage = this.dockerImage ?? "amazon/aws-otel-collector:v0.43.1";
|
|
126
|
+
const sidecar = taskDefinition.addContainer("aws-opentelemetry-collector", {
|
|
127
|
+
cpu: this.containerProps?.cpu ?? 32,
|
|
128
|
+
memoryReservationMiB: this.containerProps?.memoryReservationMiB ?? 24,
|
|
129
|
+
memoryLimitMiB: this.containerProps?.memoryLimitMiB ?? 256,
|
|
130
|
+
image: ecs.ContainerImage.fromRegistry(sidecarImage),
|
|
131
|
+
command: [commands.metricsAndTracesAndContainerResources], // This is not used when the AOT_CONFIG_CONTENT is set!
|
|
132
|
+
environment: {
|
|
133
|
+
// You can alternatively create an SSM parameter with the config, and pass it to the `secrets` option
|
|
134
|
+
AOT_CONFIG_CONTENT: this.awsOtelConfig ?? awsOtelCustomConfigYaml,
|
|
135
|
+
},
|
|
136
|
+
logging: ecs.LogDrivers.awsLogs({
|
|
137
|
+
logGroup: logGroup,
|
|
138
|
+
streamPrefix: "ecs",
|
|
139
|
+
}),
|
|
140
|
+
});
|
|
141
|
+
// A dependency should be added to all containers that export metrics.
|
|
142
|
+
// We are currently assuming that this is only the default container
|
|
143
|
+
taskDefinition.defaultContainer?.addContainerDependencies({
|
|
144
|
+
container: sidecar,
|
|
145
|
+
condition: ecs.ContainerDependencyCondition.START,
|
|
146
|
+
});
|
|
147
|
+
/*
|
|
148
|
+
* aws-otel-collector exposes these ports, and more:
|
|
149
|
+
* - udp 2000 : AWS XRay
|
|
150
|
+
* - tcp 4317 : OpenTelemetry collection GRPC
|
|
151
|
+
* - tcp 4318 : OpenTelemetry collection HTTP
|
|
152
|
+
*
|
|
153
|
+
* These are defined in the yaml config for the collector.
|
|
154
|
+
*/
|
|
155
|
+
taskDefinition.defaultContainer?.addEnvironment("AWS_XRAY_DAEMON_ADDRESS", "http://localhost:2000");
|
|
156
|
+
taskDefinition.defaultContainer?.addEnvironment("OTEL_EXPORTER_OTLP_ENDPOINT", "http://localhost:4317");
|
|
157
|
+
taskDefinition.defaultContainer?.addEnvironment("OTEL_JAVAAGENT_ENABLED", "true");
|
|
158
|
+
if (!taskDefinition.isFargateCompatible) {
|
|
159
|
+
// This extension is made for the FargateService, so this is a requirement.
|
|
160
|
+
throw new Error("This task definition can not be ran on fargate! " +
|
|
161
|
+
taskDefinition.node.id);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Grants the sidecar permissions to create logs, metrics and XRay traces by extending the task roles.
|
|
167
|
+
*/
|
|
168
|
+
class OpenTelemetryPolicies extends constructs.Construct {
|
|
169
|
+
constructor(scope, id, props) {
|
|
170
|
+
super(scope, id);
|
|
171
|
+
const awsDistroOpenTelemetryPolicyStatement = new iam.PolicyStatement({
|
|
172
|
+
effect: iam.Effect.ALLOW,
|
|
173
|
+
resources: ["*"],
|
|
174
|
+
actions: [
|
|
175
|
+
"logs:PutLogEvents",
|
|
176
|
+
"logs:CreateLogGroup",
|
|
177
|
+
"logs:CreateLogStream",
|
|
178
|
+
"logs:DescribeLogStreams",
|
|
179
|
+
"logs:DescribeLogGroups",
|
|
180
|
+
"logs:PutRetentionPolicy",
|
|
181
|
+
"xray:PutTraceSegments",
|
|
182
|
+
"xray:PutTelemetryRecords",
|
|
183
|
+
"xray:GetSamplingRules",
|
|
184
|
+
"xray:GetSamplingTargets",
|
|
185
|
+
"xray:GetSamplingStatisticSummaries",
|
|
186
|
+
"cloudwatch:PutMetricData",
|
|
187
|
+
"ec2:DescribeVolumes",
|
|
188
|
+
"ec2:DescribeTags",
|
|
189
|
+
"ssm:GetParameters",
|
|
190
|
+
],
|
|
191
|
+
});
|
|
192
|
+
props.taskDefinition.addToTaskRolePolicy(awsDistroOpenTelemetryPolicyStatement);
|
|
193
|
+
props.taskDefinition.addToExecutionRolePolicy(awsDistroOpenTelemetryPolicyStatement);
|
|
194
|
+
props.taskDefinition.executionRole?.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName("CloudWatchLogsFullAccess"));
|
|
195
|
+
props.taskDefinition.executionRole?.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName("AmazonSSMReadOnlyAccess"));
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
/**
|
|
199
|
+
* This is a modified version of `./etc/ecs/container-insights/otel-task-metrics-config.yaml`
|
|
200
|
+
* where we add `resource_to_telemetry_conversion` so the otel resource `service.name`
|
|
201
|
+
* can be added to the metric. This is useful because you can filter in metrics based on service,
|
|
202
|
+
* instead of grouping e.g. all services' memory usage under the same metric.
|
|
203
|
+
*
|
|
204
|
+
* @see https://aws-otel.github.io/docs/setup/ecs/config-through-ssm
|
|
205
|
+
* @see https://aws-otel.github.io/docs/getting-started/cloudwatch-metrics#cloudwatch-emf-exporter-awsemf
|
|
206
|
+
*/
|
|
207
|
+
const awsOtelCustomConfigYaml = readFileSync(path.resolve(__dirname, "..", "..", "assets", "open-telemetry", "otel-collector-task-metrics-config.yaml"), "utf-8")
|
|
208
|
+
.split("\n")
|
|
209
|
+
.filter((line) => !/^\s*##/.test(line)) // Skip comments starting with ##
|
|
210
|
+
.join("\n");
|
|
211
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"open-telemetry.js","sourceRoot":"","sources":["../../src/ecs/open-telemetry.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport { RetentionDays } from \"aws-cdk-lib/aws-logs\"\nimport { RemovalPolicy } from \"aws-cdk-lib\"\nimport { FargateService } from \"./fargate-service\"\nimport { readFileSync } from \"fs\"\nimport { fileURLToPath } from \"url\"\nimport * as path from \"path\"\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\nexport interface OpenTelemetryCollectorsProps {\n  service: FargateService\n\n  /** @default 6 months **/\n  logRetention?: RetentionDays\n\n  /** @default \"amazon/aws-otel-collector:v0.43.1\" */\n  dockerImage?: string\n\n  /** Should be kept as `undefined` unless you know what you are doing.\n   * This is the YAML config for the OpenTelemetry collector sidecar.\n   *\n   * An example of a config can be found at https://github.com/aws-observability/aws-otel-collector/blob/0ae198c7e7b8c43bcc8715f54e52c879c04407b6/config/ecs/container-insights/otel-task-metrics-config.yaml\n   *\n   * @default a file in `assets` tuned to work for aws and strips known high-cardinality metrics (like those containing IP addresses and ports)\n   */\n  awsOtelConfig?: string\n\n  /** Overrides for the sidecar container.\n   * You do not need to specify this.\n   *\n   * Defaults:\n   * - cpu: 32 units\n   * - memory reservation: 24 MiB\n   * - memory limit: 256 MiB\n   */\n  containerProps?: SidecarContainerProps\n}\n\n/**\n * Methods to enable collection of Open Telemetry (otel) data of a {@link FargateService}\n * using a docker container with an otel agent.\n *\n *\n * An example of a Java auto-instrumentation agent in Docker can be found\n * [in liflig-rest-baseline Dockerfile](https://github.com/capralifecycle/liflig-rest-service-baseline/blob/a29b5a472c982aa7ce04d09d0e7cfdc92a6cc977/docker/Dockerfile#L9-L29).\n *\n * The agent must be configured to output metrics to a collector.\n * That collector is what this construct provides.\n * Usually, the agent is specified in the Dockerfile or as a dependency/library,\n * and configured in the Dockerfile or in the application source code.\n *\n * Use this construct on a {@link FargateService} by constructing a new instance of {@link OpenTelemetryCollectors}\n * and calling the {@link addOpenTelemetryCollectorSidecar} method on it.\n *\n * ```ts\n * const service = FargateService(...);\n *\n * new OpenTelemetryCollectors(this, \"OtelSidecar\").addOpenTelemetryCollectorSidecar(service)\n * ```\n *\n * The sidecar exposes these ports to your service:\n * - udp 2000 : AWS XRay\n * - tcp 4317 : OpenTelemetry collection GRPC\n * - tcp 4318 : OpenTelemetry collection HTTP\n *\n * ---\n *\n * You can also disable the OpenTelemetry instrumentation agent\n * for Java-based services,\n * by setting the appropriate environment variable with {@link disableOpenTelemetryJavaAgent}:\n * ```ts\n * const service = FargateService(...);\n *\n * OpenTelemetryCollectors.disableOpenTelemetryJavaAgent(service)\n * ```\n *\n * @see OpenTelemetryCollectors.addOpenTelemetryCollectorSidecar\n */\nexport class OpenTelemetryCollectors extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    private readonly props: OpenTelemetryCollectorsProps,\n  ) {\n    super(scope, id)\n  }\n\n  /**\n   * The OpenTelemetry Java agent may run by default in the Docker image.\n   * This method will tell the agent to disable itself.\n   *\n   * You might want to do this to avoid overhead or error logs from failed\n   * connection attempts to the otel collector.\n   */\n  public disableOpenTelemetryJavaAgent() {\n    OpenTelemetryCollectors.disableOpenTelemetryJavaAgent(this.props.service)\n  }\n\n  /**\n   * The OpenTelemetry Java agent may run by default in the Docker image.\n   * This method will tell the agent to disable itself.\n   *\n   * You might want to do this to avoid overhead or error logs from failed\n   * connection attempts to the otel collector.\n   * @param service\n   */\n  public static disableOpenTelemetryJavaAgent(service: FargateService) {\n    service.taskDefinition.defaultContainer?.addEnvironment(\n      \"OTEL_JAVAAGENT_ENABLED\",\n      \"false\",\n    )\n  }\n\n  /**\n   * Adds a sidecar with an AWS Distro OpenTelemetry Collector.\n   * https://aws-otel.github.io/docs/setup/ecs\n   *\n   * You also need to add either the Java SDK for OTel or a Java agent,\n   * to capture telemetry and send to this collector.\n   */\n  public addOpenTelemetryCollectorSidecar() {\n    new OpenTelemetryPolicies(this, \"OpenTelemetryPolicies\", {\n      taskDefinition: this.props.service.taskDefinition,\n    })\n\n    this.props.service.taskDefinition.addExtension(\n      new OpenTelemetryCollectorSidecar(\n        this,\n        this.props.logRetention,\n        this.props.dockerImage,\n        this.props.awsOtelConfig,\n        this.props.containerProps,\n      ),\n    )\n  }\n}\n\nexport type SidecarContainerProps = Pick<\n  ecs.ContainerDefinitionProps,\n  \"cpu\" | \"memoryReservationMiB\" | \"memoryLimitMiB\"\n>\n\n/**\n * Adds a sidecar with an AWS Distro OpenTelemetry Collector.\n * https://aws-otel.github.io/docs/setup/ecs\n *\n * You also need to add either the Java SDK for OTel or a Java agent,\n * to capture telemetry and send to this collector.\n */\nclass OpenTelemetryCollectorSidecar implements ecs.ITaskDefinitionExtension {\n  constructor(\n    private readonly construct: constructs.Construct,\n    private readonly logRetention?: RetentionDays,\n    private readonly dockerImage?: string,\n    private readonly awsOtelConfig?: string,\n    private readonly containerProps?: SidecarContainerProps,\n  ) {}\n\n  extend(taskDefinition: ecs.TaskDefinition): void {\n    if (taskDefinition.networkMode !== ecs.NetworkMode.AWS_VPC) {\n      throw new Error(\n        \"Task NetworkMode must be AWS_VPC: \" + taskDefinition.networkMode,\n      )\n    }\n\n    const commands = {\n      metricsAndTraces: \"--config=/etc/ecs/ecs-default-config.yaml\",\n      metricsAndTracesAndContainerResources:\n        \"--config=/etc/ecs/container-insights/otel-task-metrics-config.yaml\",\n    }\n\n    const logGroup = new logs.LogGroup(this.construct, \"CollectorLogGroup\", {\n      retention: this.logRetention ?? RetentionDays.SIX_MONTHS,\n      removalPolicy: RemovalPolicy.DESTROY,\n    })\n\n    const sidecarImage = this.dockerImage ?? \"amazon/aws-otel-collector:v0.43.1\"\n    const sidecar = taskDefinition.addContainer(\"aws-opentelemetry-collector\", {\n      cpu: this.containerProps?.cpu ?? 32,\n      memoryReservationMiB: this.containerProps?.memoryReservationMiB ?? 24,\n      memoryLimitMiB: this.containerProps?.memoryLimitMiB ?? 256,\n      image: ecs.ContainerImage.fromRegistry(sidecarImage),\n      command: [commands.metricsAndTracesAndContainerResources], // This is not used when the AOT_CONFIG_CONTENT is set!\n      environment: {\n        // You can alternatively create an SSM parameter with the config, and pass it to the `secrets` option\n        AOT_CONFIG_CONTENT: this.awsOtelConfig ?? awsOtelCustomConfigYaml,\n      },\n      logging: ecs.LogDrivers.awsLogs({\n        logGroup: logGroup,\n        streamPrefix: \"ecs\",\n      }),\n    })\n\n    // A dependency should be added to all containers that export metrics.\n    // We are currently assuming that this is only the default container\n    taskDefinition.defaultContainer?.addContainerDependencies({\n      container: sidecar,\n      condition: ecs.ContainerDependencyCondition.START,\n    })\n\n    /*\n     * aws-otel-collector exposes these ports, and more:\n     * - udp 2000 : AWS XRay\n     * - tcp 4317 : OpenTelemetry collection GRPC\n     * - tcp 4318 : OpenTelemetry collection HTTP\n     *\n     * These are defined in the yaml config for the collector.\n     */\n    taskDefinition.defaultContainer?.addEnvironment(\n      \"AWS_XRAY_DAEMON_ADDRESS\",\n      \"http://localhost:2000\",\n    )\n    taskDefinition.defaultContainer?.addEnvironment(\n      \"OTEL_EXPORTER_OTLP_ENDPOINT\",\n      \"http://localhost:4317\",\n    )\n    taskDefinition.defaultContainer?.addEnvironment(\n      \"OTEL_JAVAAGENT_ENABLED\",\n      \"true\",\n    )\n\n    if (!taskDefinition.isFargateCompatible) {\n      // This extension is made for the FargateService, so this is a requirement.\n      throw new Error(\n        \"This task definition can not be ran on fargate! \" +\n          taskDefinition.node.id,\n      )\n    }\n  }\n}\n\ninterface OpenTelemetryPoliciesProps {\n  taskDefinition: ecs.TaskDefinition\n}\n\n/**\n * Grants the sidecar permissions to create logs, metrics and XRay traces by extending the task roles.\n */\nclass OpenTelemetryPolicies extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: OpenTelemetryPoliciesProps,\n  ) {\n    super(scope, id)\n\n    const awsDistroOpenTelemetryPolicyStatement = new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      resources: [\"*\"],\n      actions: [\n        \"logs:PutLogEvents\",\n        \"logs:CreateLogGroup\",\n        \"logs:CreateLogStream\",\n        \"logs:DescribeLogStreams\",\n        \"logs:DescribeLogGroups\",\n        \"logs:PutRetentionPolicy\",\n        \"xray:PutTraceSegments\",\n        \"xray:PutTelemetryRecords\",\n        \"xray:GetSamplingRules\",\n        \"xray:GetSamplingTargets\",\n        \"xray:GetSamplingStatisticSummaries\",\n        \"cloudwatch:PutMetricData\",\n        \"ec2:DescribeVolumes\",\n        \"ec2:DescribeTags\",\n        \"ssm:GetParameters\",\n      ],\n    })\n\n    props.taskDefinition.addToTaskRolePolicy(\n      awsDistroOpenTelemetryPolicyStatement,\n    )\n\n    props.taskDefinition.addToExecutionRolePolicy(\n      awsDistroOpenTelemetryPolicyStatement,\n    )\n    props.taskDefinition.executionRole?.addManagedPolicy(\n      iam.ManagedPolicy.fromAwsManagedPolicyName(\"CloudWatchLogsFullAccess\"),\n    )\n    props.taskDefinition.executionRole?.addManagedPolicy(\n      iam.ManagedPolicy.fromAwsManagedPolicyName(\"AmazonSSMReadOnlyAccess\"),\n    )\n  }\n}\n\n/**\n * This is a modified version of `./etc/ecs/container-insights/otel-task-metrics-config.yaml`\n * where we add `resource_to_telemetry_conversion` so the otel resource `service.name`\n * can be added to the metric. This is useful because you can filter in metrics based on service,\n * instead of grouping e.g. all services' memory usage under the same metric.\n *\n * @see https://aws-otel.github.io/docs/setup/ecs/config-through-ssm\n * @see https://aws-otel.github.io/docs/getting-started/cloudwatch-metrics#cloudwatch-emf-exporter-awsemf\n */\nconst awsOtelCustomConfigYaml = readFileSync(\n  path.resolve(\n    __dirname,\n    \"..\",\n    \"..\",\n    \"assets\",\n    \"open-telemetry\",\n    \"otel-collector-task-metrics-config.yaml\",\n  ),\n  \"utf-8\",\n)\n  .split(\"\\n\")\n  .filter((line) => !/^\\s*##/.test(line)) // Skip comments starting with ##\n  .join(\"\\n\")\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@liflig/cdk",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.13.0",
|
|
4
4
|
"description": "CDK library for Liflig",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"repository": {
|
|
@@ -55,7 +55,7 @@
|
|
|
55
55
|
"@commitlint/cli": "19.8.1",
|
|
56
56
|
"@commitlint/config-conventional": "19.8.1",
|
|
57
57
|
"@eslint/eslintrc": "3.3.1",
|
|
58
|
-
"@eslint/js": "9.
|
|
58
|
+
"@eslint/js": "9.32.0",
|
|
59
59
|
"@types/aws-lambda": "8.10.152",
|
|
60
60
|
"@types/jest": "30.0.0",
|
|
61
61
|
"@types/node": "24.1.0",
|
|
@@ -65,7 +65,7 @@
|
|
|
65
65
|
"aws-cdk-lib": "2.207.0",
|
|
66
66
|
"constructs": "10.4.2",
|
|
67
67
|
"esbuild": "0.25.8",
|
|
68
|
-
"eslint": "9.
|
|
68
|
+
"eslint": "9.32.0",
|
|
69
69
|
"eslint-config-prettier": "10.1.8",
|
|
70
70
|
"eslint-plugin-prettier": "5.5.3",
|
|
71
71
|
"husky": "9.1.7",
|
|
@@ -75,7 +75,7 @@
|
|
|
75
75
|
"semantic-release": "24.2.7",
|
|
76
76
|
"ts-jest": "29.4.0",
|
|
77
77
|
"tsx": "4.20.3",
|
|
78
|
-
"typedoc": "0.28.
|
|
78
|
+
"typedoc": "0.28.8",
|
|
79
79
|
"typescript": "5.8.3"
|
|
80
80
|
},
|
|
81
81
|
"dependencies": {
|