UNPKG

@cdklabs/cdk-ecs-codedeploy

Version:

CDK Constructs for performing ECS Deployments with CodeDeploy

106 lines 16.9 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.EcsAppSpec = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); /** * Represents an AppSpec to be used for ECS services. * * see: https://docs.aws.amazon.com/codedeploy/latest/userguide/reference-appspec-file-structure-resources.html#reference-appspec-file-structure-resources-ecs */ class EcsAppSpec { constructor(targetService, hooks) { this.targetService = targetService; this.hooks = hooks; } /** * Render JSON string for this AppSpec to be used * * @returns string representation of this AppSpec */ toString() { const appSpec = { version: '0.0', Resources: [{ TargetService: { Type: 'AWS::ECS::Service', Properties: { TaskDefinition: this.targetService.taskDefinition.taskDefinitionArn, LoadBalancerInfo: { ContainerName: this.targetService.containerName, ContainerPort: this.targetService.containerPort, }, PlatformVersion: this.targetService.platformVersion, NetworkConfiguration: this.configureAwsVpcNetworkingWithSecurityGroups(this.targetService.awsvpcConfiguration), CapacityProviderStrategy: this.targetService.capacityProviderStrategy?.map(capacityProviderStrategy => { return { Base: capacityProviderStrategy.base, CapacityProvider: capacityProviderStrategy.capacityProvider, Weight: capacityProviderStrategy.weight, }; }), }, }, }], ...this.hooksSection(), }; return JSON.stringify(appSpec); } hooksSection() { if (this.hooks == undefined) { return {}; } const hooks = this.hooks; if (this.hooks.beforeInstall == undefined && this.hooks.afterInstall == undefined && this.hooks.afterAllowTestTraffic == undefined && this.hooks.beforeAllowTraffic == undefined && this.hooks.afterAllowTraffic == undefined) { return {}; } const capitalize = (s) => s && s[0].toUpperCase() + s.slice(1); const hook = (name) => { const hookValue = hooks[name]; if (hookValue == undefined) { return []; } else { return [{ [capitalize(name)]: typeof hookValue === 'string' ? hookValue : hookValue.functionArn, }]; } }; const beforeInstall = hook('beforeInstall'); const afterInstall = hook('afterInstall'); const afterAllowTestTraffic = hook('afterAllowTestTraffic'); const beforeAllowTraffic = hook('beforeAllowTraffic'); const afterAllowTraffic = hook('afterAllowTraffic'); return { Hooks: [ ...beforeInstall, ...afterInstall, ...afterAllowTestTraffic, ...beforeAllowTraffic, ...afterAllowTraffic, ], }; } configureAwsVpcNetworkingWithSecurityGroups(awsvpcConfiguration) { if (!awsvpcConfiguration) { return undefined; } return { awsvpcConfiguration: { assignPublicIp: awsvpcConfiguration.assignPublicIp ? 'ENABLED' : 'DISABLED', subnets: awsvpcConfiguration.vpc.selectSubnets(awsvpcConfiguration.vpcSubnets).subnetIds, securityGroups: awsvpcConfiguration.securityGroups?.map((sg) => sg.securityGroupId), }, }; } } exports.EcsAppSpec = EcsAppSpec; _a = JSII_RTTI_SYMBOL_1; EcsAppSpec[_a] = { fqn: "@cdklabs/cdk-ecs-codedeploy.EcsAppSpec", version: "0.0.421" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/ecs-appspec/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\n\n/**\n * Represents an AppSpec to be used for ECS services.\n *\n * see: https://docs.aws.amazon.com/codedeploy/latest/userguide/reference-appspec-file-structure-resources.html#reference-appspec-file-structure-resources-ecs\n */\nexport class EcsAppSpec {\n  /**\n   * Service to target for the deployment\n   */\n  private readonly targetService: TargetService;\n  /**\n   * Optional lifecycle hooks\n   */\n  private readonly hooks?: AppSpecHooks;\n\n  constructor(targetService: TargetService, hooks?: AppSpecHooks) {\n    this.targetService = targetService;\n    this.hooks = hooks;\n  }\n\n  /**\n   * Render JSON string for this AppSpec to be used\n   *\n   * @returns string representation of this AppSpec\n   */\n  toString(): string {\n    const appSpec = {\n      version: '0.0',\n      Resources: [{\n        TargetService: {\n          Type: 'AWS::ECS::Service',\n          Properties: {\n            TaskDefinition: this.targetService.taskDefinition.taskDefinitionArn,\n            LoadBalancerInfo: {\n              ContainerName: this.targetService.containerName,\n              ContainerPort: this.targetService.containerPort,\n            },\n            PlatformVersion: this.targetService.platformVersion,\n            NetworkConfiguration: this.configureAwsVpcNetworkingWithSecurityGroups(this.targetService.awsvpcConfiguration),\n            CapacityProviderStrategy: this.targetService.capacityProviderStrategy?.map(capacityProviderStrategy => {\n              return {\n                Base: capacityProviderStrategy.base,\n                CapacityProvider: capacityProviderStrategy.capacityProvider,\n                Weight: capacityProviderStrategy.weight,\n              };\n            }),\n          },\n        },\n      }],\n      ...this.hooksSection(),\n    };\n    return JSON.stringify(appSpec);\n  }\n\n  private hooksSection() {\n    if (this.hooks == undefined) {\n      return {};\n    }\n    const hooks = this.hooks;\n    if (\n      this.hooks.beforeInstall == undefined &&\n      this.hooks.afterInstall == undefined &&\n      this.hooks.afterAllowTestTraffic == undefined &&\n      this.hooks.beforeAllowTraffic == undefined &&\n      this.hooks.afterAllowTraffic == undefined\n    ) {\n      return {};\n    }\n    const capitalize = (s: string) => s && s[0].toUpperCase() + s.slice(1);\n    const hook = (name: keyof AppSpecHooks) => {\n      const hookValue = hooks[name];\n      if (hookValue == undefined) {\n        return [];\n      } else {\n        return [{\n          [capitalize(name)]:\n            typeof hookValue === 'string'\n              ? hookValue\n              : hookValue.functionArn,\n        }];\n      }\n    };\n    const beforeInstall = hook('beforeInstall');\n    const afterInstall = hook('afterInstall');\n    const afterAllowTestTraffic = hook('afterAllowTestTraffic');\n    const beforeAllowTraffic = hook('beforeAllowTraffic');\n    const afterAllowTraffic = hook('afterAllowTraffic');\n\n    return {\n      Hooks: [\n        ...beforeInstall,\n        ...afterInstall,\n        ...afterAllowTestTraffic,\n        ...beforeAllowTraffic,\n        ...afterAllowTraffic,\n      ],\n    };\n  }\n\n  private configureAwsVpcNetworkingWithSecurityGroups(awsvpcConfiguration?: AwsvpcConfiguration) {\n    if (!awsvpcConfiguration) {\n      return undefined;\n    }\n    return {\n      awsvpcConfiguration: {\n        assignPublicIp: awsvpcConfiguration.assignPublicIp ? 'ENABLED' : 'DISABLED',\n        subnets: awsvpcConfiguration.vpc.selectSubnets(awsvpcConfiguration.vpcSubnets).subnetIds,\n        securityGroups: awsvpcConfiguration.securityGroups?.map((sg) => sg.securityGroupId),\n      },\n    };\n  }\n}\n\n/**\n * Describe the target for CodeDeploy to use when creating a deployment for an ecs.EcsDeploymentGroup.\n */\nexport interface TargetService {\n  /**\n   * The TaskDefintion to deploy to the target services.\n   */\n  readonly taskDefinition: ecs.ITaskDefinition;\n\n  /**\n   * The name of the Amazon ECS container that contains your Amazon ECS application. It must be a container specified in your Amazon ECS task definition.\n   */\n  readonly containerName: string;\n\n  /**\n   * The port on the container where traffic will be routed to.\n   */\n  readonly containerPort: number;\n\n  /**\n   * The platform version of the Fargate tasks in the deployed Amazon ECS service.\n   * see: https://docs.aws.amazon.com/AmazonECS/latest/developerguide/platform_versions.html\n   *\n   * @default LATEST\n   */\n  readonly platformVersion?: ecs.FargatePlatformVersion;\n\n  /**\n   * Network configuration for ECS services that have a network type of `awsvpc`.\n   *\n   * @default reuse current network settings for ECS service.\n   */\n  readonly awsvpcConfiguration?: AwsvpcConfiguration;\n\n  /**\n   * A list of Amazon ECS capacity providers to use for the deployment.\n   *\n   * @default reuse current capcity provider strategy for ECS service.\n   */\n  readonly capacityProviderStrategy?: ecs.CapacityProviderStrategy[];\n\n}\n\n/**\n * Network configuration for ECS services that have a network type of `awsvpc`.\n */\nexport interface AwsvpcConfiguration {\n  /**\n   * The VPC to use for the task.\n   */\n  readonly vpc: ec2.IVpc;\n\n  /**\n   * The Subnets to use for the task.\n   */\n  readonly vpcSubnets: ec2.SubnetSelection;\n\n  /**\n   * The Security Groups to use for the task.\n   */\n  readonly securityGroups: ec2.ISecurityGroup[];\n\n  /**\n   * Assign a public IP address to the task.\n   */\n  readonly assignPublicIp: boolean;\n}\n\n/**\n * Lifecycle hooks configuration\n */\nexport interface AppSpecHooks {\n  /**\n   * Lambda or ARN of a lambda to run tasks before the replacement task set is created.\n   */\n  readonly beforeInstall?: string | lambda.IFunction;\n  /**\n   * Lambda or ARN of a lambda to run tasks after the replacement task set is created and one of the target groups is associated with it.\n   */\n  readonly afterInstall?: string | lambda.IFunction;\n  /**\n   * Lambda or ARN of a lambda to run tasks after the test listener serves traffic to the replacement task set.\n   */\n  readonly afterAllowTestTraffic?: string | lambda.IFunction;\n  /**\n   * Lambda or ARN of a lambda to run tasks after the second target group is associated with the replacement task set, but before traffic is shifted to the replacement task set.\n   */\n  readonly beforeAllowTraffic?: string | lambda.IFunction;\n  /**\n   * Lambda or ARN of a lambda to run tasks after the second target group serves traffic to the replacement task set.\n   */\n  readonly afterAllowTraffic?: string | lambda.IFunction;\n}\n"]}