open-next-cdk
Version:
Deploy a NextJS app using OpenNext packaging to serverless AWS using CDK
107 lines • 16.5 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Nextjs = exports.CONFIG_ENV_JSON_PATH = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/* eslint-disable prettier/prettier */
const os = require("os");
const path = require("path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const s3 = require("aws-cdk-lib/aws-s3");
const constructs_1 = require("constructs");
const fs = require("fs-extra");
const ImageOptimizationLambda_1 = require("./ImageOptimizationLambda");
const NextjsAssetsDeployment_1 = require("./NextjsAssetsDeployment");
const NextjsBuild_1 = require("./NextjsBuild");
const NextjsDistribution_1 = require("./NextjsDistribution");
const NextjsLambda_1 = require("./NextjsLambda");
// contains server-side resolved environment vars in config bucket
exports.CONFIG_ENV_JSON_PATH = 'next-env.json';
/**
* The `Nextjs` construct is a higher level construct that makes it easy to create a NextJS app.
*
* Your standalone server application will be bundled using o(utput tracing and will be deployed to a Lambda function.
* Static assets will be deployed to an S3 bucket and served via CloudFront.
* You must use Next.js 10.3.0 or newer.
*
* Please provide a `nextjsPath` to the Next.js app inside your project.
*
* @example
* new Nextjs(this, "Web", {
* nextjsPath: path.resolve("packages/web"),
* })
*/
class Nextjs extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.props = props;
if (!props.quiet)
console.debug('┌ Building Next.js app ▼ ...');
// get dir to store temp build files in
const tempBuildDir = props.tempBuildDir
? path.resolve(path.join(props.tempBuildDir, `nextjs-cdk-build-${this.node.id}-${this.node.addr.substring(0, 4)}`))
: fs.mkdtempSync(path.join(os.tmpdir(), 'nextjs-cdk-build-'));
this.tempBuildDir = tempBuildDir;
// create static asset bucket
this.staticAssetBucket =
props.defaults?.assetDeployment?.bucket ??
new s3.Bucket(this, 'Bucket', {
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
autoDeleteObjects: true,
});
// build nextjs app
this.nextBuild = new NextjsBuild_1.NextjsBuild(this, id, { ...props, tempBuildDir });
this.serverFunction = new NextjsLambda_1.NextJsLambda(this, 'Fn', {
...props,
tempBuildDir,
nextBuild: this.nextBuild,
lambda: props.defaults?.lambda,
});
// build image optimization
this.imageOptimizationFunction = new ImageOptimizationLambda_1.ImageOptimizationLambda(this, 'ImgOptFn', {
...props,
nextBuild: this.nextBuild,
bucket: props.imageOptimizationBucket || this.bucket,
lambdaOptions: props.defaults?.lambda,
});
// deploy nextjs static assets to s3
this.assetsDeployment = new NextjsAssetsDeployment_1.NextJsAssetsDeployment(this, 'AssetDeployment', {
...props,
...props.defaults?.assetDeployment,
tempBuildDir,
nextBuild: this.nextBuild,
bucket: this.staticAssetBucket,
});
// finish static deployment BEFORE deploying new function code
// as there is some time after the new static files are uploaded but before they are rewritten
const rewriter = this.assetsDeployment.rewriter?.rewriteNode;
if (rewriter) {
this.serverFunction.lambdaFunction.node.addDependency(rewriter);
}
else {
this.serverFunction.lambdaFunction.node.addDependency(...this.assetsDeployment.deployments);
}
this.distribution = new NextjsDistribution_1.NextjsDistribution(this, 'Distribution', {
...props,
...props.defaults?.distribution,
staticAssetsBucket: this.assetsDeployment.bucket,
tempBuildDir,
nextBuild: this.nextBuild,
serverFunction: this.serverFunction.lambdaFunction,
imageOptFunction: this.imageOptimizationFunction,
});
if (!props.quiet)
console.debug('└ Finished preparing NextJS app for deployment');
}
get url() {
const customDomain = this.distribution.customDomainName;
return customDomain ? `https://${customDomain}` : this.distribution.url;
}
get bucket() {
return this.staticAssetBucket;
}
}
exports.Nextjs = Nextjs;
_a = JSII_RTTI_SYMBOL_1;
Nextjs[_a] = { fqn: "open-next-cdk.Nextjs", version: "0.0.10" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Nextjs.js","sourceRoot":"","sources":["../src/Nextjs.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable prettier/prettier */\nimport * as os from 'os';\nimport * as path from 'path';\nimport { RemovalPolicy } from 'aws-cdk-lib';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { FunctionOptions } from 'aws-cdk-lib/aws-lambda';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport * as fs from 'fs-extra';\nimport { ImageOptimizationLambda } from './ImageOptimizationLambda';\nimport { NextJsAssetsDeployment, NextjsAssetsDeploymentPropsDefaults } from './NextjsAssetsDeployment';\nimport { BaseSiteDomainProps, NextjsBaseProps } from './NextjsBase';\nimport { NextjsBuild } from './NextjsBuild';\nimport { NextjsDistribution, NextjsDistributionPropsDefaults } from './NextjsDistribution';\nimport { NextJsLambda } from './NextjsLambda';\n\n// contains server-side resolved environment vars in config bucket\nexport const CONFIG_ENV_JSON_PATH = 'next-env.json';\n\nexport interface NextjsDomainProps extends BaseSiteDomainProps { }\n\n/**\n * Defaults for created resources.\n * Why `any`? see https://github.com/aws/jsii/issues/2901\n */\nexport interface NextjsDefaultsProps {\n  /**\n   * Override static file deployment settings.\n   */\n  readonly assetDeployment?: NextjsAssetsDeploymentPropsDefaults;\n\n  /**\n   * Override server lambda function settings.\n   */\n  readonly lambda?: FunctionOptions;\n\n  /**\n   * Override CloudFront distribution settings.\n   */\n  readonly distribution?: NextjsDistributionPropsDefaults;\n}\n\nexport interface NextjsProps extends NextjsBaseProps {\n  /**\n   * Optional S3 Bucket to use, defaults to assets bucket\n   */\n  readonly imageOptimizationBucket?: s3.IBucket;\n  /**\n   * Allows you to override defaults for the resources created by this\n   * construct.\n   */\n  readonly defaults?: NextjsDefaultsProps;\n}\n\n/**\n * The `Nextjs` construct is a higher level construct that makes it easy to create a NextJS app.\n *\n * Your standalone server application will be bundled using o(utput tracing and will be deployed to a Lambda function.\n * Static assets will be deployed to an S3 bucket and served via CloudFront.\n * You must use Next.js 10.3.0 or newer.\n *\n * Please provide a `nextjsPath` to the Next.js app inside your project.\n *\n * @example\n * new Nextjs(this, \"Web\", {\n *   nextjsPath: path.resolve(\"packages/web\"),\n * })\n */\nexport class Nextjs extends Construct {\n  /**\n   * The main NextJS server handler lambda function.\n   */\n  public serverFunction: NextJsLambda;\n\n  /**\n   * The image optimization handler lambda function.\n   */\n  public imageOptimizationFunction: ImageOptimizationLambda;\n\n  /**\n   * Built NextJS project output.\n   */\n  public nextBuild: NextjsBuild;\n\n  /**\n   * Asset deployment to S3.\n   */\n  public assetsDeployment: NextJsAssetsDeployment;\n\n  /**\n   * CloudFront distribution.\n   */\n  public distribution: NextjsDistribution;\n\n  /**\n   * Where build-time assets for deployment are stored.\n   */\n  public tempBuildDir: string;\n\n  public configBucket?: s3.Bucket;\n  public lambdaFunctionUrl!: lambda.FunctionUrl;\n  public imageOptimizationLambdaFunctionUrl!: lambda.FunctionUrl;\n\n  protected staticAssetBucket: s3.IBucket;\n\n  constructor(scope: Construct, id: string, protected props: NextjsProps) {\n    super(scope, id);\n\n    if (!props.quiet) console.debug('┌ Building Next.js app ▼ ...');\n\n    // get dir to store temp build files in\n    const tempBuildDir = props.tempBuildDir\n      ? path.resolve(\n        path.join(props.tempBuildDir, `nextjs-cdk-build-${this.node.id}-${this.node.addr.substring(0, 4)}`)\n      )\n      : fs.mkdtempSync(path.join(os.tmpdir(), 'nextjs-cdk-build-'));\n\n    this.tempBuildDir = tempBuildDir;\n\n    // create static asset bucket\n    this.staticAssetBucket =\n      props.defaults?.assetDeployment?.bucket ??\n      new s3.Bucket(this, 'Bucket', {\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n      });\n\n    // build nextjs app\n    this.nextBuild = new NextjsBuild(this, id, { ...props, tempBuildDir });\n    this.serverFunction = new NextJsLambda(this, 'Fn', {\n      ...props,\n      tempBuildDir,\n      nextBuild: this.nextBuild,\n      lambda: props.defaults?.lambda,\n    });\n    // build image optimization\n    this.imageOptimizationFunction = new ImageOptimizationLambda(this, 'ImgOptFn', {\n      ...props,\n      nextBuild: this.nextBuild,\n      bucket: props.imageOptimizationBucket || this.bucket,\n      lambdaOptions: props.defaults?.lambda,\n    });\n\n    // deploy nextjs static assets to s3\n    this.assetsDeployment = new NextJsAssetsDeployment(this, 'AssetDeployment', {\n      ...props,\n      ...props.defaults?.assetDeployment,\n      tempBuildDir,\n      nextBuild: this.nextBuild,\n      bucket: this.staticAssetBucket,\n    });\n    // finish static deployment BEFORE deploying new function code\n    // as there is some time after the new static files are uploaded but before they are rewritten\n    const rewriter = this.assetsDeployment.rewriter?.rewriteNode;\n    if (rewriter) {\n      this.serverFunction.lambdaFunction.node.addDependency(rewriter);\n    } else {\n      this.serverFunction.lambdaFunction.node.addDependency(...this.assetsDeployment.deployments);\n    }\n\n    this.distribution = new NextjsDistribution(this, 'Distribution', {\n      ...props,\n      ...props.defaults?.distribution,\n      staticAssetsBucket: this.assetsDeployment.bucket,\n      tempBuildDir,\n      nextBuild: this.nextBuild,\n      serverFunction: this.serverFunction.lambdaFunction,\n      imageOptFunction: this.imageOptimizationFunction,\n    });\n\n    if (!props.quiet) console.debug('└ Finished preparing NextJS app for deployment');\n  }\n\n  public get url(): string {\n    const customDomain = this.distribution.customDomainName;\n    return customDomain ? `https://${customDomain}` : this.distribution.url;\n  }\n\n  public get bucket(): s3.IBucket {\n    return this.staticAssetBucket;\n  }\n}\n"]}