UNPKG

21.4 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.EcsTask = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const ec2 = require("@aws-cdk/aws-ec2");
8const ecs = require("@aws-cdk/aws-ecs");
9const events = require("@aws-cdk/aws-events");
10const iam = require("@aws-cdk/aws-iam");
11const cdk = require("@aws-cdk/core");
12const util_1 = require("./util");
13/**
14 * Start a task on an ECS cluster
15 */
16class EcsTask {
17 constructor(props) {
18 this.props = props;
19 try {
20 jsiiDeprecationWarnings._aws_cdk_aws_events_targets_EcsTaskProps(props);
21 }
22 catch (error) {
23 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
24 Error.captureStackTrace(error, EcsTask);
25 }
26 throw error;
27 }
28 if (props.securityGroup !== undefined && props.securityGroups !== undefined) {
29 throw new Error('Only one of SecurityGroup or SecurityGroups can be populated.');
30 }
31 this.cluster = props.cluster;
32 this.taskDefinition = props.taskDefinition;
33 this.taskCount = props.taskCount ?? 1;
34 this.platformVersion = props.platformVersion;
35 this.role = props.role ?? util_1.singletonEventRole(this.taskDefinition);
36 for (const stmt of this.createEventRolePolicyStatements()) {
37 this.role.addToPrincipalPolicy(stmt);
38 }
39 // Security groups are only configurable with the "awsvpc" network mode.
40 if (this.taskDefinition.networkMode !== ecs.NetworkMode.AWS_VPC) {
41 if (props.securityGroup !== undefined || props.securityGroups !== undefined) {
42 cdk.Annotations.of(this.taskDefinition).addWarning('security groups are ignored when network mode is not awsvpc');
43 }
44 return;
45 }
46 if (props.securityGroups) {
47 this.securityGroups = props.securityGroups;
48 return;
49 }
50 if (!cdk.Construct.isConstruct(this.taskDefinition)) {
51 throw new Error('Cannot create a security group for ECS task. ' +
52 'The task definition in ECS task is not a Construct. ' +
53 'Please pass a taskDefinition as a Construct in EcsTaskProps.');
54 }
55 let securityGroup = props.securityGroup || this.taskDefinition.node.tryFindChild('SecurityGroup');
56 securityGroup = securityGroup || new ec2.SecurityGroup(this.taskDefinition, 'SecurityGroup', { vpc: this.props.cluster.vpc });
57 this.securityGroup = securityGroup; // Maintain backwards-compatibility for customers that read the generated security group.
58 this.securityGroups = [securityGroup];
59 }
60 /**
61 * Allows using tasks as target of EventBridge events
62 */
63 bind(_rule, _id) {
64 const arn = this.cluster.clusterArn;
65 const role = this.role;
66 const containerOverrides = this.props.containerOverrides && this.props.containerOverrides
67 .map(({ containerName, ...overrides }) => ({ name: containerName, ...overrides }));
68 const input = { containerOverrides };
69 const taskCount = this.taskCount;
70 const taskDefinitionArn = this.taskDefinition.taskDefinitionArn;
71 const subnetSelection = this.props.subnetSelection || { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT };
72 const assignPublicIp = subnetSelection.subnetType === ec2.SubnetType.PUBLIC ? 'ENABLED' : 'DISABLED';
73 const baseEcsParameters = { taskCount, taskDefinitionArn };
74 const ecsParameters = this.taskDefinition.networkMode === ecs.NetworkMode.AWS_VPC
75 ? {
76 ...baseEcsParameters,
77 launchType: this.taskDefinition.isEc2Compatible ? 'EC2' : 'FARGATE',
78 platformVersion: this.platformVersion,
79 networkConfiguration: {
80 awsVpcConfiguration: {
81 subnets: this.props.cluster.vpc.selectSubnets(subnetSelection).subnetIds,
82 assignPublicIp,
83 securityGroups: this.securityGroups && this.securityGroups.map(sg => sg.securityGroupId),
84 },
85 },
86 }
87 : baseEcsParameters;
88 return {
89 arn,
90 role,
91 ecsParameters,
92 input: events.RuleTargetInput.fromObject(input),
93 targetResource: this.taskDefinition,
94 };
95 }
96 createEventRolePolicyStatements() {
97 const policyStatements = [new iam.PolicyStatement({
98 actions: ['ecs:RunTask'],
99 resources: [this.taskDefinition.taskDefinitionArn],
100 conditions: {
101 ArnEquals: { 'ecs:cluster': this.cluster.clusterArn },
102 },
103 })];
104 // If it so happens that a Task Execution Role was created for the TaskDefinition,
105 // then the EventBridge Role must have permissions to pass it (otherwise it doesn't).
106 if (this.taskDefinition.executionRole !== undefined) {
107 policyStatements.push(new iam.PolicyStatement({
108 actions: ['iam:PassRole'],
109 resources: [this.taskDefinition.executionRole.roleArn],
110 }));
111 }
112 // For Fargate task we need permission to pass the task role.
113 if (this.taskDefinition.isFargateCompatible) {
114 policyStatements.push(new iam.PolicyStatement({
115 actions: ['iam:PassRole'],
116 resources: [this.taskDefinition.taskRole.roleArn],
117 }));
118 }
119 return policyStatements;
120 }
121}
122exports.EcsTask = EcsTask;
123_a = JSII_RTTI_SYMBOL_1;
124EcsTask[_a] = { fqn: "@aws-cdk/aws-events-targets.EcsTask", version: "1.202.0" };
125//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecs-task.js","sourceRoot":"","sources":["ecs-task.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ec2 from '@aws-cdk/aws-ec2';\nimport * as ecs from '@aws-cdk/aws-ecs';\nimport * as events from '@aws-cdk/aws-events';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as cdk from '@aws-cdk/core';\nimport { ContainerOverride } from './ecs-task-properties';\nimport { singletonEventRole } from './util';\n\n/**\n * Properties to define an ECS Event Task\n */\nexport interface EcsTaskProps {\n  /**\n   * Cluster where service will be deployed\n   */\n  readonly cluster: ecs.ICluster;\n\n  /**\n   * Task Definition of the task that should be started\n   */\n  readonly taskDefinition: ecs.ITaskDefinition;\n\n  /**\n   * How many tasks should be started when this event is triggered\n   *\n   * @default 1\n   */\n  readonly taskCount?: number;\n\n  /**\n   * Container setting overrides\n   *\n   * Key is the name of the container to override, value is the\n   * values you want to override.\n   */\n  readonly containerOverrides?: ContainerOverride[];\n\n  /**\n   * In what subnets to place the task's ENIs\n   *\n   * (Only applicable in case the TaskDefinition is configured for AwsVpc networking)\n   *\n   * @default Private subnets\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Existing security group to use for the task's ENIs\n   *\n   * (Only applicable in case the TaskDefinition is configured for AwsVpc networking)\n   *\n   * @default A new security group is created\n   * @deprecated use securityGroups instead\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Existing security groups to use for the task's ENIs\n   *\n   * (Only applicable in case the TaskDefinition is configured for AwsVpc networking)\n   *\n   * @default A new security group is created\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Existing IAM role to run the ECS task\n   *\n   * @default A new IAM role is created\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * The platform version on which to run your task\n   *\n   * Unless you have specific compatibility requirements, you don't need to specify this.\n   *\n   * @see https://docs.aws.amazon.com/AmazonECS/latest/developerguide/platform_versions.html\n   *\n   * @default - ECS will set the Fargate platform version to 'LATEST'\n   */\n  readonly platformVersion?: ecs.FargatePlatformVersion;\n}\n\n/**\n * Start a task on an ECS cluster\n */\nexport class EcsTask implements events.IRuleTarget {\n  // Security group fields are public because we can generate a new security group if none is provided.\n\n  /**\n   * The security group associated with the task. Only applicable with awsvpc network mode.\n   *\n   * @default - A new security group is created.\n   * @deprecated use securityGroups instead.\n   */\n  public readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * The security groups associated with the task. Only applicable with awsvpc network mode.\n   *\n   * @default - A new security group is created.\n   */\n  public readonly securityGroups?: ec2.ISecurityGroup[];\n  private readonly cluster: ecs.ICluster;\n  private readonly taskDefinition: ecs.ITaskDefinition;\n  private readonly taskCount: number;\n  private readonly role: iam.IRole;\n  private readonly platformVersion?: ecs.FargatePlatformVersion;\n\n  constructor(private readonly props: EcsTaskProps) {\n    if (props.securityGroup !== undefined && props.securityGroups !== undefined) {\n      throw new Error('Only one of SecurityGroup or SecurityGroups can be populated.');\n    }\n\n    this.cluster = props.cluster;\n    this.taskDefinition = props.taskDefinition;\n    this.taskCount = props.taskCount ?? 1;\n    this.platformVersion = props.platformVersion;\n\n    this.role = props.role ?? singletonEventRole(this.taskDefinition);\n    for (const stmt of this.createEventRolePolicyStatements()) {\n      this.role.addToPrincipalPolicy(stmt);\n    }\n\n    // Security groups are only configurable with the \"awsvpc\" network mode.\n    if (this.taskDefinition.networkMode !== ecs.NetworkMode.AWS_VPC) {\n      if (props.securityGroup !== undefined || props.securityGroups !== undefined) {\n        cdk.Annotations.of(this.taskDefinition).addWarning('security groups are ignored when network mode is not awsvpc');\n      }\n      return;\n    }\n    if (props.securityGroups) {\n      this.securityGroups = props.securityGroups;\n      return;\n    }\n\n    if (!cdk.Construct.isConstruct(this.taskDefinition)) {\n      throw new Error('Cannot create a security group for ECS task. ' +\n        'The task definition in ECS task is not a Construct. ' +\n        'Please pass a taskDefinition as a Construct in EcsTaskProps.');\n    }\n\n    let securityGroup = props.securityGroup || this.taskDefinition.node.tryFindChild('SecurityGroup') as ec2.ISecurityGroup;\n    securityGroup = securityGroup || new ec2.SecurityGroup(this.taskDefinition, 'SecurityGroup', { vpc: this.props.cluster.vpc });\n    this.securityGroup = securityGroup; // Maintain backwards-compatibility for customers that read the generated security group.\n    this.securityGroups = [securityGroup];\n  }\n\n  /**\n   * Allows using tasks as target of EventBridge events\n   */\n  public bind(_rule: events.IRule, _id?: string): events.RuleTargetConfig {\n    const arn = this.cluster.clusterArn;\n    const role = this.role;\n    const containerOverrides = this.props.containerOverrides && this.props.containerOverrides\n      .map(({ containerName, ...overrides }) => ({ name: containerName, ...overrides }));\n    const input = { containerOverrides };\n    const taskCount = this.taskCount;\n    const taskDefinitionArn = this.taskDefinition.taskDefinitionArn;\n\n    const subnetSelection = this.props.subnetSelection || { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT };\n    const assignPublicIp = subnetSelection.subnetType === ec2.SubnetType.PUBLIC ? 'ENABLED' : 'DISABLED';\n\n    const baseEcsParameters = { taskCount, taskDefinitionArn };\n\n    const ecsParameters: events.CfnRule.EcsParametersProperty = this.taskDefinition.networkMode === ecs.NetworkMode.AWS_VPC\n      ? {\n        ...baseEcsParameters,\n        launchType: this.taskDefinition.isEc2Compatible ? 'EC2' : 'FARGATE',\n        platformVersion: this.platformVersion,\n        networkConfiguration: {\n          awsVpcConfiguration: {\n            subnets: this.props.cluster.vpc.selectSubnets(subnetSelection).subnetIds,\n            assignPublicIp,\n            securityGroups: this.securityGroups && this.securityGroups.map(sg => sg.securityGroupId),\n          },\n        },\n      }\n      : baseEcsParameters;\n\n    return {\n      arn,\n      role,\n      ecsParameters,\n      input: events.RuleTargetInput.fromObject(input),\n      targetResource: this.taskDefinition,\n    };\n  }\n\n  private createEventRolePolicyStatements(): iam.PolicyStatement[] {\n    const policyStatements = [new iam.PolicyStatement({\n      actions: ['ecs:RunTask'],\n      resources: [this.taskDefinition.taskDefinitionArn],\n      conditions: {\n        ArnEquals: { 'ecs:cluster': this.cluster.clusterArn },\n      },\n    })];\n\n    // If it so happens that a Task Execution Role was created for the TaskDefinition,\n    // then the EventBridge Role must have permissions to pass it (otherwise it doesn't).\n    if (this.taskDefinition.executionRole !== undefined) {\n      policyStatements.push(new iam.PolicyStatement({\n        actions: ['iam:PassRole'],\n        resources: [this.taskDefinition.executionRole.roleArn],\n      }));\n    }\n\n    // For Fargate task we need permission to pass the task role.\n    if (this.taskDefinition.isFargateCompatible) {\n      policyStatements.push(new iam.PolicyStatement({\n        actions: ['iam:PassRole'],\n        resources: [this.taskDefinition.taskRole.roleArn],\n      }));\n    }\n\n    return policyStatements;\n  }\n}\n"]}
\No newline at end of file