1 | ;
|
2 | var _a, _b;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.BuiltInAttributes = exports.Ec2Service = void 0;
|
5 | const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const ec2 = require("@aws-cdk/aws-ec2");
|
8 | const core_1 = require("@aws-cdk/core");
|
9 | const base_service_1 = require("../base/base-service");
|
10 | const from_service_attributes_1 = require("../base/from-service-attributes");
|
11 | const task_definition_1 = require("../base/task-definition");
|
12 | /**
|
13 | * This creates a service using the EC2 launch type on an ECS cluster.
|
14 | *
|
15 | * @resource AWS::ECS::Service
|
16 | */
|
17 | class Ec2Service extends base_service_1.BaseService {
|
18 | /**
|
19 | * Constructs a new instance of the Ec2Service class.
|
20 | */
|
21 | constructor(scope, id, props) {
|
22 | var _c;
|
23 | try {
|
24 | jsiiDeprecationWarnings._aws_cdk_aws_ecs_Ec2ServiceProps(props);
|
25 | }
|
26 | catch (error) {
|
27 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
28 | Error.captureStackTrace(error, this.constructor);
|
29 | }
|
30 | throw error;
|
31 | }
|
32 | if (props.daemon && props.desiredCount !== undefined) {
|
33 | throw new Error('Daemon mode launches one task on every instance. Don\'t supply desiredCount.');
|
34 | }
|
35 | if (props.daemon && props.maxHealthyPercent !== undefined && props.maxHealthyPercent !== 100) {
|
36 | throw new Error('Maximum percent must be 100 for daemon mode.');
|
37 | }
|
38 | if (props.minHealthyPercent !== undefined && props.maxHealthyPercent !== undefined && props.minHealthyPercent >= props.maxHealthyPercent) {
|
39 | throw new Error('Minimum healthy percent must be less than maximum healthy percent.');
|
40 | }
|
41 | if (!props.taskDefinition.isEc2Compatible) {
|
42 | throw new Error('Supplied TaskDefinition is not configured for compatibility with EC2');
|
43 | }
|
44 | if (props.securityGroup !== undefined && props.securityGroups !== undefined) {
|
45 | throw new Error('Only one of SecurityGroup or SecurityGroups can be populated.');
|
46 | }
|
47 | super(scope, id, {
|
48 | ...props,
|
49 | desiredCount: props.desiredCount,
|
50 | maxHealthyPercent: props.daemon && props.maxHealthyPercent === undefined ? 100 : props.maxHealthyPercent,
|
51 | minHealthyPercent: props.daemon && props.minHealthyPercent === undefined ? 0 : props.minHealthyPercent,
|
52 | launchType: base_service_1.LaunchType.EC2,
|
53 | enableECSManagedTags: props.enableECSManagedTags,
|
54 | }, {
|
55 | cluster: props.cluster.clusterName,
|
56 | taskDefinition: ((_c = props.deploymentController) === null || _c === void 0 ? void 0 : _c.type) === base_service_1.DeploymentControllerType.EXTERNAL ? undefined : props.taskDefinition.taskDefinitionArn,
|
57 | placementConstraints: core_1.Lazy.any({ produce: () => this.constraints }, { omitEmptyArray: true }),
|
58 | placementStrategies: core_1.Lazy.any({ produce: () => this.strategies }, { omitEmptyArray: true }),
|
59 | schedulingStrategy: props.daemon ? 'DAEMON' : 'REPLICA',
|
60 | }, props.taskDefinition);
|
61 | this.constraints = [];
|
62 | this.strategies = [];
|
63 | this.daemon = props.daemon || false;
|
64 | let securityGroups;
|
65 | if (props.securityGroup !== undefined) {
|
66 | securityGroups = [props.securityGroup];
|
67 | }
|
68 | else if (props.securityGroups !== undefined) {
|
69 | securityGroups = props.securityGroups;
|
70 | }
|
71 | if (props.taskDefinition.networkMode === task_definition_1.NetworkMode.AWS_VPC) {
|
72 | this.configureAwsVpcNetworkingWithSecurityGroups(props.cluster.vpc, props.assignPublicIp, props.vpcSubnets, securityGroups);
|
73 | }
|
74 | else {
|
75 | // Either None, Bridge or Host networking. Copy SecurityGroups from ASG.
|
76 | // We have to be smart here -- by default future Security Group rules would be created
|
77 | // in the Cluster stack. However, if the Cluster is in a different stack than us,
|
78 | // that will lead to a cyclic reference (we point to that stack for the cluster name,
|
79 | // but that stack will point to the ALB probably created right next to us).
|
80 | //
|
81 | // In that case, reference the same security groups but make sure new rules are
|
82 | // created in the current scope (i.e., this stack)
|
83 | validateNoNetworkingProps(props);
|
84 | this.connections.addSecurityGroup(...securityGroupsInThisStack(this, props.cluster.connections.securityGroups));
|
85 | }
|
86 | this.addPlacementConstraints(...props.placementConstraints || []);
|
87 | this.addPlacementStrategies(...props.placementStrategies || []);
|
88 | this.node.addValidation({
|
89 | validate: () => !this.taskDefinition.defaultContainer ? ['A TaskDefinition must have at least one essential container'] : [],
|
90 | });
|
91 | }
|
92 | /**
|
93 | * Imports from the specified service ARN.
|
94 | */
|
95 | static fromEc2ServiceArn(scope, id, ec2ServiceArn) {
|
96 | class Import extends core_1.Resource {
|
97 | constructor() {
|
98 | super(...arguments);
|
99 | this.serviceArn = ec2ServiceArn;
|
100 | this.serviceName = core_1.Stack.of(scope).splitArn(ec2ServiceArn, core_1.ArnFormat.SLASH_RESOURCE_NAME).resourceName;
|
101 | }
|
102 | }
|
103 | return new Import(scope, id);
|
104 | }
|
105 | /**
|
106 | * Imports from the specified service attrributes.
|
107 | */
|
108 | static fromEc2ServiceAttributes(scope, id, attrs) {
|
109 | try {
|
110 | jsiiDeprecationWarnings._aws_cdk_aws_ecs_Ec2ServiceAttributes(attrs);
|
111 | }
|
112 | catch (error) {
|
113 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
114 | Error.captureStackTrace(error, this.fromEc2ServiceAttributes);
|
115 | }
|
116 | throw error;
|
117 | }
|
118 | return from_service_attributes_1.fromServiceAtrributes(scope, id, attrs);
|
119 | }
|
120 | /**
|
121 | * Adds one or more placement strategies to use for tasks in the service. For more information, see
|
122 | * [Amazon ECS Task Placement Strategies](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-strategies.html).
|
123 | */
|
124 | addPlacementStrategies(...strategies) {
|
125 | try {
|
126 | jsiiDeprecationWarnings._aws_cdk_aws_ecs_PlacementStrategy(strategies);
|
127 | }
|
128 | catch (error) {
|
129 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
130 | Error.captureStackTrace(error, this.addPlacementStrategies);
|
131 | }
|
132 | throw error;
|
133 | }
|
134 | if (strategies.length > 0 && this.daemon) {
|
135 | throw new Error("Can't configure placement strategies when daemon=true");
|
136 | }
|
137 | for (const strategy of strategies) {
|
138 | this.strategies.push(...strategy.toJson());
|
139 | }
|
140 | }
|
141 | /**
|
142 | * Adds one or more placement constraints to use for tasks in the service. For more information, see
|
143 | * [Amazon ECS Task Placement Constraints](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-constraints.html).
|
144 | */
|
145 | addPlacementConstraints(...constraints) {
|
146 | try {
|
147 | jsiiDeprecationWarnings._aws_cdk_aws_ecs_PlacementConstraint(constraints);
|
148 | }
|
149 | catch (error) {
|
150 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
151 | Error.captureStackTrace(error, this.addPlacementConstraints);
|
152 | }
|
153 | throw error;
|
154 | }
|
155 | for (const constraint of constraints) {
|
156 | this.constraints.push(...constraint.toJson());
|
157 | }
|
158 | }
|
159 | /**
|
160 | * Validates this Ec2Service.
|
161 | */
|
162 | validate() {
|
163 | const ret = super.validate();
|
164 | if (!this.cluster.hasEc2Capacity) {
|
165 | ret.push('Cluster for this service needs Ec2 capacity. Call addXxxCapacity() on the cluster.');
|
166 | }
|
167 | return ret;
|
168 | }
|
169 | }
|
170 | exports.Ec2Service = Ec2Service;
|
171 | _a = JSII_RTTI_SYMBOL_1;
|
172 | Ec2Service[_a] = { fqn: "@aws-cdk/aws-ecs.Ec2Service", version: "1.156.1" };
|
173 | /**
|
174 | * Validate combinations of networking arguments.
|
175 | */
|
176 | function validateNoNetworkingProps(props) {
|
177 | if (props.vpcSubnets !== undefined
|
178 | || props.securityGroup !== undefined
|
179 | || props.securityGroups !== undefined
|
180 | || props.assignPublicIp) {
|
181 | throw new Error('vpcSubnets, securityGroup(s) and assignPublicIp can only be used in AwsVpc networking mode');
|
182 | }
|
183 | }
|
184 | /**
|
185 | * Force security group rules to be created in this stack.
|
186 | *
|
187 | * For every security group, if the scope and the group are in different stacks, return
|
188 | * a fake "imported" security group instead. This will behave as the original security group,
|
189 | * but new Ingress and Egress rule resources will be added in the current stack instead of the
|
190 | * other one.
|
191 | */
|
192 | function securityGroupsInThisStack(scope, groups) {
|
193 | const thisStack = core_1.Stack.of(scope);
|
194 | let i = 1;
|
195 | return groups.map(group => {
|
196 | if (thisStack === core_1.Stack.of(group)) {
|
197 | return group;
|
198 | } // Simple case, just return the original one
|
199 | return ec2.SecurityGroup.fromSecurityGroupId(scope, `SecurityGroup${i++}`, group.securityGroupId, {
|
200 | allowAllOutbound: group.allowAllOutbound,
|
201 | mutable: true,
|
202 | });
|
203 | });
|
204 | }
|
205 | /**
|
206 | * The built-in container instance attributes
|
207 | */
|
208 | class BuiltInAttributes {
|
209 | }
|
210 | exports.BuiltInAttributes = BuiltInAttributes;
|
211 | _b = JSII_RTTI_SYMBOL_1;
|
212 | BuiltInAttributes[_b] = { fqn: "@aws-cdk/aws-ecs.BuiltInAttributes", version: "1.156.1" };
|
213 | /**
|
214 | * The id of the instance.
|
215 | */
|
216 | BuiltInAttributes.INSTANCE_ID = 'instanceId';
|
217 | /**
|
218 | * The AvailabilityZone where the instance is running in.
|
219 | */
|
220 | BuiltInAttributes.AVAILABILITY_ZONE = 'attribute:ecs.availability-zone';
|
221 | /**
|
222 | * The AMI id the instance is using.
|
223 | */
|
224 | BuiltInAttributes.AMI_ID = 'attribute:ecs.ami-id';
|
225 | /**
|
226 | * The EC2 instance type.
|
227 | */
|
228 | BuiltInAttributes.INSTANCE_TYPE = 'attribute:ecs.instance-type';
|
229 | /**
|
230 | * The operating system of the instance.
|
231 | *
|
232 | * Either 'linux' or 'windows'.
|
233 | */
|
234 | BuiltInAttributes.OS_TYPE = 'attribute:ecs.os-type';
|
235 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2-service.js","sourceRoot":"","sources":["ec2-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ec2 from '@aws-cdk/aws-ec2';\nimport { ArnFormat, Lazy, Resource, Stack } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { BaseService, BaseServiceOptions, DeploymentControllerType, IBaseService, IService, LaunchType } from '../base/base-service';\nimport { fromServiceAtrributes } from '../base/from-service-attributes';\nimport { NetworkMode, TaskDefinition } from '../base/task-definition';\nimport { ICluster } from '../cluster';\nimport { CfnService } from '../ecs.generated';\nimport { PlacementConstraint, PlacementStrategy } from '../placement';\n\n/**\n * The properties for defining a service using the EC2 launch type.\n */\nexport interface Ec2ServiceProps extends BaseServiceOptions {\n  /**\n   * The task definition to use for tasks in the service.\n   *\n   * [disable-awslint:ref-via-interface]\n   */\n  readonly taskDefinition: TaskDefinition;\n\n  /**\n   * Specifies whether the task's elastic network interface receives a public IP address.\n   * If true, each task will receive a public IP address.\n   *\n   * This property is only used for tasks that use the awsvpc network mode.\n   *\n   * @default false\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The subnets to associate with the service.\n   *\n   * This property is only used for tasks that use the awsvpc network mode.\n   *\n   * @default - Public subnets if `assignPublicIp` is set, otherwise the first available one of Private, Isolated, Public, in that order.\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n\n  /**\n   * The security groups to associate with the service. If you do not specify a security group, a new security group is created.\n   *\n   * This property is only used for tasks that use the awsvpc network mode.\n   *\n   * @default - A new security group is created.\n   * @deprecated use securityGroups instead.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * The security groups to associate with the service. If you do not specify a security group, a new security group is created.\n   *\n   * This property is only used for tasks that use the awsvpc network mode.\n   *\n   * @default - A new security group is created.\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * The placement constraints to use for tasks in the service. For more information, see\n   * [Amazon ECS Task Placement Constraints](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-constraints.html).\n   *\n   * @default - No constraints.\n   */\n  readonly placementConstraints?: PlacementConstraint[];\n\n  /**\n   * The placement strategies to use for tasks in the service. For more information, see\n   * [Amazon ECS Task Placement Strategies](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-strategies.html).\n   *\n   * @default - No strategies.\n   */\n  readonly placementStrategies?: PlacementStrategy[];\n\n  /**\n   * Specifies whether the service will use the daemon scheduling strategy.\n   * If true, the service scheduler deploys exactly one task on each container instance in your cluster.\n   *\n   * When you are using this strategy, do not specify a desired number of tasks orany task placement strategies.\n   *\n   * @default false\n   */\n  readonly daemon?: boolean;\n}\n\n/**\n * The interface for a service using the EC2 launch type on an ECS cluster.\n */\nexport interface IEc2Service extends IService {\n\n}\n\n/**\n * The properties to import from the service using the EC2 launch type.\n */\nexport interface Ec2ServiceAttributes {\n  /**\n   * The cluster that hosts the service.\n   */\n  readonly cluster: ICluster;\n\n  /**\n   * The service ARN.\n   *\n   * @default - either this, or {@link serviceName}, is required\n   */\n  readonly serviceArn?: string;\n\n  /**\n   * The name of the service.\n   *\n   * @default - either this, or {@link serviceArn}, is required\n   */\n  readonly serviceName?: string;\n}\n\n/**\n * This creates a service using the EC2 launch type on an ECS cluster.\n *\n * @resource AWS::ECS::Service\n */\nexport class Ec2Service extends BaseService implements IEc2Service {\n\n  /**\n   * Imports from the specified service ARN.\n   */\n  public static fromEc2ServiceArn(scope: Construct, id: string, ec2ServiceArn: string): IEc2Service {\n    class Import extends Resource implements IEc2Service {\n      public readonly serviceArn = ec2ServiceArn;\n      public readonly serviceName = Stack.of(scope).splitArn(ec2ServiceArn, ArnFormat.SLASH_RESOURCE_NAME).resourceName as string;\n    }\n    return new Import(scope, id);\n  }\n\n  /**\n   * Imports from the specified service attrributes.\n   */\n  public static fromEc2ServiceAttributes(scope: Construct, id: string, attrs: Ec2ServiceAttributes): IBaseService {\n    return fromServiceAtrributes(scope, id, attrs);\n  }\n\n  private readonly constraints: CfnService.PlacementConstraintProperty[];\n  private readonly strategies: CfnService.PlacementStrategyProperty[];\n  private readonly daemon: boolean;\n\n  /**\n   * Constructs a new instance of the Ec2Service class.\n   */\n  constructor(scope: Construct, id: string, props: Ec2ServiceProps) {\n    if (props.daemon && props.desiredCount !== undefined) {\n      throw new Error('Daemon mode launches one task on every instance. Don\\'t supply desiredCount.');\n    }\n\n    if (props.daemon && props.maxHealthyPercent !== undefined && props.maxHealthyPercent !== 100) {\n      throw new Error('Maximum percent must be 100 for daemon mode.');\n    }\n\n    if (props.minHealthyPercent !== undefined && props.maxHealthyPercent !== undefined && props.minHealthyPercent >= props.maxHealthyPercent) {\n      throw new Error('Minimum healthy percent must be less than maximum healthy percent.');\n    }\n\n    if (!props.taskDefinition.isEc2Compatible) {\n      throw new Error('Supplied TaskDefinition is not configured for compatibility with EC2');\n    }\n\n    if (props.securityGroup !== undefined && props.securityGroups !== undefined) {\n      throw new Error('Only one of SecurityGroup or SecurityGroups can be populated.');\n    }\n\n    super(scope, id, {\n      ...props,\n      desiredCount: props.desiredCount,\n      maxHealthyPercent: props.daemon && props.maxHealthyPercent === undefined ? 100 : props.maxHealthyPercent,\n      minHealthyPercent: props.daemon && props.minHealthyPercent === undefined ? 0 : props.minHealthyPercent,\n      launchType: LaunchType.EC2,\n      enableECSManagedTags: props.enableECSManagedTags,\n    },\n    {\n      cluster: props.cluster.clusterName,\n      taskDefinition: props.deploymentController?.type === DeploymentControllerType.EXTERNAL ? undefined : props.taskDefinition.taskDefinitionArn,\n      placementConstraints: Lazy.any({ produce: () => this.constraints }, { omitEmptyArray: true }),\n      placementStrategies: Lazy.any({ produce: () => this.strategies }, { omitEmptyArray: true }),\n      schedulingStrategy: props.daemon ? 'DAEMON' : 'REPLICA',\n    }, props.taskDefinition);\n\n    this.constraints = [];\n    this.strategies = [];\n    this.daemon = props.daemon || false;\n\n    let securityGroups;\n    if (props.securityGroup !== undefined) {\n      securityGroups = [props.securityGroup];\n    } else if (props.securityGroups !== undefined) {\n      securityGroups = props.securityGroups;\n    }\n\n    if (props.taskDefinition.networkMode === NetworkMode.AWS_VPC) {\n      this.configureAwsVpcNetworkingWithSecurityGroups(props.cluster.vpc, props.assignPublicIp, props.vpcSubnets, securityGroups);\n    } else {\n      // Either None, Bridge or Host networking. Copy SecurityGroups from ASG.\n      // We have to be smart here -- by default future Security Group rules would be created\n      // in the Cluster stack. However, if the Cluster is in a different stack than us,\n      // that will lead to a cyclic reference (we point to that stack for the cluster name,\n      // but that stack will point to the ALB probably created right next to us).\n      //\n      // In that case, reference the same security groups but make sure new rules are\n      // created in the current scope (i.e., this stack)\n      validateNoNetworkingProps(props);\n      this.connections.addSecurityGroup(...securityGroupsInThisStack(this, props.cluster.connections.securityGroups));\n    }\n\n    this.addPlacementConstraints(...props.placementConstraints || []);\n    this.addPlacementStrategies(...props.placementStrategies || []);\n\n    this.node.addValidation({\n      validate: () => !this.taskDefinition.defaultContainer ? ['A TaskDefinition must have at least one essential container'] : [],\n    });\n  }\n\n  /**\n   * Adds one or more placement strategies to use for tasks in the service. For more information, see\n   * [Amazon ECS Task Placement Strategies](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-strategies.html).\n   */\n  public addPlacementStrategies(...strategies: PlacementStrategy[]) {\n    if (strategies.length > 0 && this.daemon) {\n      throw new Error(\"Can't configure placement strategies when daemon=true\");\n    }\n\n    for (const strategy of strategies) {\n      this.strategies.push(...strategy.toJson());\n    }\n  }\n\n  /**\n   * Adds one or more placement constraints to use for tasks in the service. For more information, see\n   * [Amazon ECS Task Placement Constraints](https://docs.aws.amazon.com/AmazonECS/latest/developerguide/task-placement-constraints.html).\n   */\n  public addPlacementConstraints(...constraints: PlacementConstraint[]) {\n    for (const constraint of constraints) {\n      this.constraints.push(...constraint.toJson());\n    }\n  }\n\n  /**\n   * Validates this Ec2Service.\n   */\n  protected validate(): string[] {\n    const ret = super.validate();\n    if (!this.cluster.hasEc2Capacity) {\n      ret.push('Cluster for this service needs Ec2 capacity. Call addXxxCapacity() on the cluster.');\n    }\n    return ret;\n  }\n}\n\n/**\n * Validate combinations of networking arguments.\n */\nfunction validateNoNetworkingProps(props: Ec2ServiceProps) {\n  if (props.vpcSubnets !== undefined\n    || props.securityGroup !== undefined\n    || props.securityGroups !== undefined\n    || props.assignPublicIp) {\n    throw new Error('vpcSubnets, securityGroup(s) and assignPublicIp can only be used in AwsVpc networking mode');\n  }\n}\n\n/**\n * Force security group rules to be created in this stack.\n *\n * For every security group, if the scope and the group are in different stacks, return\n * a fake \"imported\" security group instead. This will behave as the original security group,\n * but new Ingress and Egress rule resources will be added in the current stack instead of the\n * other one.\n */\nfunction securityGroupsInThisStack(scope: Construct, groups: ec2.ISecurityGroup[]): ec2.ISecurityGroup[] {\n  const thisStack = Stack.of(scope);\n\n  let i = 1;\n  return groups.map(group => {\n    if (thisStack === Stack.of(group)) { return group; } // Simple case, just return the original one\n\n    return ec2.SecurityGroup.fromSecurityGroupId(scope, `SecurityGroup${i++}`, group.securityGroupId, {\n      allowAllOutbound: group.allowAllOutbound,\n      mutable: true,\n    });\n  });\n}\n\n/**\n * The built-in container instance attributes\n */\nexport class BuiltInAttributes {\n  /**\n   * The id of the instance.\n   */\n  public static readonly INSTANCE_ID = 'instanceId';\n\n  /**\n   * The AvailabilityZone where the instance is running in.\n   */\n  public static readonly AVAILABILITY_ZONE = 'attribute:ecs.availability-zone';\n\n  /**\n   * The AMI id the instance is using.\n   */\n  public static readonly AMI_ID = 'attribute:ecs.ami-id';\n\n  /**\n   * The EC2 instance type.\n   */\n  public static readonly INSTANCE_TYPE = 'attribute:ecs.instance-type';\n\n  /**\n   * The operating system of the instance.\n   *\n   * Either 'linux' or 'windows'.\n   */\n  public static readonly OS_TYPE = 'attribute:ecs.os-type';\n}\n"]} |
\ | No newline at end of file |