cdk-nextjs-standalone
Version:
Deploy a NextJS app to AWS using CDK and OpenNext.
92 lines • 14.6 kB
JavaScript
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":"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","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"]}
;