UNPKG

cdk-nextjs-standalone

Version:

Deploy a NextJS app to AWS using CDK and OpenNext.

122 lines 22 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.Nextjs = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const constructs_1 = require("constructs"); const NextjsBuild_1 = require("./NextjsBuild"); const NextjsDistribution_1 = require("./NextjsDistribution"); const NextjsDomain_1 = require("./NextjsDomain"); const NextjsImage_1 = require("./NextjsImage"); const NextjsInvalidation_1 = require("./NextjsInvalidation"); const NextjsRevalidation_1 = require("./NextjsRevalidation"); const NextjsServer_1 = require("./NextjsServer"); const NextjsStaticAssets_1 = require("./NextjsStaticAssets"); /** * 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; // build nextjs app this.nextBuild = new NextjsBuild_1.NextjsBuild(this, 'Build', { nextjsPath: props.nextjsPath, buildCommand: props.buildCommand, buildPath: props.buildPath, environment: props.environment, quiet: props.quiet, skipBuild: props.skipBuild, streaming: props.streaming, ...props.overrides?.nextjs?.nextjsBuildProps, }); // deploy nextjs static assets to s3 this.staticAssets = new NextjsStaticAssets_1.NextjsStaticAssets(this, 'StaticAssets', { basePath: props.basePath, environment: props.environment, nextBuild: this.nextBuild, overrides: props.overrides?.nextjsStaticAssets, ...props.overrides?.nextjs?.nextjsStaticAssetsProps, }); this.serverFunction = new NextjsServer_1.NextjsServer(this, 'Server', { environment: props.environment, nextBuild: this.nextBuild, staticAssetBucket: this.staticAssets.bucket, overrides: props.overrides?.nextjsServer, ...props.overrides?.nextjs?.nextjsServerProps, }); // build image optimization this.imageOptimizationFunction = new NextjsImage_1.NextjsImage(this, 'Image', { bucket: props.imageOptimizationBucket || this.bucket, nextBuild: this.nextBuild, overrides: props.overrides?.nextjsImage, ...props.overrides?.nextjs?.nextjsImageProps, }); // build revalidation queue and handler function this.revalidation = new NextjsRevalidation_1.NextjsRevalidation(this, 'Revalidation', { nextBuild: this.nextBuild, serverFunction: this.serverFunction, overrides: props.overrides?.nextjsRevalidation, ...props.overrides?.nextjs?.nextjsRevalidationProps, }); if (this.props.domainProps) { this.domain = new NextjsDomain_1.NextjsDomain(this, 'Domain', { ...this.props.domainProps, overrides: props.overrides?.nextjsDomain, ...props.overrides?.nextjs?.nextjsDomainProps, }); } this.distribution = new NextjsDistribution_1.NextjsDistribution(this, 'Distribution', { nextjsPath: props.nextjsPath, basePath: props.basePath, distribution: props.distribution, streaming: props.streaming, staticAssetsBucket: this.staticAssets.bucket, nextBuild: this.nextBuild, nextDomain: this.domain, serverFunction: this.serverFunction.lambdaFunction, imageOptFunction: this.imageOptimizationFunction, overrides: props.overrides?.nextjsDistribution, ...props.overrides?.nextjs?.nextjsDistributionProps, }); if (this.domain) { this.domain.createDnsRecords(this.distribution.distribution); } if (!this.props.skipFullInvalidation) { new NextjsInvalidation_1.NextjsInvalidation(this, 'Invalidation', { distribution: this.distribution.distribution, dependencies: [], // [this.staticAssets, this.serverFunction, this.imageOptimizationFunction] overrides: props.overrides?.nextjsInvalidation, ...props.overrides?.nextjs?.nextjsInvalidationProps, }); } } /** * URL of Next.js App. */ get url() { const customDomain = this.props.domainProps?.domainName; return customDomain ? `https://${customDomain}` : this.distribution.url; } /** * Convenience method to access `Nextjs.staticAssets.bucket`. */ get bucket() { return this.staticAssets.bucket; } } exports.Nextjs = Nextjs; _a = JSII_RTTI_SYMBOL_1; Nextjs[_a] = { fqn: "cdk-nextjs-standalone.Nextjs", version: "4.2.3" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Nextjs.js","sourceRoot":"","sources":["../src/Nextjs.ts"],"names":[],"mappings":"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","sourcesContent":["import { Distribution } from 'aws-cdk-lib/aws-cloudfront';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport {\n  OptionalNextjsDistributionProps,\n  OptionalNextjsDomainProps,\n  OptionalNextjsImageProps,\n  OptionalNextjsInvalidationProps,\n  OptionalNextjsRevalidationProps,\n  OptionalNextjsServerProps,\n  OptionalNextjsStaticAssetsProps,\n} from './generated-structs';\nimport { OptionalNextjsBuildProps } from './generated-structs/OptionalNextjsBuildProps';\nimport { NextjsBuild } from './NextjsBuild';\nimport { NextjsDistribution } from './NextjsDistribution';\nimport { NextjsDomain, NextjsDomainProps } from './NextjsDomain';\nimport { NextjsImage } from './NextjsImage';\nimport { NextjsInvalidation } from './NextjsInvalidation';\nimport { NextjsOverrides } from './NextjsOverrides';\nimport { NextjsRevalidation } from './NextjsRevalidation';\nimport { NextjsServer } from './NextjsServer';\nimport { NextjsStaticAssets } from './NextjsStaticAssets';\n\nexport interface NextjsConstructOverrides {\n  readonly nextjsBuildProps?: OptionalNextjsBuildProps;\n  readonly nextjsStaticAssetsProps?: OptionalNextjsStaticAssetsProps;\n  readonly nextjsServerProps?: OptionalNextjsServerProps;\n  readonly nextjsImageProps?: OptionalNextjsImageProps;\n  readonly nextjsRevalidationProps?: OptionalNextjsRevalidationProps;\n  readonly nextjsDomainProps?: OptionalNextjsDomainProps;\n  readonly nextjsDistributionProps?: OptionalNextjsDistributionProps;\n  readonly nextjsInvalidationProps?: OptionalNextjsInvalidationProps;\n}\n\nexport interface NextjsProps {\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   * Optional build command override value.\n   * @default 'npx @opennextjs/aws@^3 build'\n   */\n  readonly buildCommand?: string;\n  /**\n   * The directory to execute `npm run build` from. By default, it is `nextjsPath`.\n   * Can be overridden, particularly useful for monorepos where `build` is expected to run\n   * at the root of the project.\n   */\n  readonly buildPath?: string;\n  /**\n   * Optional CloudFront Distribution created outside of this construct that will\n   * be used to add Next.js behaviors and origins onto. Useful with `basePath`.\n   */\n  readonly distribution?: Distribution;\n  /**\n   * Props to configure {@link NextjsDomain}. See details on how to customize at\n   * {@link NextjsDomainProps}\n   */\n  readonly domainProps?: NextjsDomainProps;\n  /**\n   * Custom environment variables to pass to the NextJS build **and** runtime.\n   */\n  readonly environment?: Record<string, string>;\n  /**\n   * Optional S3 Bucket to use, defaults to assets bucket\n   */\n  readonly imageOptimizationBucket?: s3.IBucket;\n  /**\n   * Relative path to the directory where the NextJS project is located.\n   * Can be the root of your project (`.`) or a subdirectory (`packages/web`).\n   */\n  readonly nextjsPath: string;\n  /**\n   * Override props for every construct. Enables deep customization. Use with caution as\n   * you can override all props. Recommend reviewing source code to see props\n   * you'll be overriding before using.\n   */\n  readonly overrides?: NextjsOverrides;\n  /**\n   * Less build output.\n   */\n  readonly quiet?: boolean;\n  /**\n   * Skips running Next.js build. Useful if you want to deploy `Nextjs` but\n   * haven't made any changes to Next.js app code.\n   * @default false\n   */\n  readonly skipBuild?: boolean;\n  /**\n   * By default all CloudFront cache will be invalidated on deployment.\n   * This can be set to true to skip the full cache invalidation, which\n   * could be important for some users.\n   */\n  readonly skipFullInvalidation?: boolean;\n  /**\n   * Streaming allows you to send data to the client as it's generated\n   * instead of waiting for the entire response to be generated.\n   */\n  readonly streaming?: boolean;\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: NextjsServer;\n  /**\n   * The image optimization handler lambda function.\n   */\n  public imageOptimizationFunction: NextjsImage;\n  /**\n   * Built NextJS project output.\n   */\n  public nextBuild: NextjsBuild;\n  /**\n   * Asset deployment to S3.\n   */\n  public staticAssets: NextjsStaticAssets;\n  /**\n   * Optional Route53 Hosted Zone, ACM Certificate, and Route53 DNS Records\n   */\n  public domain?: NextjsDomain;\n  /**\n   * CloudFront distribution.\n   */\n  public distribution: NextjsDistribution;\n  /**\n   * Revalidation handler and queue.\n   */\n  public revalidation: NextjsRevalidation;\n\n  public lambdaFunctionUrl!: lambda.FunctionUrl;\n  public imageOptimizationLambdaFunctionUrl!: lambda.FunctionUrl;\n\n  constructor(scope: Construct, id: string, protected props: NextjsProps) {\n    super(scope, id);\n\n    // build nextjs app\n    this.nextBuild = new NextjsBuild(this, 'Build', {\n      nextjsPath: props.nextjsPath,\n      buildCommand: props.buildCommand,\n      buildPath: props.buildPath,\n      environment: props.environment,\n      quiet: props.quiet,\n      skipBuild: props.skipBuild,\n      streaming: props.streaming,\n      ...props.overrides?.nextjs?.nextjsBuildProps,\n    });\n\n    // deploy nextjs static assets to s3\n    this.staticAssets = new NextjsStaticAssets(this, 'StaticAssets', {\n      basePath: props.basePath,\n      environment: props.environment,\n      nextBuild: this.nextBuild,\n      overrides: props.overrides?.nextjsStaticAssets,\n      ...props.overrides?.nextjs?.nextjsStaticAssetsProps,\n    });\n\n    this.serverFunction = new NextjsServer(this, 'Server', {\n      environment: props.environment,\n      nextBuild: this.nextBuild,\n      staticAssetBucket: this.staticAssets.bucket,\n      overrides: props.overrides?.nextjsServer,\n      ...props.overrides?.nextjs?.nextjsServerProps,\n    });\n    // build image optimization\n    this.imageOptimizationFunction = new NextjsImage(this, 'Image', {\n      bucket: props.imageOptimizationBucket || this.bucket,\n      nextBuild: this.nextBuild,\n      overrides: props.overrides?.nextjsImage,\n      ...props.overrides?.nextjs?.nextjsImageProps,\n    });\n\n    // build revalidation queue and handler function\n    this.revalidation = new NextjsRevalidation(this, 'Revalidation', {\n      nextBuild: this.nextBuild,\n      serverFunction: this.serverFunction,\n      overrides: props.overrides?.nextjsRevalidation,\n      ...props.overrides?.nextjs?.nextjsRevalidationProps,\n    });\n\n    if (this.props.domainProps) {\n      this.domain = new NextjsDomain(this, 'Domain', {\n        ...this.props.domainProps,\n        overrides: props.overrides?.nextjsDomain,\n        ...props.overrides?.nextjs?.nextjsDomainProps,\n      });\n    }\n    this.distribution = new NextjsDistribution(this, 'Distribution', {\n      nextjsPath: props.nextjsPath,\n      basePath: props.basePath,\n      distribution: props.distribution,\n      streaming: props.streaming,\n      staticAssetsBucket: this.staticAssets.bucket,\n      nextBuild: this.nextBuild,\n      nextDomain: this.domain,\n      serverFunction: this.serverFunction.lambdaFunction,\n      imageOptFunction: this.imageOptimizationFunction,\n      overrides: props.overrides?.nextjsDistribution,\n      ...props.overrides?.nextjs?.nextjsDistributionProps,\n    });\n    if (this.domain) {\n      this.domain.createDnsRecords(this.distribution.distribution);\n    }\n\n    if (!this.props.skipFullInvalidation) {\n      new NextjsInvalidation(this, 'Invalidation', {\n        distribution: this.distribution.distribution,\n        dependencies: [], // [this.staticAssets, this.serverFunction, this.imageOptimizationFunction]\n        overrides: props.overrides?.nextjsInvalidation,\n        ...props.overrides?.nextjs?.nextjsInvalidationProps,\n      });\n    }\n  }\n\n  /**\n   * URL of Next.js App.\n   */\n  public get url(): string {\n    const customDomain = this.props.domainProps?.domainName;\n    return customDomain ? `https://${customDomain}` : this.distribution.url;\n  }\n\n  /**\n   * Convenience method to access `Nextjs.staticAssets.bucket`.\n   */\n  public get bucket(): s3.IBucket {\n    return this.staticAssets.bucket;\n  }\n}\n"]}