UNPKG

@aws-solutions-constructs/core

Version:
192 lines 30.8 kB
"use strict"; /** * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance * with the License. A copy of the License is located at * * http://www.apache.org/licenses/LICENSE-2.0 * * or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES * OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions * and limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.CreateFargateService = CreateFargateService; exports.CheckFargateProps = CheckFargateProps; exports.CheckForConflictingServiceProps = CheckForConflictingServiceProps; exports.CheckForInvalidVpcs = CheckForInvalidVpcs; exports.getServiceVpcSecurityGroupIds = getServiceVpcSecurityGroupIds; const ec2 = require("aws-cdk-lib/aws-ec2"); const ecs = require("aws-cdk-lib/aws-ecs"); const ecr = require("aws-cdk-lib/aws-ecr"); const defaults = require(".."); const __1 = require(".."); /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function CreateFargateService(scope, id, props) { defaults.AddAwsServiceEndpoint(scope, props.constructVpc, defaults.ServiceEndpointTypes.ECR_API); defaults.AddAwsServiceEndpoint(scope, props.constructVpc, defaults.ServiceEndpointTypes.ECR_DKR); defaults.AddAwsServiceEndpoint(scope, props.constructVpc, defaults.ServiceEndpointTypes.S3); const constructContainerDefinitionProps = {}; const constructFargateServiceDefinitionProps = {}; if (!props.clientFargateServiceProps?.cluster) { // Construct Fargate Service constructFargateServiceDefinitionProps.cluster = CreateCluster(scope, `${id}-cluster`, props.constructVpc, props.clientClusterProps); } // Set up the Fargate service if (!props.clientContainerDefinitionProps?.image) { constructContainerDefinitionProps.image = CreateImage(scope, id, props.ecrRepositoryArn, props.ecrImageVersion); } // Create the Fargate Service const createTaskDefinitionResponse = CreateTaskDefinition(scope, id, props.clientFargateTaskDefinitionProps, props.clientContainerDefinitionProps, constructContainerDefinitionProps); constructFargateServiceDefinitionProps.taskDefinition = createTaskDefinitionResponse.taskDefinition; const newContainerDefinition = createTaskDefinitionResponse.containerDefinition; if (!props.clientFargateServiceProps?.vpcSubnets) { if (props.constructVpc.isolatedSubnets.length) { constructFargateServiceDefinitionProps.vpcSubnets = { subnets: props.constructVpc.isolatedSubnets, }; } else { constructFargateServiceDefinitionProps.vpcSubnets = { subnets: props.constructVpc.privateSubnets, }; } } let defaultFargateServiceProps; if (!props.clientFargateServiceProps?.securityGroups) { const serviceSecurityGroup = new ec2.SecurityGroup(scope, `${id}-sg`, { allowAllOutbound: true, disableInlineRules: false, vpc: props.constructVpc, // We add a description here so that this SG can be easily identified in tests description: 'Construct created security group' }); defaultFargateServiceProps = (0, __1.overrideProps)(defaults.DefaultFargateServiceProps(), { securityGroups: [serviceSecurityGroup] }); defaults.addCfnSuppressRules(serviceSecurityGroup, [ { id: 'W5', reason: 'Egress of 0.0.0.0/0 is default and generally considered OK', }, { id: 'W40', reason: 'Egress IPProtocol of -1 is default and generally considered OK', } ]); } else { defaultFargateServiceProps = defaults.DefaultFargateServiceProps(); } const fargateServiceProps = defaults.consolidateProps(defaultFargateServiceProps, props.clientFargateServiceProps, constructFargateServiceDefinitionProps); const newService = new ecs.FargateService(scope, `${id}-service`, fargateServiceProps); return { service: newService, containerDefinition: newContainerDefinition }; } function CreateCluster(scope, id, constructVpc, clientClusterProps) { const constructProps = { vpc: constructVpc }; return new ecs.Cluster(scope, id, defaults.consolidateProps(defaults.DefaultClusterProps(), clientClusterProps, constructProps)); } function CreateImage(scope, id, repositoryArn, imageVersion) { if (repositoryArn) { return ecs.ContainerImage.fromEcrRepository(ecr.Repository.fromRepositoryArn(scope, `${id}-container`, repositoryArn), imageVersion || "latest"); } else { throw new Error("Not Implemented - image without repo name and version"); } } function CreateTaskDefinition(scope, id, clientFargateTaskDefinitionProps, clientContainerDefinitionProps, constructContainerDefinitionProps) { const taskDefinitionProps = defaults.consolidateProps(defaults.DefaultFargateTaskDefinitionProps(), clientFargateTaskDefinitionProps); const taskDefinition = new ecs.FargateTaskDefinition(scope, `${id}-taskdef`, taskDefinitionProps); const defaultContainerDefinitionProps = defaults.consolidateProps(defaults.DefaultContainerDefinitionProps(), {}, { containerName: `${id}-container`, }); const containerDefinitionProps = defaults.consolidateProps(defaultContainerDefinitionProps, clientContainerDefinitionProps, constructContainerDefinitionProps); const containerDefinition = taskDefinition.addContainer(`${id}-container`, containerDefinitionProps); return { taskDefinition, containerDefinition }; } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function CheckFargateProps(props) { let errorMessages = ""; let errorFound = false; if (CheckForConflictingServiceProps(props)) { errorFound = true; errorMessages += "If you provide an existingFargateServiceObject, you cannot provide any props defining a new service\n"; } if (props.existingImageObject && (props.ecrImageVersion || props.ecrRepositoryArn)) { errorFound = true; errorMessages += "If you provide an existingImageObject then you cannot provide an ecrRepositoryArn nor ecrImageVersion\n"; } // Confirm no network information in Target Group Props if (props.targetGroupProps?.vpc) { errorFound = true; errorMessages += "Provide all VPC info at Construct level, not within targetGroupProps\n"; } if (props.fargateServiceProps?.taskDefinition) { errorFound = true; errorMessages += "The construct cannot accept an existing task definition in fargateServiceProps\n"; } if (props.fargateServiceProps?.cluster && props.clusterProps) { errorFound = true; errorMessages += "If you provide a cluster in fargateServiceProps then you cannot provide clusterProps\n"; } if (CheckForInvalidVpcs(props)) { errorFound = true; errorMessages += "Provide all VPC info at Construct level, not within clusterProps nor targetGroupProps\n"; } if ((props.existingFargateServiceObject || props.existingContainerDefinitionObject) && (!props.existingFargateServiceObject || !props.existingContainerDefinitionObject || !props.existingVpc)) { errorFound = true; errorMessages += "If an existing Service is indicated by supplying either existingFargateServiceObject or existingContainerDefinitionObject, then existingFargateServiceObject, existingContainerDefinitionObject, and existingVpc must all be provided\n"; } if (errorFound) { throw new Error(errorMessages); } } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function CheckForConflictingServiceProps(props) { if (props.existingFargateServiceObject && (props.existingImageObject || props.ecrImageVersion || props.containerDefinitionProps || props.fargateTaskDefinitionProps || props.ecrRepositoryArn || props.fargateServiceProps || props.clusterProps)) { return true; } return false; } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function CheckForInvalidVpcs(props) { if (props.clusterProps?.vpc || props.targetGroupProps?.vpc) { return true; } return false; } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function getServiceVpcSecurityGroupIds(service) { const securityGroupIds = []; service.connections.securityGroups.forEach(element => securityGroupIds.push(element.securityGroupId)); return securityGroupIds; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-helper.js","sourceRoot":"","sources":["fargate-helper.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\n/*\n *  The functions found here in the core library are for internal use and can be changed\n *  or removed outside of a major release. We recommend against calling them directly from client code.\n */\n\nimport { Construct } from \"constructs\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as ecs from \"aws-cdk-lib/aws-ecs\";\nimport * as ecr from \"aws-cdk-lib/aws-ecr\";\nimport * as defaults from \"..\";\nimport { overrideProps } from \"..\";\n\nexport interface CreateFargateServiceResponse {\n  readonly service: ecs.FargateService,\n  readonly containerDefinition: ecs.ContainerDefinition\n}\n\nexport interface CreateFargateServiceProps {\n  readonly constructVpc: ec2.IVpc,\n  readonly clientClusterProps?: ecs.ClusterProps,\n  readonly ecrRepositoryArn?: string,\n  readonly ecrImageVersion?: string,\n  readonly clientFargateTaskDefinitionProps?: ecs.FargateTaskDefinitionProps | any,\n  readonly clientContainerDefinitionProps?: ecs.ContainerDefinitionProps | any,\n  readonly clientFargateServiceProps?: ecs.FargateServiceProps | any\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function CreateFargateService(\n  scope: Construct,\n  id: string,\n  props: CreateFargateServiceProps\n): CreateFargateServiceResponse {\n  defaults.AddAwsServiceEndpoint(\n    scope,\n    props.constructVpc,\n    defaults.ServiceEndpointTypes.ECR_API\n  );\n  defaults.AddAwsServiceEndpoint(\n    scope,\n    props.constructVpc,\n    defaults.ServiceEndpointTypes.ECR_DKR\n  );\n  defaults.AddAwsServiceEndpoint(\n    scope,\n    props.constructVpc,\n    defaults.ServiceEndpointTypes.S3\n  );\n\n  const constructContainerDefinitionProps: any = {};\n  const constructFargateServiceDefinitionProps: any = {};\n\n  if (!props.clientFargateServiceProps?.cluster) {\n    // Construct Fargate Service\n    constructFargateServiceDefinitionProps.cluster = CreateCluster(\n      scope,\n      `${id}-cluster`,\n      props.constructVpc,\n      props.clientClusterProps\n    );\n  }\n\n  // Set up the Fargate service\n  if (!props.clientContainerDefinitionProps?.image) {\n    constructContainerDefinitionProps.image = CreateImage(\n      scope,\n      id,\n      props.ecrRepositoryArn,\n      props.ecrImageVersion\n    );\n  }\n\n  // Create the Fargate Service\n  const createTaskDefinitionResponse = CreateTaskDefinition(\n    scope,\n    id,\n    props.clientFargateTaskDefinitionProps,\n    props.clientContainerDefinitionProps,\n    constructContainerDefinitionProps\n  );\n  constructFargateServiceDefinitionProps.taskDefinition = createTaskDefinitionResponse.taskDefinition;\n  const newContainerDefinition = createTaskDefinitionResponse.containerDefinition;\n\n  if (!props.clientFargateServiceProps?.vpcSubnets) {\n    if (props.constructVpc.isolatedSubnets.length) {\n      constructFargateServiceDefinitionProps.vpcSubnets = {\n        subnets: props.constructVpc.isolatedSubnets,\n      };\n    } else {\n      constructFargateServiceDefinitionProps.vpcSubnets = {\n        subnets: props.constructVpc.privateSubnets,\n      };\n    }\n  }\n\n  let defaultFargateServiceProps;\n\n  if (!props.clientFargateServiceProps?.securityGroups) {\n    const serviceSecurityGroup = new ec2.SecurityGroup(scope, `${id}-sg`, {\n      allowAllOutbound: true,\n      disableInlineRules: false,\n      vpc: props.constructVpc,\n      // We add a description here so that this SG can be easily identified in tests\n      description: 'Construct created security group'\n    });\n    defaultFargateServiceProps = overrideProps(defaults.DefaultFargateServiceProps(), { securityGroups: [ serviceSecurityGroup ]});\n    defaults.addCfnSuppressRules(serviceSecurityGroup, [\n      {\n        id: 'W5',\n        reason: 'Egress of 0.0.0.0/0 is default and generally considered OK',\n      },\n      {\n        id: 'W40',\n        reason: 'Egress IPProtocol of -1 is default and generally considered OK',\n      }\n    ]);\n  } else {\n    defaultFargateServiceProps = defaults.DefaultFargateServiceProps();\n  }\n\n  const fargateServiceProps = defaults.consolidateProps(\n    defaultFargateServiceProps,\n    props.clientFargateServiceProps,\n    constructFargateServiceDefinitionProps\n  );\n\n  const newService = new ecs.FargateService(\n    scope,\n    `${id}-service`,\n    fargateServiceProps,\n  );\n\n  return { service: newService, containerDefinition: newContainerDefinition };\n}\n\nfunction CreateCluster(\n  scope: Construct,\n  id: string,\n  constructVpc: ec2.IVpc,\n  clientClusterProps?: ecs.ClusterProps\n): ecs.ICluster {\n  const constructProps = { vpc: constructVpc };\n  return new ecs.Cluster(\n    scope,\n    id,\n    defaults.consolidateProps(\n      defaults.DefaultClusterProps(),\n      clientClusterProps,\n      constructProps\n    )\n  );\n}\n\nfunction CreateImage(\n  scope: Construct,\n  id: string,\n  repositoryArn?: string,\n  imageVersion?: string\n): ecs.ContainerImage {\n  if (repositoryArn) {\n    return ecs.ContainerImage.fromEcrRepository(\n      ecr.Repository.fromRepositoryArn(scope, `${id}-container`, repositoryArn),\n      imageVersion || \"latest\"\n    );\n  } else {\n    throw new Error(\"Not Implemented - image without repo name and version\");\n  }\n}\n\ninterface CreateTaskDefinitionResponse {\n  taskDefinition: ecs.FargateTaskDefinition\n  containerDefinition: ecs.ContainerDefinition\n}\n\nfunction CreateTaskDefinition(\n  scope: Construct,\n  id: string,\n  clientFargateTaskDefinitionProps?: ecs.FargateTaskDefinitionProps,\n  clientContainerDefinitionProps?: ecs.ContainerDefinitionProps,\n  constructContainerDefinitionProps?: ecs.ContainerDefinitionProps\n): CreateTaskDefinitionResponse {\n  const taskDefinitionProps = defaults.consolidateProps(\n    defaults.DefaultFargateTaskDefinitionProps(),\n    clientFargateTaskDefinitionProps\n  );\n  const taskDefinition = new ecs.FargateTaskDefinition(\n    scope,\n    `${id}-taskdef`,\n    taskDefinitionProps\n  );\n\n  const defaultContainerDefinitionProps = defaults.consolidateProps(defaults.DefaultContainerDefinitionProps(), {}, {\n    containerName: `${id}-container`,\n  });\n  const containerDefinitionProps = defaults.consolidateProps(\n    defaultContainerDefinitionProps,\n    clientContainerDefinitionProps,\n    constructContainerDefinitionProps,\n  );\n  const containerDefinition = taskDefinition.addContainer(`${id}-container`, containerDefinitionProps);\n  return { taskDefinition, containerDefinition };\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function CheckFargateProps(props: any) {\n  let errorMessages = \"\";\n  let errorFound = false;\n\n  if (CheckForConflictingServiceProps(props)) {\n    errorFound = true;\n    errorMessages +=\n      \"If you provide an existingFargateServiceObject, you cannot provide any props defining a new service\\n\";\n  }\n\n  if (\n    props.existingImageObject &&\n    (props.ecrImageVersion || props.ecrRepositoryArn)\n  ) {\n    errorFound = true;\n    errorMessages +=\n      \"If you provide an existingImageObject then you cannot provide an ecrRepositoryArn nor ecrImageVersion\\n\";\n  }\n\n  // Confirm no network information in Target Group Props\n  if (props.targetGroupProps?.vpc) {\n    errorFound = true;\n    errorMessages +=\n      \"Provide all VPC info at Construct level, not within targetGroupProps\\n\";\n  }\n\n  if (props.fargateServiceProps?.taskDefinition) {\n    errorFound = true;\n    errorMessages +=\n      \"The construct cannot accept an existing task definition in fargateServiceProps\\n\";\n  }\n\n  if (props.fargateServiceProps?.cluster && props.clusterProps) {\n    errorFound = true;\n    errorMessages +=\n      \"If you provide a cluster in fargateServiceProps then you cannot provide clusterProps\\n\";\n  }\n\n  if (CheckForInvalidVpcs(props)) {\n    errorFound = true;\n    errorMessages +=\n      \"Provide all VPC info at Construct level, not within clusterProps nor targetGroupProps\\n\";\n  }\n\n  if (\n    (props.existingFargateServiceObject ||\n      props.existingContainerDefinitionObject) &&\n    (!props.existingFargateServiceObject ||\n      !props.existingContainerDefinitionObject ||\n      !props.existingVpc)\n  ) {\n    errorFound = true;\n    errorMessages +=\n      \"If an existing Service is indicated by supplying either existingFargateServiceObject or existingContainerDefinitionObject, then existingFargateServiceObject, existingContainerDefinitionObject, and existingVpc must all be provided\\n\";\n  }\n\n  if (errorFound) {\n    throw new Error(errorMessages);\n  }\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function CheckForConflictingServiceProps(props: any): boolean {\n  if (props.existingFargateServiceObject &&\n    (props.existingImageObject ||\n      props.ecrImageVersion ||\n      props.containerDefinitionProps ||\n      props.fargateTaskDefinitionProps ||\n      props.ecrRepositoryArn ||\n      props.fargateServiceProps ||\n      props.clusterProps)\n  ) {\n    return true;\n  }\n  return false;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function CheckForInvalidVpcs(props: any): boolean {\n  if (props.clusterProps?.vpc || props.targetGroupProps?.vpc) {\n    return true;\n  }\n  return false;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function getServiceVpcSecurityGroupIds(service: ecs.FargateService): string[] {\n  const securityGroupIds: string[] = [];\n\n  service.connections.securityGroups.forEach(element => securityGroupIds.push(element.securityGroupId));\n\n  return securityGroupIds;\n}\n"]}