UNPKG

106 kBJavaScriptView Raw
1"use strict";
2var _a, _b;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.AsgCapacityProvider = exports.ExecuteCommandLogging = exports.Cluster = exports.MachineImageType = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const autoscaling = require("@aws-cdk/aws-autoscaling");
8const cloudwatch = require("@aws-cdk/aws-cloudwatch");
9const ec2 = require("@aws-cdk/aws-ec2");
10const iam = require("@aws-cdk/aws-iam");
11const cloudmap = require("@aws-cdk/aws-servicediscovery");
12const core_1 = require("@aws-cdk/core");
13const amis_1 = require("./amis");
14const instance_drain_hook_1 = require("./drain-hook/instance-drain-hook");
15const ecs_canned_metrics_generated_1 = require("./ecs-canned-metrics.generated");
16const ecs_generated_1 = require("./ecs.generated");
17// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
18// eslint-disable-next-line
19const core_2 = require("@aws-cdk/core");
20/**
21 * The machine image type
22 */
23var MachineImageType;
24(function (MachineImageType) {
25 /**
26 * Amazon ECS-optimized Amazon Linux 2 AMI
27 */
28 MachineImageType[MachineImageType["AMAZON_LINUX_2"] = 0] = "AMAZON_LINUX_2";
29 /**
30 * Bottlerocket AMI
31 */
32 MachineImageType[MachineImageType["BOTTLEROCKET"] = 1] = "BOTTLEROCKET";
33})(MachineImageType = exports.MachineImageType || (exports.MachineImageType = {}));
34/**
35 * A regional grouping of one or more container instances on which you can run tasks and services.
36 */
37class Cluster extends core_1.Resource {
38 /**
39 * Constructs a new instance of the Cluster class.
40 */
41 constructor(scope, id, props = {}) {
42 var _c;
43 super(scope, id, {
44 physicalName: props.clusterName,
45 });
46 /**
47 * Manage the allowed network connections for the cluster with Security Groups.
48 */
49 this.connections = new ec2.Connections();
50 /**
51 * The names of both ASG and Fargate capacity providers associated with the cluster.
52 */
53 this._capacityProviderNames = [];
54 /**
55 * Specifies whether the cluster has EC2 instance capacity.
56 */
57 this._hasEc2Capacity = false;
58 try {
59 jsiiDeprecationWarnings._aws_cdk_aws_ecs_ClusterProps(props);
60 }
61 catch (error) {
62 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
63 Error.captureStackTrace(error, this.constructor);
64 }
65 throw error;
66 }
67 /**
68 * clusterSettings needs to be undefined if containerInsights is not explicitly set in order to allow any
69 * containerInsights settings on the account to apply. See:
70 * https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-clustersettings.html#cfn-ecs-cluster-clustersettings-value
71 */
72 let clusterSettings = undefined;
73 if (props.containerInsights !== undefined) {
74 clusterSettings = [{ name: 'containerInsights', value: props.containerInsights ? ContainerInsights.ENABLED : ContainerInsights.DISABLED }];
75 }
76 this._capacityProviderNames = (_c = props.capacityProviders) !== null && _c !== void 0 ? _c : [];
77 if (props.enableFargateCapacityProviders) {
78 this.enableFargateCapacityProviders();
79 }
80 if (props.executeCommandConfiguration) {
81 if ((props.executeCommandConfiguration.logging === ExecuteCommandLogging.OVERRIDE) !==
82 (props.executeCommandConfiguration.logConfiguration !== undefined)) {
83 throw new Error('Execute command log configuration must only be specified when logging is OVERRIDE.');
84 }
85 this._executeCommandConfiguration = props.executeCommandConfiguration;
86 }
87 const cluster = new ecs_generated_1.CfnCluster(this, 'Resource', {
88 clusterName: this.physicalName,
89 clusterSettings,
90 configuration: this._executeCommandConfiguration && this.renderExecuteCommandConfiguration(),
91 });
92 this.clusterArn = this.getResourceArnAttribute(cluster.attrArn, {
93 service: 'ecs',
94 resource: 'cluster',
95 resourceName: this.physicalName,
96 });
97 this.clusterName = this.getResourceNameAttribute(cluster.ref);
98 this.vpc = props.vpc || new ec2.Vpc(this, 'Vpc', { maxAzs: 2 });
99 this._defaultCloudMapNamespace = props.defaultCloudMapNamespace !== undefined
100 ? this.addDefaultCloudMapNamespace(props.defaultCloudMapNamespace)
101 : undefined;
102 this._autoscalingGroup = props.capacity !== undefined
103 ? this.addCapacity('DefaultAutoScalingGroup', props.capacity)
104 : undefined;
105 // Only create cluster capacity provider associations if there are any EC2
106 // capacity providers. Ordinarily we'd just add the construct to the tree
107 // since it's harmless, but we'd prefer not to add unexpected new
108 // resources to the stack which could surprise users working with
109 // brown-field CDK apps and stacks.
110 core_1.Aspects.of(this).add(new MaybeCreateCapacityProviderAssociations(this, id, this._capacityProviderNames));
111 }
112 /**
113 * Import an existing cluster to the stack from its attributes.
114 */
115 static fromClusterAttributes(scope, id, attrs) {
116 try {
117 jsiiDeprecationWarnings._aws_cdk_aws_ecs_ClusterAttributes(attrs);
118 }
119 catch (error) {
120 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
121 Error.captureStackTrace(error, this.fromClusterAttributes);
122 }
123 throw error;
124 }
125 return new ImportedCluster(scope, id, attrs);
126 }
127 /**
128 * Import an existing cluster to the stack from the cluster ARN.
129 * This does not provide access to the vpc, hasEc2Capacity, or connections -
130 * use the `fromClusterAttributes` method to access those properties.
131 */
132 static fromClusterArn(scope, id, clusterArn) {
133 const stack = core_1.Stack.of(scope);
134 const arn = stack.splitArn(clusterArn, core_1.ArnFormat.SLASH_RESOURCE_NAME);
135 const clusterName = arn.resourceName;
136 if (!clusterName) {
137 throw new Error(`Missing required Cluster Name from Cluster ARN: ${clusterArn}`);
138 }
139 const errorSuffix = 'is not available for a Cluster imported using fromClusterArn(), please use fromClusterAttributes() instead.';
140 class Import extends core_1.Resource {
141 constructor() {
142 super(...arguments);
143 this.clusterArn = clusterArn;
144 this.clusterName = clusterName;
145 }
146 get hasEc2Capacity() {
147 throw new Error(`hasEc2Capacity ${errorSuffix}`);
148 }
149 get connections() {
150 throw new Error(`connections ${errorSuffix}`);
151 }
152 get vpc() {
153 throw new Error(`vpc ${errorSuffix}`);
154 }
155 }
156 return new Import(scope, id, {
157 environmentFromArn: clusterArn,
158 });
159 }
160 /**
161 * Enable the Fargate capacity providers for this cluster.
162 */
163 enableFargateCapacityProviders() {
164 for (const provider of ['FARGATE', 'FARGATE_SPOT']) {
165 if (!this._capacityProviderNames.includes(provider)) {
166 this._capacityProviderNames.push(provider);
167 }
168 }
169 }
170 renderExecuteCommandConfiguration() {
171 var _c, _d, _e, _f;
172 return {
173 executeCommandConfiguration: {
174 kmsKeyId: (_d = (_c = this._executeCommandConfiguration) === null || _c === void 0 ? void 0 : _c.kmsKey) === null || _d === void 0 ? void 0 : _d.keyArn,
175 logConfiguration: ((_e = this._executeCommandConfiguration) === null || _e === void 0 ? void 0 : _e.logConfiguration) && this.renderExecuteCommandLogConfiguration(),
176 logging: (_f = this._executeCommandConfiguration) === null || _f === void 0 ? void 0 : _f.logging,
177 },
178 };
179 }
180 renderExecuteCommandLogConfiguration() {
181 var _c, _d, _e;
182 const logConfiguration = (_c = this._executeCommandConfiguration) === null || _c === void 0 ? void 0 : _c.logConfiguration;
183 if ((logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.s3EncryptionEnabled) && !(logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.s3Bucket)) {
184 throw new Error('You must specify an S3 bucket name in the execute command log configuration to enable S3 encryption.');
185 }
186 if ((logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.cloudWatchEncryptionEnabled) && !(logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.cloudWatchLogGroup)) {
187 throw new Error('You must specify a CloudWatch log group in the execute command log configuration to enable CloudWatch encryption.');
188 }
189 return {
190 cloudWatchEncryptionEnabled: logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.cloudWatchEncryptionEnabled,
191 cloudWatchLogGroupName: (_d = logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.cloudWatchLogGroup) === null || _d === void 0 ? void 0 : _d.logGroupName,
192 s3BucketName: (_e = logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.s3Bucket) === null || _e === void 0 ? void 0 : _e.bucketName,
193 s3EncryptionEnabled: logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.s3EncryptionEnabled,
194 s3KeyPrefix: logConfiguration === null || logConfiguration === void 0 ? void 0 : logConfiguration.s3KeyPrefix,
195 };
196 }
197 /**
198 * Add an AWS Cloud Map DNS namespace for this cluster.
199 * NOTE: HttpNamespaces are not supported, as ECS always requires a DNSConfig when registering an instance to a Cloud
200 * Map service.
201 */
202 addDefaultCloudMapNamespace(options) {
203 try {
204 jsiiDeprecationWarnings._aws_cdk_aws_ecs_CloudMapNamespaceOptions(options);
205 }
206 catch (error) {
207 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
208 Error.captureStackTrace(error, this.addDefaultCloudMapNamespace);
209 }
210 throw error;
211 }
212 if (this._defaultCloudMapNamespace !== undefined) {
213 throw new Error('Can only add default namespace once.');
214 }
215 const namespaceType = options.type !== undefined
216 ? options.type
217 : cloudmap.NamespaceType.DNS_PRIVATE;
218 const sdNamespace = namespaceType === cloudmap.NamespaceType.DNS_PRIVATE ?
219 new cloudmap.PrivateDnsNamespace(this, 'DefaultServiceDiscoveryNamespace', {
220 name: options.name,
221 vpc: this.vpc,
222 }) :
223 new cloudmap.PublicDnsNamespace(this, 'DefaultServiceDiscoveryNamespace', {
224 name: options.name,
225 });
226 this._defaultCloudMapNamespace = sdNamespace;
227 return sdNamespace;
228 }
229 /**
230 * Getter for namespace added to cluster
231 */
232 get defaultCloudMapNamespace() {
233 return this._defaultCloudMapNamespace;
234 }
235 /**
236 * It is highly recommended to use {@link Cluster.addAsgCapacityProvider} instead of this method.
237 *
238 * This method adds compute capacity to a cluster by creating an AutoScalingGroup with the specified options.
239 *
240 * Returns the AutoScalingGroup so you can add autoscaling settings to it.
241 */
242 addCapacity(id, options) {
243 var _c, _d;
244 try {
245 jsiiDeprecationWarnings._aws_cdk_aws_ecs_AddCapacityOptions(options);
246 }
247 catch (error) {
248 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
249 Error.captureStackTrace(error, this.addCapacity);
250 }
251 throw error;
252 }
253 // Do 2-way defaulting here: if the machineImageType is BOTTLEROCKET, pick the right AMI.
254 // Otherwise, determine the machineImageType from the given AMI.
255 const machineImage = (_c = options.machineImage) !== null && _c !== void 0 ? _c : (options.machineImageType === MachineImageType.BOTTLEROCKET ? new amis_1.BottleRocketImage({
256 architecture: options.instanceType.architecture,
257 }) : new amis_1.EcsOptimizedAmi());
258 const machineImageType = (_d = options.machineImageType) !== null && _d !== void 0 ? _d : (isBottleRocketImage(machineImage) ? MachineImageType.BOTTLEROCKET : MachineImageType.AMAZON_LINUX_2);
259 const autoScalingGroup = new autoscaling.AutoScalingGroup(this, id, {
260 vpc: this.vpc,
261 machineImage,
262 updateType: options.updateType || autoscaling.UpdateType.REPLACING_UPDATE,
263 ...options,
264 });
265 this.addAutoScalingGroup(autoScalingGroup, {
266 machineImageType: machineImageType,
267 ...options,
268 });
269 return autoScalingGroup;
270 }
271 /**
272 * This method adds an Auto Scaling Group Capacity Provider to a cluster.
273 *
274 * @param provider the capacity provider to add to this cluster.
275 */
276 addAsgCapacityProvider(provider, options = {}) {
277 try {
278 jsiiDeprecationWarnings._aws_cdk_aws_ecs_AsgCapacityProvider(provider);
279 jsiiDeprecationWarnings._aws_cdk_aws_ecs_AddAutoScalingGroupCapacityOptions(options);
280 }
281 catch (error) {
282 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
283 Error.captureStackTrace(error, this.addAsgCapacityProvider);
284 }
285 throw error;
286 }
287 // Don't add the same capacity provider more than once.
288 if (this._capacityProviderNames.includes(provider.capacityProviderName)) {
289 return;
290 }
291 this._hasEc2Capacity = true;
292 this.configureAutoScalingGroup(provider.autoScalingGroup, {
293 ...options,
294 machineImageType: provider.machineImageType,
295 // Don't enable the instance-draining lifecycle hook if managed termination protection is enabled
296 taskDrainTime: provider.enableManagedTerminationProtection ? core_1.Duration.seconds(0) : options.taskDrainTime,
297 });
298 this._capacityProviderNames.push(provider.capacityProviderName);
299 }
300 /**
301 * This method adds compute capacity to a cluster using the specified AutoScalingGroup.
302 *
303 * @deprecated Use {@link Cluster.addAsgCapacityProvider} instead.
304 * @param autoScalingGroup the ASG to add to this cluster.
305 * [disable-awslint:ref-via-interface] is needed in order to install the ECS
306 * agent by updating the ASGs user data.
307 */
308 addAutoScalingGroup(autoScalingGroup, options = {}) {
309 try {
310 jsiiDeprecationWarnings.print("@aws-cdk/aws-ecs.Cluster#addAutoScalingGroup", "Use {@link Cluster.addAsgCapacityProvider} instead.");
311 jsiiDeprecationWarnings._aws_cdk_aws_ecs_AddAutoScalingGroupCapacityOptions(options);
312 }
313 catch (error) {
314 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
315 Error.captureStackTrace(error, this.addAutoScalingGroup);
316 }
317 throw error;
318 }
319 this._hasEc2Capacity = true;
320 this.connections.connections.addSecurityGroup(...autoScalingGroup.connections.securityGroups);
321 this.configureAutoScalingGroup(autoScalingGroup, options);
322 }
323 configureAutoScalingGroup(autoScalingGroup, options = {}) {
324 if (autoScalingGroup.osType === ec2.OperatingSystemType.WINDOWS) {
325 this.configureWindowsAutoScalingGroup(autoScalingGroup, options);
326 }
327 else {
328 // Tie instances to cluster
329 switch (options.machineImageType) {
330 // Bottlerocket AMI
331 case MachineImageType.BOTTLEROCKET: {
332 autoScalingGroup.addUserData(
333 // Connect to the cluster
334 // Source: https://github.com/bottlerocket-os/bottlerocket/blob/develop/QUICKSTART-ECS.md#connecting-to-your-cluster
335 '[settings.ecs]', `cluster = "${this.clusterName}"`);
336 // Enabling SSM
337 // Source: https://github.com/bottlerocket-os/bottlerocket/blob/develop/QUICKSTART-ECS.md#enabling-ssm
338 autoScalingGroup.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'));
339 // required managed policy
340 autoScalingGroup.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonEC2ContainerServiceforEC2Role'));
341 break;
342 }
343 default:
344 // Amazon ECS-optimized AMI for Amazon Linux 2
345 autoScalingGroup.addUserData(`echo ECS_CLUSTER=${this.clusterName} >> /etc/ecs/ecs.config`);
346 if (!options.canContainersAccessInstanceRole) {
347 // Deny containers access to instance metadata service
348 // Source: https://docs.aws.amazon.com/AmazonECS/latest/developerguide/instance_IAM_role.html
349 autoScalingGroup.addUserData('sudo iptables --insert FORWARD 1 --in-interface docker+ --destination 169.254.169.254/32 --jump DROP');
350 autoScalingGroup.addUserData('sudo service iptables save');
351 // The following is only for AwsVpc networking mode, but doesn't hurt for the other modes.
352 autoScalingGroup.addUserData('echo ECS_AWSVPC_BLOCK_IMDS=true >> /etc/ecs/ecs.config');
353 }
354 if (autoScalingGroup.spotPrice && options.spotInstanceDraining) {
355 autoScalingGroup.addUserData('echo ECS_ENABLE_SPOT_INSTANCE_DRAINING=true >> /etc/ecs/ecs.config');
356 }
357 }
358 }
359 // ECS instances must be able to do these things
360 // Source: https://docs.aws.amazon.com/AmazonECS/latest/developerguide/instance_IAM_role.html
361 // But, scoped down to minimal permissions required.
362 // Notes:
363 // - 'ecs:CreateCluster' removed. The cluster already exists.
364 autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({
365 actions: [
366 'ecs:DeregisterContainerInstance',
367 'ecs:RegisterContainerInstance',
368 'ecs:Submit*',
369 ],
370 resources: [
371 this.clusterArn,
372 ],
373 }));
374 autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({
375 actions: [
376 // These act on a cluster instance, and the instance doesn't exist until the service starts.
377 // Thus, scope to the cluster using a condition.
378 // See: https://docs.aws.amazon.com/IAM/latest/UserGuide/list_amazonelasticcontainerservice.html
379 'ecs:Poll',
380 'ecs:StartTelemetrySession',
381 ],
382 resources: ['*'],
383 conditions: {
384 ArnEquals: { 'ecs:cluster': this.clusterArn },
385 },
386 }));
387 autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({
388 actions: [
389 // These do not support resource constraints, and must be resource '*'
390 'ecs:DiscoverPollEndpoint',
391 'ecr:GetAuthorizationToken',
392 // Preserved for backwards compatibility.
393 // Users are able to enable cloudwatch agent using CDK. Existing
394 // customers might be installing CW agent as part of user-data so if we
395 // remove these permissions we will break that customer use cases.
396 'logs:CreateLogStream',
397 'logs:PutLogEvents',
398 ],
399 resources: ['*'],
400 }));
401 // 0 disables, otherwise forward to underlying implementation which picks the sane default
402 if (!options.taskDrainTime || options.taskDrainTime.toSeconds() !== 0) {
403 new instance_drain_hook_1.InstanceDrainHook(autoScalingGroup, 'DrainECSHook', {
404 autoScalingGroup,
405 cluster: this,
406 drainTime: options.taskDrainTime,
407 topicEncryptionKey: options.topicEncryptionKey,
408 });
409 }
410 }
411 /**
412 * This method enables the Fargate or Fargate Spot capacity providers on the cluster.
413 *
414 * @param provider the capacity provider to add to this cluster.
415 * @deprecated Use {@link enableFargateCapacityProviders} instead.
416 * @see {@link addAsgCapacityProvider} to add an Auto Scaling Group capacity provider to the cluster.
417 */
418 addCapacityProvider(provider) {
419 try {
420 jsiiDeprecationWarnings.print("@aws-cdk/aws-ecs.Cluster#addCapacityProvider", "Use {@link enableFargateCapacityProviders} instead.");
421 }
422 catch (error) {
423 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
424 Error.captureStackTrace(error, this.addCapacityProvider);
425 }
426 throw error;
427 }
428 if (!(provider === 'FARGATE' || provider === 'FARGATE_SPOT')) {
429 throw new Error('CapacityProvider not supported');
430 }
431 if (!this._capacityProviderNames.includes(provider)) {
432 this._capacityProviderNames.push(provider);
433 }
434 }
435 configureWindowsAutoScalingGroup(autoScalingGroup, options = {}) {
436 // clear the cache of the agent
437 autoScalingGroup.addUserData('Remove-Item -Recurse C:\\ProgramData\\Amazon\\ECS\\Cache');
438 // pull the latest ECS Tools
439 autoScalingGroup.addUserData('Import-Module ECSTools');
440 // set the cluster name environment variable
441 autoScalingGroup.addUserData(`[Environment]::SetEnvironmentVariable("ECS_CLUSTER", "${this.clusterName}", "Machine")`);
442 autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable("ECS_ENABLE_AWSLOGS_EXECUTIONROLE_OVERRIDE", "true", "Machine")');
443 // tslint:disable-next-line: max-line-length
444 autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable("ECS_AVAILABLE_LOGGING_DRIVERS", \'["json-file","awslogs"]\', "Machine")');
445 // enable instance draining
446 if (autoScalingGroup.spotPrice && options.spotInstanceDraining) {
447 autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable("ECS_ENABLE_SPOT_INSTANCE_DRAINING", "true", "Machine")');
448 }
449 // enable task iam role
450 if (!options.canContainersAccessInstanceRole) {
451 autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable("ECS_ENABLE_TASK_IAM_ROLE", "true", "Machine")');
452 autoScalingGroup.addUserData(`Initialize-ECSAgent -Cluster '${this.clusterName}' -EnableTaskIAMRole`);
453 }
454 else {
455 autoScalingGroup.addUserData(`Initialize-ECSAgent -Cluster '${this.clusterName}'`);
456 }
457 }
458 /**
459 * Getter for autoscaling group added to cluster
460 */
461 get autoscalingGroup() {
462 return this._autoscalingGroup;
463 }
464 /**
465 * Whether the cluster has EC2 capacity associated with it
466 */
467 get hasEc2Capacity() {
468 return this._hasEc2Capacity;
469 }
470 /**
471 * Getter for execute command configuration associated with the cluster.
472 */
473 get executeCommandConfiguration() {
474 return this._executeCommandConfiguration;
475 }
476 /**
477 * This method returns the CloudWatch metric for this clusters CPU reservation.
478 *
479 * @default average over 5 minutes
480 */
481 metricCpuReservation(props) {
482 return this.cannedMetric(ecs_canned_metrics_generated_1.ECSMetrics.cpuReservationAverage, props);
483 }
484 /**
485 * This method returns the CloudWatch metric for this clusters CPU utilization.
486 *
487 * @default average over 5 minutes
488 */
489 metricCpuUtilization(props) {
490 return this.cannedMetric(ecs_canned_metrics_generated_1.ECSMetrics.cpuUtilizationAverage, props);
491 }
492 /**
493 * This method returns the CloudWatch metric for this clusters memory reservation.
494 *
495 * @default average over 5 minutes
496 */
497 metricMemoryReservation(props) {
498 return this.cannedMetric(ecs_canned_metrics_generated_1.ECSMetrics.memoryReservationAverage, props);
499 }
500 /**
501 * This method returns the CloudWatch metric for this clusters memory utilization.
502 *
503 * @default average over 5 minutes
504 */
505 metricMemoryUtilization(props) {
506 return this.cannedMetric(ecs_canned_metrics_generated_1.ECSMetrics.memoryUtilizationAverage, props);
507 }
508 /**
509 * This method returns the specifed CloudWatch metric for this cluster.
510 */
511 metric(metricName, props) {
512 return new cloudwatch.Metric({
513 namespace: 'AWS/ECS',
514 metricName,
515 dimensionsMap: { ClusterName: this.clusterName },
516 ...props,
517 }).attachTo(this);
518 }
519 cannedMetric(fn, props) {
520 return new cloudwatch.Metric({
521 ...fn({ ClusterName: this.clusterName }),
522 ...props,
523 }).attachTo(this);
524 }
525}
526exports.Cluster = Cluster;
527_a = JSII_RTTI_SYMBOL_1;
528Cluster[_a] = { fqn: "@aws-cdk/aws-ecs.Cluster", version: "1.156.1" };
529/**
530 * An Cluster that has been imported
531 */
532class ImportedCluster extends core_1.Resource {
533 /**
534 * Constructs a new instance of the ImportedCluster class.
535 */
536 constructor(scope, id, props) {
537 var _c;
538 super(scope, id);
539 /**
540 * Security group of the cluster instances
541 */
542 this.connections = new ec2.Connections();
543 this.clusterName = props.clusterName;
544 this.vpc = props.vpc;
545 this.hasEc2Capacity = props.hasEc2Capacity !== false;
546 this._defaultCloudMapNamespace = props.defaultCloudMapNamespace;
547 this._executeCommandConfiguration = props.executeCommandConfiguration;
548 this.clusterArn = (_c = props.clusterArn) !== null && _c !== void 0 ? _c : core_1.Stack.of(this).formatArn({
549 service: 'ecs',
550 resource: 'cluster',
551 resourceName: props.clusterName,
552 });
553 this.connections = new ec2.Connections({
554 securityGroups: props.securityGroups,
555 });
556 }
557 get defaultCloudMapNamespace() {
558 return this._defaultCloudMapNamespace;
559 }
560 get executeCommandConfiguration() {
561 return this._executeCommandConfiguration;
562 }
563}
564var ContainerInsights;
565(function (ContainerInsights) {
566 /**
567 * Enable CloudWatch Container Insights for the cluster
568 */
569 ContainerInsights["ENABLED"] = "enabled";
570 /**
571 * Disable CloudWatch Container Insights for the cluster
572 */
573 ContainerInsights["DISABLED"] = "disabled";
574})(ContainerInsights || (ContainerInsights = {}));
575/**
576 * The log settings to use to for logging the execute command session. For more information, see
577 * [Logging] https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-executecommandconfiguration.html#cfn-ecs-cluster-executecommandconfiguration-logging
578 */
579var ExecuteCommandLogging;
580(function (ExecuteCommandLogging) {
581 /**
582 * The execute command session is not logged.
583 */
584 ExecuteCommandLogging["NONE"] = "NONE";
585 /**
586 * The awslogs configuration in the task definition is used. If no logging parameter is specified, it defaults to this value. If no awslogs log driver is configured in the task definition, the output won't be logged.
587 */
588 ExecuteCommandLogging["DEFAULT"] = "DEFAULT";
589 /**
590 * Specify the logging details as a part of logConfiguration.
591 */
592 ExecuteCommandLogging["OVERRIDE"] = "OVERRIDE";
593})(ExecuteCommandLogging = exports.ExecuteCommandLogging || (exports.ExecuteCommandLogging = {}));
594/**
595 * An Auto Scaling Group Capacity Provider. This allows an ECS cluster to target
596 * a specific EC2 Auto Scaling Group for the placement of tasks. Optionally (and
597 * recommended), ECS can manage the number of instances in the ASG to fit the
598 * tasks, and can ensure that instances are not prematurely terminated while
599 * there are still tasks running on them.
600 */
601class AsgCapacityProvider extends core_2.Construct {
602 constructor(scope, id, props) {
603 var _c;
604 super(scope, id);
605 try {
606 jsiiDeprecationWarnings._aws_cdk_aws_ecs_AsgCapacityProviderProps(props);
607 }
608 catch (error) {
609 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
610 Error.captureStackTrace(error, this.constructor);
611 }
612 throw error;
613 }
614 this.autoScalingGroup = props.autoScalingGroup;
615 this.machineImageType = (_c = props.machineImageType) !== null && _c !== void 0 ? _c : MachineImageType.AMAZON_LINUX_2;
616 this.enableManagedTerminationProtection =
617 props.enableManagedTerminationProtection === undefined ? true : props.enableManagedTerminationProtection;
618 if (this.enableManagedTerminationProtection) {
619 this.autoScalingGroup.protectNewInstancesFromScaleIn();
620 }
621 if (props.capacityProviderName) {
622 if (!(/^(?!aws|ecs|fargate).+/gm.test(props.capacityProviderName))) {
623 throw new Error(`Invalid Capacity Provider Name: ${props.capacityProviderName}, If a name is specified, it cannot start with aws, ecs, or fargate.`);
624 }
625 }
626 const capacityProvider = new ecs_generated_1.CfnCapacityProvider(this, id, {
627 name: props.capacityProviderName,
628 autoScalingGroupProvider: {
629 autoScalingGroupArn: this.autoScalingGroup.autoScalingGroupName,
630 managedScaling: props.enableManagedScaling === false ? undefined : {
631 status: 'ENABLED',
632 targetCapacity: props.targetCapacityPercent || 100,
633 maximumScalingStepSize: props.maximumScalingStepSize,
634 minimumScalingStepSize: props.minimumScalingStepSize,
635 },
636 managedTerminationProtection: this.enableManagedTerminationProtection ? 'ENABLED' : 'DISABLED',
637 },
638 });
639 this.capacityProviderName = capacityProvider.ref;
640 }
641}
642exports.AsgCapacityProvider = AsgCapacityProvider;
643_b = JSII_RTTI_SYMBOL_1;
644AsgCapacityProvider[_b] = { fqn: "@aws-cdk/aws-ecs.AsgCapacityProvider", version: "1.156.1" };
645/**
646 * A visitor that adds a capacity provider association to a Cluster only if
647 * the caller created any EC2 Capacity Providers.
648 */
649class MaybeCreateCapacityProviderAssociations {
650 constructor(scope, id, capacityProviders) {
651 this.scope = scope;
652 this.id = id;
653 this.capacityProviders = capacityProviders;
654 }
655 visit(node) {
656 if (node instanceof Cluster) {
657 if (this.capacityProviders.length > 0 && !this.resource) {
658 const resource = new ecs_generated_1.CfnClusterCapacityProviderAssociations(this.scope, this.id, {
659 cluster: node.clusterName,
660 defaultCapacityProviderStrategy: [],
661 capacityProviders: core_1.Lazy.list({ produce: () => this.capacityProviders }),
662 });
663 this.resource = resource;
664 }
665 }
666 }
667}
668function isBottleRocketImage(image) {
669 return image instanceof amis_1.BottleRocketImage;
670}
671//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cluster.js","sourceRoot":"","sources":["cluster.ts"],"names":[],"mappings":"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","sourcesContent":["import * as autoscaling from '@aws-cdk/aws-autoscaling';\nimport * as cloudwatch from '@aws-cdk/aws-cloudwatch';\nimport * as ec2 from '@aws-cdk/aws-ec2';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as kms from '@aws-cdk/aws-kms';\nimport * as logs from '@aws-cdk/aws-logs';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cloudmap from '@aws-cdk/aws-servicediscovery';\nimport { Duration, Lazy, IResource, Resource, Stack, Aspects, IAspect, IConstruct, ArnFormat } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { BottleRocketImage, EcsOptimizedAmi } from './amis';\nimport { InstanceDrainHook } from './drain-hook/instance-drain-hook';\nimport { ECSMetrics } from './ecs-canned-metrics.generated';\nimport { CfnCluster, CfnCapacityProvider, CfnClusterCapacityProviderAssociations } from './ecs.generated';\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from '@aws-cdk/core';\n\n/**\n * The properties used to define an ECS cluster.\n */\nexport interface ClusterProps {\n  /**\n   * The name for the cluster.\n   *\n   * @default CloudFormation-generated name\n   */\n  readonly clusterName?: string;\n\n  /**\n   * The VPC where your ECS instances will be running or your ENIs will be deployed\n   *\n   * @default - creates a new VPC with two AZs\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * The service discovery namespace created in this cluster\n   *\n   * @default - no service discovery namespace created, you can use `addDefaultCloudMapNamespace` to add a\n   * default service discovery namespace later.\n   */\n  readonly defaultCloudMapNamespace?: CloudMapNamespaceOptions;\n\n  /**\n   * The ec2 capacity to add to the cluster\n   *\n   * @default - no EC2 capacity will be added, you can use `addCapacity` to add capacity later.\n   */\n  readonly capacity?: AddCapacityOptions;\n\n  /**\n   * The capacity providers to add to the cluster\n   *\n   * @default - None. Currently only FARGATE and FARGATE_SPOT are supported.\n   * @deprecated Use {@link ClusterProps.enableFargateCapacityProviders} instead.\n   */\n  readonly capacityProviders?: string[];\n\n  /**\n   * Whether to enable Fargate Capacity Providers\n   *\n   * @default false\n   */\n  readonly enableFargateCapacityProviders?: boolean;\n\n  /**\n   * If true CloudWatch Container Insights will be enabled for the cluster\n   *\n   * @default - Container Insights will be disabled for this cluser.\n   */\n  readonly containerInsights?: boolean;\n\n  /**\n   * The execute command configuration for the cluster\n   *\n   * @default - no configuration will be provided.\n   */\n  readonly executeCommandConfiguration?: ExecuteCommandConfiguration;\n}\n\n/**\n * The machine image type\n */\nexport enum MachineImageType {\n  /**\n   * Amazon ECS-optimized Amazon Linux 2 AMI\n   */\n  AMAZON_LINUX_2,\n  /**\n   * Bottlerocket AMI\n   */\n  BOTTLEROCKET\n}\n\n/**\n * A regional grouping of one or more container instances on which you can run tasks and services.\n */\nexport class Cluster extends Resource implements ICluster {\n  /**\n   * Import an existing cluster to the stack from its attributes.\n   */\n  public static fromClusterAttributes(scope: Construct, id: string, attrs: ClusterAttributes): ICluster {\n    return new ImportedCluster(scope, id, attrs);\n  }\n\n  /**\n   * Import an existing cluster to the stack from the cluster ARN.\n   * This does not provide access to the vpc, hasEc2Capacity, or connections -\n   * use the `fromClusterAttributes` method to access those properties.\n   */\n  public static fromClusterArn(scope: Construct, id: string, clusterArn: string): ICluster {\n    const stack = Stack.of(scope);\n    const arn = stack.splitArn(clusterArn, ArnFormat.SLASH_RESOURCE_NAME);\n    const clusterName = arn.resourceName;\n\n    if (!clusterName) {\n      throw new Error(`Missing required Cluster Name from Cluster ARN: ${clusterArn}`);\n    }\n\n    const errorSuffix = 'is not available for a Cluster imported using fromClusterArn(), please use fromClusterAttributes() instead.';\n\n    class Import extends Resource implements ICluster {\n      public readonly clusterArn = clusterArn;\n      public readonly clusterName = clusterName!;\n      get hasEc2Capacity(): boolean {\n        throw new Error(`hasEc2Capacity ${errorSuffix}`);\n      }\n      get connections(): ec2.Connections {\n        throw new Error(`connections ${errorSuffix}`);\n      }\n      get vpc(): ec2.IVpc {\n        throw new Error(`vpc ${errorSuffix}`);\n      }\n    }\n\n    return new Import(scope, id, {\n      environmentFromArn: clusterArn,\n    });\n  }\n\n  /**\n   * Manage the allowed network connections for the cluster with Security Groups.\n   */\n  public readonly connections: ec2.Connections = new ec2.Connections();\n\n  /**\n   * The VPC associated with the cluster.\n   */\n  public readonly vpc: ec2.IVpc;\n\n  /**\n   * The Amazon Resource Name (ARN) that identifies the cluster.\n   */\n  public readonly clusterArn: string;\n\n  /**\n   * The name of the cluster.\n   */\n  public readonly clusterName: string;\n\n  /**\n   * The names of both ASG and Fargate capacity providers associated with the cluster.\n   */\n  private _capacityProviderNames: string[] = [];\n\n  /**\n   * The AWS Cloud Map namespace to associate with the cluster.\n   */\n  private _defaultCloudMapNamespace?: cloudmap.INamespace;\n\n  /**\n   * Specifies whether the cluster has EC2 instance capacity.\n   */\n  private _hasEc2Capacity: boolean = false;\n\n  /**\n   * The autoscaling group for added Ec2 capacity\n   */\n  private _autoscalingGroup?: autoscaling.IAutoScalingGroup;\n\n  /**\n   * The execute command configuration for the cluster\n   */\n  private _executeCommandConfiguration?: ExecuteCommandConfiguration;\n\n  /**\n   * Constructs a new instance of the Cluster class.\n   */\n  constructor(scope: Construct, id: string, props: ClusterProps = {}) {\n    super(scope, id, {\n      physicalName: props.clusterName,\n    });\n\n    /**\n     * clusterSettings needs to be undefined if containerInsights is not explicitly set in order to allow any\n     * containerInsights settings on the account to apply.  See:\n     * https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-clustersettings.html#cfn-ecs-cluster-clustersettings-value\n    */\n    let clusterSettings = undefined;\n    if (props.containerInsights !== undefined) {\n      clusterSettings = [{ name: 'containerInsights', value: props.containerInsights ? ContainerInsights.ENABLED : ContainerInsights.DISABLED }];\n    }\n\n    this._capacityProviderNames = props.capacityProviders ?? [];\n    if (props.enableFargateCapacityProviders) {\n      this.enableFargateCapacityProviders();\n    }\n\n    if (props.executeCommandConfiguration) {\n      if ((props.executeCommandConfiguration.logging === ExecuteCommandLogging.OVERRIDE) !==\n        (props.executeCommandConfiguration.logConfiguration !== undefined)) {\n        throw new Error('Execute command log configuration must only be specified when logging is OVERRIDE.');\n      }\n      this._executeCommandConfiguration = props.executeCommandConfiguration;\n    }\n\n    const cluster = new CfnCluster(this, 'Resource', {\n      clusterName: this.physicalName,\n      clusterSettings,\n      configuration: this._executeCommandConfiguration && this.renderExecuteCommandConfiguration(),\n    });\n\n    this.clusterArn = this.getResourceArnAttribute(cluster.attrArn, {\n      service: 'ecs',\n      resource: 'cluster',\n      resourceName: this.physicalName,\n    });\n    this.clusterName = this.getResourceNameAttribute(cluster.ref);\n\n    this.vpc = props.vpc || new ec2.Vpc(this, 'Vpc', { maxAzs: 2 });\n\n\n    this._defaultCloudMapNamespace = props.defaultCloudMapNamespace !== undefined\n      ? this.addDefaultCloudMapNamespace(props.defaultCloudMapNamespace)\n      : undefined;\n\n    this._autoscalingGroup = props.capacity !== undefined\n      ? this.addCapacity('DefaultAutoScalingGroup', props.capacity)\n      : undefined;\n\n    // Only create cluster capacity provider associations if there are any EC2\n    // capacity providers. Ordinarily we'd just add the construct to the tree\n    // since it's harmless, but we'd prefer not to add unexpected new\n    // resources to the stack which could surprise users working with\n    // brown-field CDK apps and stacks.\n    Aspects.of(this).add(new MaybeCreateCapacityProviderAssociations(this, id, this._capacityProviderNames));\n  }\n\n  /**\n   * Enable the Fargate capacity providers for this cluster.\n   */\n  public enableFargateCapacityProviders() {\n    for (const provider of ['FARGATE', 'FARGATE_SPOT']) {\n      if (!this._capacityProviderNames.includes(provider)) {\n        this._capacityProviderNames.push(provider);\n      }\n    }\n  }\n\n  private renderExecuteCommandConfiguration() : CfnCluster.ClusterConfigurationProperty {\n    return {\n      executeCommandConfiguration: {\n        kmsKeyId: this._executeCommandConfiguration?.kmsKey?.keyArn,\n        logConfiguration: this._executeCommandConfiguration?.logConfiguration && this.renderExecuteCommandLogConfiguration(),\n        logging: this._executeCommandConfiguration?.logging,\n      },\n    };\n  }\n\n  private renderExecuteCommandLogConfiguration(): CfnCluster.ExecuteCommandLogConfigurationProperty {\n    const logConfiguration = this._executeCommandConfiguration?.logConfiguration;\n    if (logConfiguration?.s3EncryptionEnabled && !logConfiguration?.s3Bucket) {\n      throw new Error('You must specify an S3 bucket name in the execute command log configuration to enable S3 encryption.');\n    }\n    if (logConfiguration?.cloudWatchEncryptionEnabled && !logConfiguration?.cloudWatchLogGroup) {\n      throw new Error('You must specify a CloudWatch log group in the execute command log configuration to enable CloudWatch encryption.');\n    }\n    return {\n      cloudWatchEncryptionEnabled: logConfiguration?.cloudWatchEncryptionEnabled,\n      cloudWatchLogGroupName: logConfiguration?.cloudWatchLogGroup?.logGroupName,\n      s3BucketName: logConfiguration?.s3Bucket?.bucketName,\n      s3EncryptionEnabled: logConfiguration?.s3EncryptionEnabled,\n      s3KeyPrefix: logConfiguration?.s3KeyPrefix,\n    };\n  }\n\n  /**\n   * Add an AWS Cloud Map DNS namespace for this cluster.\n   * NOTE: HttpNamespaces are not supported, as ECS always requires a DNSConfig when registering an instance to a Cloud\n   * Map service.\n   */\n  public addDefaultCloudMapNamespace(options: CloudMapNamespaceOptions): cloudmap.INamespace {\n    if (this._defaultCloudMapNamespace !== undefined) {\n      throw new Error('Can only add default namespace once.');\n    }\n\n    const namespaceType = options.type !== undefined\n      ? options.type\n      : cloudmap.NamespaceType.DNS_PRIVATE;\n\n    const sdNamespace = namespaceType === cloudmap.NamespaceType.DNS_PRIVATE ?\n      new cloudmap.PrivateDnsNamespace(this, 'DefaultServiceDiscoveryNamespace', {\n        name: options.name,\n        vpc: this.vpc,\n      }) :\n      new cloudmap.PublicDnsNamespace(this, 'DefaultServiceDiscoveryNamespace', {\n        name: options.name,\n      });\n\n    this._defaultCloudMapNamespace = sdNamespace;\n\n    return sdNamespace;\n  }\n\n  /**\n   * Getter for namespace added to cluster\n   */\n  public get defaultCloudMapNamespace(): cloudmap.INamespace | undefined {\n    return this._defaultCloudMapNamespace;\n  }\n\n  /**\n   * It is highly recommended to use {@link Cluster.addAsgCapacityProvider} instead of this method.\n   *\n   * This method adds compute capacity to a cluster by creating an AutoScalingGroup with the specified options.\n   *\n   * Returns the AutoScalingGroup so you can add autoscaling settings to it.\n   */\n  public addCapacity(id: string, options: AddCapacityOptions): autoscaling.AutoScalingGroup {\n    // Do 2-way defaulting here: if the machineImageType is BOTTLEROCKET, pick the right AMI.\n    // Otherwise, determine the machineImageType from the given AMI.\n    const machineImage = options.machineImage ??\n      (options.machineImageType === MachineImageType.BOTTLEROCKET ? new BottleRocketImage({\n        architecture: options.instanceType.architecture,\n      }) : new EcsOptimizedAmi());\n\n    const machineImageType = options.machineImageType ??\n      (isBottleRocketImage(machineImage) ? MachineImageType.BOTTLEROCKET : MachineImageType.AMAZON_LINUX_2);\n\n    const autoScalingGroup = new autoscaling.AutoScalingGroup(this, id, {\n      vpc: this.vpc,\n      machineImage,\n      updateType: options.updateType || autoscaling.UpdateType.REPLACING_UPDATE,\n      ...options,\n    });\n\n    this.addAutoScalingGroup(autoScalingGroup, {\n      machineImageType: machineImageType,\n      ...options,\n    });\n\n    return autoScalingGroup;\n  }\n\n  /**\n   * This method adds an Auto Scaling Group Capacity Provider to a cluster.\n   *\n   * @param provider the capacity provider to add to this cluster.\n   */\n  public addAsgCapacityProvider(provider: AsgCapacityProvider, options: AddAutoScalingGroupCapacityOptions= {}) {\n    // Don't add the same capacity provider more than once.\n    if (this._capacityProviderNames.includes(provider.capacityProviderName)) {\n      return;\n    }\n    this._hasEc2Capacity = true;\n    this.configureAutoScalingGroup(provider.autoScalingGroup, {\n      ...options,\n      machineImageType: provider.machineImageType,\n      // Don't enable the instance-draining lifecycle hook if managed termination protection is enabled\n      taskDrainTime: provider.enableManagedTerminationProtection ? Duration.seconds(0) : options.taskDrainTime,\n    });\n\n    this._capacityProviderNames.push(provider.capacityProviderName);\n  }\n\n  /**\n   * This method adds compute capacity to a cluster using the specified AutoScalingGroup.\n   *\n   * @deprecated Use {@link Cluster.addAsgCapacityProvider} instead.\n   * @param autoScalingGroup the ASG to add to this cluster.\n   * [disable-awslint:ref-via-interface] is needed in order to install the ECS\n   * agent by updating the ASGs user data.\n   */\n  public addAutoScalingGroup(autoScalingGroup: autoscaling.AutoScalingGroup, options: AddAutoScalingGroupCapacityOptions = {}) {\n    this._hasEc2Capacity = true;\n    this.connections.connections.addSecurityGroup(...autoScalingGroup.connections.securityGroups);\n    this.configureAutoScalingGroup(autoScalingGroup, options);\n  }\n\n  private configureAutoScalingGroup(autoScalingGroup: autoscaling.AutoScalingGroup, options: AddAutoScalingGroupCapacityOptions = {}) {\n    if (autoScalingGroup.osType === ec2.OperatingSystemType.WINDOWS) {\n      this.configureWindowsAutoScalingGroup(autoScalingGroup, options);\n    } else {\n      // Tie instances to cluster\n      switch (options.machineImageType) {\n        // Bottlerocket AMI\n        case MachineImageType.BOTTLEROCKET: {\n          autoScalingGroup.addUserData(\n            // Connect to the cluster\n            // Source: https://github.com/bottlerocket-os/bottlerocket/blob/develop/QUICKSTART-ECS.md#connecting-to-your-cluster\n            '[settings.ecs]',\n            `cluster = \"${this.clusterName}\"`,\n          );\n          // Enabling SSM\n          // Source: https://github.com/bottlerocket-os/bottlerocket/blob/develop/QUICKSTART-ECS.md#enabling-ssm\n          autoScalingGroup.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'));\n          // required managed policy\n          autoScalingGroup.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AmazonEC2ContainerServiceforEC2Role'));\n          break;\n        }\n        default:\n          // Amazon ECS-optimized AMI for Amazon Linux 2\n          autoScalingGroup.addUserData(`echo ECS_CLUSTER=${this.clusterName} >> /etc/ecs/ecs.config`);\n          if (!options.canContainersAccessInstanceRole) {\n            // Deny containers access to instance metadata service\n            // Source: https://docs.aws.amazon.com/AmazonECS/latest/developerguide/instance_IAM_role.html\n            autoScalingGroup.addUserData('sudo iptables --insert FORWARD 1 --in-interface docker+ --destination 169.254.169.254/32 --jump DROP');\n            autoScalingGroup.addUserData('sudo service iptables save');\n            // The following is only for AwsVpc networking mode, but doesn't hurt for the other modes.\n            autoScalingGroup.addUserData('echo ECS_AWSVPC_BLOCK_IMDS=true >> /etc/ecs/ecs.config');\n          }\n\n          if (autoScalingGroup.spotPrice && options.spotInstanceDraining) {\n            autoScalingGroup.addUserData('echo ECS_ENABLE_SPOT_INSTANCE_DRAINING=true >> /etc/ecs/ecs.config');\n          }\n      }\n    }\n\n    // ECS instances must be able to do these things\n    // Source: https://docs.aws.amazon.com/AmazonECS/latest/developerguide/instance_IAM_role.html\n    // But, scoped down to minimal permissions required.\n    //  Notes:\n    //   - 'ecs:CreateCluster' removed. The cluster already exists.\n    autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'ecs:DeregisterContainerInstance',\n        'ecs:RegisterContainerInstance',\n        'ecs:Submit*',\n      ],\n      resources: [\n        this.clusterArn,\n      ],\n    }));\n    autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        // These act on a cluster instance, and the instance doesn't exist until the service starts.\n        // Thus, scope to the cluster using a condition.\n        // See: https://docs.aws.amazon.com/IAM/latest/UserGuide/list_amazonelasticcontainerservice.html\n        'ecs:Poll',\n        'ecs:StartTelemetrySession',\n      ],\n      resources: ['*'],\n      conditions: {\n        ArnEquals: { 'ecs:cluster': this.clusterArn },\n      },\n    }));\n    autoScalingGroup.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        // These do not support resource constraints, and must be resource '*'\n        'ecs:DiscoverPollEndpoint',\n        'ecr:GetAuthorizationToken',\n        // Preserved for backwards compatibility.\n        // Users are able to enable cloudwatch agent using CDK. Existing\n        // customers might be installing CW agent as part of user-data so if we\n        // remove these permissions we will break that customer use cases.\n        'logs:CreateLogStream',\n        'logs:PutLogEvents',\n      ],\n      resources: ['*'],\n    }));\n\n    // 0 disables, otherwise forward to underlying implementation which picks the sane default\n    if (!options.taskDrainTime || options.taskDrainTime.toSeconds() !== 0) {\n      new InstanceDrainHook(autoScalingGroup, 'DrainECSHook', {\n        autoScalingGroup,\n        cluster: this,\n        drainTime: options.taskDrainTime,\n        topicEncryptionKey: options.topicEncryptionKey,\n      });\n    }\n  }\n\n  /**\n   * This method enables the Fargate or Fargate Spot capacity providers on the cluster.\n   *\n   * @param provider the capacity provider to add to this cluster.\n   * @deprecated Use {@link enableFargateCapacityProviders} instead.\n   * @see {@link addAsgCapacityProvider} to add an Auto Scaling Group capacity provider to the cluster.\n   */\n  public addCapacityProvider(provider: string) {\n    if (!(provider === 'FARGATE' || provider === 'FARGATE_SPOT')) {\n      throw new Error('CapacityProvider not supported');\n    }\n\n    if (!this._capacityProviderNames.includes(provider)) {\n      this._capacityProviderNames.push(provider);\n    }\n  }\n\n  private configureWindowsAutoScalingGroup(autoScalingGroup: autoscaling.AutoScalingGroup, options: AddAutoScalingGroupCapacityOptions = {}) {\n    // clear the cache of the agent\n    autoScalingGroup.addUserData('Remove-Item -Recurse C:\\\\ProgramData\\\\Amazon\\\\ECS\\\\Cache');\n\n    // pull the latest ECS Tools\n    autoScalingGroup.addUserData('Import-Module ECSTools');\n\n    // set the cluster name environment variable\n    autoScalingGroup.addUserData(`[Environment]::SetEnvironmentVariable(\"ECS_CLUSTER\", \"${this.clusterName}\", \"Machine\")`);\n    autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable(\"ECS_ENABLE_AWSLOGS_EXECUTIONROLE_OVERRIDE\", \"true\", \"Machine\")');\n    // tslint:disable-next-line: max-line-length\n    autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable(\"ECS_AVAILABLE_LOGGING_DRIVERS\", \\'[\"json-file\",\"awslogs\"]\\', \"Machine\")');\n\n    // enable instance draining\n    if (autoScalingGroup.spotPrice && options.spotInstanceDraining) {\n      autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable(\"ECS_ENABLE_SPOT_INSTANCE_DRAINING\", \"true\", \"Machine\")');\n    }\n\n    // enable task iam role\n    if (!options.canContainersAccessInstanceRole) {\n      autoScalingGroup.addUserData('[Environment]::SetEnvironmentVariable(\"ECS_ENABLE_TASK_IAM_ROLE\", \"true\", \"Machine\")');\n      autoScalingGroup.addUserData(`Initialize-ECSAgent -Cluster '${this.clusterName}' -EnableTaskIAMRole`);\n    } else {\n      autoScalingGroup.addUserData(`Initialize-ECSAgent -Cluster '${this.clusterName}'`);\n    }\n  }\n\n  /**\n   * Getter for autoscaling group added to cluster\n   */\n  public get autoscalingGroup(): autoscaling.IAutoScalingGroup | undefined {\n    return this._autoscalingGroup;\n  }\n\n  /**\n   * Whether the cluster has EC2 capacity associated with it\n   */\n  public get hasEc2Capacity(): boolean {\n    return this._hasEc2Capacity;\n  }\n\n  /**\n   * Getter for execute command configuration associated with the cluster.\n   */\n  public get executeCommandConfiguration(): ExecuteCommandConfiguration | undefined {\n    return this._executeCommandConfiguration;\n  }\n\n  /**\n   * This method returns the CloudWatch metric for this clusters CPU reservation.\n   *\n   * @default average over 5 minutes\n   */\n  public metricCpuReservation(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.cannedMetric(ECSMetrics.cpuReservationAverage, props);\n  }\n\n  /**\n   * This method returns the CloudWatch metric for this clusters CPU utilization.\n   *\n   * @default average over 5 minutes\n   */\n  public metricCpuUtilization(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.cannedMetric(ECSMetrics.cpuUtilizationAverage, props);\n  }\n\n  /**\n   * This method returns the CloudWatch metric for this clusters memory reservation.\n   *\n   * @default average over 5 minutes\n   */\n  public metricMemoryReservation(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.cannedMetric(ECSMetrics.memoryReservationAverage, props);\n  }\n\n  /**\n   * This method returns the CloudWatch metric for this clusters memory utilization.\n   *\n   * @default average over 5 minutes\n   */\n  public metricMemoryUtilization(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.cannedMetric(ECSMetrics.memoryUtilizationAverage, props);\n  }\n\n  /**\n   * This method returns the specifed CloudWatch metric for this cluster.\n   */\n  public metric(metricName: string, props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return new cloudwatch.Metric({\n      namespace: 'AWS/ECS',\n      metricName,\n      dimensionsMap: { ClusterName: this.clusterName },\n      ...props,\n    }).attachTo(this);\n  }\n\n  private cannedMetric(\n    fn: (dims: { ClusterName: string }) => cloudwatch.MetricProps,\n    props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return new cloudwatch.Metric({\n      ...fn({ ClusterName: this.clusterName }),\n      ...props,\n    }).attachTo(this);\n  }\n}\n\n/**\n * A regional grouping of one or more container instances on which you can run tasks and services.\n */\nexport interface ICluster extends IResource {\n  /**\n   * The name of the cluster.\n   * @attribute\n   */\n  readonly clusterName: string;\n\n  /**\n   * The Amazon Resource Name (ARN) that identifies the cluster.\n   * @attribute\n   */\n  readonly clusterArn: string;\n\n  /**\n   * The VPC associated with the cluster.\n   */\n  readonly vpc: ec2.IVpc;\n\n  /**\n   * Manage the allowed network connections for the cluster with Security Groups.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Specifies whether the cluster has EC2 instance capacity.\n   */\n  readonly hasEc2Capacity: boolean;\n\n  /**\n   * The AWS Cloud Map namespace to associate with the cluster.\n   */\n  readonly defaultCloudMapNamespace?: cloudmap.INamespace;\n\n  /**\n   * The autoscaling group added to the cluster if capacity is associated to the cluster\n   */\n  readonly autoscalingGroup?: autoscaling.IAutoScalingGroup;\n\n  /**\n   * The execute command configuration for the cluster\n   */\n  readonly executeCommandConfiguration?: ExecuteCommandConfiguration;\n}\n\n/**\n * The properties to import from the ECS cluster.\n */\nexport interface ClusterAttributes {\n  /**\n   * The name of the cluster.\n   */\n  readonly clusterName: string;\n\n  /**\n   * The Amazon Resource Name (ARN) that identifies the cluster.\n   *\n   * @default Derived from clusterName\n   */\n  readonly clusterArn?: string;\n\n  /**\n   * The VPC associated with the cluster.\n   */\n  readonly vpc: ec2.IVpc;\n\n  /**\n   * The security groups associated with the container instances registered to the cluster.\n   */\n  readonly securityGroups: ec2.ISecurityGroup[];\n\n  /**\n   * Specifies whether the cluster has EC2 instance capacity.\n   *\n   * @default true\n   */\n  readonly hasEc2Capacity?: boolean;\n\n  /**\n   * The AWS Cloud Map namespace to associate with the cluster.\n   *\n   * @default - No default namespace\n   */\n  readonly defaultCloudMapNamespace?: cloudmap.INamespace;\n\n  /**\n   * Autoscaling group added to the cluster if capacity is added\n   *\n   * @default - No default autoscaling group\n   */\n  readonly autoscalingGroup?: autoscaling.IAutoScalingGroup;\n\n  /**\n   * The execute command configuration for the cluster\n   *\n   * @default - none.\n   */\n  readonly executeCommandConfiguration?: ExecuteCommandConfiguration;\n}\n\n/**\n * An Cluster that has been imported\n */\nclass ImportedCluster extends Resource implements ICluster {\n  /**\n   * Name of the cluster\n   */\n  public readonly clusterName: string;\n\n  /**\n   * ARN of the cluster\n   */\n  public readonly clusterArn: string;\n\n  /**\n   * VPC that the cluster instances are running in\n   */\n  public readonly vpc: ec2.IVpc;\n\n  /**\n   * Security group of the cluster instances\n   */\n  public readonly connections = new ec2.Connections();\n\n  /**\n   * Whether the cluster has EC2 capacity\n   */\n  public readonly hasEc2Capacity: boolean;\n\n  /**\n   * Cloudmap namespace created in the cluster\n   */\n  private _defaultCloudMapNamespace?: cloudmap.INamespace;\n\n  /**\n   * The execute command configuration for the cluster\n   */\n  private _executeCommandConfiguration?: ExecuteCommandConfiguration;\n\n  /**\n   * Constructs a new instance of the ImportedCluster class.\n   */\n  constructor(scope: Construct, id: string, props: ClusterAttributes) {\n    super(scope, id);\n    this.clusterName = props.clusterName;\n    this.vpc = props.vpc;\n    this.hasEc2Capacity = props.hasEc2Capacity !== false;\n    this._defaultCloudMapNamespace = props.defaultCloudMapNamespace;\n    this._executeCommandConfiguration = props.executeCommandConfiguration;\n\n    this.clusterArn = props.clusterArn ?? Stack.of(this).formatArn({\n      service: 'ecs',\n      resource: 'cluster',\n      resourceName: props.clusterName,\n    });\n\n    this.connections = new ec2.Connections({\n      securityGroups: props.securityGroups,\n    });\n  }\n\n  public get defaultCloudMapNamespace(): cloudmap.INamespace | undefined {\n    return this._defaultCloudMapNamespace;\n  }\n\n  public get executeCommandConfiguration(): ExecuteCommandConfiguration | undefined {\n    return this._executeCommandConfiguration;\n  }\n}\n\n/**\n * The properties for adding an AutoScalingGroup.\n */\nexport interface AddAutoScalingGroupCapacityOptions {\n  /**\n   * Specifies whether the containers can access the container instance role.\n   *\n   * @default false\n   */\n  readonly canContainersAccessInstanceRole?: boolean;\n\n  /**\n   * The time period to wait before force terminating an instance that is draining.\n   *\n   * This creates a Lambda function that is used by a lifecycle hook for the\n   * AutoScalingGroup that will delay instance termination until all ECS tasks\n   * have drained from the instance. Set to 0 to disable task draining.\n   *\n   * Set to 0 to disable task draining.\n   *\n   * @deprecated The lifecycle draining hook is not configured if using the EC2 Capacity Provider. Enable managed termination protection instead.\n   * @default Duration.minutes(5)\n   */\n  readonly taskDrainTime?: Duration;\n\n  /**\n   * Specify whether to enable Automated Draining for Spot Instances running Amazon ECS Services.\n   * For more information, see [Using Spot Instances](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/container-instance-spot.html).\n   *\n   * @default false\n   */\n  readonly spotInstanceDraining?: boolean\n\n  /**\n   * If {@link AddAutoScalingGroupCapacityOptions.taskDrainTime} is non-zero, then the ECS cluster creates an\n   * SNS Topic to as part of a system to drain instances of tasks when the instance is being shut down.\n   * If this property is provided, then this key will be used to encrypt the contents of that SNS Topic.\n   * See [SNS Data Encryption](https://docs.aws.amazon.com/sns/latest/dg/sns-data-encryption.html) for more information.\n   *\n   * @default The SNS Topic will not be encrypted.\n   */\n  readonly topicEncryptionKey?: kms.IKey;\n\n  /**\n   * What type of machine image this is\n   *\n   * Depending on the setting, different UserData will automatically be added\n   * to the `AutoScalingGroup` to configure it properly for use with ECS.\n   *\n   * If you create an `AutoScalingGroup` yourself and are adding it via\n   * `addAutoScalingGroup()`, you must specify this value. If you are adding an\n   * `autoScalingGroup` via `addCapacity`, this value will be determined\n   * from the `machineImage` you pass.\n   *\n   * @default - Automatically determined from `machineImage`, if available, otherwise `MachineImageType.AMAZON_LINUX_2`.\n   */\n  readonly machineImageType?: MachineImageType;\n}\n\n/**\n * The properties for adding instance capacity to an AutoScalingGroup.\n */\nexport interface AddCapacityOptions extends AddAutoScalingGroupCapacityOptions, autoscaling.CommonAutoScalingGroupProps {\n  /**\n   * The EC2 instance type to use when launching instances into the AutoScalingGroup.\n   */\n  readonly instanceType: ec2.InstanceType;\n\n  /**\n   * The ECS-optimized AMI variant to use\n   *\n   * The default is to use an ECS-optimized AMI of Amazon Linux 2 which is\n   * automatically updated to the latest version on every deployment. This will\n   * replace the instances in the AutoScalingGroup. Make sure you have not disabled\n   * task draining, to avoid downtime when the AMI updates.\n   *\n   * To use an image that does not update on every deployment, pass:\n   *\n   * ```ts\n   * const machineImage = ecs.EcsOptimizedImage.amazonLinux2(ecs.AmiHardwareType.STANDARD, {\n   *   cachedInContext: true,\n   * });\n   * ```\n   *\n   * For more information, see [Amazon ECS-optimized\n   * AMIs](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/ecs-optimized_AMI.html).\n   *\n   * You must define either `machineImage` or `machineImageType`, not both.\n   *\n   * @default - Automatically updated, ECS-optimized Amazon Linux 2\n   */\n  readonly machineImage?: ec2.IMachineImage;\n}\n\n/**\n * The options for creating an AWS Cloud Map namespace.\n */\nexport interface CloudMapNamespaceOptions {\n  /**\n   * The name of the namespace, such as example.com.\n   */\n  readonly name: string;\n\n  /**\n   * The type of CloudMap Namespace to create.\n   *\n   * @default PrivateDns\n   */\n  readonly type?: cloudmap.NamespaceType;\n\n  /**\n   * The VPC to associate the namespace with. This property is required for private DNS namespaces.\n   *\n   * @default VPC of the cluster for Private DNS Namespace, otherwise none\n   */\n  readonly vpc?: ec2.IVpc;\n}\n\nenum ContainerInsights {\n  /**\n   * Enable CloudWatch Container Insights for the cluster\n   */\n\n  ENABLED = 'enabled',\n\n  /**\n   * Disable CloudWatch Container Insights for the cluster\n   */\n  DISABLED = 'disabled',\n}\n\n/**\n * A Capacity Provider strategy to use for the service.\n *\n * NOTE: defaultCapacityProviderStrategy on cluster not currently supported.\n */\nexport interface CapacityProviderStrategy {\n  /**\n   * The name of the capacity provider.\n   */\n  readonly capacityProvider: string;\n\n  /**\n   * The base value designates how many tasks, at a minimum, to run on the specified capacity provider. Only one\n   * capacity provider in a capacity provider strategy can have a base defined. If no value is specified, the default\n   * value of 0 is used.\n   *\n   * @default - none\n   */\n  readonly base?: number;\n\n  /**\n   * The weight value designates the relative percentage of the total number of tasks launched that should use the\n   * specified\ncapacity provider. The weight value is taken into consideration after the base value, if defined, is satisfied.\n   *\n   * @default - 0\n   */\n  readonly weight?: number;\n}\n\n/**\n * The details of the execute command configuration. For more information, see\n * [ExecuteCommandConfiguration] https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-executecommandconfiguration.html\n */\nexport interface ExecuteCommandConfiguration {\n  /**\n   * The AWS Key Management Service key ID to encrypt the data between the local client and the container.\n   *\n   * @default - none\n   */\n  readonly kmsKey?: kms.IKey,\n\n  /**\n   * The log configuration for the results of the execute command actions. The logs can be sent to CloudWatch Logs or an Amazon S3 bucket.\n   *\n   * @default - none\n   */\n  readonly logConfiguration?: ExecuteCommandLogConfiguration,\n\n  /**\n   * The log settings to use for logging the execute command session.\n   *\n   * @default - none\n   */\n  readonly logging?: ExecuteCommandLogging,\n}\n\n/**\n * The log settings to use to for logging the execute command session. For more information, see\n * [Logging] https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-executecommandconfiguration.html#cfn-ecs-cluster-executecommandconfiguration-logging\n */\nexport enum ExecuteCommandLogging {\n  /**\n   * The execute command session is not logged.\n   */\n  NONE = 'NONE',\n\n  /**\n   * The awslogs configuration in the task definition is used. If no logging parameter is specified, it defaults to this value. If no awslogs log driver is configured in the task definition, the output won't be logged.\n   */\n  DEFAULT = 'DEFAULT',\n\n  /**\n   * Specify the logging details as a part of logConfiguration.\n   */\n  OVERRIDE = 'OVERRIDE',\n}\n\n/**\n * The log configuration for the results of the execute command actions. The logs can be sent to CloudWatch Logs and/ or an Amazon S3 bucket.\n * For more information, see [ExecuteCommandLogConfiguration] https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-ecs-cluster-executecommandlogconfiguration.html\n */\nexport interface ExecuteCommandLogConfiguration {\n  /**\n   * Whether or not to enable encryption on the CloudWatch logs.\n   *\n   * @default - encryption will be disabled.\n   */\n  readonly cloudWatchEncryptionEnabled?: boolean,\n\n  /**\n   * The name of the CloudWatch log group to send logs to. The CloudWatch log group must already be created.\n   * @default - none\n   */\n  readonly cloudWatchLogGroup?: logs.ILogGroup,\n\n  /**\n   * The name of the S3 bucket to send logs to. The S3 bucket must already be created.\n   *\n   * @default - none\n   */\n  readonly s3Bucket?: s3.IBucket,\n\n  /**\n   * Whether or not to enable encryption on the CloudWatch logs.\n   *\n   * @default - encryption will be disabled.\n   */\n  readonly s3EncryptionEnabled?: boolean,\n\n  /**\n   * An optional folder in the S3 bucket to place logs in.\n   *\n   * @default - none\n   */\n  readonly s3KeyPrefix?: string\n}\n\n/**\n * The options for creating an Auto Scaling Group Capacity Provider.\n */\nexport interface AsgCapacityProviderProps extends AddAutoScalingGroupCapacityOptions {\n  /**\n   * The name of the capacity provider. If a name is specified,\n   * it cannot start with `aws`, `ecs`, or `fargate`. If no name is specified,\n   * a default name in the CFNStackName-CFNResourceName-RandomString format is used.\n   *\n   * @default CloudFormation-generated name\n   */\n  readonly capacityProviderName?: string;\n\n  /**\n   * The autoscaling group to add as a Capacity Provider.\n   */\n  readonly autoScalingGroup: autoscaling.IAutoScalingGroup;\n\n  /**\n   * Whether to enable managed scaling\n   *\n   * @default true\n   */\n  readonly enableManagedScaling?: boolean;\n\n  /**\n   * Whether to enable managed termination protection\n   *\n   * @default true\n   */\n  readonly enableManagedTerminationProtection?: boolean;\n\n  /**\n   * Maximum scaling step size. In most cases this should be left alone.\n   *\n   * @default 1000\n   */\n  readonly maximumScalingStepSize?: number;\n\n  /**\n   * Minimum scaling step size. In most cases this should be left alone.\n   *\n   * @default 1\n   */\n  readonly minimumScalingStepSize?: number;\n\n  /**\n   * Target capacity percent. In most cases this should be left alone.\n   *\n   * @default 100\n   */\n  readonly targetCapacityPercent?: number;\n}\n\n/**\n * An Auto Scaling Group Capacity Provider. This allows an ECS cluster to target\n * a specific EC2 Auto Scaling Group for the placement of tasks. Optionally (and\n * recommended), ECS can manage the number of instances in the ASG to fit the\n * tasks, and can ensure that instances are not prematurely terminated while\n * there are still tasks running on them.\n */\nexport class AsgCapacityProvider extends CoreConstruct {\n  /**\n   * Capacity provider name\n   * @default Chosen by CloudFormation\n   */\n  readonly capacityProviderName: string;\n\n  /**\n   * Auto Scaling Group\n   */\n  readonly autoScalingGroup: autoscaling.AutoScalingGroup;\n\n  /**\n   * Auto Scaling Group machineImageType.\n   */\n  readonly machineImageType: MachineImageType;\n\n  /**\n   * Whether managed termination protection is enabled\n   */\n  readonly enableManagedTerminationProtection?: boolean;\n\n  constructor(scope: Construct, id: string, props: AsgCapacityProviderProps) {\n    super(scope, id);\n\n    this.autoScalingGroup = props.autoScalingGroup as autoscaling.AutoScalingGroup;\n\n    this.machineImageType = props.machineImageType ?? MachineImageType.AMAZON_LINUX_2;\n\n    this.enableManagedTerminationProtection =\n      props.enableManagedTerminationProtection === undefined ? true : props.enableManagedTerminationProtection;\n\n    if (this.enableManagedTerminationProtection) {\n      this.autoScalingGroup.protectNewInstancesFromScaleIn();\n    }\n    if (props.capacityProviderName) {\n      if (!(/^(?!aws|ecs|fargate).+/gm.test(props.capacityProviderName))) {\n        throw new Error(`Invalid Capacity Provider Name: ${props.capacityProviderName}, If a name is specified, it cannot start with aws, ecs, or fargate.`);\n      }\n    }\n    const capacityProvider = new CfnCapacityProvider(this, id, {\n      name: props.capacityProviderName,\n      autoScalingGroupProvider: {\n        autoScalingGroupArn: this.autoScalingGroup.autoScalingGroupName,\n        managedScaling: props.enableManagedScaling === false ? undefined : {\n          status: 'ENABLED',\n          targetCapacity: props.targetCapacityPercent || 100,\n          maximumScalingStepSize: props.maximumScalingStepSize,\n          minimumScalingStepSize: props.minimumScalingStepSize,\n        },\n        managedTerminationProtection: this.enableManagedTerminationProtection ? 'ENABLED' : 'DISABLED',\n      },\n    });\n\n    this.capacityProviderName = capacityProvider.ref;\n  }\n}\n\n/**\n * A visitor that adds a capacity provider association to a Cluster only if\n * the caller created any EC2 Capacity Providers.\n */\nclass MaybeCreateCapacityProviderAssociations implements IAspect {\n  private scope: CoreConstruct;\n  private id: string;\n  private capacityProviders: string[]\n  private resource?: CfnClusterCapacityProviderAssociations\n\n  constructor(scope: CoreConstruct, id: string, capacityProviders: string[] ) {\n    this.scope = scope;\n    this.id = id;\n    this.capacityProviders = capacityProviders;\n  }\n\n  public visit(node: IConstruct): void {\n    if (node instanceof Cluster) {\n      if (this.capacityProviders.length > 0 && !this.resource) {\n        const resource = new CfnClusterCapacityProviderAssociations(this.scope, this.id, {\n          cluster: node.clusterName,\n          defaultCapacityProviderStrategy: [],\n          capacityProviders: Lazy.list({ produce: () => this.capacityProviders }),\n        });\n        this.resource = resource;\n      }\n    }\n  }\n}\n\n\nfunction isBottleRocketImage(image: ec2.IMachineImage) {\n  return image instanceof BottleRocketImage;\n}\n"]}
\No newline at end of file