deploy-time-build
Version:
Build during CDK deployment.
221 lines (216 loc) • 29.7 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.NodejsBuild = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
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_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");
/**
* Build Node.js app and optionally publish the artifact to an S3 bucket.
*/
class NodejsBuild extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
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('nodejs22.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: '25648b21-2c40-4f09-aa65-b6bbb0c44659', // generated for this construct
lambdaPurpose: 'NodejsBuildCustomResourceHandler',
timeout: aws_cdk_lib_1.Duration.minutes(5),
});
const nodejsVersion = props.nodejsVersion ?? 18;
let buildImage = 'aws/codebuild/standard:7.0';
// See: https://docs.aws.amazon.com/codebuild/latest/userguide/available-runtimes.html#linux-runtimes
switch (nodejsVersion) {
case 12:
case 14:
buildImage = 'aws/codebuild/standard:5.0';
break;
case 16:
buildImage = 'aws/codebuild/standard:6.0';
break;
case 18:
case 20:
case 22:
buildImage = 'aws/codebuild/standard:7.0';
break;
default:
aws_cdk_lib_1.Annotations.of(this).addWarning(`Possibly unsupported Node.js version: ${nodejsVersion}. Currently 12, 14, 16, 18, 20, and 22 are supported.`);
}
const outputEnvFile = props.outputEnvFile ?? false;
const envFileKeyOutputKey = 'envFileKey';
const project = new aws_codebuild_1.Project(this, 'Project', {
environment: { buildImage: aws_codebuild_1.LinuxBuildImage.fromCodeBuildImageId(buildImage) },
buildSpec: aws_codebuild_1.BuildSpec.fromObject({
version: '0.2',
env: {
shell: 'bash',
},
phases: {
install: {
'runtime-versions': {
nodejs: nodejsVersion,
},
},
build: {
commands: [
'current_dir=$(pwd)',
// Iterate a json array using jq
// https://www.starkandwayne.com/blog/bash-for-loop-over-json-array-using-jq/index.html
`
echo "$input"
for obj in $(echo "$input" | jq -r '.[] | @base64'); do
decoded=$(echo "$obj" | base64 --decode)
assetUrl=$(echo "$decoded" | jq -r '.assetUrl')
extractPath=$(echo "$decoded" | jq -r '.extractPath')
commands=$(echo "$decoded" | jq -r '.commands')
# Download the zip file
aws s3 cp "$assetUrl" temp.zip
# Extract the zip file to the extractPath directory
mkdir -p "$extractPath"
unzip temp.zip -d "$extractPath"
# Remove the zip file
rm temp.zip
# Run the specified commands in the extractPath directory
cd "$extractPath"
ls -la
eval "$commands"
cd "$current_dir"
ls -la
done
`,
'ls -la',
'cd "$workingDirectory"',
'eval "$buildCommands"',
'ls -la',
'cd "$current_dir"',
'cd "$outputSourceDirectory"',
'aws s3 sync . "s3://$destinationBucketName/$destinationKeyPrefix" --delete',
// Invalidate CloudFront cache if distribution is specified
`
if [[ -n "$distributionId" ]]
then
INVALIDATION_OUTPUT=$(aws cloudfront create-invalidation --distribution-id "$distributionId" --paths "$distributionPath" --output json)
INVALIDATION_ID=$(echo "$INVALIDATION_OUTPUT" | jq -r '.Invalidation.Id')
aws cloudfront wait invalidation-completed --distribution-id "$distributionId" --id "$INVALIDATION_ID"
fi
`,
// Upload .env if required
`
if [[ $outputEnvFile == "true" ]]
then
# Split the comma-separated string into an array
for var_name in \${envNames//,/ }
do
echo "Element: $var_name"
var_value="\${!var_name}"
echo "$var_name=$var_value" >> tmp.env
done
aws s3 cp tmp.env "s3://$assetBucketName/$envFileKey"
fi
`,
],
},
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="NodejsBuild 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": "$logicalResourceId",
"Status": "$STATUS",
"Reason": "$REASON",
"Data": {
"${envFileKeyOutputKey}": "$envFileKey"
}
}
EOF
curl -i -X PUT -H 'Content-Type:' -d "@payload.json" "$responseURL"
`,
],
},
},
}),
});
handler.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ['codebuild:StartBuild'],
resources: [project.projectArn],
}));
this.grantPrincipal = project.grantPrincipal;
props.destinationBucket.grantReadWrite(project);
if (props.distribution) {
project.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ['cloudfront:GetInvalidation', 'cloudfront:CreateInvalidation'],
resources: [props.distribution.distributionArn],
}));
}
const commonExclude = ['.DS_Store', '.git', 'node_modules'];
const assets = props.assets.map((assetProps) => {
const asset = new aws_s3_assets_1.Asset(this, `Source-${assetProps.path.replace('/', '')}`, {
...assetProps,
...(props.excludeCommonFiles ?? true ? { exclude: [...commonExclude, ...(assetProps.exclude ?? [])] } : {}),
});
asset.grantRead(project);
return asset;
});
const bucket = assets[0].bucket;
if (outputEnvFile) {
// use the asset bucket that are created by CDK bootstrap to store .env file
bucket.grantWrite(project);
}
const sources = props.assets.map((s, i) => ({
sourceBucketName: assets[i].s3BucketName,
sourceObjectKey: assets[i].s3ObjectKey,
extractPath: s.extractPath ?? (0, path_1.basename)(s.path),
commands: s.commands,
}));
const workingDirectory = props.workingDirectory ?? sources[0].extractPath;
const properties = {
type: 'NodejsBuild',
sources,
destinationBucketName: props.destinationBucket.bucketName,
destinationKeyPrefix: props.destinationKeyPrefix ?? '/',
distributionId: props.distribution?.distributionId,
assetBucketName: bucket.bucketName,
workingDirectory,
// join paths for CodeBuild (Linux) platform
outputSourceDirectory: path_1.posix.join(workingDirectory, props.outputSourceDirectory),
environment: props.buildEnvironment,
buildCommands: props.buildCommands ?? ['npm run build'],
codeBuildProjectName: project.projectName,
outputEnvFile,
};
const custom = new aws_cdk_lib_1.CustomResource(this, 'Resource', {
serviceToken: handler.functionArn,
resourceType: 'Custom::CDKNodejsBuild',
properties,
});
if (project.role) {
custom.node.addDependency(project.role);
}
if (props.outputEnvFile) {
new aws_cdk_lib_1.CfnOutput(this, 'DownloadEnvFile', { value: `aws s3 cp ${bucket.s3UrlForObject(custom.getAttString(envFileKeyOutputKey))} .env.local` });
}
}
}
exports.NodejsBuild = NodejsBuild;
_a = JSII_RTTI_SYMBOL_1;
NodejsBuild[_a] = { fqn: "deploy-time-build.NodejsBuild", version: "0.4.5" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nodejs-build.js","sourceRoot":"","sources":["../src/nodejs-build.ts"],"names":[],"mappings":"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","sourcesContent":["import { basename, join, posix } from 'path';\nimport { Annotations, CfnOutput, CustomResource, Duration } from 'aws-cdk-lib';\nimport { IDistribution } from 'aws-cdk-lib/aws-cloudfront';\nimport { BuildSpec, LinuxBuildImage, Project } from 'aws-cdk-lib/aws-codebuild';\nimport { IGrantable, IPrincipal, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, Runtime, RuntimeFamily, SingletonFunction } from 'aws-cdk-lib/aws-lambda';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Asset, AssetProps } from 'aws-cdk-lib/aws-s3-assets';\nimport { Construct } from 'constructs';\nimport { NodejsBuildResourceProps } from './types';\n\nexport interface AssetConfig extends AssetProps {\n  /**\n   * Shell commands executed right after the asset zip is extracted to the build environment.\n   * @default No command is executed.\n   */\n  readonly commands?: string[];\n\n  /**\n   * Relative path from a build directory to the directory where the asset is extracted.\n   * @default basename of the asset path.\n   */\n  readonly extractPath?: string;\n}\n\nexport interface NodejsBuildProps {\n  /**\n   * The AssetProps from which s3-assets are created and copied to the build environment.\n   */\n  readonly assets: AssetConfig[];\n  /**\n   * Environment variables injected to the build environment.\n   * You can use CDK deploy-time values as well as literals.\n   * @default {}\n   */\n  readonly buildEnvironment?: { [key: string]: string };\n  /**\n   * S3 Bucket to which your build artifacts are finally deployed.\n   */\n  readonly destinationBucket: IBucket;\n  /**\n   * Key prefix to deploy your build artifact.\n   * @default '/'\n   */\n  readonly destinationKeyPrefix?: string;\n  /**\n   * The distribution you are using to publish you build artifact.\n   * If any specified, the caches are invalidated on new artifact deployments.\n   * @default No distribution\n   */\n  readonly distribution?: IDistribution;\n  /**\n   * Shell commands to build your project. They are executed on the working directory you specified.\n   * @default ['npm run build']\n   */\n  readonly buildCommands?: string[];\n  /**\n   * Relative path from the build directory to the directory where build commands run.\n   * @default assetProps[0].extractPath\n   */\n  readonly workingDirectory?: string;\n  /**\n   * Relative path from the working directory to the directory where the build artifacts are output.\n   */\n  readonly outputSourceDirectory: string;\n  /**\n   * The version of Node.js to use in a build environment. Available versions: 12, 14, 16, 18, 20, and 22.\n   * @default 18\n   */\n  readonly nodejsVersion?: number;\n  /**\n   * If true, a .env file is uploaded to an S3 bucket with values of `buildEnvironment` property.\n   * You can copy it to your local machine by running the command in the stack output.\n   * @default false\n   */\n  readonly outputEnvFile?: boolean;\n  /**\n   * If true, common unnecessary files/directories such as .DS_Store, .git, node_modules, etc are excluded\n   * from the assets by default.\n   * @default true\n   */\n  readonly excludeCommonFiles?: boolean;\n}\n\n/**\n * Build Node.js app and optionally publish the artifact to an S3 bucket.\n */\nexport class NodejsBuild extends Construct implements IGrantable {\n  public readonly grantPrincipal: IPrincipal;\n\n  constructor(scope: Construct, id: string, props: NodejsBuildProps) {\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('nodejs22.x', RuntimeFamily.NODEJS),\n      code: Code.fromAsset(join(__dirname, '..', 'lambda', 'trigger-codebuild', 'dist')),\n      handler: 'index.handler',\n      uuid: '25648b21-2c40-4f09-aa65-b6bbb0c44659', // generated for this construct\n      lambdaPurpose: 'NodejsBuildCustomResourceHandler',\n      timeout: Duration.minutes(5),\n    });\n\n    const nodejsVersion = props.nodejsVersion ?? 18;\n    let buildImage = 'aws/codebuild/standard:7.0';\n    // See: https://docs.aws.amazon.com/codebuild/latest/userguide/available-runtimes.html#linux-runtimes\n    switch (nodejsVersion) {\n      case 12:\n      case 14:\n        buildImage = 'aws/codebuild/standard:5.0';\n        break;\n      case 16:\n        buildImage = 'aws/codebuild/standard:6.0';\n        break;\n      case 18:\n      case 20:\n      case 22:\n        buildImage = 'aws/codebuild/standard:7.0';\n        break;\n      default:\n        Annotations.of(this).addWarning(`Possibly unsupported Node.js version: ${nodejsVersion}. Currently 12, 14, 16, 18, 20, and 22 are supported.`);\n    }\n\n    const outputEnvFile = props.outputEnvFile ?? false;\n    const envFileKeyOutputKey = 'envFileKey';\n\n    const project = new Project(this, 'Project', {\n      environment: { buildImage: LinuxBuildImage.fromCodeBuildImageId(buildImage) },\n      buildSpec: BuildSpec.fromObject({\n        version: '0.2',\n        env: {\n          shell: 'bash',\n        },\n        phases: {\n          install: {\n            'runtime-versions': {\n              nodejs: nodejsVersion,\n            },\n          },\n          build: {\n            commands: [\n              'current_dir=$(pwd)',\n              // Iterate a json array using jq\n              // https://www.starkandwayne.com/blog/bash-for-loop-over-json-array-using-jq/index.html\n              `\necho \"$input\"\nfor obj in $(echo \"$input\" | jq -r '.[] | @base64'); do\n  decoded=$(echo \"$obj\" | base64 --decode)\n  assetUrl=$(echo \"$decoded\" | jq -r '.assetUrl')\n  extractPath=$(echo \"$decoded\" | jq -r '.extractPath')\n  commands=$(echo \"$decoded\" | jq -r '.commands')\n\n  # Download the zip file\n  aws s3 cp \"$assetUrl\" temp.zip\n\n  # Extract the zip file to the extractPath directory\n  mkdir -p \"$extractPath\"\n  unzip temp.zip -d \"$extractPath\"\n\n  # Remove the zip file\n  rm temp.zip\n\n  # Run the specified commands in the extractPath directory\n  cd \"$extractPath\"\n  ls -la\n  eval \"$commands\"\n  cd \"$current_dir\"\n  ls -la\ndone\n              `,\n              'ls -la',\n              'cd \"$workingDirectory\"',\n              'eval \"$buildCommands\"',\n              'ls -la',\n              'cd \"$current_dir\"',\n              'cd \"$outputSourceDirectory\"',\n              'aws s3 sync . \"s3://$destinationBucketName/$destinationKeyPrefix\" --delete',\n              // Invalidate CloudFront cache if distribution is specified\n              `\nif [[ -n \"$distributionId\" ]]\nthen\nINVALIDATION_OUTPUT=$(aws cloudfront create-invalidation --distribution-id \"$distributionId\" --paths \"$distributionPath\" --output json)\nINVALIDATION_ID=$(echo \"$INVALIDATION_OUTPUT\" | jq -r '.Invalidation.Id')\naws cloudfront wait invalidation-completed --distribution-id \"$distributionId\" --id \"$INVALIDATION_ID\"\nfi\n`,\n              // Upload .env if required\n              `\nif [[ $outputEnvFile == \"true\" ]]\nthen\n  # Split the comma-separated string into an array\n  for var_name in \\${envNames//,/ }\n  do\n      echo \"Element: $var_name\"\n      var_value=\"\\${!var_name}\"\n      echo \"$var_name=$var_value\" >> tmp.env\n  done\n\n  aws s3 cp tmp.env \"s3://$assetBucketName/$envFileKey\"\nfi\n              `,\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=\"NodejsBuild 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\": \"$logicalResourceId\",\n  \"Status\": \"$STATUS\",\n  \"Reason\": \"$REASON\",\n  \"Data\": {\n    \"${envFileKeyOutputKey}\": \"$envFileKey\"\n  }\n}\nEOF\ncurl -i -X PUT -H 'Content-Type:' -d \"@payload.json\" \"$responseURL\"\n              `,\n            ],\n          },\n        },\n      }),\n    });\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    props.destinationBucket.grantReadWrite(project);\n    if (props.distribution) {\n      project.addToRolePolicy(\n        new PolicyStatement({\n          actions: ['cloudfront:GetInvalidation', 'cloudfront:CreateInvalidation'],\n          resources: [props.distribution.distributionArn],\n        })\n      );\n    }\n\n    const commonExclude = ['.DS_Store', '.git', 'node_modules'];\n    const assets = props.assets.map((assetProps) => {\n      const asset = new Asset(this, `Source-${assetProps.path.replace('/', '')}`, {\n        ...assetProps,\n        ...(props.excludeCommonFiles ?? true ? { exclude: [...commonExclude, ...(assetProps.exclude ?? [])] } : {}),\n      });\n      asset.grantRead(project);\n      return asset;\n    });\n\n    const bucket = assets[0].bucket;\n    if (outputEnvFile) {\n      // use the asset bucket that are created by CDK bootstrap to store .env file\n      bucket.grantWrite(project);\n    }\n\n    const sources: NodejsBuildResourceProps['sources'] = props.assets.map((s, i) => ({\n      sourceBucketName: assets[i].s3BucketName,\n      sourceObjectKey: assets[i].s3ObjectKey,\n      extractPath: s.extractPath ?? basename(s.path),\n      commands: s.commands,\n    }));\n\n    const workingDirectory = props.workingDirectory ?? sources[0].extractPath;\n    const properties: NodejsBuildResourceProps = {\n      type: 'NodejsBuild',\n      sources,\n      destinationBucketName: props.destinationBucket.bucketName,\n      destinationKeyPrefix: props.destinationKeyPrefix ?? '/',\n      distributionId: props.distribution?.distributionId,\n      assetBucketName: bucket.bucketName,\n      workingDirectory,\n      // join paths for CodeBuild (Linux) platform\n      outputSourceDirectory: posix.join(workingDirectory, props.outputSourceDirectory),\n      environment: props.buildEnvironment,\n      buildCommands: props.buildCommands ?? ['npm run build'],\n      codeBuildProjectName: project.projectName,\n      outputEnvFile,\n    };\n\n    const custom = new CustomResource(this, 'Resource', {\n      serviceToken: handler.functionArn,\n      resourceType: 'Custom::CDKNodejsBuild',\n      properties,\n    });\n    if (project.role) {\n      custom.node.addDependency(project.role);\n    }\n\n    if (props.outputEnvFile) {\n      new CfnOutput(this, 'DownloadEnvFile', { value: `aws s3 cp ${bucket.s3UrlForObject(custom.getAttString(envFileKeyOutputKey))} .env.local` });\n    }\n  }\n}\n"]}