UNPKG

cdk-nextjs

Version:

Deploy Next.js apps on AWS with CDK

243 lines 47.3 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.NextjsBuild = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const node_child_process_1 = require("node:child_process"); const node_fs_1 = require("node:fs"); const node_path_1 = require("node:path"); const posix_1 = require("node:path/posix"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_ecr_assets_1 = require("aws-cdk-lib/aws-ecr-assets"); const aws_lambda_1 = require("aws-cdk-lib/aws-lambda"); const constructs_1 = require("constructs"); const constants_1 = require("../constants"); /** * Builds Next.js assets. * @link https://nextjs.org/docs/pages/api-reference/next-config-js/output */ class NextjsBuild extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); /** * Repository name for the builder image. */ this.builderImageRepo = "cdk-nextjs/builder"; this.containerRuntime = process.env.CDK_DOCKER || "docker"; this.builderImageAlias = `${this.builderImageRepo}:${this.node.addr.slice(30)}`; this.relativePathToPackage = props.relativePathToPackage || "."; this.props = props; this.relativePathToEntrypoint = this.getRelativeEntrypointPath(); if (!props.builderImageProps?.skipBuild) { this.createBuilderImage(); } this.buildImageDigest = this.getBuilderImageDigest(); this.buildId = this.getBuildId(); this.publicDirEntries = this.getPublicDirEntries(); if (props.nextjsType === constants_1.NextjsType.GLOBAL_CONTAINERS || props.nextjsType === constants_1.NextjsType.REGIONAL_CONTAINERS) { this.imageForNextjsContainers = this.createImageForNextjsContainers(); } else { this.imageForNextjsFunctions = this.createImageForNextjsFunctions(); } this.imageForNextjsAssetsDeployment = this.createImageForNextjsAssetsDeployment(); } getRelativeEntrypointPath() { // joinPosix b/c this will be used in linux container return (0, posix_1.join)(this.props.relativePathToPackage || "", "server.js"); } /** * A builder or base image needs to be created so that the same image can be * built `FROM` for `NextjsFunctions` or `NextjsContainers` and `NextjsAssetsDeployment`. * This image doesn't need to be uploaded to ECR so we're "manually" creating * it with `execSync` and other images will be built `FROM` it. */ createBuilderImage() { const buildCommand = this.props.buildCommand || "npm run build"; const { buildArgs = { BUILD_COMMAND: buildCommand, RELATIVE_PATH_TO_PACKAGE: this.relativePathToPackage, ...this.props.builderImageProps?.buildArgs, }, envVarNames = [], exclude = [ "**/node_modules", ".git", "**/cdk.out", "**/.next", ".gitignore", "*.md", ], file = "builder.Dockerfile", platform, } = this.props.builderImageProps || {}; // to be added to user provided build context before builder image is built const filePathsToCopy = [ (0, node_path_1.join)(__dirname, "cdk-nextjs-cache-handler.cjs"), ]; // to be removed from user provided build context after builder image is built const filePathsToRemove = [ (0, node_path_1.join)(this.props.buildContext, "cdk-nextjs-cache-handler.cjs"), ]; // if custom file (Dockerfile) is not specified then use library's default builder.Dockerfile + .dockerignore if (!this.props.builderImageProps?.file) { filePathsToCopy.push((0, node_path_1.join)(__dirname, file)); filePathsToRemove.push((0, node_path_1.join)(this.props.buildContext, file)); const excludeFileStr = exclude?.join("\n"); const dockerignoreFilePath = (0, node_path_1.join)(this.props.buildContext, ".dockerignore"); (0, node_fs_1.writeFileSync)(dockerignoreFilePath, excludeFileStr); filePathsToRemove.push(dockerignoreFilePath); } for (const filePathToCopy of filePathsToCopy) { (0, node_fs_1.cpSync)(filePathToCopy, (0, node_path_1.join)(this.props.buildContext, (0, node_path_1.basename)(filePathToCopy))); } const buildArgsStr = this.createBuildArgStr(buildArgs); this.injectBuilderDockerfileEnvVars((0, node_path_1.join)(this.props.buildContext, file), envVarNames); const command = this.props.builderImageProps?.command || `${this.containerRuntime} build ${platform ? `--platform ${platform.platform}` : ""} --file ${file} --tag ${this.builderImageAlias} ${buildArgsStr} .`; let error; try { console.log(`Building image with command: ${command} in directory: ${this.props.buildContext}`); (0, node_child_process_1.execSync)(command, { stdio: "inherit", cwd: this.props.buildContext, env: process.env, }); } catch (err) { error = err; } finally { for (const filePathToRemove of filePathsToRemove) { (0, node_fs_1.rmSync)(filePathToRemove); } } if (error) throw error; } injectBuilderDockerfileEnvVars(builderDockerfilePath, envVarNames) { const envVars = {}; for (const envVarName of envVarNames) { if (process.env[envVarName]) { envVars[envVarName] = process.env[envVarName]; } } const content = Object.entries(envVars) .map(([name, value]) => `${name}="${value}"`) .join(" "); const oldFile = (0, node_fs_1.readFileSync)(builderDockerfilePath).toString(); const newFile = oldFile.replace(constants_1.INJECT_CDK_NEXTJS_BUILD_ENV_VARS, content); (0, node_fs_1.writeFileSync)(builderDockerfilePath, newFile); } createBuildArgStr(buildArgs) { return Object.entries(buildArgs).reduce((acc, [key, value]) => { return `${acc} --build-arg ${key}="${value}"`; }, ""); } getBuilderImageDigest() { const digest = (0, node_child_process_1.execSync)(`${this.containerRuntime} images --no-trunc --quiet ${this.builderImageAlias}`, { encoding: "utf-8" }); return digest.slice(0, -1); // remove trailing \n } getPublicDirEntries() { const publicDirPath = (0, posix_1.join)("/app", this.props.relativePathToPackage || "", "public"); if ((0, node_fs_1.existsSync)(publicDirPath)) { const publicDirEntriesString = (0, node_child_process_1.execSync)(`${this.containerRuntime} run ${this.builderImageAlias} node -e "console.log(JSON.stringify(fs.readdirSync('${publicDirPath}', { withFileTypes: true }).map((e) => ({ name: e.name, isDirectory: e.isDirectory()}))))"`, { encoding: "utf-8" }); return JSON.parse(publicDirEntriesString); } else { return []; } } getBuildId() { const buildIdPath = (0, posix_1.join)("/app", this.props.relativePathToPackage || "", ".next", "BUILD_ID"); const buildId = (0, node_child_process_1.execSync)(`${this.containerRuntime} run ${this.builderImageAlias} /bin/sh -c "cat ${buildIdPath}"`, { encoding: "utf-8" }); return buildId; } createImageForNextjsContainers() { const dockerfileNamePrefix = this.props.nextjsType === constants_1.NextjsType.GLOBAL_CONTAINERS ? "global" : "regional"; const dockerfileName = `${dockerfileNamePrefix}-containers.Dockerfile`; // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile // which is in lib/nextjs-build folder. const buildContext = this.props.overrides?.containersImageBuildContext ?? (0, node_path_1.join)(__dirname, "..", "..", "lib", "nextjs-build"); const dockerImageAsset = new aws_ecr_assets_1.DockerImageAsset(this, "Image", { directory: buildContext, extraHash: this.buildImageDigest, // rebuild when builder hash changes file: dockerfileName, ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER, ...this.props.overrides?.nextjsContainersDockerImageAssetProps, buildArgs: { [constants_1.BUILD_ID_ARG_NAME]: this.buildId, [constants_1.BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias, [constants_1.CACHE_PATH_ARG_NAME]: constants_1.CACHE_PATH, [constants_1.DATA_CACHE_PATH_ARG_NAME]: constants_1.DATA_CACHE_PATH, [constants_1.PUBLIC_PATH_ARG_NAME]: constants_1.PUBLIC_PATH, [constants_1.IMAGE_CACHE_PATH_ARG_NAME]: constants_1.IMAGE_CACHE_PATH, [constants_1.MOUNT_PATH_ARG_NAME]: constants_1.MOUNT_PATH, [constants_1.RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage, ...this.props.overrides?.nextjsContainersDockerImageAssetProps ?.buildArgs, }, }); return dockerImageAsset; } createImageForNextjsFunctions() { const dockerfileName = "global-functions.Dockerfile"; // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile // which is in lib/nextjs-build folder. const buildContext = this.props.overrides?.functionsImageBuildContext ?? (0, node_path_1.join)(__dirname, "..", "..", "lib", "nextjs-build"); const dockerImageCode = aws_lambda_1.DockerImageCode.fromImageAsset(buildContext, { cmd: ["node", this.relativePathToEntrypoint], extraHash: this.buildImageDigest, // rebuild when builder hash changes file: dockerfileName, ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER, ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps, buildArgs: { [constants_1.BUILD_ID_ARG_NAME]: this.buildId, [constants_1.BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias, [constants_1.CACHE_PATH_ARG_NAME]: constants_1.CACHE_PATH, [constants_1.DATA_CACHE_PATH_ARG_NAME]: constants_1.DATA_CACHE_PATH, [constants_1.PUBLIC_PATH_ARG_NAME]: constants_1.PUBLIC_PATH, [constants_1.IMAGE_CACHE_PATH_ARG_NAME]: constants_1.IMAGE_CACHE_PATH, [constants_1.MOUNT_PATH_ARG_NAME]: constants_1.MOUNT_PATH, [constants_1.RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage, ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps?.buildArgs, }, }); // TODO: how to clean up temp dir? // rmSync(tempDir, { recursive: true }); return dockerImageCode; } createImageForNextjsAssetsDeployment() { const dockerfileName = "assets-deployment.Dockerfile"; /** * Path to bundled custom resource code */ const buildContext = this.props.overrides?.assetsDeploymentImageBuildContext ?? (0, node_path_1.join)(__dirname, "..", "..", "assets", "lambdas", "assets-deployment", "assets-deployment.lambda"); // cdk-nextjs/builder-{hash} already contains Next.js built code which // we'll copy into final image. But we also need lambda code to run // asset deployment tasks const dockerImageCode = aws_lambda_1.DockerImageCode.fromImageAsset(buildContext, { extraHash: this.buildImageDigest, // rebuild when builder hash changes file: dockerfileName, ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER, ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps, buildArgs: { [constants_1.BUILD_ID_ARG_NAME]: this.buildId, [constants_1.BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias, [constants_1.PUBLIC_PATH_ARG_NAME]: constants_1.PUBLIC_PATH, [constants_1.RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage, ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps ?.buildArgs, }, }); return dockerImageCode; } } exports.NextjsBuild = NextjsBuild; _a = JSII_RTTI_SYMBOL_1; NextjsBuild[_a] = { fqn: "cdk-nextjs.NextjsBuild", version: "0.4.10" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-build.js","sourceRoot":"","sources":["../../src/nextjs-build/nextjs-build.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync } from \"node:child_process\";\nimport {\n  cpSync,\n  existsSync,\n  readFileSync,\n  rmSync,\n  writeFileSync,\n} from \"node:fs\";\nimport { basename, join } from \"node:path\";\nimport { join as joinPosix } from \"node:path/posix\";\nimport { IgnoreMode } from \"aws-cdk-lib\";\nimport { DockerImageAsset, Platform } from \"aws-cdk-lib/aws-ecr-assets\";\nimport { AssetImageCodeProps, DockerImageCode } from \"aws-cdk-lib/aws-lambda\";\nimport { Construct } from \"constructs\";\nimport {\n  BUILDER_IMAGE_ALIAS_ARG_NAME,\n  MOUNT_PATH,\n  IMAGE_CACHE_PATH,\n  IMAGE_CACHE_PATH_ARG_NAME,\n  MOUNT_PATH_ARG_NAME,\n  NextjsType,\n  PUBLIC_PATH,\n  PUBLIC_PATH_ARG_NAME,\n  RELATIVE_PATH_TO_PACKAGE_ARG_NAME,\n  BUILD_ID_ARG_NAME,\n  DATA_CACHE_PATH_ARG_NAME,\n  DATA_CACHE_PATH,\n  CACHE_PATH,\n  CACHE_PATH_ARG_NAME,\n  INJECT_CDK_NEXTJS_BUILD_ENV_VARS,\n} from \"../constants\";\nimport { OptionalDockerImageAssetProps } from \"../generated-structs/OptionalDockerImageAssetProps\";\nimport { NextjsBaseProps } from \"../root-constructs/nextjs-base-construct\";\n\nexport interface BuilderImageProps {\n  /**\n   * Build Args to be passed to `docker build` command.\n   * @see https://docs.docker.com/build/building/variables/#build-arguments\n   */\n  readonly buildArgs?: Record<string, string>;\n  /**\n   * `docker build ...` command to run in {@link NextBaseProps.buildContext}.\n   * Default interpolates other props. If you override, other props will have\n   * no effect on command.\n   */\n  readonly command?: string;\n  /**\n   * Environment variables names to pass from host to container during build process.\n   *\n   * These variable names will be set before the build command in builder.Dockerfile\n   * like: `API_KEY=\"MY_API_KEY\" npm run build`\n   *\n   * @example [\"MY_API_KEY\"]\n   */\n  readonly envVarNames?: string[];\n  /**\n   * Lines in .dockerignore file which will be created in your {@link NextBaseProps.buildContext}\n   * @default [\"node_modules\", \".git\", \".gitignore\", \".md\"]\n   */\n  readonly exclude?: string[];\n  /**\n   * Name of Dockerfile in builder build context. If specified, you are responsible\n   * for ensuring it exists in build context before construct is instantiated.\n   * @default \"builder.Dockerfile\"\n   */\n  readonly file?: string;\n  readonly platform?: Platform;\n  /**\n   * Skip building the builder image.\n   * @default false\n   */\n  readonly skipBuild?: boolean;\n}\n\nexport interface NextjsBuildOverrides {\n  readonly nextjsContainersDockerImageAssetProps?: OptionalDockerImageAssetProps;\n  readonly nextjsFunctionsAssetImageCodeProps?: AssetImageCodeProps;\n  readonly nextjsAssetDeploymentAssetImageCodeProps?: AssetImageCodeProps;\n  /**\n   * Default folder for build context is the \"lib/nextjs-build\" folder in the\n   * installed cdk-nextjs library which has the \"global-functions.Dockerfile\".\n   * Note, if you specify this then you're responsible for ensuring the dockerfile\n   * is present in the build context directory and any referenced files are\n   * present as well. You can specify dockerfile name with adjacent\n   * `nextjsFunctionsAssetImageCodeProps.file` property.\n   * @default \"cdk-nextjs/lib/nextjs-build\"\n   */\n  readonly functionsImageBuildContext?: string;\n  /**\n   * Default folder for build context is the \"lib/nextjs-build\" folder in the\n   * installed cdk-nextjs library which has the \"assets-deployment.Dockerfile\".\n   * Note, if you specify this then you're responsible for ensuring the dockerfile\n   * is present in the build context directory and any referenced files are\n   * present as well. You can specify dockerfile name with adjacent\n   * `nextjsAssetDeploymentAssetImageCodeProps.file` property.\n   * @default \"cdk-nextjs/lib/nextjs-build\"\n   */\n  readonly assetsDeploymentImageBuildContext?: string;\n  /**\n   * Default folder for build context is the \"assets/lambdas/assets-deployment/assets-deployment.lambda\" folder in the\n   * installed cdk-nextjs library which has the \"{...}-containers.Dockerfile\".\n   * Note, if you specify this then you're responsible for ensuring the dockerfile\n   * is present in the build context directory and any referenced files are\n   * present as well. You can specify dockerfile name with adjacent\n   * `nextjsContainersDockerImageAssetProps.file` property.\n   * @default \"cdk-nextjs/lib/nextjs-build\"\n   */\n  readonly containersImageBuildContext?: string;\n}\n\nexport interface NextjsBuildProps {\n  /**\n   * @see {@link NextjsBaseProps[\"buildCommand\"]}\n   */\n  readonly buildCommand: NextjsBaseProps[\"buildCommand\"];\n  /**\n   * @see {@link NextjsBaseProps[\"buildContext\"]}\n   */\n  readonly buildContext: NextjsBaseProps[\"buildContext\"];\n  /**\n   *\n   */\n  readonly builderImageProps?: BuilderImageProps;\n  /**\n   * @see {@link NextjsBaseProps.relativePathToPackage}\n   */\n  readonly relativePathToPackage?: NextjsBaseProps[\"relativePathToPackage\"];\n  readonly nextjsType: NextjsType;\n  readonly overrides?: NextjsBuildOverrides;\n}\n\nexport interface PublicDirEntry {\n  readonly name: string;\n  readonly isDirectory: boolean;\n}\n\n/**\n * Builds Next.js assets.\n * @link https://nextjs.org/docs/pages/api-reference/next-config-js/output\n */\nexport class NextjsBuild extends Construct {\n  /**\n   * Image alias of builder image Next.js app which is built for other images to be\n   * built `FROM`. This image isn't built with CDK Assets construct b/c it\n   * doesn't need to be uploaded to ECR. We still need to include slice of\n   * `node.addr` in tag in case multiple cdk-nextjs constructs are used.\n   */\n  builderImageAlias: string;\n  /**\n   * Unique id for Next.js build. Used to partition EFS FileSystem.\n   */\n  buildId: string;\n  /**\n   * Hash of builder image which will change whenever the image changes. Useful\n   * for passing to properties of custom resources that depend upon the builder\n   * image to re-run when build image changes.\n   */\n  buildImageDigest: string;\n  /**\n   * Docker image built if using Fargate.\n   */\n  imageForNextjsContainers?: DockerImageAsset;\n  /**\n   * Docker image built if using Lambda.\n   */\n  imageForNextjsFunctions?: DockerImageCode;\n  /**\n   * Docker image built for `NextjsAssetsDeployment`\n   */\n  imageForNextjsAssetsDeployment: DockerImageCode;\n  /**\n   * Absolute path to public. Use by CloudFront/ALB to create behaviors/rules\n   * @example \"/Users/john/myapp/public\"\n   */\n  publicDirEntries: PublicDirEntry[];\n  /**\n   * The entrypoint JavaScript file used as an argument for Node.js to run the\n   * Next.js standalone server relative to the standalone directory.\n   * @example \"./server.js\"\n   * @example \"./packages/ui/server.js\" (monorepo)\n   */\n  relativePathToEntrypoint: string;\n\n  /**\n   * Repository name for the builder image.\n   */\n  private builderImageRepo = \"cdk-nextjs/builder\";\n\n  private containerRuntime = process.env.CDK_DOCKER || \"docker\";\n  private props: NextjsBuildProps;\n  private relativePathToPackage: string;\n\n  constructor(scope: Construct, id: string, props: NextjsBuildProps) {\n    super(scope, id);\n    this.builderImageAlias = `${this.builderImageRepo}:${this.node.addr.slice(30)}`;\n    this.relativePathToPackage = props.relativePathToPackage || \".\";\n    this.props = props;\n    this.relativePathToEntrypoint = this.getRelativeEntrypointPath();\n    if (!props.builderImageProps?.skipBuild) {\n      this.createBuilderImage();\n    }\n    this.buildImageDigest = this.getBuilderImageDigest();\n    this.buildId = this.getBuildId();\n    this.publicDirEntries = this.getPublicDirEntries();\n    if (\n      props.nextjsType === NextjsType.GLOBAL_CONTAINERS ||\n      props.nextjsType === NextjsType.REGIONAL_CONTAINERS\n    ) {\n      this.imageForNextjsContainers = this.createImageForNextjsContainers();\n    } else {\n      this.imageForNextjsFunctions = this.createImageForNextjsFunctions();\n    }\n    this.imageForNextjsAssetsDeployment =\n      this.createImageForNextjsAssetsDeployment();\n  }\n\n  private getRelativeEntrypointPath() {\n    // joinPosix b/c this will be used in linux container\n    return joinPosix(this.props.relativePathToPackage || \"\", \"server.js\");\n  }\n  /**\n   * A builder or base image needs to be created so that the same image can be\n   * built `FROM` for `NextjsFunctions` or `NextjsContainers` and `NextjsAssetsDeployment`.\n   * This image doesn't need to be uploaded to ECR so we're \"manually\" creating\n   * it with `execSync` and other images will be built `FROM` it.\n   */\n  private createBuilderImage() {\n    const buildCommand = this.props.buildCommand || \"npm run build\";\n    const {\n      buildArgs = {\n        BUILD_COMMAND: buildCommand,\n        RELATIVE_PATH_TO_PACKAGE: this.relativePathToPackage,\n        ...this.props.builderImageProps?.buildArgs,\n      },\n      envVarNames = [],\n      exclude = [\n        \"**/node_modules\",\n        \".git\",\n        \"**/cdk.out\",\n        \"**/.next\",\n        \".gitignore\",\n        \"*.md\",\n      ],\n      file = \"builder.Dockerfile\",\n      platform,\n    } = this.props.builderImageProps || {};\n\n    // to be added to user provided build context before builder image is built\n    const filePathsToCopy: string[] = [\n      join(__dirname, \"cdk-nextjs-cache-handler.cjs\"),\n    ];\n    // to be removed from user provided build context after builder image is built\n    const filePathsToRemove: string[] = [\n      join(this.props.buildContext, \"cdk-nextjs-cache-handler.cjs\"),\n    ];\n\n    // if custom file (Dockerfile) is not specified then use library's default builder.Dockerfile + .dockerignore\n    if (!this.props.builderImageProps?.file) {\n      filePathsToCopy.push(join(__dirname, file));\n      filePathsToRemove.push(join(this.props.buildContext, file));\n      const excludeFileStr = exclude?.join(\"\\n\");\n      const dockerignoreFilePath = join(\n        this.props.buildContext,\n        \".dockerignore\",\n      );\n      writeFileSync(dockerignoreFilePath, excludeFileStr);\n      filePathsToRemove.push(dockerignoreFilePath);\n    }\n\n    for (const filePathToCopy of filePathsToCopy) {\n      cpSync(\n        filePathToCopy,\n        join(this.props.buildContext, basename(filePathToCopy)),\n      );\n    }\n\n    const buildArgsStr = this.createBuildArgStr(buildArgs);\n    this.injectBuilderDockerfileEnvVars(\n      join(this.props.buildContext, file),\n      envVarNames,\n    );\n    const command =\n      this.props.builderImageProps?.command ||\n      `${this.containerRuntime} build ${platform ? `--platform ${platform.platform}` : \"\"} --file ${file} --tag ${this.builderImageAlias} ${buildArgsStr} .`;\n    let error: unknown;\n    try {\n      console.log(\n        `Building image with command: ${command} in directory: ${this.props.buildContext}`,\n      );\n      execSync(command, {\n        stdio: \"inherit\",\n        cwd: this.props.buildContext,\n        env: process.env,\n      });\n    } catch (err) {\n      error = err;\n    } finally {\n      for (const filePathToRemove of filePathsToRemove) {\n        rmSync(filePathToRemove);\n      }\n    }\n    if (error) throw error;\n  }\n  private injectBuilderDockerfileEnvVars(\n    builderDockerfilePath: string,\n    envVarNames: string[],\n  ) {\n    const envVars: Record<string, string> = {};\n    for (const envVarName of envVarNames) {\n      if (process.env[envVarName]) {\n        envVars[envVarName] = process.env[envVarName];\n      }\n    }\n    const content = Object.entries(envVars)\n      .map(([name, value]) => `${name}=\"${value}\"`)\n      .join(\" \");\n    const oldFile = readFileSync(builderDockerfilePath).toString();\n    const newFile = oldFile.replace(INJECT_CDK_NEXTJS_BUILD_ENV_VARS, content);\n    writeFileSync(builderDockerfilePath, newFile);\n  }\n  private createBuildArgStr(\n    buildArgs: Required<BuilderImageProps>[\"buildArgs\"],\n  ) {\n    return Object.entries(buildArgs).reduce((acc, [key, value]) => {\n      return `${acc} --build-arg ${key}=\"${value}\"`;\n    }, \"\");\n  }\n  private getBuilderImageDigest() {\n    const digest = execSync(\n      `${this.containerRuntime} images --no-trunc --quiet ${this.builderImageAlias}`,\n      { encoding: \"utf-8\" },\n    );\n    return digest.slice(0, -1); // remove trailing \\n\n  }\n  private getPublicDirEntries(): PublicDirEntry[] {\n    const publicDirPath = joinPosix(\n      \"/app\",\n      this.props.relativePathToPackage || \"\",\n      \"public\",\n    );\n    if (existsSync(publicDirPath)) {\n      const publicDirEntriesString = execSync(\n        `${this.containerRuntime} run ${this.builderImageAlias} node -e \"console.log(JSON.stringify(fs.readdirSync('${publicDirPath}', { withFileTypes: true }).map((e) => ({ name: e.name, isDirectory: e.isDirectory()}))))\"`,\n        { encoding: \"utf-8\" },\n      );\n      return JSON.parse(publicDirEntriesString) as PublicDirEntry[];\n    } else {\n      return [];\n    }\n  }\n  private getBuildId() {\n    const buildIdPath = joinPosix(\n      \"/app\",\n      this.props.relativePathToPackage || \"\",\n      \".next\",\n      \"BUILD_ID\",\n    );\n    const buildId = execSync(\n      `${this.containerRuntime} run ${this.builderImageAlias} /bin/sh -c \"cat ${buildIdPath}\"`,\n      { encoding: \"utf-8\" },\n    );\n    return buildId;\n  }\n  private createImageForNextjsContainers() {\n    const dockerfileNamePrefix =\n      this.props.nextjsType === NextjsType.GLOBAL_CONTAINERS\n        ? \"global\"\n        : \"regional\";\n    const dockerfileName = `${dockerfileNamePrefix}-containers.Dockerfile`;\n    // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll\n    // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile\n    // which is in lib/nextjs-build folder.\n    const buildContext =\n      this.props.overrides?.containersImageBuildContext ??\n      join(__dirname, \"..\", \"..\", \"lib\", \"nextjs-build\");\n    const dockerImageAsset = new DockerImageAsset(this, \"Image\", {\n      directory: buildContext,\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsContainersDockerImageAssetProps,\n      buildArgs: {\n        [BUILD_ID_ARG_NAME]: this.buildId,\n        [BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias,\n        [CACHE_PATH_ARG_NAME]: CACHE_PATH,\n        [DATA_CACHE_PATH_ARG_NAME]: DATA_CACHE_PATH,\n        [PUBLIC_PATH_ARG_NAME]: PUBLIC_PATH,\n        [IMAGE_CACHE_PATH_ARG_NAME]: IMAGE_CACHE_PATH,\n        [MOUNT_PATH_ARG_NAME]: MOUNT_PATH,\n        [RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage,\n        ...this.props.overrides?.nextjsContainersDockerImageAssetProps\n          ?.buildArgs,\n      },\n    });\n    return dockerImageAsset;\n  }\n\n  private createImageForNextjsFunctions() {\n    const dockerfileName = \"global-functions.Dockerfile\";\n    // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll\n    // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile\n    // which is in lib/nextjs-build folder.\n    const buildContext =\n      this.props.overrides?.functionsImageBuildContext ??\n      join(__dirname, \"..\", \"..\", \"lib\", \"nextjs-build\");\n    const dockerImageCode = DockerImageCode.fromImageAsset(buildContext, {\n      cmd: [\"node\", this.relativePathToEntrypoint],\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps,\n      buildArgs: {\n        [BUILD_ID_ARG_NAME]: this.buildId,\n        [BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias,\n        [CACHE_PATH_ARG_NAME]: CACHE_PATH,\n        [DATA_CACHE_PATH_ARG_NAME]: DATA_CACHE_PATH,\n        [PUBLIC_PATH_ARG_NAME]: PUBLIC_PATH,\n        [IMAGE_CACHE_PATH_ARG_NAME]: IMAGE_CACHE_PATH,\n        [MOUNT_PATH_ARG_NAME]: MOUNT_PATH,\n        [RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage,\n        ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps?.buildArgs,\n      },\n    });\n    // TODO: how to clean up temp dir?\n    // rmSync(tempDir, { recursive: true });\n    return dockerImageCode;\n  }\n\n  private createImageForNextjsAssetsDeployment() {\n    const dockerfileName = \"assets-deployment.Dockerfile\";\n    /**\n     * Path to bundled custom resource code\n     */\n    const buildContext =\n      this.props.overrides?.assetsDeploymentImageBuildContext ??\n      join(\n        __dirname,\n        \"..\",\n        \"..\",\n        \"assets\",\n        \"lambdas\",\n        \"assets-deployment\",\n        \"assets-deployment.lambda\",\n      );\n    // cdk-nextjs/builder-{hash} already contains Next.js built code which\n    // we'll copy into final image. But we also need lambda code to run\n    // asset deployment tasks\n    const dockerImageCode = DockerImageCode.fromImageAsset(buildContext, {\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps,\n      buildArgs: {\n        [BUILD_ID_ARG_NAME]: this.buildId,\n        [BUILDER_IMAGE_ALIAS_ARG_NAME]: this.builderImageAlias,\n        [PUBLIC_PATH_ARG_NAME]: PUBLIC_PATH,\n        [RELATIVE_PATH_TO_PACKAGE_ARG_NAME]: this.relativePathToPackage,\n        ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps\n          ?.buildArgs,\n      },\n    });\n    return dockerImageCode;\n  }\n}\n"]}