UNPKG

deploy-time-build

Version:
221 lines (216 loc) 29.7 kB
"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"]}