deploy-time-build
Version:
Build during CDK deployment.
209 lines • 32.9 kB
JavaScript
"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"]}