UNPKG

deploy-time-build

Version:
209 lines 32.9 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.ContainerImageBuild = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const fs_1 = require("fs"); const path_1 = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild"); const aws_ecr_1 = require("aws-cdk-lib/aws-ecr"); const aws_ecs_1 = require("aws-cdk-lib/aws-ecs"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_lambda_1 = require("aws-cdk-lib/aws-lambda"); const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets"); const constructs_1 = require("constructs"); const singleton_project_1 = require("./singleton-project"); /** * Build a container image and push it to an ECR repository on deploy-time. */ class ContainerImageBuild extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.props = props; const handler = new aws_lambda_1.SingletonFunction(this, 'CustomResourceHandler', { // Use raw string to avoid from tightening CDK version requirement runtime: new aws_lambda_1.Runtime('nodejs20.x', aws_lambda_1.RuntimeFamily.NODEJS), code: aws_lambda_1.Code.fromAsset((0, path_1.join)(__dirname, '../lambda/trigger-codebuild/dist')), handler: 'index.handler', uuid: 'db740fd5-5436-4a84-8a09-e6dfcd01f4f3', // generated for this construct lambdaPurpose: 'DeployTimeBuildCustomResourceHandler', timeout: aws_cdk_lib_1.Duration.minutes(5), }); // Use buildx for cross-platform image build const armImage = aws_codebuild_1.LinuxArmBuildImage.fromCodeBuildImageId('aws/codebuild/amazonlinux2-aarch64-standard:3.0'); const x64Image = aws_codebuild_1.LinuxBuildImage.fromCodeBuildImageId('aws/codebuild/standard:7.0'); // Select the build image based on the target platform const isArm64 = props.platform?.platform === 'linux/arm64'; const buildImage = isArm64 ? armImage : x64Image; let repository = props.repository; if (repository === undefined) { repository = new aws_ecr_1.Repository(this, 'Repository', { removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY }); repository.node.defaultChild.addPropertyOverride('EmptyOnDelete', true); } const repositoryUri = repository.repositoryUri; const imageArtifactName = 'artifact'; const project = new singleton_project_1.SingletonProject(this, 'Project', { uuid: 'e83729fe-b156-4e70-9bec-452b15847a30', projectPurpose: isArm64 ? 'ContainerImageBuildArm64' : 'ContainerImageBuildAmd64', environment: { computeType: aws_codebuild_1.ComputeType.SMALL, buildImage: buildImage, privileged: true, }, vpc: props.vpc, buildSpec: aws_codebuild_1.BuildSpec.fromObject({ version: '0.2', phases: { build: { commands: [ 'current_dir=$(pwd)', 'echo "$input"', 'mkdir workdir', 'cd workdir', 'aws s3 cp "$sourceS3Url" temp.zip', 'unzip temp.zip', 'ls -la', 'aws ecr get-login-password | docker login --username AWS --password-stdin $repositoryAuthUri', // for accessing ECR public 'aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin public.ecr.aws', 'docker buildx ls', 'eval "$buildCommand"', `docker tag ${imageArtifactName}:latest ${repositoryUri}:$imageTag`, `docker push ${repositoryUri}:$imageTag`, ], }, post_build: { commands: [ 'echo Build completed on `date`', ` STATUS='SUCCESS' if [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ] # Test if the build is failing then STATUS='FAILED' REASON="ContainerImageBuild failed. See CloudWatch Log stream for the detailed reason: https://$AWS_REGION.console.aws.amazon.com/cloudwatch/home?region=$AWS_REGION#logsV2:log-groups/log-group/\\$252Faws\\$252Fcodebuild\\$252F$projectName/log-events/$CODEBUILD_LOG_PATH" fi cat <<EOF > payload.json { "StackId": "$stackId", "RequestId": "$requestId", "LogicalResourceId":"$logicalResourceId", "PhysicalResourceId": "$imageTag", "Status": "$STATUS", "Reason": "$REASON", "Data": { "ImageTag": "$imageTag" } } EOF curl -v -i -X PUT -H 'Content-Type:' -d "@payload.json" "$responseURL" `, ], }, }, }), }).project; project.role.addManagedPolicy(aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly')); repository.grantPullPush(project); repository.grant(project, 'ecr:DescribeImages'); handler.addToRolePolicy(new aws_iam_1.PolicyStatement({ actions: ['codebuild:StartBuild'], resources: [project.projectArn], })); this.grantPrincipal = project.grantPrincipal; let assetExclude = props.exclude; // automatically read .dockerignore for convenience if (assetExclude === undefined && (0, fs_1.existsSync)((0, path_1.join)(props.directory, '.dockerignore'))) { assetExclude = (0, fs_1.readFileSync)((0, path_1.join)(props.directory, '.dockerignore')).toString().split('\n'); } const asset = new aws_s3_assets_1.Asset(this, 'Source', { ...props, exclude: assetExclude, path: props.directory, }); asset.grantRead(project); const buildCommandOptions = { ...props, platform: props.platform?.platform }; buildCommandOptions.outputs ?? (buildCommandOptions.outputs = []); // we don't use push=true here because CodeBuild Docker server does not seem to work properly with the configuration. buildCommandOptions.outputs.push('type=image', `name=${imageArtifactName}`); if (props.zstdCompression) { // to enable zstd compression, buildx directly pushes the artifact image to a registry // https://aws.amazon.com/blogs/containers/reducing-aws-fargate-startup-times-with-zstd-compressed-container-images/ buildCommandOptions.outputs.push('oci-mediatypes=true', 'compression=zstd', 'force-compression=true', 'compression-level=3'); } const buildCommand = this.getDockerBuildCommand(buildCommandOptions); const properties = { type: 'ContainerImageBuild', buildCommand: buildCommand, repositoryUri, imageTag: props.tag, tagPrefix: props.tagPrefix, codeBuildProjectName: project.projectName, sourceS3Url: asset.s3ObjectUrl, }; const custom = new aws_cdk_lib_1.CustomResource(this, 'Resource', { serviceToken: handler.functionArn, resourceType: 'Custom::CDKContainerImageBuild', properties, }); custom.node.addDependency(project); this.repository = repository; this.imageTag = custom.getAttString('ImageTag'); } /** * Get the instance of {@link DockerImageCode} for a Lambda function image. */ toLambdaDockerImageCode() { if (this.props.zstdCompression) { throw new Error('You cannot enable zstdCompression for a Lambda image.'); } return aws_lambda_1.DockerImageCode.fromEcr(this.repository, { tagOrDigest: this.imageTag }); } /** * Get the instance of {@link ContainerImage} for an ECS task definition. */ toEcsDockerImageCode() { return aws_ecs_1.ContainerImage.fromEcrRepository(this.repository, this.imageTag); } getDockerBuildCommand(options) { // the members of props differs with CDK version. // By regarding props as any, we can use props that are not available in older cdk versions. // logic is copied from packages/cdk-assets/lib/private/docker.ts const cacheOptionToFlag = (option) => { let flag = `type=${option.type}`; if (option.params) { flag += ',' + Object.entries(option.params) .map(([k, v]) => `${k}=${v}`) .join(','); } return flag; }; const flatten = (x) => { return Array.prototype.concat([], ...x); }; const dockerBuildCommand = [ 'docker buildx build', ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])), ...flatten(Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`])), ...(options.buildSsh ? ['--ssh', options.buildSsh] : []), ...(options.target ? ['--target', options.target] : []), ...(options.file ? ['--file', options.file] : []), ...(options.networkMode ? ['--network', options.networkMode] : []), ...(options.platform ? ['--platform', options.platform] : []), ...(options.outputs ? ['--output', options.outputs.join(',')] : []), ...(options.cacheFrom ? [...options.cacheFrom.map((cacheFrom) => ['--cache-from', cacheOptionToFlag(cacheFrom)]).flat()] : []), ...(options.cacheTo ? ['--cache-to', cacheOptionToFlag(options.cacheTo)] : []), ...(options.cacheDisabled ? ['--no-cache'] : []), '--provenance=false', '.', ]; return dockerBuildCommand.join(' '); } } exports.ContainerImageBuild = ContainerImageBuild; _a = JSII_RTTI_SYMBOL_1; ContainerImageBuild[_a] = { fqn: "deploy-time-build.ContainerImageBuild", version: "0.3.37" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container-image-build.js","sourceRoot":"","sources":["../src/container-image-build.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, readFileSync } from 'fs';\nimport { join } from 'path';\nimport { CfnResource, CustomResource, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { BuildSpec, ComputeType, LinuxArmBuildImage, LinuxBuildImage } from 'aws-cdk-lib/aws-codebuild';\nimport { IVpc } from 'aws-cdk-lib/aws-ec2';\nimport { IRepository, Repository } from 'aws-cdk-lib/aws-ecr';\nimport { DockerImageAssetProps } from 'aws-cdk-lib/aws-ecr-assets';\nimport { ContainerImage } from 'aws-cdk-lib/aws-ecs';\nimport { IGrantable, IPrincipal, ManagedPolicy, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, DockerImageCode, Runtime, RuntimeFamily, SingletonFunction } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Construct } from 'constructs';\nimport { SingletonProject } from './singleton-project';\nimport { ContainerImageBuildResourceProps } from './types';\n\nexport interface ContainerImageBuildProps extends DockerImageAssetProps {\n  /**\n   * The tag when to push the image\n   * @default use assetHash as tag\n   */\n  readonly tag?: string;\n\n  /**\n   * Prefix to add to the image tag\n   * @default no prefix\n   */\n  readonly tagPrefix?: string;\n\n  /**\n   * The ECR repository to push the image.\n   * @default create a new ECR repository\n   */\n  readonly repository?: IRepository;\n\n  /**\n   * Use zstd for compressing a container image.\n   * @default false\n   */\n  readonly zstdCompression?: boolean;\n\n  /**\n   * The VPC where your build job will be deployed.\n   * This VPC must have private subnets with NAT Gateways.\n   *\n   * Use this property when you want to control the outbound IP addresses that base images are pulled from.\n   * @default No VPC used.\n   */\n  readonly vpc?: IVpc;\n}\n\n/**\n * Build a container image and push it to an ECR repository on deploy-time.\n */\nexport class ContainerImageBuild extends Construct implements IGrantable {\n  public readonly grantPrincipal: IPrincipal;\n  public readonly repository: IRepository;\n  public readonly imageTag: string;\n\n  constructor(scope: Construct, id: string, private readonly props: ContainerImageBuildProps) {\n    super(scope, id);\n\n    const handler = new SingletonFunction(this, 'CustomResourceHandler', {\n      // Use raw string to avoid from tightening CDK version requirement\n      runtime: new Runtime('nodejs20.x', RuntimeFamily.NODEJS),\n      code: Code.fromAsset(join(__dirname, '../lambda/trigger-codebuild/dist')),\n      handler: 'index.handler',\n      uuid: 'db740fd5-5436-4a84-8a09-e6dfcd01f4f3', // generated for this construct\n      lambdaPurpose: 'DeployTimeBuildCustomResourceHandler',\n      timeout: Duration.minutes(5),\n    });\n\n    // Use buildx for cross-platform image build\n    const armImage = LinuxArmBuildImage.fromCodeBuildImageId('aws/codebuild/amazonlinux2-aarch64-standard:3.0');\n    const x64Image = LinuxBuildImage.fromCodeBuildImageId('aws/codebuild/standard:7.0');\n    // Select the build image based on the target platform\n    const isArm64 = props.platform?.platform === 'linux/arm64';\n    const buildImage = isArm64 ? armImage : x64Image;\n\n    let repository = props.repository;\n    if (repository === undefined) {\n      repository = new Repository(this, 'Repository', { removalPolicy: RemovalPolicy.DESTROY });\n      (repository.node.defaultChild as CfnResource).addPropertyOverride('EmptyOnDelete', true);\n    }\n    const repositoryUri = repository.repositoryUri;\n    const imageArtifactName = 'artifact';\n\n    const project = new SingletonProject(this, 'Project', {\n      uuid: 'e83729fe-b156-4e70-9bec-452b15847a30',\n      projectPurpose: isArm64 ? 'ContainerImageBuildArm64' : 'ContainerImageBuildAmd64',\n      environment: {\n        computeType: ComputeType.SMALL,\n        buildImage: buildImage,\n        privileged: true,\n      },\n      vpc: props.vpc,\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        phases: {\n          build: {\n            commands: [\n              'current_dir=$(pwd)',\n              'echo \"$input\"',\n              'mkdir workdir',\n              'cd workdir',\n              'aws s3 cp \"$sourceS3Url\" temp.zip',\n              'unzip temp.zip',\n              'ls -la',\n              'aws ecr get-login-password | docker login --username AWS --password-stdin $repositoryAuthUri',\n              // for accessing ECR public\n              'aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin public.ecr.aws',\n              'docker buildx ls',\n              'eval \"$buildCommand\"',\n              `docker tag ${imageArtifactName}:latest ${repositoryUri}:$imageTag`,\n              `docker push ${repositoryUri}:$imageTag`,\n            ],\n          },\n          post_build: {\n            commands: [\n              'echo Build completed on `date`',\n              `\nSTATUS='SUCCESS'\nif [ $CODEBUILD_BUILD_SUCCEEDING -ne 1 ] # Test if the build is failing\nthen\nSTATUS='FAILED'\nREASON=\"ContainerImageBuild failed. See CloudWatch Log stream for the detailed reason: \nhttps://$AWS_REGION.console.aws.amazon.com/cloudwatch/home?region=$AWS_REGION#logsV2:log-groups/log-group/\\\\$252Faws\\\\$252Fcodebuild\\\\$252F$projectName/log-events/$CODEBUILD_LOG_PATH\"\nfi\ncat <<EOF > payload.json\n{\n  \"StackId\": \"$stackId\",\n  \"RequestId\": \"$requestId\",\n  \"LogicalResourceId\":\"$logicalResourceId\",\n  \"PhysicalResourceId\": \"$imageTag\",\n  \"Status\": \"$STATUS\",\n  \"Reason\": \"$REASON\",\n  \"Data\": {\n    \"ImageTag\": \"$imageTag\"\n  }\n}\nEOF\ncurl -v -i -X PUT -H 'Content-Type:' -d \"@payload.json\" \"$responseURL\"\n              `,\n            ],\n          },\n        },\n      }),\n    }).project;\n\n    project.role!.addManagedPolicy(ManagedPolicy.fromAwsManagedPolicyName('AmazonElasticContainerRegistryPublicReadOnly'));\n    repository.grantPullPush(project);\n    repository.grant(project, 'ecr:DescribeImages');\n\n    handler.addToRolePolicy(\n      new PolicyStatement({\n        actions: ['codebuild:StartBuild'],\n        resources: [project.projectArn],\n      }),\n    );\n\n    this.grantPrincipal = project.grantPrincipal;\n\n    let assetExclude = props.exclude;\n    // automatically read .dockerignore for convenience\n    if (assetExclude === undefined && existsSync(join(props.directory, '.dockerignore'))) {\n      assetExclude = readFileSync(join(props.directory, '.dockerignore')).toString().split('\\n');\n    }\n    const asset = new Asset(this, 'Source', {\n      ...props,\n      exclude: assetExclude,\n      path: props.directory,\n    });\n    asset.grantRead(project);\n\n    const buildCommandOptions = { ...props, platform: props.platform?.platform } as any;\n    buildCommandOptions.outputs ??= [];\n    // we don't use push=true here because CodeBuild Docker server does not seem to work properly with the configuration.\n    buildCommandOptions.outputs.push('type=image', `name=${imageArtifactName}`);\n    if (props.zstdCompression) {\n      // to enable zstd compression, buildx directly pushes the artifact image to a registry\n      // https://aws.amazon.com/blogs/containers/reducing-aws-fargate-startup-times-with-zstd-compressed-container-images/\n      buildCommandOptions.outputs.push('oci-mediatypes=true', 'compression=zstd', 'force-compression=true', 'compression-level=3');\n    }\n    const buildCommand = this.getDockerBuildCommand(buildCommandOptions);\n\n    const properties: ContainerImageBuildResourceProps = {\n      type: 'ContainerImageBuild',\n      buildCommand: buildCommand,\n      repositoryUri,\n      imageTag: props.tag,\n      tagPrefix: props.tagPrefix,\n      codeBuildProjectName: project.projectName,\n      sourceS3Url: asset.s3ObjectUrl,\n    };\n\n    const custom = new CustomResource(this, 'Resource', {\n      serviceToken: handler.functionArn,\n      resourceType: 'Custom::CDKContainerImageBuild',\n      properties,\n    });\n    custom.node.addDependency(project);\n\n    this.repository = repository;\n    this.imageTag = custom.getAttString('ImageTag');\n  }\n\n  /**\n   * Get the instance of {@link DockerImageCode} for a Lambda function image.\n   */\n  public toLambdaDockerImageCode() {\n    if (this.props.zstdCompression) {\n      throw new Error('You cannot enable zstdCompression for a Lambda image.');\n    }\n    return DockerImageCode.fromEcr(this.repository, { tagOrDigest: this.imageTag });\n  }\n\n  /**\n   * Get the instance of {@link ContainerImage} for an ECS task definition.\n   */\n  public toEcsDockerImageCode() {\n    return ContainerImage.fromEcrRepository(this.repository, this.imageTag);\n  }\n\n  private getDockerBuildCommand(options: any) {\n    // the members of props differs with CDK version.\n    // By regarding props as any, we can use props that are not available in older cdk versions.\n\n    // logic is copied from packages/cdk-assets/lib/private/docker.ts\n    const cacheOptionToFlag = (option: any): string => {\n      let flag = `type=${option.type}`;\n      if (option.params) {\n        flag +=\n          ',' +\n          Object.entries(option.params)\n            .map(([k, v]) => `${k}=${v}`)\n            .join(',');\n      }\n      return flag;\n    };\n    const flatten = (x: string[][]) => {\n      return Array.prototype.concat([], ...x);\n    };\n\n    const dockerBuildCommand = [\n      'docker buildx build',\n      ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])),\n      ...flatten(Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`])),\n      ...(options.buildSsh ? ['--ssh', options.buildSsh] : []),\n      ...(options.target ? ['--target', options.target] : []),\n      ...(options.file ? ['--file', options.file] : []),\n      ...(options.networkMode ? ['--network', options.networkMode] : []),\n      ...(options.platform ? ['--platform', options.platform] : []),\n      ...(options.outputs ? ['--output', options.outputs.join(',')] : []),\n      ...(options.cacheFrom ? [...options.cacheFrom.map((cacheFrom: any) => ['--cache-from', cacheOptionToFlag(cacheFrom)]).flat()] : []),\n      ...(options.cacheTo ? ['--cache-to', cacheOptionToFlag(options.cacheTo)] : []),\n      ...(options.cacheDisabled ? ['--no-cache'] : []),\n      '--provenance=false',\n      '.',\n    ];\n    return dockerBuildCommand.join(' ');\n  }\n}\n"]}