cdk-nextjs
Version:
Deploy Next.js apps on AWS with CDK
258 lines • 45.4 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.NextjsDistribution = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
const constructs_1 = require("constructs");
const constants_1 = require("./constants");
class NextjsDistribution extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
/**
* Common security headers applied by default to all origins
* @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-response-headers-policies.html#managed-response-headers-policies-security
*/
this.commonSecurityHeadersBehavior = {
contentTypeOptions: { override: false },
frameOptions: {
frameOption: aws_cloudfront_1.HeadersFrameOption.SAMEORIGIN,
override: false,
},
referrerPolicy: {
override: false,
referrerPolicy: aws_cloudfront_1.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,
},
strictTransportSecurity: {
accessControlMaxAge: aws_cdk_lib_1.Duration.days(365),
includeSubdomains: true,
override: false,
preload: true,
},
xssProtection: { override: false, protection: true, modeBlock: true },
};
this.props = props;
this.staticOrigin = this.createStaticOrigin();
this.isFunctionCompute = props.nextjsType === constants_1.NextjsType.GLOBAL_FUNCTIONS;
this.dynamicOrigin = this.createDynamicOrigin();
this.dynamicOriginResponsePolicy = this.createDynamicOriginRequestPolicy();
this.dynamicCloudFrontFunctionAssociations =
this.createDynamicCloudFrontFunctionAssociations();
this.staticBehaviorOptions = this.createStaticBehaviorOptions();
this.dynamicBehaviorOptions = this.createDynamicBehaviorOptions();
this.imageBehaviorOptions = this.createImageBehaviorOptions();
this.distribution = this.getDistribution();
this.addStaticBehaviors();
this.addDynamicBehaviors();
}
createStaticOrigin() {
const s3Origin = aws_cloudfront_origins_1.S3BucketOrigin.withOriginAccessControl(this.props.assetsBucket, this.props.overrides?.s3BucketOriginProps);
return s3Origin;
}
createDynamicOrigin() {
if (this.isFunctionCompute) {
if (!this.props.functionUrl)
throw new Error("Missing NextjsDistributionProps.functionUrl");
return aws_cloudfront_origins_1.FunctionUrlOrigin.withOriginAccessControl(this.props.functionUrl, this.props.overrides?.dynamicFunctionUrlOriginWithOACProps);
}
else {
const loadBalancer = this.props.loadBalancer;
if (!loadBalancer)
throw new Error("Missing NextjsDistributionProps.loadBalancer");
return aws_cloudfront_origins_1.VpcOrigin.withApplicationLoadBalancer(loadBalancer, {
protocolPolicy: this.props.certificate
? aws_cloudfront_1.OriginProtocolPolicy.HTTPS_ONLY
: aws_cloudfront_1.OriginProtocolPolicy.HTTP_ONLY,
...this.props.overrides?.dynamicVpcOriginWithEndpointProps,
});
}
}
/**
* Lambda Function URLs "expect the `Host` header to contain the origin domain
* name, not the domain name of the CloudFront distribution."
* @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-origin-request-policies.html#managed-origin-request-policy-all-viewer-except-host-header
*/
createDynamicOriginRequestPolicy() {
return this.isFunctionCompute
? aws_cloudfront_1.OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER
: aws_cloudfront_1.OriginRequestPolicy.ALL_VIEWER;
}
/**
* Ensures Next.js `request.url` will be correct domain instead of URL of
* compute (Lambda or Fargate)
* @see https://open-next.js.org/advanced/workaround#workaround-set-x-forwarded-host-header-aws-specific
*/
createDynamicCloudFrontFunctionAssociations() {
const associations = [];
if (this.isFunctionCompute) {
const cloudFrontFn = new aws_cloudfront_1.Function(this, "CloudFrontFn", {
code: aws_cloudfront_1.FunctionCode.fromInline(`
function handler(event) {
var request = event.request;
request.headers["x-forwarded-host"] = request.headers.host;
return request;
}
`),
});
associations.push({
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST,
function: cloudFrontFn,
});
}
return associations;
}
createStaticBehaviorOptions() {
const staticBehaviorOptions = this.props.overrides?.staticBehaviorOptions;
const responseHeadersPolicy = staticBehaviorOptions?.responseHeadersPolicy ??
new aws_cloudfront_1.ResponseHeadersPolicy(this, "StaticResponseHeadersPolicy", {
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
comment: `Nextjs Static Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.staticResponseHeadersPolicyProps,
});
return {
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
cachedMethods: aws_cloudfront_1.CachedMethods.CACHE_GET_HEAD_OPTIONS,
cachePolicy: aws_cloudfront_1.CachePolicy.CACHING_OPTIMIZED,
origin: this.staticOrigin,
responseHeadersPolicy,
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
...staticBehaviorOptions,
};
}
createDynamicBehaviorOptions() {
const dynamicBehaviorOptions = this.props.overrides?.dynamicBehaviorOptions;
// create default cache policy if not provided
const cachePolicy = dynamicBehaviorOptions?.cachePolicy ??
new aws_cloudfront_1.CachePolicy(this, "DynamicCachePolicy", {
queryStringBehavior: aws_cloudfront_1.CacheQueryStringBehavior.all(),
headerBehavior: aws_cloudfront_1.CacheHeaderBehavior.allowList("accept", "rsc", "next-router-prefetch", "next-router-state-tree", "next-url", "x-prerender-revalidate"),
cookieBehavior: aws_cloudfront_1.CacheCookieBehavior.all(),
enableAcceptEncodingBrotli: true,
enableAcceptEncodingGzip: true,
comment: `Nextjs Dynamic Cache Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.dynamicCachePolicyProps,
});
const responseHeadersPolicy = dynamicBehaviorOptions?.responseHeadersPolicy ??
new aws_cloudfront_1.ResponseHeadersPolicy(this, "DynamicResponseHeadersPolicy", {
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
comment: `Nextjs Dynamic Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.dynamicBehaviorOptions?.responseHeadersPolicy,
});
const behaviorOptions = {
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_ALL,
cachePolicy,
functionAssociations: this.dynamicCloudFrontFunctionAssociations,
origin: this.dynamicOrigin,
originRequestPolicy: this.dynamicOriginResponsePolicy,
responseHeadersPolicy,
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
...dynamicBehaviorOptions,
};
return behaviorOptions;
}
createImageBehaviorOptions() {
const imageBehaviorOptions = this.props.overrides?.imageBehaviorOptions;
// add default cache policy if not provided
const cachePolicy = imageBehaviorOptions?.cachePolicy ??
new aws_cloudfront_1.CachePolicy(this, "ImageCachePolicy", {
// SECURITY NOTE: by default we don't include cookies in cache for
// images b/c it significantly improves image perf for most sites BUT
// if you have private images locked behind auth implemented with cookies
// you need to override this.
queryStringBehavior: aws_cloudfront_1.CacheQueryStringBehavior.all(),
headerBehavior: aws_cloudfront_1.CacheHeaderBehavior.allowList("accept"),
cookieBehavior: aws_cloudfront_1.CacheCookieBehavior.none(),
enableAcceptEncodingBrotli: true,
enableAcceptEncodingGzip: true,
comment: `Nextjs Image Cache Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.imageCachePolicyProps,
});
// add default response headers policy if not provided
const responseHeadersPolicy = imageBehaviorOptions?.responseHeadersPolicy ??
new aws_cloudfront_1.ResponseHeadersPolicy(this, "ImageResponseHeadersPolicy", {
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
comment: `Nextjs Image Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.imageResponseHeadersPolicyProps,
});
return {
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
cachedMethods: aws_cloudfront_1.CachedMethods.CACHE_GET_HEAD_OPTIONS,
functionAssociations: this.dynamicCloudFrontFunctionAssociations,
origin: this.dynamicOrigin,
originRequestPolicy: this.dynamicOriginResponsePolicy,
cachePolicy,
responseHeadersPolicy,
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
...imageBehaviorOptions,
};
}
/**
* Creates or uses user specified CloudFront Distribution
*/
getDistribution() {
let distribution;
if (this.props.distribution) {
distribution = this.props.distribution;
}
else {
distribution = new aws_cloudfront_1.Distribution(this, "Distribution", {
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2021,
defaultBehavior: this.dynamicBehaviorOptions,
// best to use HTTP 2 and 3 for compatability (HTTP 2) and performance (HTTP3)
// CloudFront will choose best option for client
httpVersion: aws_cloudfront_1.HttpVersion.HTTP2_AND_3,
comment: `cdk-nextjs Distribution for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
...this.props.overrides?.distributionProps,
});
}
return distribution;
}
addDynamicBehaviors() {
// Image Behavior
this.distribution.addBehavior(this.getPathPattern("_next/image*"), this.imageBehaviorOptions.origin, this.imageBehaviorOptions);
// Root Path Behaviors
if (this.props.basePath) {
// because we already have a basePath we don't use / instead we use /base-path
this.distribution.addBehavior(this.props.basePath, this.dynamicBehaviorOptions.origin, this.dynamicBehaviorOptions);
// when basePath is set, we emulate the "default behavior" (*) for the site as `/base-path/*`
this.distribution.addBehavior(this.getPathPattern("*"), this.dynamicBehaviorOptions.origin, this.dynamicBehaviorOptions);
}
else {
// if no base path, then default behavior will handle all other paths
}
}
addStaticBehaviors() {
this.distribution.addBehavior(this.getPathPattern("_next/static*"), this.staticOrigin, this.staticBehaviorOptions);
// 22 = 25 (max) - 1 (_next/image) - 1 (_next/static) - 1 (*)
if (this.props.publicDirEntries.length >= 22) {
throw new Error(`Too many public/ files in Next.js build. CloudFront limits Distributions to 25 Cache Behaviors. See documented limit here: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/cloudfront-limits.html#limits-web-distributions. Try including all public files into 1 top level directory (i.e. static/*).`);
}
for (const publicFile of this.props.publicDirEntries) {
const pathPattern = publicFile.isDirectory
? `${publicFile.name}/*`
: publicFile.name;
if (!/^[a-zA-Z0-9_\-.*$/~"'@:+?&]+$/.test(pathPattern)) {
throw new Error(`Invalid CloudFront Distribution Cache Behavior Path Pattern: ${pathPattern}. Please see documentation here: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/distribution-web-values-specify.html#DownloadDistValuesPathPattern`);
}
const finalPathPattern = this.getPathPattern(pathPattern);
this.distribution.addBehavior(finalPathPattern, this.staticOrigin, this.staticBehaviorOptions);
}
}
/**
* Optionally prepends base path to given path pattern.
*/
getPathPattern(pathPattern) {
if (this.props.basePath) {
return `${this.props.basePath}/${pathPattern}`;
}
else {
return pathPattern;
}
}
}
exports.NextjsDistribution = NextjsDistribution;
_a = JSII_RTTI_SYMBOL_1;
NextjsDistribution[_a] = { fqn: "cdk-nextjs.NextjsDistribution", version: "0.4.10" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-distribution.js","sourceRoot":"","sources":["../src/nextjs-distribution.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Stack } from \"aws-cdk-lib\";\nimport { ICertificate } from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n  AddBehaviorOptions,\n  AllowedMethods,\n  BehaviorOptions,\n  CacheCookieBehavior,\n  CacheHeaderBehavior,\n  CachePolicy,\n  CachePolicyProps,\n  CacheQueryStringBehavior,\n  CachedMethods,\n  Function as CloudFrontFunction,\n  Distribution,\n  FunctionAssociation,\n  FunctionCode,\n  FunctionEventType,\n  HeadersFrameOption,\n  HeadersReferrerPolicy,\n  HttpVersion,\n  IOrigin,\n  IOriginRequestPolicy,\n  OriginProtocolPolicy,\n  OriginRequestPolicy,\n  ResponseHeadersPolicy,\n  ResponseHeadersPolicyProps,\n  ResponseSecurityHeadersBehavior,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {\n  FunctionUrlOrigin,\n  FunctionUrlOriginWithOACProps,\n  S3BucketOrigin,\n  VpcOrigin,\n  VpcOriginWithEndpointProps,\n} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport { ApplicationLoadBalancer } from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { IFunctionUrl } from \"aws-cdk-lib/aws-lambda\";\nimport { IBucket } from \"aws-cdk-lib/aws-s3\";\nimport { Construct } from \"constructs\";\nimport { NextjsType } from \"./constants\";\nimport { OptionalDistributionProps } from \"./generated-structs/OptionalDistributionProps\";\nimport { OptionalS3OriginBucketWithOACProps } from \"./generated-structs/OptionalS3OriginBucketWithOACProps\";\nimport { PublicDirEntry } from \"./nextjs-build/nextjs-build\";\n\nexport interface NextjsDistributionOverrides {\n  readonly distributionProps?: OptionalDistributionProps;\n  readonly imageBehaviorOptions?: AddBehaviorOptions;\n  readonly imageCachePolicyProps?: CachePolicyProps;\n  readonly imageResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;\n  readonly dynamicBehaviorOptions?: AddBehaviorOptions;\n  readonly dynamicCachePolicyProps?: CachePolicyProps;\n  readonly dynamicResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;\n  readonly dynamicFunctionUrlOriginWithOACProps?: FunctionUrlOriginWithOACProps;\n  readonly dynamicVpcOriginWithEndpointProps?: VpcOriginWithEndpointProps;\n  readonly staticBehaviorOptions?: AddBehaviorOptions;\n  readonly staticResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;\n  readonly s3BucketOriginProps?: OptionalS3OriginBucketWithOACProps;\n}\n\nexport interface NextjsDistributionProps {\n  /**\n   * Bucket containing static assets.\n   * Must be provided if you want to serve static files.\n   */\n  readonly assetsBucket: IBucket;\n  readonly basePath?: string;\n  /**\n   * Optional but only applicable for `NextjsType.GLOBAL_CONTAINERS`\n   */\n  readonly certificate?: ICertificate;\n  readonly distribution?: Distribution;\n  /**\n   * Required if `NextjsType.GLOBAL_FUNCTIONS`\n   */\n  readonly functionUrl?: IFunctionUrl;\n  /**\n   * Required if `NextjsType.GLOBAL_CONTAINERS` or `NextjsType.REGIONAL_CONTAINERS`\n   */\n  readonly loadBalancer?: ApplicationLoadBalancer;\n  readonly nextjsType: NextjsType;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsDistributionOverrides;\n  /**\n   * Entries (files/directories) within Next.js app's public directory. Used to\n   * add static behaviors to distribution.\n   */\n  readonly publicDirEntries: PublicDirEntry[];\n}\n\nexport class NextjsDistribution extends Construct {\n  distribution: Distribution;\n\n  private props: NextjsDistributionProps;\n  /**\n   * Common security headers applied by default to all origins\n   * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-response-headers-policies.html#managed-response-headers-policies-security\n   */\n  private commonSecurityHeadersBehavior: ResponseSecurityHeadersBehavior = {\n    contentTypeOptions: { override: false },\n    frameOptions: {\n      frameOption: HeadersFrameOption.SAMEORIGIN,\n      override: false,\n    },\n    referrerPolicy: {\n      override: false,\n      referrerPolicy: HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,\n    },\n\n    strictTransportSecurity: {\n      accessControlMaxAge: Duration.days(365),\n      includeSubdomains: true,\n      override: false,\n      preload: true,\n    },\n    xssProtection: { override: false, protection: true, modeBlock: true },\n  };\n  private staticOrigin: IOrigin;\n  private dynamicOrigin: IOrigin;\n  private dynamicOriginResponsePolicy: IOriginRequestPolicy;\n  private dynamicCloudFrontFunctionAssociations: FunctionAssociation[];\n  private isFunctionCompute: boolean;\n  private staticBehaviorOptions: BehaviorOptions;\n  private dynamicBehaviorOptions: BehaviorOptions;\n  private imageBehaviorOptions: BehaviorOptions;\n\n  constructor(scope: Construct, id: string, props: NextjsDistributionProps) {\n    super(scope, id);\n    this.props = props;\n    this.staticOrigin = this.createStaticOrigin();\n    this.isFunctionCompute = props.nextjsType === NextjsType.GLOBAL_FUNCTIONS;\n    this.dynamicOrigin = this.createDynamicOrigin();\n    this.dynamicOriginResponsePolicy = this.createDynamicOriginRequestPolicy();\n    this.dynamicCloudFrontFunctionAssociations =\n      this.createDynamicCloudFrontFunctionAssociations();\n    this.staticBehaviorOptions = this.createStaticBehaviorOptions();\n    this.dynamicBehaviorOptions = this.createDynamicBehaviorOptions();\n    this.imageBehaviorOptions = this.createImageBehaviorOptions();\n    this.distribution = this.getDistribution();\n    this.addStaticBehaviors();\n    this.addDynamicBehaviors();\n  }\n\n  private createStaticOrigin(): IOrigin {\n    const s3Origin = S3BucketOrigin.withOriginAccessControl(\n      this.props.assetsBucket,\n      this.props.overrides?.s3BucketOriginProps,\n    );\n    return s3Origin;\n  }\n  private createDynamicOrigin(): IOrigin {\n    if (this.isFunctionCompute) {\n      if (!this.props.functionUrl)\n        throw new Error(\"Missing NextjsDistributionProps.functionUrl\");\n      return FunctionUrlOrigin.withOriginAccessControl(\n        this.props.functionUrl,\n        this.props.overrides?.dynamicFunctionUrlOriginWithOACProps,\n      );\n    } else {\n      const loadBalancer = this.props.loadBalancer;\n      if (!loadBalancer)\n        throw new Error(\"Missing NextjsDistributionProps.loadBalancer\");\n      return VpcOrigin.withApplicationLoadBalancer(loadBalancer, {\n        protocolPolicy: this.props.certificate\n          ? OriginProtocolPolicy.HTTPS_ONLY\n          : OriginProtocolPolicy.HTTP_ONLY,\n        ...this.props.overrides?.dynamicVpcOriginWithEndpointProps,\n      });\n    }\n  }\n  /**\n   * Lambda Function URLs \"expect the `Host` header to contain the origin domain\n   * name, not the domain name of the CloudFront distribution.\"\n   * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-origin-request-policies.html#managed-origin-request-policy-all-viewer-except-host-header\n   */\n  private createDynamicOriginRequestPolicy(): IOriginRequestPolicy {\n    return this.isFunctionCompute\n      ? OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER\n      : OriginRequestPolicy.ALL_VIEWER;\n  }\n  /**\n   * Ensures Next.js `request.url` will be correct domain instead of URL of\n   * compute (Lambda or Fargate)\n   * @see https://open-next.js.org/advanced/workaround#workaround-set-x-forwarded-host-header-aws-specific\n   */\n  private createDynamicCloudFrontFunctionAssociations(): FunctionAssociation[] {\n    const associations: FunctionAssociation[] = [];\n    if (this.isFunctionCompute) {\n      const cloudFrontFn = new CloudFrontFunction(this, \"CloudFrontFn\", {\n        code: FunctionCode.fromInline(`\n          function handler(event) {\n            var request = event.request;\n            request.headers[\"x-forwarded-host\"] = request.headers.host;\n            return request;\n          }\n          `),\n      });\n      associations.push({\n        eventType: FunctionEventType.VIEWER_REQUEST,\n        function: cloudFrontFn,\n      });\n    }\n    return associations;\n  }\n  private createStaticBehaviorOptions(): BehaviorOptions {\n    const staticBehaviorOptions = this.props.overrides?.staticBehaviorOptions;\n    const responseHeadersPolicy =\n      staticBehaviorOptions?.responseHeadersPolicy ??\n      new ResponseHeadersPolicy(this, \"StaticResponseHeadersPolicy\", {\n        securityHeadersBehavior: this.commonSecurityHeadersBehavior,\n        comment: `Nextjs Static Response Headers Policy for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.staticResponseHeadersPolicyProps,\n      });\n    return {\n      allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n      cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n      origin: this.staticOrigin,\n      responseHeadersPolicy,\n      viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      ...staticBehaviorOptions,\n    };\n  }\n  private createDynamicBehaviorOptions(): BehaviorOptions {\n    const dynamicBehaviorOptions = this.props.overrides?.dynamicBehaviorOptions;\n    // create default cache policy if not provided\n    const cachePolicy =\n      dynamicBehaviorOptions?.cachePolicy ??\n      new CachePolicy(this, \"DynamicCachePolicy\", {\n        queryStringBehavior: CacheQueryStringBehavior.all(),\n        headerBehavior: CacheHeaderBehavior.allowList(\n          \"accept\",\n          \"rsc\",\n          \"next-router-prefetch\",\n          \"next-router-state-tree\",\n          \"next-url\",\n          \"x-prerender-revalidate\",\n        ),\n        cookieBehavior: CacheCookieBehavior.all(),\n        enableAcceptEncodingBrotli: true,\n        enableAcceptEncodingGzip: true,\n        comment: `Nextjs Dynamic Cache Policy for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.dynamicCachePolicyProps,\n      });\n    const responseHeadersPolicy =\n      dynamicBehaviorOptions?.responseHeadersPolicy ??\n      new ResponseHeadersPolicy(this, \"DynamicResponseHeadersPolicy\", {\n        securityHeadersBehavior: this.commonSecurityHeadersBehavior,\n        comment: `Nextjs Dynamic Response Headers Policy for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.dynamicBehaviorOptions?.responseHeadersPolicy,\n      });\n    const behaviorOptions: BehaviorOptions = {\n      allowedMethods: AllowedMethods.ALLOW_ALL,\n      cachePolicy,\n      functionAssociations: this.dynamicCloudFrontFunctionAssociations,\n      origin: this.dynamicOrigin,\n      originRequestPolicy: this.dynamicOriginResponsePolicy,\n      responseHeadersPolicy,\n      viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      ...dynamicBehaviorOptions,\n    };\n    return behaviorOptions;\n  }\n  private createImageBehaviorOptions(): BehaviorOptions {\n    const imageBehaviorOptions = this.props.overrides?.imageBehaviorOptions;\n    // add default cache policy if not provided\n    const cachePolicy =\n      imageBehaviorOptions?.cachePolicy ??\n      new CachePolicy(this, \"ImageCachePolicy\", {\n        // SECURITY NOTE: by default we don't include cookies in cache for\n        // images b/c it significantly improves image perf for most sites BUT\n        // if you have private images locked behind auth implemented with cookies\n        // you need to override this.\n        queryStringBehavior: CacheQueryStringBehavior.all(),\n        headerBehavior: CacheHeaderBehavior.allowList(\"accept\"),\n        cookieBehavior: CacheCookieBehavior.none(),\n        enableAcceptEncodingBrotli: true,\n        enableAcceptEncodingGzip: true,\n        comment: `Nextjs Image Cache Policy for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.imageCachePolicyProps,\n      });\n    // add default response headers policy if not provided\n    const responseHeadersPolicy =\n      imageBehaviorOptions?.responseHeadersPolicy ??\n      new ResponseHeadersPolicy(this, \"ImageResponseHeadersPolicy\", {\n        securityHeadersBehavior: this.commonSecurityHeadersBehavior,\n        comment: `Nextjs Image Response Headers Policy for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.imageResponseHeadersPolicyProps,\n      });\n    return {\n      allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n      functionAssociations: this.dynamicCloudFrontFunctionAssociations,\n      origin: this.dynamicOrigin,\n      originRequestPolicy: this.dynamicOriginResponsePolicy,\n      cachePolicy,\n      responseHeadersPolicy,\n      viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      ...imageBehaviorOptions,\n    };\n  }\n  /**\n   * Creates or uses user specified CloudFront Distribution\n   */\n  private getDistribution(): Distribution {\n    let distribution: Distribution;\n    if (this.props.distribution) {\n      distribution = this.props.distribution;\n    } else {\n      distribution = new Distribution(this, \"Distribution\", {\n        minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2021,\n        defaultBehavior: this.dynamicBehaviorOptions,\n        // best to use HTTP 2 and 3 for compatability (HTTP 2) and performance (HTTP3)\n        // CloudFront will choose best option for client\n        httpVersion: HttpVersion.HTTP2_AND_3,\n        comment: `cdk-nextjs Distribution for ${Stack.of(this).stackName}`,\n        ...this.props.overrides?.distributionProps,\n      });\n    }\n    return distribution;\n  }\n  private addDynamicBehaviors() {\n    // Image Behavior\n    this.distribution.addBehavior(\n      this.getPathPattern(\"_next/image*\"),\n      this.imageBehaviorOptions.origin,\n      this.imageBehaviorOptions,\n    );\n    // Root Path Behaviors\n    if (this.props.basePath) {\n      // because we already have a basePath we don't use / instead we use /base-path\n      this.distribution.addBehavior(\n        this.props.basePath,\n        this.dynamicBehaviorOptions.origin,\n        this.dynamicBehaviorOptions,\n      );\n      // when basePath is set, we emulate the \"default behavior\" (*) for the site as `/base-path/*`\n      this.distribution.addBehavior(\n        this.getPathPattern(\"*\"),\n        this.dynamicBehaviorOptions.origin,\n        this.dynamicBehaviorOptions,\n      );\n    } else {\n      // if no base path, then default behavior will handle all other paths\n    }\n  }\n  private addStaticBehaviors() {\n    this.distribution.addBehavior(\n      this.getPathPattern(\"_next/static*\"),\n      this.staticOrigin,\n      this.staticBehaviorOptions,\n    );\n    // 22 = 25 (max) - 1 (_next/image) - 1 (_next/static) - 1 (*)\n    if (this.props.publicDirEntries.length >= 22) {\n      throw new Error(\n        `Too many public/ files in Next.js build. CloudFront limits Distributions to 25 Cache Behaviors. See documented limit here: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/cloudfront-limits.html#limits-web-distributions. Try including all public files into 1 top level directory (i.e. static/*).`,\n      );\n    }\n    for (const publicFile of this.props.publicDirEntries) {\n      const pathPattern = publicFile.isDirectory\n        ? `${publicFile.name}/*`\n        : publicFile.name;\n      if (!/^[a-zA-Z0-9_\\-.*$/~\"'@:+?&]+$/.test(pathPattern)) {\n        throw new Error(\n          `Invalid CloudFront Distribution Cache Behavior Path Pattern: ${pathPattern}. Please see documentation here: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/distribution-web-values-specify.html#DownloadDistValuesPathPattern`,\n        );\n      }\n      const finalPathPattern = this.getPathPattern(pathPattern);\n      this.distribution.addBehavior(\n        finalPathPattern,\n        this.staticOrigin,\n        this.staticBehaviorOptions,\n      );\n    }\n  }\n  /**\n   * Optionally prepends base path to given path pattern.\n   */\n  private getPathPattern(pathPattern: string) {\n    if (this.props.basePath) {\n      return `${this.props.basePath}/${pathPattern}`;\n    } else {\n      return pathPattern;\n    }\n  }\n}\n"]}