UNPKG

open-next-cdk

Version:

Deploy a NextJS app using OpenNext packaging to serverless AWS using CDK

179 lines 27.3 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.makeTokenPlaceholder = exports.TOKEN_PLACEHOLDER_END = exports.TOKEN_PLACEHOLDER_BEGIN = exports.getBuildCmdEnvironment = exports.createArchive = exports.NextjsBuild = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const path = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const constructs_1 = require("constructs"); const spawn = require("cross-spawn"); const fs = require("fs-extra"); const NextjsAssetsDeployment_1 = require("./NextjsAssetsDeployment"); const NEXTJS_BUILD_DIR = '.open-next'; const NEXTJS_STATIC_DIR = 'assets'; const NEXTJS_BUILD_MIDDLEWARE_FN_DIR = 'middleware-function'; const NEXTJS_BUILD_IMAGE_FN_DIR = 'image-optimization-function'; const NEXTJS_BUILD_SERVER_FN_DIR = 'server-function'; /** * Represents a built NextJS application. * This construct runs `npm build` in standalone output mode inside your `nextjsPath`. * This construct can be used by higher level constructs or used directly. */ class NextjsBuild extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.props = props; const nextJsPath = props.nextJsPath || props.nextjsPath; const openNextPath = props.openNextPath; if (nextJsPath) { if (openNextPath) throw new Error(`Cannot supply both nextJsPath and openNextPath`); if (!fs.existsSync(nextJsPath)) throw new Error(`NextJS application not found at "${nextJsPath}"`); // build app if (nextJsPath) { this.runNpmBuild(nextJsPath); } this.openNextPath = path.join(nextJsPath, NEXTJS_BUILD_DIR); if (!fs.existsSync(this.openNextPath)) throw new Error(`OpenNext package failed to build at "${this.openNextPath}"`); } else if (openNextPath) { this.openNextPath = path.resolve(openNextPath); if (!fs.existsSync(this.openNextPath)) throw new Error(`OpenNext package not found at "${this.openNextPath}"`); } else { throw new Error(`Must supply either nextJsPath or openNextPath`); } // our outputs this.nextStaticDir = this._getNextStaticDir(); this.nextImageFnDir = this._getOutputDir(NEXTJS_BUILD_IMAGE_FN_DIR); this.nextServerFnDir = this._getOutputDir(NEXTJS_BUILD_SERVER_FN_DIR); this.nextMiddlewareFnDir = this._getOutputDir(NEXTJS_BUILD_MIDDLEWARE_FN_DIR, true); } runNpmBuild(nextjsPath) { const { isPlaceholder, quiet } = this.props; if (isPlaceholder) { if (!quiet) console.debug(`Skipping build for placeholder NextjsBuild at ${nextjsPath}`); return; } // validate site path exists if (!fs.existsSync(nextjsPath)) { throw new Error(`Invalid nextjsPath ${nextjsPath} - directory does not exist at "${path.resolve(nextjsPath)}"`); } // Ensure that the site has a build script defined if (!fs.existsSync(path.join(nextjsPath, 'package.json'))) { throw new Error(`No package.json found at "${nextjsPath}".`); } const packageJson = fs.readJsonSync(path.join(nextjsPath, 'package.json')); if (!packageJson.scripts || !packageJson.scripts.build) { throw new Error(`No "build" script found within package.json in "${nextjsPath}".`); } // build environment vars const buildEnv = { ...process.env, ...getBuildCmdEnvironment(this.props.environment), ...(this.props.nodeEnv ? { NODE_ENV: this.props.nodeEnv } : {}), }; const buildPath = this.props.buildPath ?? nextjsPath; const buildCommand = this.props.buildCommand ?? 'npx --yes open-next@1 build'; // run build console.debug(`├ Running "${buildCommand}" in`, buildPath); const cmdParts = buildCommand.split(/\s+/); const buildResult = spawn.sync(cmdParts[0], cmdParts.slice(1), { cwd: buildPath, stdio: this.props.quiet ? 'ignore' : 'inherit', env: buildEnv, shell: true, }); if (buildResult.status !== 0) { throw new Error('The app "build" script failed.'); } } readPublicFileList() { const publicDir = this._getNextStaticDir(); if (!fs.existsSync(publicDir)) return []; return NextjsAssetsDeployment_1.listDirectory(publicDir).map((file) => path.join('/', path.relative(publicDir, file))); } _getNextBuildDir() { return this.openNextPath; } _getOutputDir(subdir, suppressMissing = false) { const { isPlaceholder } = this.props; const nextDir = this._getNextBuildDir(); const standaloneDir = path.join(nextDir, subdir); if (!suppressMissing && !isPlaceholder && !fs.existsSync(standaloneDir)) { throw new Error(`Could not find ${standaloneDir} directory.`); } return standaloneDir; } // contains static files _getNextStaticDir() { return path.join(this._getNextBuildDir(), NEXTJS_STATIC_DIR); } } exports.NextjsBuild = NextjsBuild; _a = JSII_RTTI_SYMBOL_1; NextjsBuild[_a] = { fqn: "open-next-cdk.NextjsBuild", version: "0.0.10" }; // zip up a directory and return path to zip file function createArchive({ directory, zipFileName, zipOutDir, fileGlob = '.', compressionLevel = 1, quiet, }) { // if directory is empty, can skip if (!fs.existsSync(directory) || fs.readdirSync(directory).length === 0) return null; zipOutDir = path.resolve(zipOutDir); fs.mkdirpSync(zipOutDir); // get output path const zipFilePath = path.join(zipOutDir, zipFileName); // delete existing zip file if (fs.existsSync(zipFilePath)) { fs.unlinkSync(zipFilePath); } // run script to create zipfile, preserving symlinks for node_modules (e.g. pnpm structure) let result; const isWindows = process.platform === 'win32'; if (isWindows) { const psCompressionLevel = compressionLevel === 0 ? 'NoCompression' : 'Fastest'; result = spawn.sync('powershell.exe', [ '-NoLogo', '-NoProfile', '-NonInteractive', '-Command', `Compress-Archive -Path '${directory}\\*' -DestinationPath '${zipFilePath}' -CompressionLevel ${psCompressionLevel}`, ], { stdio: 'inherit' }); } else { result = spawn.sync('bash', // getting ENOENT when specifying 'node' here for some reason [ quiet ? '-c' : '-xc', [`cd '${directory}'`, `zip -ryq${compressionLevel} '${zipFilePath}' ${fileGlob}`].join('&&'), ], { stdio: 'inherit' }); } if (result.status !== 0) { throw new Error(`There was a problem generating the package for ${zipFileName} with ${directory}: ${result.error}`); } // check output if (!fs.existsSync(zipFilePath)) { throw new Error(`There was a problem generating the archive for ${directory}; the archive is missing in ${zipFilePath}.`); } return zipFilePath; } exports.createArchive = createArchive; function getBuildCmdEnvironment(siteEnvironment) { // Generate environment placeholders to be replaced // ie. environment => { API_URL: api.url } // environment => API_URL="{NEXT{! API_URL !}}" // const buildCmdEnvironment = {}; Object.entries(siteEnvironment || {}).forEach(([key, value]) => { buildCmdEnvironment[key] = aws_cdk_lib_1.Token.isUnresolved(value) ? exports.makeTokenPlaceholder(key) : value; }); return buildCmdEnvironment; } exports.getBuildCmdEnvironment = getBuildCmdEnvironment; exports.TOKEN_PLACEHOLDER_BEGIN = '{NEXT{! '; exports.TOKEN_PLACEHOLDER_END = ' !}}'; exports.makeTokenPlaceholder = (value) => exports.TOKEN_PLACEHOLDER_BEGIN + value.toString() + exports.TOKEN_PLACEHOLDER_END; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsBuild.js","sourceRoot":"","sources":["../src/NextjsBuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Token } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport * as spawn from 'cross-spawn';\nimport * as fs from 'fs-extra';\nimport { listDirectory } from './NextjsAssetsDeployment';\nimport { CompressionLevel, NextjsBaseProps } from './NextjsBase';\n\nconst NEXTJS_BUILD_DIR = '.open-next';\nconst NEXTJS_STATIC_DIR = 'assets';\nconst NEXTJS_BUILD_MIDDLEWARE_FN_DIR = 'middleware-function';\nconst NEXTJS_BUILD_IMAGE_FN_DIR = 'image-optimization-function';\nconst NEXTJS_BUILD_SERVER_FN_DIR = 'server-function';\n\nexport interface NextjsBuildProps extends NextjsBaseProps {}\n\n/**\n * Represents a built NextJS application.\n * This construct runs `npm build` in standalone output mode inside your `nextjsPath`.\n * This construct can be used by higher level constructs or used directly.\n */\nexport class NextjsBuild extends Construct {\n  // build output directories\n  /**\n   * Contains code for middleware. Not currently used.\n   */\n  public nextMiddlewareFnDir?: string;\n  /**\n   * Contains server code and dependencies.\n   */\n  public nextServerFnDir: string;\n  /**\n   * Contains function for processessing image requests.\n   * Should be arm64.\n   */\n  public nextImageFnDir: string;\n  /**\n   * Static files containing client-side code.\n   */\n  public nextStaticDir: string;\n\n  public props: NextjsBuildProps;\n\n  readonly openNextPath: string;\n\n  constructor(scope: Construct, id: string, props: NextjsBuildProps) {\n    super(scope, id);\n    this.props = props;\n\n    const nextJsPath = props.nextJsPath || props.nextjsPath;\n    const openNextPath = props.openNextPath;\n\n    if (nextJsPath) {\n      if (openNextPath) throw new Error(`Cannot supply both nextJsPath and openNextPath`);\n      if (!fs.existsSync(nextJsPath)) throw new Error(`NextJS application not found at \"${nextJsPath}\"`);\n\n      // build app\n      if (nextJsPath) {\n        this.runNpmBuild(nextJsPath);\n      }\n\n      this.openNextPath = path.join(nextJsPath, NEXTJS_BUILD_DIR);\n      if (!fs.existsSync(this.openNextPath))\n        throw new Error(`OpenNext package failed to build at \"${this.openNextPath}\"`);\n    } else if (openNextPath) {\n      this.openNextPath = path.resolve(openNextPath);\n      if (!fs.existsSync(this.openNextPath)) throw new Error(`OpenNext package not found at \"${this.openNextPath}\"`);\n    } else {\n      throw new Error(`Must supply either nextJsPath or openNextPath`);\n    }\n\n    // our outputs\n    this.nextStaticDir = this._getNextStaticDir();\n    this.nextImageFnDir = this._getOutputDir(NEXTJS_BUILD_IMAGE_FN_DIR);\n    this.nextServerFnDir = this._getOutputDir(NEXTJS_BUILD_SERVER_FN_DIR);\n    this.nextMiddlewareFnDir = this._getOutputDir(NEXTJS_BUILD_MIDDLEWARE_FN_DIR, true);\n  }\n\n  private runNpmBuild(nextjsPath: string) {\n    const { isPlaceholder, quiet } = this.props;\n\n    if (isPlaceholder) {\n      if (!quiet) console.debug(`Skipping build for placeholder NextjsBuild at ${nextjsPath}`);\n      return;\n    }\n\n    // validate site path exists\n    if (!fs.existsSync(nextjsPath)) {\n      throw new Error(`Invalid nextjsPath ${nextjsPath} - directory does not exist at \"${path.resolve(nextjsPath)}\"`);\n    }\n    // Ensure that the site has a build script defined\n    if (!fs.existsSync(path.join(nextjsPath, 'package.json'))) {\n      throw new Error(`No package.json found at \"${nextjsPath}\".`);\n    }\n    const packageJson = fs.readJsonSync(path.join(nextjsPath, 'package.json'));\n    if (!packageJson.scripts || !packageJson.scripts.build) {\n      throw new Error(`No \"build\" script found within package.json in \"${nextjsPath}\".`);\n    }\n\n    // build environment vars\n    const buildEnv = {\n      ...process.env,\n      ...getBuildCmdEnvironment(this.props.environment),\n      ...(this.props.nodeEnv ? { NODE_ENV: this.props.nodeEnv } : {}),\n    };\n\n    const buildPath = this.props.buildPath ?? nextjsPath;\n    const buildCommand = this.props.buildCommand ?? 'npx --yes open-next@1 build';\n    // run build\n    console.debug(`├ Running \"${buildCommand}\" in`, buildPath);\n    const cmdParts = buildCommand.split(/\\s+/);\n    const buildResult = spawn.sync(cmdParts[0], cmdParts.slice(1), {\n      cwd: buildPath,\n      stdio: this.props.quiet ? 'ignore' : 'inherit',\n      env: buildEnv,\n      shell: true,\n    });\n    if (buildResult.status !== 0) {\n      throw new Error('The app \"build\" script failed.');\n    }\n  }\n\n  readPublicFileList() {\n    const publicDir = this._getNextStaticDir();\n    if (!fs.existsSync(publicDir)) return [];\n    return listDirectory(publicDir).map((file) => path.join('/', path.relative(publicDir, file)));\n  }\n\n  private _getNextBuildDir() {\n    return this.openNextPath;\n  }\n\n  private _getOutputDir(subdir: string, suppressMissing = false) {\n    const { isPlaceholder } = this.props;\n\n    const nextDir = this._getNextBuildDir();\n    const standaloneDir = path.join(nextDir, subdir);\n\n    if (!suppressMissing && !isPlaceholder && !fs.existsSync(standaloneDir)) {\n      throw new Error(`Could not find ${standaloneDir} directory.`);\n    }\n    return standaloneDir;\n  }\n\n  // contains static files\n  private _getNextStaticDir() {\n    return path.join(this._getNextBuildDir(), NEXTJS_STATIC_DIR);\n  }\n}\n\nexport interface CreateArchiveArgs {\n  readonly compressionLevel?: CompressionLevel;\n  readonly directory: string;\n  readonly zipFileName: string;\n  readonly zipOutDir: string;\n  readonly fileGlob?: string;\n  readonly quiet?: boolean;\n}\n\n// zip up a directory and return path to zip file\nexport function createArchive({\n  directory,\n  zipFileName,\n  zipOutDir,\n  fileGlob = '.',\n  compressionLevel = 1,\n  quiet,\n}: CreateArchiveArgs): string | null {\n  // if directory is empty, can skip\n  if (!fs.existsSync(directory) || fs.readdirSync(directory).length === 0) return null;\n\n  zipOutDir = path.resolve(zipOutDir);\n  fs.mkdirpSync(zipOutDir);\n  // get output path\n  const zipFilePath = path.join(zipOutDir, zipFileName);\n\n  // delete existing zip file\n  if (fs.existsSync(zipFilePath)) {\n    fs.unlinkSync(zipFilePath);\n  }\n\n  // run script to create zipfile, preserving symlinks for node_modules (e.g. pnpm structure)\n  let result;\n  const isWindows = process.platform === 'win32';\n  if (isWindows) {\n    const psCompressionLevel = compressionLevel === 0 ? 'NoCompression' : 'Fastest';\n    result = spawn.sync(\n      'powershell.exe',\n      [\n        '-NoLogo',\n        '-NoProfile',\n        '-NonInteractive',\n        '-Command',\n        `Compress-Archive -Path '${directory}\\\\*' -DestinationPath '${zipFilePath}' -CompressionLevel ${psCompressionLevel}`,\n      ],\n      { stdio: 'inherit' }\n    );\n  } else {\n    result = spawn.sync(\n      'bash', // getting ENOENT when specifying 'node' here for some reason\n      [\n        quiet ? '-c' : '-xc',\n        [`cd '${directory}'`, `zip -ryq${compressionLevel} '${zipFilePath}' ${fileGlob}`].join('&&'),\n      ],\n      { stdio: 'inherit' }\n    );\n  }\n  if (result.status !== 0) {\n    throw new Error(`There was a problem generating the package for ${zipFileName} with ${directory}: ${result.error}`);\n  }\n  // check output\n  if (!fs.existsSync(zipFilePath)) {\n    throw new Error(\n      `There was a problem generating the archive for ${directory}; the archive is missing in ${zipFilePath}.`\n    );\n  }\n\n  return zipFilePath;\n}\n\nexport function getBuildCmdEnvironment(siteEnvironment?: { [key: string]: string }): Record<string, string> {\n  // Generate environment placeholders to be replaced\n  // ie. environment => { API_URL: api.url }\n  //     environment => API_URL=\"{NEXT{! API_URL !}}\"\n  //\n  const buildCmdEnvironment: Record<string, string> = {};\n  Object.entries(siteEnvironment || {}).forEach(([key, value]) => {\n    buildCmdEnvironment[key] = Token.isUnresolved(value) ? makeTokenPlaceholder(key) : value;\n  });\n\n  return buildCmdEnvironment;\n}\n\nexport const TOKEN_PLACEHOLDER_BEGIN = '{NEXT{! ';\nexport const TOKEN_PLACEHOLDER_END = ' !}}';\nexport const makeTokenPlaceholder = (value: string): string =>\n  TOKEN_PLACEHOLDER_BEGIN + value.toString() + TOKEN_PLACEHOLDER_END;\n"]}