UNPKG

cdk-nextjs-standalone

Version:

Deploy a NextJS app to AWS using CDK and OpenNext.

92 lines 14.6 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.NextjsStaticAssets = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const fs = require("node:fs"); const node_os_1 = require("node:os"); const node_path_1 = require("node:path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const s3 = require("aws-cdk-lib/aws-s3"); const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets"); const constructs_1 = require("constructs"); const constants_1 = require("./constants"); const NextjsBucketDeployment_1 = require("./NextjsBucketDeployment"); /** * Uploads Nextjs built static and public files to S3. * * Will inject resolved environment variables that are unresolved at synthesis * in CloudFormation Custom Resource. */ class NextjsStaticAssets extends constructs_1.Construct { get buildEnvVars() { const buildEnvVars = {}; for (const [k, v] of Object.entries(this.props.environment || {})) { if (k.startsWith('NEXT_PUBLIC')) { buildEnvVars[k] = v; } } return buildEnvVars; } constructor(scope, id, props) { super(scope, id); this.props = props; this.bucket = this.createBucket(); // when `cdk deploy "NonNextjsStack" --exclusively` is run, don't bundle assets since they will not exist if (aws_cdk_lib_1.Stack.of(this).bundlingRequired) { const asset = this.createAsset(); this.createBucketDeployment(asset); } } createBucket() { return (this.props.bucket ?? new s3.Bucket(this, 'Bucket', { removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, autoDeleteObjects: true, enforceSSL: true, blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL, encryption: s3.BucketEncryption.S3_MANAGED, ...this.props.overrides?.bucketProps, })); } createAsset() { // create temporary directory to join open-next's static output with cache output const tmpAssetsDir = fs.mkdtempSync((0, node_path_1.resolve)((0, node_os_1.tmpdir)(), 'cdk-nextjs-assets-')); fs.cpSync(this.props.nextBuild.nextStaticDir, tmpAssetsDir, { recursive: true }); fs.cpSync(this.props.nextBuild.nextCacheDir, (0, node_path_1.resolve)(tmpAssetsDir, constants_1.CACHE_BUCKET_KEY_PREFIX), { recursive: true }); const asset = new aws_s3_assets_1.Asset(this, 'Asset', { path: tmpAssetsDir, ...this.props.overrides?.assetProps, }); fs.rmSync(tmpAssetsDir, { recursive: true }); return asset; } createBucketDeployment(asset) { const basePath = this.props.basePath?.replace(/^\//, ''); // remove leading slash (if present) const allFiles = basePath ? `${basePath}/**/*` : '**/*'; const staticFiles = basePath ? `${basePath}/_next/static/**/*'` : '_next/static/**/*'; return new NextjsBucketDeployment_1.NextjsBucketDeployment(this, 'BucketDeployment', { asset, destinationBucket: this.bucket, destinationKeyPrefix: basePath, debug: true, // only put env vars that are placeholders in custom resource properties // to be replaced. other env vars were injected at build time. substitutionConfig: NextjsBucketDeployment_1.NextjsBucketDeployment.getSubstitutionConfig(this.buildEnvVars), prune: this.props.prune, // defaults to false putConfig: { [allFiles]: { CacheControl: 'public, max-age=0, must-revalidate', }, [staticFiles]: { CacheControl: 'public, max-age=31536000, immutable', }, }, ...this.props.overrides?.nextjsBucketDeploymentProps, }); } } exports.NextjsStaticAssets = NextjsStaticAssets; _a = JSII_RTTI_SYMBOL_1; NextjsStaticAssets[_a] = { fqn: "cdk-nextjs-standalone.NextjsStaticAssets", version: "4.2.3" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsStaticAssets.js","sourceRoot":"","sources":["../src/NextjsStaticAssets.ts"],"names":[],"mappings":";;;;;AAAA,8BAA8B;AAC9B,qCAAiC;AACjC,yCAAoC;AACpC,6CAAmD;AACnD,yCAAyC;AACzC,6DAAkD;AAClD,2CAAuC;AACvC,2CAAsD;AAEtD,qEAAkE;AA2ClE;;;;;GAKG;AACH,MAAa,kBAAmB,SAAQ,sBAAS;IAQ/C,IAAY,YAAY;QACtB,MAAM,YAAY,GAA2B,EAAE,CAAC;QAChD,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,EAAE,CAAC,EAAE,CAAC;YAClE,IAAI,CAAC,CAAC,UAAU,CAAC,aAAa,CAAC,EAAE,CAAC;gBAChC,YAAY,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;YACtB,CAAC;QACH,CAAC;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA8B;QACtE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QAElC,yGAAyG;QACzG,IAAI,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,gBAAgB,EAAE,CAAC;YACpC,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YACjC,IAAI,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC;QACrC,CAAC;IACH,CAAC;IAEO,YAAY;QAClB,OAAO,CACL,IAAI,CAAC,KAAK,CAAC,MAAM;YACjB,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,QAAQ,EAAE;gBAC5B,aAAa,EAAE,2BAAa,CAAC,OAAO;gBACpC,iBAAiB,EAAE,IAAI;gBACvB,UAAU,EAAE,IAAI;gBAChB,iBAAiB,EAAE,EAAE,CAAC,iBAAiB,CAAC,SAAS;gBACjD,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;gBAC1C,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,WAAW;aACrC,CAAC,CACH,CAAC;IACJ,CAAC;IAEO,WAAW;QACjB,iFAAiF;QACjF,MAAM,YAAY,GAAG,EAAE,CAAC,WAAW,CAAC,IAAA,mBAAO,EAAC,IAAA,gBAAM,GAAE,EAAE,oBAAoB,CAAC,CAAC,CAAC;QAC7E,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,aAAa,EAAE,YAAY,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACjF,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,YAAY,EAAE,IAAA,mBAAO,EAAC,YAAY,EAAE,mCAAuB,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAClH,MAAM,KAAK,GAAG,IAAI,qBAAK,CAAC,IAAI,EAAE,OAAO,EAAE;YACrC,IAAI,EAAE,YAAY;YAClB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,UAAU;SACpC,CAAC,CAAC;QACH,EAAE,CAAC,MAAM,CAAC,YAAY,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7C,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,sBAAsB,CAAC,KAAY;QACzC,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,CAAC,oCAAoC;QAC9F,MAAM,QAAQ,GAAG,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;QACxD,MAAM,WAAW,GAAG,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,qBAAqB,CAAC,CAAC,CAAC,mBAAmB,CAAC;QAEtF,OAAO,IAAI,+CAAsB,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC1D,KAAK;YACL,iBAAiB,EAAE,IAAI,CAAC,MAAM;YAC9B,oBAAoB,EAAE,QAAQ;YAC9B,KAAK,EAAE,IAAI;YACX,wEAAwE;YACxE,8DAA8D;YAC9D,kBAAkB,EAAE,+CAAsB,CAAC,qBAAqB,CAAC,IAAI,CAAC,YAAY,CAAC;YACnF,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,oBAAoB;YAC7C,SAAS,EAAE;gBACT,CAAC,QAAQ,CAAC,EAAE;oBACV,YAAY,EAAE,oCAAoC;iBACnD;gBACD,CAAC,WAAW,CAAC,EAAE;oBACb,YAAY,EAAE,qCAAqC;iBACpD;aACF;YACD,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,2BAA2B;SACrD,CAAC,CAAC;IACL,CAAC;;AAlFH,gDAmFC","sourcesContent":["import * as fs from 'node:fs';\nimport { tmpdir } from 'node:os';\nimport { resolve } from 'node:path';\nimport { RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Construct } from 'constructs';\nimport { CACHE_BUCKET_KEY_PREFIX } from './constants';\nimport { OptionalAssetProps, OptionalNextjsBucketDeploymentProps } from './generated-structs';\nimport { NextjsBucketDeployment } from './NextjsBucketDeployment';\nimport { NextjsBuild } from './NextjsBuild';\n\nexport interface NextjsStaticAssetOverrides {\n  readonly bucketProps?: s3.BucketProps;\n  readonly nextjsBucketDeploymentProps?: OptionalNextjsBucketDeploymentProps;\n  readonly assetProps?: OptionalAssetProps;\n}\n\nexport interface NextjsStaticAssetsProps {\n  /**\n   * Optional value to prefix the Next.js site under a /prefix path on CloudFront.\n   * Usually used when you deploy multiple Next.js sites on same domain using /sub-path\n   *\n   * Note, you'll need to set [basePath](https://nextjs.org/docs/app/api-reference/next-config-js/basePath)\n   * in your `next.config.ts` to this value and ensure any files in `public`\n   * folder have correct prefix.\n   * @example \"/my-base-path\"\n   */\n  readonly basePath?: string;\n  /**\n   * Define your own bucket to store static assets.\n   */\n  readonly bucket?: s3.IBucket | undefined;\n  /**\n   * Custom environment variables to pass to the NextJS build and runtime.\n   */\n  readonly environment?: Record<string, string>;\n  /**\n   * The `NextjsBuild` instance representing the built Nextjs application.\n   */\n  readonly nextBuild: NextjsBuild;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsStaticAssetOverrides;\n  /**\n   * If `true` (default), then removes old static assets after upload new static assets.\n   * @default true\n   */\n  readonly prune?: boolean;\n}\n\n/**\n * Uploads Nextjs built static and public files to S3.\n *\n * Will inject resolved environment variables that are unresolved at synthesis\n * in CloudFormation Custom Resource.\n */\nexport class NextjsStaticAssets extends Construct {\n  /**\n   * Bucket containing assets.\n   */\n  bucket: s3.IBucket;\n\n  protected props: NextjsStaticAssetsProps;\n\n  private get buildEnvVars() {\n    const buildEnvVars: Record<string, string> = {};\n    for (const [k, v] of Object.entries(this.props.environment || {})) {\n      if (k.startsWith('NEXT_PUBLIC')) {\n        buildEnvVars[k] = v;\n      }\n    }\n    return buildEnvVars;\n  }\n\n  constructor(scope: Construct, id: string, props: NextjsStaticAssetsProps) {\n    super(scope, id);\n    this.props = props;\n\n    this.bucket = this.createBucket();\n\n    // when `cdk deploy \"NonNextjsStack\" --exclusively` is run, don't bundle assets since they will not exist\n    if (Stack.of(this).bundlingRequired) {\n      const asset = this.createAsset();\n      this.createBucketDeployment(asset);\n    }\n  }\n\n  private createBucket(): s3.IBucket {\n    return (\n      this.props.bucket ??\n      new s3.Bucket(this, 'Bucket', {\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n        enforceSSL: true,\n        blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n        encryption: s3.BucketEncryption.S3_MANAGED,\n        ...this.props.overrides?.bucketProps,\n      })\n    );\n  }\n\n  private createAsset(): Asset {\n    // create temporary directory to join open-next's static output with cache output\n    const tmpAssetsDir = fs.mkdtempSync(resolve(tmpdir(), 'cdk-nextjs-assets-'));\n    fs.cpSync(this.props.nextBuild.nextStaticDir, tmpAssetsDir, { recursive: true });\n    fs.cpSync(this.props.nextBuild.nextCacheDir, resolve(tmpAssetsDir, CACHE_BUCKET_KEY_PREFIX), { recursive: true });\n    const asset = new Asset(this, 'Asset', {\n      path: tmpAssetsDir,\n      ...this.props.overrides?.assetProps,\n    });\n    fs.rmSync(tmpAssetsDir, { recursive: true });\n    return asset;\n  }\n\n  private createBucketDeployment(asset: Asset) {\n    const basePath = this.props.basePath?.replace(/^\\//, ''); // remove leading slash (if present)\n    const allFiles = basePath ? `${basePath}/**/*` : '**/*';\n    const staticFiles = basePath ? `${basePath}/_next/static/**/*'` : '_next/static/**/*';\n\n    return new NextjsBucketDeployment(this, 'BucketDeployment', {\n      asset,\n      destinationBucket: this.bucket,\n      destinationKeyPrefix: basePath,\n      debug: true,\n      // only put env vars that are placeholders in custom resource properties\n      // to be replaced. other env vars were injected at build time.\n      substitutionConfig: NextjsBucketDeployment.getSubstitutionConfig(this.buildEnvVars),\n      prune: this.props.prune, // defaults to false\n      putConfig: {\n        [allFiles]: {\n          CacheControl: 'public, max-age=0, must-revalidate',\n        },\n        [staticFiles]: {\n          CacheControl: 'public, max-age=31536000, immutable',\n        },\n      },\n      ...this.props.overrides?.nextjsBucketDeploymentProps,\n    });\n  }\n}\n"]}