cdk-nextjs-standalone
Version:
Deploy a NextJS app to AWS using CDK and OpenNext.
109 lines • 17.9 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.NextjsServer = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const node_crypto_1 = require("node:crypto");
const node_fs_1 = 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 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");
const constants_1 = require("./constants");
const NextjsBucketDeployment_1 = require("./NextjsBucketDeployment");
const common_lambda_props_1 = require("./utils/common-lambda-props");
const create_archive_1 = require("./utils/create-archive");
/**
* Build a lambda function from a NextJS application to handle server-side rendering, API routes, and image optimization.
*/
class NextjsServer extends constructs_1.Construct {
get environment() {
return {
...this.props.environment,
...this.props.lambda?.environment,
CACHE_BUCKET_NAME: this.props.staticAssetBucket.bucketName,
CACHE_BUCKET_REGION: aws_cdk_lib_1.Stack.of(this.props.staticAssetBucket).region,
CACHE_BUCKET_KEY_PREFIX: constants_1.CACHE_BUCKET_KEY_PREFIX,
};
}
constructor(scope, id, props) {
super(scope, id);
this.props = props;
// must create code asset separately (typically it is implicitly created in
//`Function` construct) b/c we need to substitute unresolved env vars
const sourceAsset = this.createSourceCodeAsset();
// source and destination assets are defined separately so that source
// assets are immutable (easier debugging). Technically we could overwrite
// source asset
const destinationAsset = this.createDestinationCodeAsset();
const bucketDeployment = this.createBucketDeployment(sourceAsset, destinationAsset);
this.lambdaFunction = this.createFunction(destinationAsset);
// don't update lambda function until bucket deployment is complete
this.lambdaFunction.node.addDependency(bucketDeployment);
}
createSourceCodeAsset() {
const archivePath = (0, create_archive_1.createArchive)({
directory: this.props.nextBuild.nextServerFnDir,
quiet: this.props.quiet,
zipFileName: 'server-fn.zip',
});
const asset = new aws_s3_assets_1.Asset(this, 'SourceCodeAsset', {
path: archivePath,
...this.props.overrides?.sourceCodeAssetProps,
});
// new Asset() creates copy of zip into cdk.out/. This cleans up tmp folder
(0, node_fs_1.rmSync)(archivePath, { recursive: true });
return asset;
}
createDestinationCodeAsset() {
// create dummy directory to upload with random values so it's uploaded each time
// TODO: look into caching?
const assetsTmpDir = (0, node_fs_1.mkdtempSync)((0, node_path_1.resolve)((0, node_os_1.tmpdir)(), 'bucket-deployment-dest-asset-'));
// this code will never run b/c we explicitly declare dependency between
// lambda function and bucket deployment.
(0, node_fs_1.writeFileSync)((0, node_path_1.resolve)(assetsTmpDir, 'index.mjs'), `export function handler() { return '${(0, node_crypto_1.randomUUID)()}' }`);
const destinationAsset = new aws_s3_assets_1.Asset(this, 'DestinationCodeAsset', {
path: assetsTmpDir,
...this.props.overrides?.destinationCodeAssetProps,
});
(0, node_fs_1.rmSync)(assetsTmpDir, { recursive: true });
return destinationAsset;
}
createBucketDeployment(sourceAsset, destinationAsset) {
const bucketDeployment = new NextjsBucketDeployment_1.NextjsBucketDeployment(this, 'BucketDeployment', {
asset: sourceAsset,
debug: true,
destinationBucket: destinationAsset.bucket,
destinationKeyPrefix: destinationAsset.s3ObjectKey,
prune: false, // not applicable b/c zip: true
// this.props.environment is for build time, not this.environment which is for runtime
substitutionConfig: NextjsBucketDeployment_1.NextjsBucketDeployment.getSubstitutionConfig(this.props.environment || {}),
zip: true,
...this.props.overrides?.nextjsBucketDeploymentProps,
});
return bucketDeployment;
}
createFunction(asset) {
// until after the build time env vars in code zip asset are substituted
const fn = new aws_lambda_1.Function(this, 'Fn', {
...(0, common_lambda_props_1.getCommonFunctionProps)(this),
code: aws_lambda_1.Code.fromBucket(asset.bucket, asset.s3ObjectKey),
handler: 'index.handler',
description: 'Next.js Server Handler',
...this.props.lambda,
// `environment` needs to go after `this.props.lambda` b/c if
// `this.props.lambda.environment` is defined, it will override
// CACHE_* environment variables which are required
environment: { ...this.environment, ...this.props.lambda?.environment },
...this.props.overrides?.functionProps,
});
this.props.staticAssetBucket.grantReadWrite(fn);
return fn;
}
}
exports.NextjsServer = NextjsServer;
_a = JSII_RTTI_SYMBOL_1;
NextjsServer[_a] = { fqn: "cdk-nextjs-standalone.NextjsServer", version: "4.2.3" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsServer.js","sourceRoot":"","sources":["../src/NextjsServer.ts"],"names":[],"mappings":"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","sourcesContent":["import { randomUUID } from 'node:crypto';\nimport { mkdtempSync, rmSync, writeFileSync } from 'node:fs';\nimport { tmpdir } from 'node:os';\nimport { resolve } from 'node:path';\nimport { Stack } from 'aws-cdk-lib';\nimport { Code, Function, FunctionOptions } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket, IBucket } 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, OptionalFunctionProps, OptionalNextjsBucketDeploymentProps } from './generated-structs';\nimport { NextjsProps } from './Nextjs';\nimport { NextjsBucketDeployment } from './NextjsBucketDeployment';\nimport { NextjsBuild } from './NextjsBuild';\nimport { getCommonFunctionProps } from './utils/common-lambda-props';\nimport { createArchive } from './utils/create-archive';\n\nexport interface NextjsServerOverrides {\n  readonly sourceCodeAssetProps?: OptionalAssetProps;\n  readonly destinationCodeAssetProps?: OptionalAssetProps;\n  readonly functionProps?: OptionalFunctionProps;\n  readonly nextjsBucketDeploymentProps?: OptionalNextjsBucketDeploymentProps;\n}\n\nexport type EnvironmentVars = Record<string, string>;\n\nexport interface NextjsServerProps {\n  /**\n   * @see {@link NextjsProps.environment}\n   */\n  readonly environment?: NextjsProps['environment'];\n  /**\n   * Override function properties.\n   */\n  readonly lambda?: FunctionOptions;\n  /**\n   * @see {@link NextjsBuild}\n   */\n  readonly nextBuild: NextjsBuild;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsServerOverrides;\n  /**\n   * @see {@link NextjsProps.quiet}\n   */\n  readonly quiet?: NextjsProps['quiet'];\n  /**\n   * Static asset bucket. Function needs bucket to read from cache.\n   */\n  readonly staticAssetBucket: IBucket;\n}\n\n/**\n * Build a lambda function from a NextJS application to handle server-side rendering, API routes, and image optimization.\n */\nexport class NextjsServer extends Construct {\n  configBucket?: Bucket;\n  lambdaFunction: Function;\n\n  private props: NextjsServerProps;\n  private get environment(): Record<string, string> {\n    return {\n      ...this.props.environment,\n      ...this.props.lambda?.environment,\n      CACHE_BUCKET_NAME: this.props.staticAssetBucket.bucketName,\n      CACHE_BUCKET_REGION: Stack.of(this.props.staticAssetBucket).region,\n      CACHE_BUCKET_KEY_PREFIX,\n    };\n  }\n\n  constructor(scope: Construct, id: string, props: NextjsServerProps) {\n    super(scope, id);\n    this.props = props;\n\n    // must create code asset separately (typically it is implicitly created in\n    //`Function` construct) b/c we need to substitute unresolved env vars\n    const sourceAsset = this.createSourceCodeAsset();\n    // source and destination assets are defined separately so that source\n    // assets are immutable (easier debugging). Technically we could overwrite\n    // source asset\n    const destinationAsset = this.createDestinationCodeAsset();\n    const bucketDeployment = this.createBucketDeployment(sourceAsset, destinationAsset);\n    this.lambdaFunction = this.createFunction(destinationAsset);\n    // don't update lambda function until bucket deployment is complete\n    this.lambdaFunction.node.addDependency(bucketDeployment);\n  }\n\n  private createSourceCodeAsset() {\n    const archivePath = createArchive({\n      directory: this.props.nextBuild.nextServerFnDir,\n      quiet: this.props.quiet,\n      zipFileName: 'server-fn.zip',\n    });\n    const asset = new Asset(this, 'SourceCodeAsset', {\n      path: archivePath,\n      ...this.props.overrides?.sourceCodeAssetProps,\n    });\n    // new Asset() creates copy of zip into cdk.out/. This cleans up tmp folder\n    rmSync(archivePath, { recursive: true });\n    return asset;\n  }\n\n  private createDestinationCodeAsset() {\n    // create dummy directory to upload with random values so it's uploaded each time\n    // TODO: look into caching?\n    const assetsTmpDir = mkdtempSync(resolve(tmpdir(), 'bucket-deployment-dest-asset-'));\n    // this code will never run b/c we explicitly declare dependency between\n    // lambda function and bucket deployment.\n    writeFileSync(resolve(assetsTmpDir, 'index.mjs'), `export function handler() { return '${randomUUID()}' }`);\n    const destinationAsset = new Asset(this, 'DestinationCodeAsset', {\n      path: assetsTmpDir,\n      ...this.props.overrides?.destinationCodeAssetProps,\n    });\n    rmSync(assetsTmpDir, { recursive: true });\n    return destinationAsset;\n  }\n\n  private createBucketDeployment(sourceAsset: Asset, destinationAsset: Asset) {\n    const bucketDeployment = new NextjsBucketDeployment(this, 'BucketDeployment', {\n      asset: sourceAsset,\n      debug: true,\n      destinationBucket: destinationAsset.bucket,\n      destinationKeyPrefix: destinationAsset.s3ObjectKey,\n      prune: false, // not applicable b/c zip: true\n      // this.props.environment is for build time, not this.environment which is for runtime\n      substitutionConfig: NextjsBucketDeployment.getSubstitutionConfig(this.props.environment || {}),\n      zip: true,\n      ...this.props.overrides?.nextjsBucketDeploymentProps,\n    });\n    return bucketDeployment;\n  }\n\n  private createFunction(asset: Asset) {\n    // until after the build time env vars in code zip asset are substituted\n    const fn = new Function(this, 'Fn', {\n      ...getCommonFunctionProps(this),\n      code: Code.fromBucket(asset.bucket, asset.s3ObjectKey),\n      handler: 'index.handler',\n      description: 'Next.js Server Handler',\n      ...this.props.lambda,\n      // `environment` needs to go after `this.props.lambda` b/c if\n      // `this.props.lambda.environment` is defined, it will override\n      // CACHE_* environment variables which are required\n      environment: { ...this.environment, ...this.props.lambda?.environment },\n      ...this.props.overrides?.functionProps,\n    });\n    this.props.staticAssetBucket.grantReadWrite(fn);\n\n    return fn;\n  }\n}\n"]}