cdk-nextjs 0.1.0 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +13834 -0
- package/.prettierrc +0 -0
- package/API.md +9655 -0
- package/LICENSE +202 -0
- package/README.md +164 -2
- package/THIRD-PARTY-LICENSES.md +31 -0
- package/assets/lambdas/assets-deployment/assets-deployment.lambda/assets-deployment.Dockerfile +18 -0
- package/assets/lambdas/assets-deployment/assets-deployment.lambda/index.js +8833 -0
- package/assets/lambdas/revalidate/revalidate.lambda/index.js +67 -0
- package/assets/lambdas/sign-fn-url/sign-fn-url.edge-lambda/index.js +2024 -0
- package/docs/breaking-changes.md +8 -0
- package/docs/cdk-nextjs-NextjsGlobalContainers.png +0 -0
- package/docs/cdk-nextjs-NextjsGlobalFunctions.png +0 -0
- package/docs/cdk-nextjs-NextjsRegionalContainers.png +0 -0
- package/docs/cdk-nextjs.drawio +163 -0
- package/docs/notes.md +26 -0
- package/examples/README.md +15 -0
- package/lib/common.d.ts +23 -0
- package/lib/common.js +28 -0
- package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.d.ts +106 -0
- package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.js +3 -0
- package/lib/generated-structs/OptionalCloudFrontFunctionProps.d.ts +43 -0
- package/lib/generated-structs/OptionalCloudFrontFunctionProps.js +3 -0
- package/lib/generated-structs/OptionalClusterProps.d.ts +62 -0
- package/lib/generated-structs/OptionalClusterProps.js +3 -0
- package/lib/generated-structs/OptionalDistributionProps.d.ts +155 -0
- package/lib/generated-structs/OptionalDistributionProps.js +3 -0
- package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +124 -0
- package/lib/generated-structs/OptionalDockerImageAssetProps.js +3 -0
- package/lib/generated-structs/OptionalDockerImageFunctionProps.d.ts +399 -0
- package/lib/generated-structs/OptionalDockerImageFunctionProps.js +3 -0
- package/lib/generated-structs/OptionalEdgeFunctionProps.d.ts +428 -0
- package/lib/generated-structs/OptionalEdgeFunctionProps.js +3 -0
- package/lib/generated-structs/OptionalFunctionProps.d.ts +422 -0
- package/lib/generated-structs/OptionalFunctionProps.js +3 -0
- package/lib/generated-structs/OptionalFunctionUrlProps.d.ts +30 -0
- package/lib/generated-structs/OptionalFunctionUrlProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +40 -0
- package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +26 -0
- package/lib/generated-structs/OptionalNextjsBuildProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +39 -0
- package/lib/generated-structs/OptionalNextjsContainersProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsDistributionProps.d.ts +45 -0
- package/lib/generated-structs/OptionalNextjsDistributionProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsFileSystemProps.d.ts +10 -0
- package/lib/generated-structs/OptionalNextjsFileSystemProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +11 -0
- package/lib/generated-structs/OptionalNextjsInvalidationProps.js +3 -0
- package/lib/generated-structs/OptionalNextjsVpcProps.d.ts +16 -0
- package/lib/generated-structs/OptionalNextjsVpcProps.js +3 -0
- package/lib/generated-structs/OptionalS3OriginBucketWithOACProps.d.ts +70 -0
- package/lib/generated-structs/OptionalS3OriginBucketWithOACProps.js +3 -0
- package/lib/generated-structs/OptionalVpcProps.d.ts +224 -0
- package/lib/generated-structs/OptionalVpcProps.js +3 -0
- package/lib/index.d.ts +35 -0
- package/lib/index.js +32 -0
- package/lib/lambdas/assets-deployment/assets-deployment-function.d.ts +13 -0
- package/lib/lambdas/assets-deployment/assets-deployment-function.js +22 -0
- package/lib/lambdas/assets-deployment/assets-deployment.lambda.d.ts +2 -0
- package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +63 -0
- package/lib/lambdas/assets-deployment/common.d.ts +8 -0
- package/lib/lambdas/assets-deployment/common.js +32 -0
- package/lib/lambdas/assets-deployment/fs-to-fs.d.ts +2 -0
- package/lib/lambdas/assets-deployment/fs-to-fs.js +9 -0
- package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +2 -0
- package/lib/lambdas/assets-deployment/fs-to-s3.js +45 -0
- package/lib/lambdas/assets-deployment/prune-s3.d.ts +15 -0
- package/lib/lambdas/assets-deployment/prune-s3.js +42 -0
- package/lib/lambdas/assets-deployment/s3.d.ts +2 -0
- package/lib/lambdas/assets-deployment/s3.js +7 -0
- package/lib/lambdas/assets-deployment/utils.d.ts +18 -0
- package/lib/lambdas/assets-deployment/utils.js +35 -0
- package/lib/lambdas/revalidate/revalidate-function.d.ts +13 -0
- package/lib/lambdas/revalidate/revalidate-function.js +22 -0
- package/lib/lambdas/revalidate/revalidate.lambda.d.ts +2 -0
- package/lib/lambdas/revalidate/revalidate.lambda.js +53 -0
- package/lib/lambdas/sign-fn-url/sign-fn-url-function.d.ts +13 -0
- package/lib/lambdas/sign-fn-url/sign-fn-url-function.js +23 -0
- package/lib/lambdas/sign-fn-url/sign-fn-url.edge-lambda.d.ts +9 -0
- package/lib/lambdas/sign-fn-url/sign-fn-url.edge-lambda.js +35 -0
- package/lib/lambdas/sign-fn-url/sign-request.d.ts +28 -0
- package/lib/lambdas/sign-fn-url/sign-request.js +119 -0
- package/lib/lambdas/sign-fn-url/sign-request.test.d.ts +1 -0
- package/lib/lambdas/sign-fn-url/sign-request.test.js +129 -0
- package/lib/nextjs-assets-deployment.d.ts +116 -0
- package/lib/nextjs-assets-deployment.js +93 -0
- package/lib/nextjs-build/add-cache-handler.d.ts +1 -0
- package/lib/nextjs-build/add-cache-handler.js +23 -0
- package/lib/nextjs-build/add-cache-handler.mjs +18 -0
- package/lib/nextjs-build/builder.Dockerfile +25 -0
- package/lib/nextjs-build/cache-handler.cjs +21080 -0
- package/lib/nextjs-build/cache-handler.d.ts +6 -0
- package/lib/nextjs-build/cache-handler.js +22 -0
- package/lib/nextjs-build/global-containers.Dockerfile +45 -0
- package/lib/nextjs-build/global-functions.Dockerfile +46 -0
- package/lib/nextjs-build/nextjs-build.d.ts +150 -0
- package/lib/nextjs-build/nextjs-build.js +220 -0
- package/lib/nextjs-build/regional-containers.Dockerfile +45 -0
- package/lib/nextjs-build/symlink-full-route-cache.d.ts +1 -0
- package/lib/nextjs-build/symlink-full-route-cache.js +37 -0
- package/lib/nextjs-build/symlink-full-route-cache.mjs +23 -0
- package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +8 -0
- package/lib/nextjs-compute/nextjs-compute-base-props.js +3 -0
- package/lib/nextjs-compute/nextjs-containers.d.ts +43 -0
- package/lib/nextjs-compute/nextjs-containers.js +149 -0
- package/lib/nextjs-compute/nextjs-functions.d.ts +23 -0
- package/lib/nextjs-compute/nextjs-functions.js +57 -0
- package/lib/nextjs-distribution.d.ts +107 -0
- package/lib/nextjs-distribution.js +331 -0
- package/lib/nextjs-file-system.d.ts +42 -0
- package/lib/nextjs-file-system.js +74 -0
- package/lib/nextjs-invalidation.d.ts +19 -0
- package/lib/nextjs-invalidation.js +52 -0
- package/lib/nextjs-revalidation.d.ts +30 -0
- package/lib/nextjs-revalidation.js +65 -0
- package/lib/nextjs-static-assets.d.ts +21 -0
- package/lib/nextjs-static-assets.js +33 -0
- package/lib/nextjs-vpc.d.ts +42 -0
- package/lib/nextjs-vpc.js +64 -0
- package/lib/root-constructs/nextjs-base-overrides.d.ts +26 -0
- package/lib/root-constructs/nextjs-base-overrides.js +3 -0
- package/lib/root-constructs/nextjs-base-props.d.ts +56 -0
- package/lib/root-constructs/nextjs-base-props.js +3 -0
- package/lib/root-constructs/nextjs-global-containers.d.ts +74 -0
- package/lib/root-constructs/nextjs-global-containers.js +125 -0
- package/lib/root-constructs/nextjs-global-functions.d.ts +76 -0
- package/lib/root-constructs/nextjs-global-functions.js +131 -0
- package/lib/root-constructs/nextjs-regional-containers.d.ts +43 -0
- package/lib/root-constructs/nextjs-regional-containers.js +92 -0
- package/package.json +163 -2
- package/dist/index.js +0 -1
|
@@ -0,0 +1,331 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.NextjsDistribution = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
7
|
+
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
|
|
8
|
+
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
|
|
9
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
10
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
11
|
+
const constructs_1 = require("constructs");
|
|
12
|
+
const common_1 = require("./common");
|
|
13
|
+
const sign_fn_url_function_1 = require("./lambdas/sign-fn-url/sign-fn-url-function");
|
|
14
|
+
class NextjsDistribution extends constructs_1.Construct {
|
|
15
|
+
constructor(scope, id, props) {
|
|
16
|
+
super(scope, id);
|
|
17
|
+
/**
|
|
18
|
+
* Common security headers applied by default to all origins
|
|
19
|
+
* @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-response-headers-policies.html#managed-response-headers-policies-security
|
|
20
|
+
*/
|
|
21
|
+
this.commonSecurityHeadersBehavior = {
|
|
22
|
+
contentTypeOptions: { override: false },
|
|
23
|
+
frameOptions: {
|
|
24
|
+
frameOption: aws_cloudfront_1.HeadersFrameOption.SAMEORIGIN,
|
|
25
|
+
override: false,
|
|
26
|
+
},
|
|
27
|
+
referrerPolicy: {
|
|
28
|
+
override: false,
|
|
29
|
+
referrerPolicy: aws_cloudfront_1.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,
|
|
30
|
+
},
|
|
31
|
+
strictTransportSecurity: {
|
|
32
|
+
accessControlMaxAge: aws_cdk_lib_1.Duration.days(365),
|
|
33
|
+
includeSubdomains: true,
|
|
34
|
+
override: false,
|
|
35
|
+
preload: true,
|
|
36
|
+
},
|
|
37
|
+
xssProtection: { override: false, protection: true, modeBlock: true },
|
|
38
|
+
};
|
|
39
|
+
this.props = props;
|
|
40
|
+
this.staticOrigin = this.createStaticOrigin();
|
|
41
|
+
this.isFunctionCompute = props.nextjsType === common_1.NextjsType.GLOBAL_FUNCTIONS;
|
|
42
|
+
this.dynamicOrigin = this.createDynamicOrigin();
|
|
43
|
+
this.dynamicOriginResponsePolicy = this.createDynamicOriginRequestPolicy();
|
|
44
|
+
this.dynamicCloudFrontFunctionAssociations =
|
|
45
|
+
this.createDynamicCloudFrontFunctionAssociations();
|
|
46
|
+
if (this.isFunctionCompute) {
|
|
47
|
+
this.edgeLambdas = this.createEdgeLambdas();
|
|
48
|
+
}
|
|
49
|
+
this.staticBehaviorOptions = this.createStaticBehaviorOptions();
|
|
50
|
+
this.dynamicBehaviorOptions = this.createDynamicBehaviorOptions();
|
|
51
|
+
this.imageBehaviorOptions = this.createImageBehaviorOptions();
|
|
52
|
+
this.distribution = this.getDistribution();
|
|
53
|
+
this.addStaticBehaviors();
|
|
54
|
+
this.addDynamicBehaviors();
|
|
55
|
+
if (this.isFunctionCompute) {
|
|
56
|
+
// this.addLambdaOac(); // TODO: wait for POST body encryption feature for Lambda OAC
|
|
57
|
+
}
|
|
58
|
+
new aws_cdk_lib_1.CfnOutput(this, "DistributionDomainName", {
|
|
59
|
+
value: this.distribution.domainName,
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
createStaticOrigin() {
|
|
63
|
+
const s3Origin = aws_cloudfront_origins_1.S3BucketOrigin.withOriginAccessControl(this.props.assetsBucket, this.props.overrides?.s3BucketOriginProps);
|
|
64
|
+
return s3Origin;
|
|
65
|
+
}
|
|
66
|
+
createDynamicOrigin() {
|
|
67
|
+
let protocolPolicy;
|
|
68
|
+
if (this.isFunctionCompute) {
|
|
69
|
+
protocolPolicy = aws_cloudfront_1.OriginProtocolPolicy.HTTPS_ONLY;
|
|
70
|
+
}
|
|
71
|
+
else {
|
|
72
|
+
protocolPolicy = this.props.certificate
|
|
73
|
+
? aws_cloudfront_1.OriginProtocolPolicy.HTTPS_ONLY
|
|
74
|
+
: aws_cloudfront_1.OriginProtocolPolicy.HTTP_ONLY;
|
|
75
|
+
}
|
|
76
|
+
// WAITING FOR CLOUDFRONT FEATURE: CloudFront Lambda Function URL OAC
|
|
77
|
+
// doesn't support POST with body since CloudFront doesn't include sha256
|
|
78
|
+
// hash of body. see: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/private-content-restricting-access-to-lambda.html
|
|
79
|
+
// We could monkey patch `fetch` to include sha256 in request for server
|
|
80
|
+
// actions, but better solution is for cloudfront to support it
|
|
81
|
+
//
|
|
82
|
+
// TODO: use L2 construct when released: https://github.com/aws/aws-cdk/issues/31629
|
|
83
|
+
/**
|
|
84
|
+
* Given stack id: "arn:aws:cloudformation:us-east-1:905418358903:stack/lh-stickb-idp/4bf74be0-e880-11ee-aea9-0affc6185b25",
|
|
85
|
+
* returns "4bf74be0"
|
|
86
|
+
*/
|
|
87
|
+
// const uniqueStackIdPart = Fn.select(
|
|
88
|
+
// 0,
|
|
89
|
+
// Fn.split("-", Fn.select(2, Fn.split("/", `${Aws.STACK_ID}`))),
|
|
90
|
+
// );
|
|
91
|
+
// const lambdaOac = new CfnOriginAccessControl(this, "OAC", {
|
|
92
|
+
// originAccessControlConfig: {
|
|
93
|
+
// name: `OAC-Lambda-${uniqueStackIdPart}`,
|
|
94
|
+
// originAccessControlOriginType: "lambda",
|
|
95
|
+
// signingBehavior: "always",
|
|
96
|
+
// signingProtocol: "sigv4",
|
|
97
|
+
// },
|
|
98
|
+
// });
|
|
99
|
+
// const cfnDistribution = this.distribution.node
|
|
100
|
+
// .defaultChild as CfnDistribution;
|
|
101
|
+
// cfnDistribution.addPropertyOverride(
|
|
102
|
+
// "DistributionConfig.Origins.0.OriginAccessControlId",
|
|
103
|
+
// lambdaOac.getAtt("Id"),
|
|
104
|
+
// );
|
|
105
|
+
return new aws_cloudfront_origins_1.HttpOrigin(aws_cdk_lib_1.Fn.parseDomainName(this.props.dynamicUrl), {
|
|
106
|
+
protocolPolicy,
|
|
107
|
+
...this.props.overrides?.dynamicHttpOriginProps,
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Lambda Function URLs "expect the `Host` header to contain the origin domain
|
|
112
|
+
* name, not the domain name of the CloudFront distribution."
|
|
113
|
+
* @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-origin-request-policies.html#managed-origin-request-policy-all-viewer-except-host-header
|
|
114
|
+
*/
|
|
115
|
+
createDynamicOriginRequestPolicy() {
|
|
116
|
+
return this.isFunctionCompute
|
|
117
|
+
? aws_cloudfront_1.OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER
|
|
118
|
+
: aws_cloudfront_1.OriginRequestPolicy.ALL_VIEWER;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Ensures Next.js `request.url` will be correct domain instead of URL of
|
|
122
|
+
* compute option (App Runner, Fargate, or Lambda)
|
|
123
|
+
* @see https://open-next.js.org/advanced/workaround#workaround-set-x-forwarded-host-header-aws-specific
|
|
124
|
+
*/
|
|
125
|
+
createDynamicCloudFrontFunctionAssociations() {
|
|
126
|
+
const associations = [];
|
|
127
|
+
if (this.isFunctionCompute) {
|
|
128
|
+
const cloudFrontFn = new aws_cloudfront_1.Function(this, "CloudFrontFn", {
|
|
129
|
+
code: aws_cloudfront_1.FunctionCode.fromInline(`
|
|
130
|
+
function handler(event) {
|
|
131
|
+
var request = event.request;
|
|
132
|
+
request.headers["x-forwarded-host"] = request.headers.host;
|
|
133
|
+
return request;
|
|
134
|
+
}
|
|
135
|
+
`),
|
|
136
|
+
});
|
|
137
|
+
associations.push({
|
|
138
|
+
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST,
|
|
139
|
+
function: cloudFrontFn,
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
return associations;
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Required to sign requests so that we can use IAM_AUTH for Lambda Function URL
|
|
146
|
+
* to prevent public access. Once CloudFront Lambda OAC is released, we can
|
|
147
|
+
* use infra configuration for this instead of custom code.
|
|
148
|
+
*/
|
|
149
|
+
createEdgeLambdas() {
|
|
150
|
+
if (!this.props.functionArn)
|
|
151
|
+
throw new Error("functionArn is required");
|
|
152
|
+
const edgeFn = new sign_fn_url_function_1.SignFnUrlFunction(this, "SignFnUrl", {
|
|
153
|
+
currentVersionOptions: {
|
|
154
|
+
retryAttempts: 0, // fail fast when trying to delete b/c replicated functions take long time to delete
|
|
155
|
+
},
|
|
156
|
+
initialPolicy: [
|
|
157
|
+
new aws_iam_1.PolicyStatement({
|
|
158
|
+
actions: ["lambda:InvokeFunctionUrl"],
|
|
159
|
+
resources: [this.props.functionArn],
|
|
160
|
+
}),
|
|
161
|
+
],
|
|
162
|
+
code: aws_lambda_1.Code.fromInline("export function handler() {}"),
|
|
163
|
+
handler: "handler",
|
|
164
|
+
runtime: aws_lambda_1.Runtime.NODEJS_LATEST,
|
|
165
|
+
...this.props.overrides?.edgeFunctionProps,
|
|
166
|
+
});
|
|
167
|
+
edgeFn.currentVersion.grantInvoke(new aws_iam_1.ServicePrincipal("edgelambda.amazonaws.com"));
|
|
168
|
+
edgeFn.currentVersion.grantInvoke(new aws_iam_1.ServicePrincipal("lambda.amazonaws.com"));
|
|
169
|
+
return [
|
|
170
|
+
{
|
|
171
|
+
eventType: aws_cloudfront_1.LambdaEdgeEventType.ORIGIN_REQUEST,
|
|
172
|
+
functionVersion: edgeFn.currentVersion,
|
|
173
|
+
includeBody: true,
|
|
174
|
+
},
|
|
175
|
+
];
|
|
176
|
+
}
|
|
177
|
+
createStaticBehaviorOptions() {
|
|
178
|
+
const staticBehaviorOptions = this.props.overrides?.staticBehaviorOptions;
|
|
179
|
+
const responseHeadersPolicy = staticBehaviorOptions?.responseHeadersPolicy ??
|
|
180
|
+
new aws_cloudfront_1.ResponseHeadersPolicy(this, "StaticResponseHeadersPolicy", {
|
|
181
|
+
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
|
|
182
|
+
comment: `Nextjs Static Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
183
|
+
...this.props.overrides?.staticResponseHeadersPolicyProps,
|
|
184
|
+
});
|
|
185
|
+
return {
|
|
186
|
+
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
|
|
187
|
+
cachedMethods: aws_cloudfront_1.CachedMethods.CACHE_GET_HEAD_OPTIONS,
|
|
188
|
+
cachePolicy: aws_cloudfront_1.CachePolicy.CACHING_OPTIMIZED,
|
|
189
|
+
origin: this.staticOrigin,
|
|
190
|
+
responseHeadersPolicy,
|
|
191
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
192
|
+
...staticBehaviorOptions,
|
|
193
|
+
};
|
|
194
|
+
}
|
|
195
|
+
createDynamicBehaviorOptions() {
|
|
196
|
+
const dynamicBehaviorOptions = this.props.overrides?.dynamicBehaviorOptions;
|
|
197
|
+
// create default cache policy if not provided
|
|
198
|
+
const cachePolicy = dynamicBehaviorOptions?.cachePolicy ??
|
|
199
|
+
new aws_cloudfront_1.CachePolicy(this, "DynamicCachePolicy", {
|
|
200
|
+
queryStringBehavior: aws_cloudfront_1.CacheQueryStringBehavior.all(),
|
|
201
|
+
headerBehavior: aws_cloudfront_1.CacheHeaderBehavior.allowList("accept", "rsc", "next-router-prefetch", "next-router-state-tree", "next-url", "x-prerender-revalidate"),
|
|
202
|
+
cookieBehavior: aws_cloudfront_1.CacheCookieBehavior.all(),
|
|
203
|
+
enableAcceptEncodingBrotli: true,
|
|
204
|
+
enableAcceptEncodingGzip: true,
|
|
205
|
+
comment: `Nextjs Dynamic Cache Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
206
|
+
...this.props.overrides?.dynamicCachePolicyProps,
|
|
207
|
+
});
|
|
208
|
+
const responseHeadersPolicy = dynamicBehaviorOptions?.responseHeadersPolicy ??
|
|
209
|
+
new aws_cloudfront_1.ResponseHeadersPolicy(this, "DynamicResponseHeadersPolicy", {
|
|
210
|
+
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
|
|
211
|
+
comment: `Nextjs Dynamic Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
212
|
+
...this.props.overrides?.dynamicBehaviorOptions?.responseHeadersPolicy,
|
|
213
|
+
});
|
|
214
|
+
const behaviorOptions = {
|
|
215
|
+
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_ALL,
|
|
216
|
+
cachePolicy,
|
|
217
|
+
edgeLambdas: this.edgeLambdas,
|
|
218
|
+
functionAssociations: this.dynamicCloudFrontFunctionAssociations,
|
|
219
|
+
origin: this.dynamicOrigin,
|
|
220
|
+
originRequestPolicy: this.dynamicOriginResponsePolicy,
|
|
221
|
+
responseHeadersPolicy,
|
|
222
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
223
|
+
...dynamicBehaviorOptions,
|
|
224
|
+
};
|
|
225
|
+
return behaviorOptions;
|
|
226
|
+
}
|
|
227
|
+
createImageBehaviorOptions() {
|
|
228
|
+
const imageBehaviorOptions = this.props.overrides?.imageBehaviorOptions;
|
|
229
|
+
// add default cache policy if not provided
|
|
230
|
+
const cachePolicy = imageBehaviorOptions?.cachePolicy ??
|
|
231
|
+
new aws_cloudfront_1.CachePolicy(this, "ImageCachePolicy", {
|
|
232
|
+
// SECURITY NOTE: by default we don't include cookies in cache for
|
|
233
|
+
// images b/c it significantly improves image perf for most sites BUT
|
|
234
|
+
// if you have private images locked behind auth implemented with cookies
|
|
235
|
+
// you need to override this.
|
|
236
|
+
queryStringBehavior: aws_cloudfront_1.CacheQueryStringBehavior.all(),
|
|
237
|
+
headerBehavior: aws_cloudfront_1.CacheHeaderBehavior.allowList("accept"),
|
|
238
|
+
cookieBehavior: aws_cloudfront_1.CacheCookieBehavior.none(),
|
|
239
|
+
enableAcceptEncodingBrotli: true,
|
|
240
|
+
enableAcceptEncodingGzip: true,
|
|
241
|
+
comment: `Nextjs Image Cache Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
242
|
+
...this.props.overrides?.imageCachePolicyProps,
|
|
243
|
+
});
|
|
244
|
+
// add default response headers policy if not provided
|
|
245
|
+
const responseHeadersPolicy = imageBehaviorOptions?.responseHeadersPolicy ??
|
|
246
|
+
new aws_cloudfront_1.ResponseHeadersPolicy(this, "ImageResponseHeadersPolicy", {
|
|
247
|
+
securityHeadersBehavior: this.commonSecurityHeadersBehavior,
|
|
248
|
+
comment: `Nextjs Image Response Headers Policy for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
249
|
+
...this.props.overrides?.imageResponseHeadersPolicyProps,
|
|
250
|
+
});
|
|
251
|
+
return {
|
|
252
|
+
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
|
|
253
|
+
cachedMethods: aws_cloudfront_1.CachedMethods.CACHE_GET_HEAD_OPTIONS,
|
|
254
|
+
edgeLambdas: this.edgeLambdas,
|
|
255
|
+
functionAssociations: this.dynamicCloudFrontFunctionAssociations,
|
|
256
|
+
origin: this.dynamicOrigin,
|
|
257
|
+
originRequestPolicy: this.dynamicOriginResponsePolicy,
|
|
258
|
+
cachePolicy,
|
|
259
|
+
responseHeadersPolicy,
|
|
260
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
261
|
+
...imageBehaviorOptions,
|
|
262
|
+
};
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* Creates or uses user specified CloudFront Distribution
|
|
266
|
+
*/
|
|
267
|
+
getDistribution() {
|
|
268
|
+
let distribution;
|
|
269
|
+
if (this.props.distribution) {
|
|
270
|
+
distribution = this.props.distribution;
|
|
271
|
+
}
|
|
272
|
+
else {
|
|
273
|
+
distribution = new aws_cloudfront_1.Distribution(this, "Distribution", {
|
|
274
|
+
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2021,
|
|
275
|
+
defaultBehavior: this.dynamicBehaviorOptions,
|
|
276
|
+
// best to use HTTP 2 and 3 for compatability (HTTP 2) and performance (HTTP3)
|
|
277
|
+
// CloudFront will choose best option for client
|
|
278
|
+
httpVersion: aws_cloudfront_1.HttpVersion.HTTP2_AND_3,
|
|
279
|
+
comment: `cdk-nextjs Distribution for ${aws_cdk_lib_1.Stack.of(this).stackName}`,
|
|
280
|
+
...this.props.overrides?.distributionProps,
|
|
281
|
+
});
|
|
282
|
+
}
|
|
283
|
+
return distribution;
|
|
284
|
+
}
|
|
285
|
+
addDynamicBehaviors() {
|
|
286
|
+
// Image Behavior
|
|
287
|
+
this.distribution.addBehavior(this.getPathPattern("_next/image*"), this.imageBehaviorOptions.origin, this.imageBehaviorOptions);
|
|
288
|
+
// Root Path Behaviors
|
|
289
|
+
if (this.props.basePath) {
|
|
290
|
+
// because we already have a basePath we don't use / instead we use /base-path
|
|
291
|
+
this.distribution.addBehavior(this.props.basePath, this.dynamicBehaviorOptions.origin, this.dynamicBehaviorOptions);
|
|
292
|
+
// when basePath is set, we emulate the "default behavior" (*) for the site as `/base-path/*`
|
|
293
|
+
this.distribution.addBehavior(this.getPathPattern("*"), this.dynamicBehaviorOptions.origin, this.dynamicBehaviorOptions);
|
|
294
|
+
}
|
|
295
|
+
else {
|
|
296
|
+
// if no base path, then default behavior will handle all other paths
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
addStaticBehaviors() {
|
|
300
|
+
this.distribution.addBehavior("_next/static*", this.staticOrigin, this.staticBehaviorOptions);
|
|
301
|
+
// 22 = 25 (max) - 1 (_next/image) - 1 (_next/static) - 1 (*)
|
|
302
|
+
if (this.props.publicDirEntries.length >= 22) {
|
|
303
|
+
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/*).`);
|
|
304
|
+
}
|
|
305
|
+
for (const publicFile of this.props.publicDirEntries) {
|
|
306
|
+
const pathPattern = publicFile.isDirectory
|
|
307
|
+
? `${publicFile.name}/*`
|
|
308
|
+
: publicFile.name;
|
|
309
|
+
if (!/^[a-zA-Z0-9_\-.*$/~"'@:+?&]+$/.test(pathPattern)) {
|
|
310
|
+
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`);
|
|
311
|
+
}
|
|
312
|
+
const finalPathPattern = this.getPathPattern(pathPattern);
|
|
313
|
+
this.distribution.addBehavior(finalPathPattern, this.staticOrigin, this.staticBehaviorOptions);
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
/**
|
|
317
|
+
* Optionally prepends base path to given path pattern.
|
|
318
|
+
*/
|
|
319
|
+
getPathPattern(pathPattern) {
|
|
320
|
+
if (this.props.basePath) {
|
|
321
|
+
return `${this.props.basePath}/${pathPattern}`;
|
|
322
|
+
}
|
|
323
|
+
else {
|
|
324
|
+
return pathPattern;
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
exports.NextjsDistribution = NextjsDistribution;
|
|
329
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
330
|
+
NextjsDistribution[_a] = { fqn: "cdk-nextjs.NextjsDistribution", version: "0.1.1" };
|
|
331
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-distribution.js","sourceRoot":"","sources":["../src/nextjs-distribution.ts"],"names":[],"mappings":";;;;;AAAA,6CAA6D;AAE7D,+DA4BoC;AACpC,+EAI4C;AAC5C,iDAAwE;AACxE,uDAAuD;AAEvD,2CAAuC;AACvC,qCAAsC;AAItC,qFAA+E;AAkD/E,MAAa,kBAAmB,SAAQ,sBAAS;IAqC/C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA8B;QACtE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAlCnB;;;WAGG;QACK,kCAA6B,GAAoC;YACvE,kBAAkB,EAAE,EAAE,QAAQ,EAAE,KAAK,EAAE;YACvC,YAAY,EAAE;gBACZ,WAAW,EAAE,mCAAkB,CAAC,UAAU;gBAC1C,QAAQ,EAAE,KAAK;aAChB;YACD,cAAc,EAAE;gBACd,QAAQ,EAAE,KAAK;gBACf,cAAc,EAAE,sCAAqB,CAAC,+BAA+B;aACtE;YAED,uBAAuB,EAAE;gBACvB,mBAAmB,EAAE,sBAAQ,CAAC,IAAI,CAAC,GAAG,CAAC;gBACvC,iBAAiB,EAAE,IAAI;gBACvB,QAAQ,EAAE,KAAK;gBACf,OAAO,EAAE,IAAI;aACd;YACD,aAAa,EAAE,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE;SACtE,CAAC;QAaA,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC9C,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC,UAAU,KAAK,mBAAU,CAAC,gBAAgB,CAAC;QAC1E,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAChD,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,gCAAgC,EAAE,CAAC;QAC3E,IAAI,CAAC,qCAAqC;YACxC,IAAI,CAAC,2CAA2C,EAAE,CAAC;QACrD,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC9C,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,2BAA2B,EAAE,CAAC;QAChE,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,4BAA4B,EAAE,CAAC;QAClE,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QAC9D,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAC3C,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,qFAAqF;QACvF,CAAC;QACD,IAAI,uBAAS,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC5C,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,UAAU;SACpC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB;QACxB,MAAM,QAAQ,GAAG,uCAAc,CAAC,uBAAuB,CACrD,IAAI,CAAC,KAAK,CAAC,YAAY,EACvB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,mBAAmB,CAC1C,CAAC;QACF,OAAO,QAAQ,CAAC;IAClB,CAAC;IACO,mBAAmB;QACzB,IAAI,cAAoC,CAAC;QACzC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,cAAc,GAAG,qCAAoB,CAAC,UAAU,CAAC;QACnD,CAAC;aAAM,CAAC;YACN,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW;gBACrC,CAAC,CAAC,qCAAoB,CAAC,UAAU;gBACjC,CAAC,CAAC,qCAAoB,CAAC,SAAS,CAAC;QACrC,CAAC;QACD,qEAAqE;QACrE,yEAAyE;QACzE,0IAA0I;QAC1I,wEAAwE;QACxE,+DAA+D;QAC/D,EAAE;QACF,oFAAoF;QACpF;;;WAGG;QACH,uCAAuC;QACvC,OAAO;QACP,mEAAmE;QACnE,KAAK;QACL,8DAA8D;QAC9D,iCAAiC;QACjC,+CAA+C;QAC/C,+CAA+C;QAC/C,iCAAiC;QACjC,gCAAgC;QAChC,OAAO;QACP,MAAM;QACN,iDAAiD;QACjD,sCAAsC;QACtC,uCAAuC;QACvC,0DAA0D;QAC1D,4BAA4B;QAC5B,KAAK;QACL,OAAO,IAAI,mCAAU,CAAC,gBAAE,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,EAAE;YAC/D,cAAc;YACd,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,sBAAsB;SAChD,CAAC,CAAC;IACL,CAAC;IACD;;;;OAIG;IACK,gCAAgC;QACtC,OAAO,IAAI,CAAC,iBAAiB;YAC3B,CAAC,CAAC,oCAAmB,CAAC,6BAA6B;YACnD,CAAC,CAAC,oCAAmB,CAAC,UAAU,CAAC;IACrC,CAAC;IACD;;;;OAIG;IACK,2CAA2C;QACjD,MAAM,YAAY,GAA0B,EAAE,CAAC;QAC/C,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,MAAM,YAAY,GAAG,IAAI,yBAAkB,CAAC,IAAI,EAAE,cAAc,EAAE;gBAChE,IAAI,EAAE,6BAAY,CAAC,UAAU,CAAC;;;;;;WAM3B,CAAC;aACL,CAAC,CAAC;YACH,YAAY,CAAC,IAAI,CAAC;gBAChB,SAAS,EAAE,kCAAiB,CAAC,cAAc;gBAC3C,QAAQ,EAAE,YAAY;aACvB,CAAC,CAAC;QACL,CAAC;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IACD;;;;OAIG;IACK,iBAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW;YAAE,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;QACxE,MAAM,MAAM,GAAG,IAAI,wCAAiB,CAAC,IAAI,EAAE,WAAW,EAAE;YACtD,qBAAqB,EAAE;gBACrB,aAAa,EAAE,CAAC,EAAE,oFAAoF;aACvG;YACD,aAAa,EAAE;gBACb,IAAI,yBAAe,CAAC;oBAClB,OAAO,EAAE,CAAC,0BAA0B,CAAC;oBACrC,SAAS,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC;iBACpC,CAAC;aACH;YACD,IAAI,EAAE,iBAAI,CAAC,UAAU,CAAC,8BAA8B,CAAC;YACrD,OAAO,EAAE,SAAS;YAClB,OAAO,EAAE,oBAAO,CAAC,aAAa;YAC9B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,iBAAiB;SAC3C,CAAC,CAAC;QACH,MAAM,CAAC,cAAc,CAAC,WAAW,CAC/B,IAAI,0BAAgB,CAAC,0BAA0B,CAAC,CACjD,CAAC;QACF,MAAM,CAAC,cAAc,CAAC,WAAW,CAC/B,IAAI,0BAAgB,CAAC,sBAAsB,CAAC,CAC7C,CAAC;QACF,OAAO;YACL;gBACE,SAAS,EAAE,oCAAmB,CAAC,cAAc;gBAC7C,eAAe,EAAE,MAAM,CAAC,cAAc;gBACtC,WAAW,EAAE,IAAI;aAClB;SACF,CAAC;IACJ,CAAC;IACO,2BAA2B;QACjC,MAAM,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,qBAAqB,CAAC;QAC1E,MAAM,qBAAqB,GACzB,qBAAqB,EAAE,qBAAqB;YAC5C,IAAI,sCAAqB,CAAC,IAAI,EAAE,6BAA6B,EAAE;gBAC7D,uBAAuB,EAAE,IAAI,CAAC,6BAA6B;gBAC3D,OAAO,EAAE,6CAA6C,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBAChF,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,gCAAgC;aAC1D,CAAC,CAAC;QACL,OAAO;YACL,cAAc,EAAE,+BAAc,CAAC,sBAAsB;YACrD,aAAa,EAAE,8BAAa,CAAC,sBAAsB;YACnD,WAAW,EAAE,4BAAW,CAAC,iBAAiB;YAC1C,MAAM,EAAE,IAAI,CAAC,YAAY;YACzB,qBAAqB;YACrB,oBAAoB,EAAE,qCAAoB,CAAC,iBAAiB;YAC5D,GAAG,qBAAqB;SACzB,CAAC;IACJ,CAAC;IACO,4BAA4B;QAClC,MAAM,sBAAsB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,sBAAsB,CAAC;QAC5E,8CAA8C;QAC9C,MAAM,WAAW,GACf,sBAAsB,EAAE,WAAW;YACnC,IAAI,4BAAW,CAAC,IAAI,EAAE,oBAAoB,EAAE;gBAC1C,mBAAmB,EAAE,yCAAwB,CAAC,GAAG,EAAE;gBACnD,cAAc,EAAE,oCAAmB,CAAC,SAAS,CAC3C,QAAQ,EACR,KAAK,EACL,sBAAsB,EACtB,wBAAwB,EACxB,UAAU,EACV,wBAAwB,CACzB;gBACD,cAAc,EAAE,oCAAmB,CAAC,GAAG,EAAE;gBACzC,0BAA0B,EAAE,IAAI;gBAChC,wBAAwB,EAAE,IAAI;gBAC9B,OAAO,EAAE,mCAAmC,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBACtE,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,uBAAuB;aACjD,CAAC,CAAC;QACL,MAAM,qBAAqB,GACzB,sBAAsB,EAAE,qBAAqB;YAC7C,IAAI,sCAAqB,CAAC,IAAI,EAAE,8BAA8B,EAAE;gBAC9D,uBAAuB,EAAE,IAAI,CAAC,6BAA6B;gBAC3D,OAAO,EAAE,8CAA8C,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBACjF,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,sBAAsB,EAAE,qBAAqB;aACvE,CAAC,CAAC;QACL,MAAM,eAAe,GAAoB;YACvC,cAAc,EAAE,+BAAc,CAAC,SAAS;YACxC,WAAW;YACX,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,oBAAoB,EAAE,IAAI,CAAC,qCAAqC;YAChE,MAAM,EAAE,IAAI,CAAC,aAAa;YAC1B,mBAAmB,EAAE,IAAI,CAAC,2BAA2B;YACrD,qBAAqB;YACrB,oBAAoB,EAAE,qCAAoB,CAAC,iBAAiB;YAC5D,GAAG,sBAAsB;SAC1B,CAAC;QACF,OAAO,eAAe,CAAC;IACzB,CAAC;IACO,0BAA0B;QAChC,MAAM,oBAAoB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,oBAAoB,CAAC;QACxE,2CAA2C;QAC3C,MAAM,WAAW,GACf,oBAAoB,EAAE,WAAW;YACjC,IAAI,4BAAW,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBACxC,kEAAkE;gBAClE,qEAAqE;gBACrE,yEAAyE;gBACzE,6BAA6B;gBAC7B,mBAAmB,EAAE,yCAAwB,CAAC,GAAG,EAAE;gBACnD,cAAc,EAAE,oCAAmB,CAAC,SAAS,CAAC,QAAQ,CAAC;gBACvD,cAAc,EAAE,oCAAmB,CAAC,IAAI,EAAE;gBAC1C,0BAA0B,EAAE,IAAI;gBAChC,wBAAwB,EAAE,IAAI;gBAC9B,OAAO,EAAE,iCAAiC,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBACpE,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,qBAAqB;aAC/C,CAAC,CAAC;QACL,sDAAsD;QACtD,MAAM,qBAAqB,GACzB,oBAAoB,EAAE,qBAAqB;YAC3C,IAAI,sCAAqB,CAAC,IAAI,EAAE,4BAA4B,EAAE;gBAC5D,uBAAuB,EAAE,IAAI,CAAC,6BAA6B;gBAC3D,OAAO,EAAE,4CAA4C,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBAC/E,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,+BAA+B;aACzD,CAAC,CAAC;QACL,OAAO;YACL,cAAc,EAAE,+BAAc,CAAC,sBAAsB;YACrD,aAAa,EAAE,8BAAa,CAAC,sBAAsB;YACnD,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,oBAAoB,EAAE,IAAI,CAAC,qCAAqC;YAChE,MAAM,EAAE,IAAI,CAAC,aAAa;YAC1B,mBAAmB,EAAE,IAAI,CAAC,2BAA2B;YACrD,WAAW;YACX,qBAAqB;YACrB,oBAAoB,EAAE,qCAAoB,CAAC,iBAAiB;YAC5D,GAAG,oBAAoB;SACxB,CAAC;IACJ,CAAC;IACD;;OAEG;IACK,eAAe;QACrB,IAAI,YAA0B,CAAC;QAC/B,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;YAC5B,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC;QACzC,CAAC;aAAM,CAAC;YACN,YAAY,GAAG,IAAI,6BAAY,CAAC,IAAI,EAAE,cAAc,EAAE;gBACpD,sBAAsB,EAAE,uCAAsB,CAAC,aAAa;gBAC5D,eAAe,EAAE,IAAI,CAAC,sBAAsB;gBAC5C,8EAA8E;gBAC9E,gDAAgD;gBAChD,WAAW,EAAE,4BAAW,CAAC,WAAW;gBACpC,OAAO,EAAE,+BAA+B,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE;gBAClE,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,iBAAiB;aAC3C,CAAC,CAAC;QACL,CAAC;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IACO,mBAAmB;QACzB,iBAAiB;QACjB,IAAI,CAAC,YAAY,CAAC,WAAW,CAC3B,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,EACnC,IAAI,CAAC,oBAAoB,CAAC,MAAM,EAChC,IAAI,CAAC,oBAAoB,CAC1B,CAAC;QACF,sBAAsB;QACtB,IAAI,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;YACxB,8EAA8E;YAC9E,IAAI,CAAC,YAAY,CAAC,WAAW,CAC3B,IAAI,CAAC,KAAK,CAAC,QAAQ,EACnB,IAAI,CAAC,sBAAsB,CAAC,MAAM,EAClC,IAAI,CAAC,sBAAsB,CAC5B,CAAC;YACF,6FAA6F;YAC7F,IAAI,CAAC,YAAY,CAAC,WAAW,CAC3B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EACxB,IAAI,CAAC,sBAAsB,CAAC,MAAM,EAClC,IAAI,CAAC,sBAAsB,CAC5B,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,qEAAqE;QACvE,CAAC;IACH,CAAC;IACO,kBAAkB;QACxB,IAAI,CAAC,YAAY,CAAC,WAAW,CAC3B,eAAe,EACf,IAAI,CAAC,YAAY,EACjB,IAAI,CAAC,qBAAqB,CAC3B,CAAC;QACF,6DAA6D;QAC7D,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,IAAI,EAAE,EAAE,CAAC;YAC7C,MAAM,IAAI,KAAK,CACb,2TAA2T,CAC5T,CAAC;QACJ,CAAC;QACD,KAAK,MAAM,UAAU,IAAI,IAAI,CAAC,KAAK,CAAC,gBAAgB,EAAE,CAAC;YACrD,MAAM,WAAW,GAAG,UAAU,CAAC,WAAW;gBACxC,CAAC,CAAC,GAAG,UAAU,CAAC,IAAI,IAAI;gBACxB,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC;YACpB,IAAI,CAAC,+BAA+B,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;gBACvD,MAAM,IAAI,KAAK,CACb,gEAAgE,WAAW,wKAAwK,CACpP,CAAC;YACJ,CAAC;YACD,MAAM,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAC1D,IAAI,CAAC,YAAY,CAAC,WAAW,CAC3B,gBAAgB,EAChB,IAAI,CAAC,YAAY,EACjB,IAAI,CAAC,qBAAqB,CAC3B,CAAC;QACJ,CAAC;IACH,CAAC;IACD;;OAEG;IACK,cAAc,CAAC,WAAmB;QACxC,IAAI,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;YACxB,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,IAAI,WAAW,EAAE,CAAC;QACjD,CAAC;aAAM,CAAC;YACN,OAAO,WAAW,CAAC;QACrB,CAAC;IACH,CAAC;;AA7WH,gDA8WC","sourcesContent":["import { CfnOutput, Duration, Fn, 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  LambdaEdgeEventType,\n  OriginProtocolPolicy,\n  OriginRequestPolicy,\n  ResponseHeadersPolicy,\n  ResponseHeadersPolicyProps,\n  ResponseSecurityHeadersBehavior,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {\n  HttpOrigin,\n  HttpOriginProps,\n  S3BucketOrigin,\n} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport { PolicyStatement, ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\nimport { Code, Runtime } from \"aws-cdk-lib/aws-lambda\";\nimport { IBucket } from \"aws-cdk-lib/aws-s3\";\nimport { Construct } from \"constructs\";\nimport { NextjsType } from \"./common\";\nimport { OptionalDistributionProps } from \"./generated-structs/OptionalDistributionProps\";\nimport { OptionalFunctionProps } from \"./generated-structs/OptionalFunctionProps\";\nimport { OptionalS3OriginBucketWithOACProps } from \"./generated-structs/OptionalS3OriginBucketWithOACProps\";\nimport { SignFnUrlFunction } from \"./lambdas/sign-fn-url/sign-fn-url-function\";\nimport { PublicDirEntry } from \"./nextjs-build/nextjs-build\";\n\nexport interface NextjsDistributionOverrides {\n  readonly edgeFunctionProps?: OptionalFunctionProps;\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 dynamicHttpOriginProps?: HttpOriginProps;\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   * Dynamic (Next.js server) URL to add behavior to distribution\n   */\n  readonly dynamicUrl: string;\n  /**\n   * Required if `NextjsType.GLOBAL_FUNCTIONS`\n   */\n  readonly functionArn?: string;\n  readonly nextjsType: NextjsType;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsDistributionOverrides;\n  /**\n   * Path to directory of Next.js app's public directory. Used to add static\n   * 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 edgeLambdas?: AddBehaviorOptions[\"edgeLambdas\"];\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    if (this.isFunctionCompute) {\n      this.edgeLambdas = this.createEdgeLambdas();\n    }\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    if (this.isFunctionCompute) {\n      // this.addLambdaOac(); // TODO: wait for POST body encryption feature for Lambda OAC\n    }\n    new CfnOutput(this, \"DistributionDomainName\", {\n      value: this.distribution.domainName,\n    });\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(): HttpOrigin {\n    let protocolPolicy: OriginProtocolPolicy;\n    if (this.isFunctionCompute) {\n      protocolPolicy = OriginProtocolPolicy.HTTPS_ONLY;\n    } else {\n      protocolPolicy = this.props.certificate\n        ? OriginProtocolPolicy.HTTPS_ONLY\n        : OriginProtocolPolicy.HTTP_ONLY;\n    }\n    // WAITING FOR CLOUDFRONT FEATURE: CloudFront Lambda Function URL OAC\n    // doesn't support POST with body since CloudFront doesn't include sha256\n    // hash of body. see: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/private-content-restricting-access-to-lambda.html\n    // We could monkey patch `fetch` to include sha256 in request for server\n    // actions, but better solution is for cloudfront to support it\n    //\n    // TODO: use L2 construct when released: https://github.com/aws/aws-cdk/issues/31629\n    /**\n     * Given stack id: \"arn:aws:cloudformation:us-east-1:905418358903:stack/lh-stickb-idp/4bf74be0-e880-11ee-aea9-0affc6185b25\",\n     * returns \"4bf74be0\"\n     */\n    // const uniqueStackIdPart = Fn.select(\n    //   0,\n    //   Fn.split(\"-\", Fn.select(2, Fn.split(\"/\", `${Aws.STACK_ID}`))),\n    // );\n    // const lambdaOac = new CfnOriginAccessControl(this, \"OAC\", {\n    //   originAccessControlConfig: {\n    //     name: `OAC-Lambda-${uniqueStackIdPart}`,\n    //     originAccessControlOriginType: \"lambda\",\n    //     signingBehavior: \"always\",\n    //     signingProtocol: \"sigv4\",\n    //   },\n    // });\n    // const cfnDistribution = this.distribution.node\n    //   .defaultChild as CfnDistribution;\n    // cfnDistribution.addPropertyOverride(\n    //   \"DistributionConfig.Origins.0.OriginAccessControlId\",\n    //   lambdaOac.getAtt(\"Id\"),\n    // );\n    return new HttpOrigin(Fn.parseDomainName(this.props.dynamicUrl), {\n      protocolPolicy,\n      ...this.props.overrides?.dynamicHttpOriginProps,\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 option (App Runner, Fargate, or Lambda)\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  /**\n   * Required to sign requests so that we can use IAM_AUTH for Lambda Function URL\n   * to prevent public access. Once CloudFront Lambda OAC is released, we can\n   * use infra configuration for this instead of custom code.\n   */\n  private createEdgeLambdas(): AddBehaviorOptions[\"edgeLambdas\"] {\n    if (!this.props.functionArn) throw new Error(\"functionArn is required\");\n    const edgeFn = new SignFnUrlFunction(this, \"SignFnUrl\", {\n      currentVersionOptions: {\n        retryAttempts: 0, // fail fast when trying to delete b/c replicated functions take long time to delete\n      },\n      initialPolicy: [\n        new PolicyStatement({\n          actions: [\"lambda:InvokeFunctionUrl\"],\n          resources: [this.props.functionArn],\n        }),\n      ],\n      code: Code.fromInline(\"export function handler() {}\"),\n      handler: \"handler\",\n      runtime: Runtime.NODEJS_LATEST,\n      ...this.props.overrides?.edgeFunctionProps,\n    });\n    edgeFn.currentVersion.grantInvoke(\n      new ServicePrincipal(\"edgelambda.amazonaws.com\"),\n    );\n    edgeFn.currentVersion.grantInvoke(\n      new ServicePrincipal(\"lambda.amazonaws.com\"),\n    );\n    return [\n      {\n        eventType: LambdaEdgeEventType.ORIGIN_REQUEST,\n        functionVersion: edgeFn.currentVersion,\n        includeBody: true,\n      },\n    ];\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      edgeLambdas: this.edgeLambdas,\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      edgeLambdas: this.edgeLambdas,\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      \"_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"]}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Connections, IVpc } from "aws-cdk-lib/aws-ec2";
|
|
2
|
+
import { AccessPoint, AccessPointProps, FileSystem, FileSystemProps } from "aws-cdk-lib/aws-efs";
|
|
3
|
+
import { IRole } from "aws-cdk-lib/aws-iam";
|
|
4
|
+
import { Construct } from "constructs";
|
|
5
|
+
export interface NextjsFileSystemOverrides {
|
|
6
|
+
readonly fileSystemProps?: FileSystemProps;
|
|
7
|
+
readonly accessPointProps?: AccessPointProps;
|
|
8
|
+
}
|
|
9
|
+
export interface NextjsFileSystemProps {
|
|
10
|
+
readonly overrides?: NextjsFileSystemOverrides;
|
|
11
|
+
readonly vpc: IVpc;
|
|
12
|
+
}
|
|
13
|
+
export interface AllowComputeProps {
|
|
14
|
+
readonly connections: Connections;
|
|
15
|
+
readonly role: IRole;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Next.js Network File System enabling sharing of image optimization cache,
|
|
19
|
+
* data cach, and pages cache.
|
|
20
|
+
*/
|
|
21
|
+
export declare class NextjsFileSystem extends Construct {
|
|
22
|
+
fileSystem: FileSystem;
|
|
23
|
+
accessPoint: AccessPoint;
|
|
24
|
+
private props;
|
|
25
|
+
constructor(scope: Construct, id: string, props: NextjsFileSystemProps);
|
|
26
|
+
/**
|
|
27
|
+
* Creates EFS File System
|
|
28
|
+
*
|
|
29
|
+
* Note, the resource policy for the File System will include the boolean
|
|
30
|
+
* condition, `"elasticfilesystem:AccessedViaMountTarget": "true"` which from
|
|
31
|
+
* CDK [docs](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_efs-readme.html#permissions)
|
|
32
|
+
* says, "only allow access to clients using IAM authentication and deny access
|
|
33
|
+
* to anonymous clients".
|
|
34
|
+
* @see https://docs.aws.amazon.com/efs/latest/ug/access-control-block-public-access.html
|
|
35
|
+
*
|
|
36
|
+
* Ideally we could add IAM string condition `elasticfilesystem:AccessPointArn`
|
|
37
|
+
* to the resource policy but this causes circular dependency.
|
|
38
|
+
*/
|
|
39
|
+
private createFileSystem;
|
|
40
|
+
private createAccessPoint;
|
|
41
|
+
allowCompute({ connections, role }: AllowComputeProps): void;
|
|
42
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.NextjsFileSystem = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
7
|
+
const aws_efs_1 = require("aws-cdk-lib/aws-efs");
|
|
8
|
+
const constructs_1 = require("constructs");
|
|
9
|
+
/**
|
|
10
|
+
* Next.js Network File System enabling sharing of image optimization cache,
|
|
11
|
+
* data cach, and pages cache.
|
|
12
|
+
*/
|
|
13
|
+
class NextjsFileSystem extends constructs_1.Construct {
|
|
14
|
+
constructor(scope, id, props) {
|
|
15
|
+
super(scope, id);
|
|
16
|
+
this.props = props;
|
|
17
|
+
this.fileSystem = this.createFileSystem();
|
|
18
|
+
this.accessPoint = this.createAccessPoint();
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Creates EFS File System
|
|
22
|
+
*
|
|
23
|
+
* Note, the resource policy for the File System will include the boolean
|
|
24
|
+
* condition, `"elasticfilesystem:AccessedViaMountTarget": "true"` which from
|
|
25
|
+
* CDK [docs](https://docs.aws.amazon.com/cdk/api/v2/docs/aws-cdk-lib.aws_efs-readme.html#permissions)
|
|
26
|
+
* says, "only allow access to clients using IAM authentication and deny access
|
|
27
|
+
* to anonymous clients".
|
|
28
|
+
* @see https://docs.aws.amazon.com/efs/latest/ug/access-control-block-public-access.html
|
|
29
|
+
*
|
|
30
|
+
* Ideally we could add IAM string condition `elasticfilesystem:AccessPointArn`
|
|
31
|
+
* to the resource policy but this causes circular dependency.
|
|
32
|
+
*/
|
|
33
|
+
createFileSystem() {
|
|
34
|
+
const fileSystem = new aws_efs_1.FileSystem(this, "FileSystem", {
|
|
35
|
+
encrypted: true,
|
|
36
|
+
lifecyclePolicy: aws_efs_1.LifecyclePolicy.AFTER_30_DAYS,
|
|
37
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
38
|
+
vpc: this.props.vpc,
|
|
39
|
+
allowAnonymousAccess: false,
|
|
40
|
+
...this.props.overrides?.fileSystemProps,
|
|
41
|
+
});
|
|
42
|
+
return fileSystem;
|
|
43
|
+
}
|
|
44
|
+
createAccessPoint() {
|
|
45
|
+
const uid = "1001";
|
|
46
|
+
const gid = "1001";
|
|
47
|
+
const accessPoint = new aws_efs_1.AccessPoint(this, "AccessPoint", {
|
|
48
|
+
// as /next/cache doesn't exist in a new efs filesystem, the efs will
|
|
49
|
+
// create the directory with the following options
|
|
50
|
+
createAcl: {
|
|
51
|
+
ownerGid: gid,
|
|
52
|
+
ownerUid: uid,
|
|
53
|
+
permissions: "755",
|
|
54
|
+
},
|
|
55
|
+
fileSystem: this.fileSystem,
|
|
56
|
+
path: "/next/cache",
|
|
57
|
+
// enforce POSIX identity so container wil access file system with this identity
|
|
58
|
+
posixUser: {
|
|
59
|
+
gid,
|
|
60
|
+
uid,
|
|
61
|
+
},
|
|
62
|
+
...this.props.overrides?.accessPointProps,
|
|
63
|
+
});
|
|
64
|
+
return accessPoint;
|
|
65
|
+
}
|
|
66
|
+
allowCompute({ connections, role }) {
|
|
67
|
+
this.fileSystem.connections.allowDefaultPortFrom(connections);
|
|
68
|
+
this.fileSystem.grantReadWrite(role);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
exports.NextjsFileSystem = NextjsFileSystem;
|
|
72
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
73
|
+
NextjsFileSystem[_a] = { fqn: "cdk-nextjs.NextjsFileSystem", version: "0.1.1" };
|
|
74
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV4dGpzLWZpbGUtc3lzdGVtLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL25leHRqcy1maWxlLXN5c3RlbS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUE0QztBQUU1QyxpREFNNkI7QUFFN0IsMkNBQXVDO0FBaUJ2Qzs7O0dBR0c7QUFDSCxNQUFhLGdCQUFpQixTQUFRLHNCQUFTO0lBSzdDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBNEI7UUFDcEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUNqQixJQUFJLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQztRQUNuQixJQUFJLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1FBQzFDLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUM7SUFDOUMsQ0FBQztJQUNEOzs7Ozs7Ozs7Ozs7T0FZRztJQUNLLGdCQUFnQjtRQUN0QixNQUFNLFVBQVUsR0FBRyxJQUFJLG9CQUFVLENBQUMsSUFBSSxFQUFFLFlBQVksRUFBRTtZQUNwRCxTQUFTLEVBQUUsSUFBSTtZQUNmLGVBQWUsRUFBRSx5QkFBZSxDQUFDLGFBQWE7WUFDOUMsYUFBYSxFQUFFLDJCQUFhLENBQUMsT0FBTztZQUNwQyxHQUFHLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHO1lBQ25CLG9CQUFvQixFQUFFLEtBQUs7WUFDM0IsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxlQUFlO1NBQ3pDLENBQUMsQ0FBQztRQUNILE9BQU8sVUFBVSxDQUFDO0lBQ3BCLENBQUM7SUFDTyxpQkFBaUI7UUFDdkIsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDO1FBQ25CLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQztRQUNuQixNQUFNLFdBQVcsR0FBRyxJQUFJLHFCQUFXLENBQUMsSUFBSSxFQUFFLGFBQWEsRUFBRTtZQUN2RCxxRUFBcUU7WUFDckUsa0RBQWtEO1lBQ2xELFNBQVMsRUFBRTtnQkFDVCxRQUFRLEVBQUUsR0FBRztnQkFDYixRQUFRLEVBQUUsR0FBRztnQkFDYixXQUFXLEVBQUUsS0FBSzthQUNuQjtZQUNELFVBQVUsRUFBRSxJQUFJLENBQUMsVUFBVTtZQUMzQixJQUFJLEVBQUUsYUFBYTtZQUNuQixnRkFBZ0Y7WUFDaEYsU0FBUyxFQUFFO2dCQUNULEdBQUc7Z0JBQ0gsR0FBRzthQUNKO1lBQ0QsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxnQkFBZ0I7U0FDMUMsQ0FBQyxDQUFDO1FBQ0gsT0FBTyxXQUFXLENBQUM7SUFDckIsQ0FBQztJQUNELFlBQVksQ0FBQyxFQUFFLFdBQVcsRUFBRSxJQUFJLEVBQXFCO1FBQ25ELElBQUksQ0FBQyxVQUFVLENBQUMsV0FBVyxDQUFDLG9CQUFvQixDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQzlELElBQUksQ0FBQyxVQUFVLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7O0FBNURILDRDQTZEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlbW92YWxQb2xpY3kgfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7IENvbm5lY3Rpb25zLCBJVnBjIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzJcIjtcbmltcG9ydCB7XG4gIEFjY2Vzc1BvaW50LFxuICBBY2Nlc3NQb2ludFByb3BzLFxuICBGaWxlU3lzdGVtLFxuICBGaWxlU3lzdGVtUHJvcHMsXG4gIExpZmVjeWNsZVBvbGljeSxcbn0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lZnNcIjtcbmltcG9ydCB7IElSb2xlIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmV4dGpzRmlsZVN5c3RlbU92ZXJyaWRlcyB7XG4gIHJlYWRvbmx5IGZpbGVTeXN0ZW1Qcm9wcz86IEZpbGVTeXN0ZW1Qcm9wcztcbiAgcmVhZG9ubHkgYWNjZXNzUG9pbnRQcm9wcz86IEFjY2Vzc1BvaW50UHJvcHM7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmV4dGpzRmlsZVN5c3RlbVByb3BzIHtcbiAgcmVhZG9ubHkgb3ZlcnJpZGVzPzogTmV4dGpzRmlsZVN5c3RlbU92ZXJyaWRlcztcbiAgcmVhZG9ubHkgdnBjOiBJVnBjO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEFsbG93Q29tcHV0ZVByb3BzIHtcbiAgcmVhZG9ubHkgY29ubmVjdGlvbnM6IENvbm5lY3Rpb25zO1xuICByZWFkb25seSByb2xlOiBJUm9sZTtcbn1cblxuLyoqXG4gKiBOZXh0LmpzIE5ldHdvcmsgRmlsZSBTeXN0ZW0gZW5hYmxpbmcgc2hhcmluZyBvZiBpbWFnZSBvcHRpbWl6YXRpb24gY2FjaGUsXG4gKiBkYXRhIGNhY2gsIGFuZCBwYWdlcyBjYWNoZS5cbiAqL1xuZXhwb3J0IGNsYXNzIE5leHRqc0ZpbGVTeXN0ZW0gZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBmaWxlU3lzdGVtOiBGaWxlU3lzdGVtO1xuICBhY2Nlc3NQb2ludDogQWNjZXNzUG9pbnQ7XG4gIHByaXZhdGUgcHJvcHM6IE5leHRqc0ZpbGVTeXN0ZW1Qcm9wcztcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogTmV4dGpzRmlsZVN5c3RlbVByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcbiAgICB0aGlzLnByb3BzID0gcHJvcHM7XG4gICAgdGhpcy5maWxlU3lzdGVtID0gdGhpcy5jcmVhdGVGaWxlU3lzdGVtKCk7XG4gICAgdGhpcy5hY2Nlc3NQb2ludCA9IHRoaXMuY3JlYXRlQWNjZXNzUG9pbnQoKTtcbiAgfVxuICAvKipcbiAgICogQ3JlYXRlcyBFRlMgRmlsZSBTeXN0ZW1cbiAgICpcbiAgICogTm90ZSwgdGhlIHJlc291cmNlIHBvbGljeSBmb3IgdGhlIEZpbGUgU3lzdGVtIHdpbGwgaW5jbHVkZSB0aGUgYm9vbGVhblxuICAgKiBjb25kaXRpb24sIGBcImVsYXN0aWNmaWxlc3lzdGVtOkFjY2Vzc2VkVmlhTW91bnRUYXJnZXRcIjogXCJ0cnVlXCJgIHdoaWNoIGZyb21cbiAgICogQ0RLIFtkb2NzXShodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vY2RrL2FwaS92Mi9kb2NzL2F3cy1jZGstbGliLmF3c19lZnMtcmVhZG1lLmh0bWwjcGVybWlzc2lvbnMpXG4gICAqIHNheXMsIFwib25seSBhbGxvdyBhY2Nlc3MgdG8gY2xpZW50cyB1c2luZyBJQU0gYXV0aGVudGljYXRpb24gYW5kIGRlbnkgYWNjZXNzXG4gICAqIHRvIGFub255bW91cyBjbGllbnRzXCIuXG4gICAqIEBzZWUgaHR0cHM6Ly9kb2NzLmF3cy5hbWF6b24uY29tL2Vmcy9sYXRlc3QvdWcvYWNjZXNzLWNvbnRyb2wtYmxvY2stcHVibGljLWFjY2Vzcy5odG1sXG4gICAqXG4gICAqIElkZWFsbHkgd2UgY291bGQgYWRkIElBTSBzdHJpbmcgY29uZGl0aW9uIGBlbGFzdGljZmlsZXN5c3RlbTpBY2Nlc3NQb2ludEFybmBcbiAgICogdG8gdGhlIHJlc291cmNlIHBvbGljeSBidXQgdGhpcyBjYXVzZXMgY2lyY3VsYXIgZGVwZW5kZW5jeS5cbiAgICovXG4gIHByaXZhdGUgY3JlYXRlRmlsZVN5c3RlbSgpIHtcbiAgICBjb25zdCBmaWxlU3lzdGVtID0gbmV3IEZpbGVTeXN0ZW0odGhpcywgXCJGaWxlU3lzdGVtXCIsIHtcbiAgICAgIGVuY3J5cHRlZDogdHJ1ZSxcbiAgICAgIGxpZmVjeWNsZVBvbGljeTogTGlmZWN5Y2xlUG9saWN5LkFGVEVSXzMwX0RBWVMsXG4gICAgICByZW1vdmFsUG9saWN5OiBSZW1vdmFsUG9saWN5LkRFU1RST1ksXG4gICAgICB2cGM6IHRoaXMucHJvcHMudnBjLFxuICAgICAgYWxsb3dBbm9ueW1vdXNBY2Nlc3M6IGZhbHNlLFxuICAgICAgLi4udGhpcy5wcm9wcy5vdmVycmlkZXM/LmZpbGVTeXN0ZW1Qcm9wcyxcbiAgICB9KTtcbiAgICByZXR1cm4gZmlsZVN5c3RlbTtcbiAgfVxuICBwcml2YXRlIGNyZWF0ZUFjY2Vzc1BvaW50KCkge1xuICAgIGNvbnN0IHVpZCA9IFwiMTAwMVwiO1xuICAgIGNvbnN0IGdpZCA9IFwiMTAwMVwiO1xuICAgIGNvbnN0IGFjY2Vzc1BvaW50ID0gbmV3IEFjY2Vzc1BvaW50KHRoaXMsIFwiQWNjZXNzUG9pbnRcIiwge1xuICAgICAgLy8gYXMgL25leHQvY2FjaGUgZG9lc24ndCBleGlzdCBpbiBhIG5ldyBlZnMgZmlsZXN5c3RlbSwgdGhlIGVmcyB3aWxsXG4gICAgICAvLyBjcmVhdGUgdGhlIGRpcmVjdG9yeSB3aXRoIHRoZSBmb2xsb3dpbmcgb3B0aW9uc1xuICAgICAgY3JlYXRlQWNsOiB7XG4gICAgICAgIG93bmVyR2lkOiBnaWQsXG4gICAgICAgIG93bmVyVWlkOiB1aWQsXG4gICAgICAgIHBlcm1pc3Npb25zOiBcIjc1NVwiLFxuICAgICAgfSxcbiAgICAgIGZpbGVTeXN0ZW06IHRoaXMuZmlsZVN5c3RlbSxcbiAgICAgIHBhdGg6IFwiL25leHQvY2FjaGVcIixcbiAgICAgIC8vIGVuZm9yY2UgUE9TSVggaWRlbnRpdHkgc28gY29udGFpbmVyIHdpbCBhY2Nlc3MgZmlsZSBzeXN0ZW0gd2l0aCB0aGlzIGlkZW50aXR5XG4gICAgICBwb3NpeFVzZXI6IHtcbiAgICAgICAgZ2lkLFxuICAgICAgICB1aWQsXG4gICAgICB9LFxuICAgICAgLi4udGhpcy5wcm9wcy5vdmVycmlkZXM/LmFjY2Vzc1BvaW50UHJvcHMsXG4gICAgfSk7XG4gICAgcmV0dXJuIGFjY2Vzc1BvaW50O1xuICB9XG4gIGFsbG93Q29tcHV0ZSh7IGNvbm5lY3Rpb25zLCByb2xlIH06IEFsbG93Q29tcHV0ZVByb3BzKSB7XG4gICAgdGhpcy5maWxlU3lzdGVtLmNvbm5lY3Rpb25zLmFsbG93RGVmYXVsdFBvcnRGcm9tKGNvbm5lY3Rpb25zKTtcbiAgICB0aGlzLmZpbGVTeXN0ZW0uZ3JhbnRSZWFkV3JpdGUocm9sZSk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { IDistribution } from "aws-cdk-lib/aws-cloudfront";
|
|
2
|
+
import { AwsCustomResourceProps } from "aws-cdk-lib/custom-resources";
|
|
3
|
+
import { Construct } from "constructs";
|
|
4
|
+
export interface NextjsInvalidationOverrides {
|
|
5
|
+
readonly awsCustomResourceProps?: AwsCustomResourceProps;
|
|
6
|
+
}
|
|
7
|
+
export interface NextjsInvalidationProps {
|
|
8
|
+
/**
|
|
9
|
+
* CloudFront Distribution to invalidate
|
|
10
|
+
*/
|
|
11
|
+
readonly distribution: IDistribution;
|
|
12
|
+
/**
|
|
13
|
+
* Override props for every construct.
|
|
14
|
+
*/
|
|
15
|
+
readonly overrides?: NextjsInvalidationOverrides;
|
|
16
|
+
}
|
|
17
|
+
export declare class NextjsInvalidation extends Construct {
|
|
18
|
+
constructor(scope: Construct, id: string, props: NextjsInvalidationProps);
|
|
19
|
+
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.NextjsInvalidation = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
7
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
8
|
+
const custom_resources_1 = require("aws-cdk-lib/custom-resources");
|
|
9
|
+
const constructs_1 = require("constructs");
|
|
10
|
+
class NextjsInvalidation extends constructs_1.Construct {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id);
|
|
13
|
+
const awsSdkCall = {
|
|
14
|
+
// make `physicalResourceId` change each time to invalidate CloudFront
|
|
15
|
+
// distribution on each change
|
|
16
|
+
physicalResourceId: custom_resources_1.PhysicalResourceId.of(`${props.distribution.distributionId}-${Date.now()}`),
|
|
17
|
+
action: "CreateInvalidationCommand",
|
|
18
|
+
service: "@aws-sdk/client-cloudfront",
|
|
19
|
+
parameters: {
|
|
20
|
+
DistributionId: props.distribution.distributionId,
|
|
21
|
+
InvalidationBatch: {
|
|
22
|
+
CallerReference: new Date().toISOString(),
|
|
23
|
+
Paths: {
|
|
24
|
+
Quantity: 1,
|
|
25
|
+
Items: ["/*"],
|
|
26
|
+
},
|
|
27
|
+
},
|
|
28
|
+
},
|
|
29
|
+
};
|
|
30
|
+
new custom_resources_1.AwsCustomResource(this, "AwsCR", {
|
|
31
|
+
onCreate: awsSdkCall,
|
|
32
|
+
onUpdate: awsSdkCall,
|
|
33
|
+
policy: custom_resources_1.AwsCustomResourcePolicy.fromStatements([
|
|
34
|
+
new aws_iam_1.PolicyStatement({
|
|
35
|
+
actions: ["cloudfront:CreateInvalidation"],
|
|
36
|
+
resources: [
|
|
37
|
+
aws_cdk_lib_1.Stack.of(this).formatArn({
|
|
38
|
+
resource: `distribution/${props.distribution.distributionId}`,
|
|
39
|
+
service: "cloudfront",
|
|
40
|
+
region: "",
|
|
41
|
+
}),
|
|
42
|
+
],
|
|
43
|
+
}),
|
|
44
|
+
]),
|
|
45
|
+
...props.overrides?.awsCustomResourceProps,
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
exports.NextjsInvalidation = NextjsInvalidation;
|
|
50
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
51
|
+
NextjsInvalidation[_a] = { fqn: "cdk-nextjs.NextjsInvalidation", version: "0.1.1" };
|
|
52
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV4dGpzLWludmFsaWRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9uZXh0anMtaW52YWxpZGF0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkNBQW9DO0FBRXBDLGlEQUFzRDtBQUN0RCxtRUFNc0M7QUFDdEMsMkNBQXVDO0FBaUJ2QyxNQUFhLGtCQUFtQixTQUFRLHNCQUFTO0lBQy9DLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBOEI7UUFDdEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUNqQixNQUFNLFVBQVUsR0FBZTtZQUM3QixzRUFBc0U7WUFDdEUsOEJBQThCO1lBQzlCLGtCQUFrQixFQUFFLHFDQUFrQixDQUFDLEVBQUUsQ0FDdkMsR0FBRyxLQUFLLENBQUMsWUFBWSxDQUFDLGNBQWMsSUFBSSxJQUFJLENBQUMsR0FBRyxFQUFFLEVBQUUsQ0FDckQ7WUFDRCxNQUFNLEVBQUUsMkJBQTJCO1lBQ25DLE9BQU8sRUFBRSw0QkFBNEI7WUFDckMsVUFBVSxFQUFFO2dCQUNWLGNBQWMsRUFBRSxLQUFLLENBQUMsWUFBWSxDQUFDLGNBQWM7Z0JBQ2pELGlCQUFpQixFQUFFO29CQUNqQixlQUFlLEVBQUUsSUFBSSxJQUFJLEVBQUUsQ0FBQyxXQUFXLEVBQUU7b0JBQ3pDLEtBQUssRUFBRTt3QkFDTCxRQUFRLEVBQUUsQ0FBQzt3QkFDWCxLQUFLLEVBQUUsQ0FBQyxJQUFJLENBQUM7cUJBQ2Q7aUJBQ0Y7YUFDRjtTQUNGLENBQUM7UUFDRixJQUFJLG9DQUFpQixDQUFDLElBQUksRUFBRSxPQUFPLEVBQUU7WUFDbkMsUUFBUSxFQUFFLFVBQVU7WUFDcEIsUUFBUSxFQUFFLFVBQVU7WUFDcEIsTUFBTSxFQUFFLDBDQUF1QixDQUFDLGNBQWMsQ0FBQztnQkFDN0MsSUFBSSx5QkFBZSxDQUFDO29CQUNsQixPQUFPLEVBQUUsQ0FBQywrQkFBK0IsQ0FBQztvQkFDMUMsU0FBUyxFQUFFO3dCQUNULG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsQ0FBQzs0QkFDdkIsUUFBUSxFQUFFLGdCQUFnQixLQUFLLENBQUMsWUFBWSxDQUFDLGNBQWMsRUFBRTs0QkFDN0QsT0FBTyxFQUFFLFlBQVk7NEJBQ3JCLE1BQU0sRUFBRSxFQUFFO3lCQUNYLENBQUM7cUJBQ0g7aUJBQ0YsQ0FBQzthQUNILENBQUM7WUFDRixHQUFHLEtBQUssQ0FBQyxTQUFTLEVBQUUsc0JBQXNCO1NBQzNDLENBQUMsQ0FBQztJQUNMLENBQUM7O0FBdkNILGdEQXdDQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFN0YWNrIH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBJRGlzdHJpYnV0aW9uIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1jbG91ZGZyb250XCI7XG5pbXBvcnQgeyBQb2xpY3lTdGF0ZW1lbnQgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiO1xuaW1wb3J0IHtcbiAgQXdzQ3VzdG9tUmVzb3VyY2UsXG4gIEF3c1Nka0NhbGwsXG4gIEF3c0N1c3RvbVJlc291cmNlUG9saWN5LFxuICBQaHlzaWNhbFJlc291cmNlSWQsXG4gIEF3c0N1c3RvbVJlc291cmNlUHJvcHMsXG59IGZyb20gXCJhd3MtY2RrLWxpYi9jdXN0b20tcmVzb3VyY2VzXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIE5leHRqc0ludmFsaWRhdGlvbk92ZXJyaWRlcyB7XG4gIHJlYWRvbmx5IGF3c0N1c3RvbVJlc291cmNlUHJvcHM/OiBBd3NDdXN0b21SZXNvdXJjZVByb3BzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5leHRqc0ludmFsaWRhdGlvblByb3BzIHtcbiAgLyoqXG4gICAqIENsb3VkRnJvbnQgRGlzdHJpYnV0aW9uIHRvIGludmFsaWRhdGVcbiAgICovXG4gIHJlYWRvbmx5IGRpc3RyaWJ1dGlvbjogSURpc3RyaWJ1dGlvbjtcbiAgLyoqXG4gICAqIE92ZXJyaWRlIHByb3BzIGZvciBldmVyeSBjb25zdHJ1Y3QuXG4gICAqL1xuICByZWFkb25seSBvdmVycmlkZXM/OiBOZXh0anNJbnZhbGlkYXRpb25PdmVycmlkZXM7XG59XG5cbmV4cG9ydCBjbGFzcyBOZXh0anNJbnZhbGlkYXRpb24gZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogTmV4dGpzSW52YWxpZGF0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuICAgIGNvbnN0IGF3c1Nka0NhbGw6IEF3c1Nka0NhbGwgPSB7XG4gICAgICAvLyBtYWtlIGBwaHlzaWNhbFJlc291cmNlSWRgIGNoYW5nZSBlYWNoIHRpbWUgdG8gaW52YWxpZGF0ZSBDbG91ZEZyb250XG4gICAgICAvLyBkaXN0cmlidXRpb24gb24gZWFjaCBjaGFuZ2VcbiAgICAgIHBoeXNpY2FsUmVzb3VyY2VJZDogUGh5c2ljYWxSZXNvdXJjZUlkLm9mKFxuICAgICAgICBgJHtwcm9wcy5kaXN0cmlidXRpb24uZGlzdHJpYnV0aW9uSWR9LSR7RGF0ZS5ub3coKX1gLFxuICAgICAgKSxcbiAgICAgIGFjdGlvbjogXCJDcmVhdGVJbnZhbGlkYXRpb25Db21tYW5kXCIsXG4gICAgICBzZXJ2aWNlOiBcIkBhd3Mtc2RrL2NsaWVudC1jbG91ZGZyb250XCIsXG4gICAgICBwYXJhbWV0ZXJzOiB7XG4gICAgICAgIERpc3RyaWJ1dGlvbklkOiBwcm9wcy5kaXN0cmlidXRpb24uZGlzdHJpYnV0aW9uSWQsXG4gICAgICAgIEludmFsaWRhdGlvbkJhdGNoOiB7XG4gICAgICAgICAgQ2FsbGVyUmVmZXJlbmNlOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICAgICAgUGF0aHM6IHtcbiAgICAgICAgICAgIFF1YW50aXR5OiAxLFxuICAgICAgICAgICAgSXRlbXM6IFtcIi8qXCJdLFxuICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH07XG4gICAgbmV3IEF3c0N1c3RvbVJlc291cmNlKHRoaXMsIFwiQXdzQ1JcIiwge1xuICAgICAgb25DcmVhdGU6IGF3c1Nka0NhbGwsXG4gICAgICBvblVwZGF0ZTogYXdzU2RrQ2FsbCxcbiAgICAgIHBvbGljeTogQXdzQ3VzdG9tUmVzb3VyY2VQb2xpY3kuZnJvbVN0YXRlbWVudHMoW1xuICAgICAgICBuZXcgUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICBhY3Rpb25zOiBbXCJjbG91ZGZyb250OkNyZWF0ZUludmFsaWRhdGlvblwiXSxcbiAgICAgICAgICByZXNvdXJjZXM6IFtcbiAgICAgICAgICAgIFN0YWNrLm9mKHRoaXMpLmZvcm1hdEFybih7XG4gICAgICAgICAgICAgIHJlc291cmNlOiBgZGlzdHJpYnV0aW9uLyR7cHJvcHMuZGlzdHJpYnV0aW9uLmRpc3RyaWJ1dGlvbklkfWAsXG4gICAgICAgICAgICAgIHNlcnZpY2U6IFwiY2xvdWRmcm9udFwiLFxuICAgICAgICAgICAgICByZWdpb246IFwiXCIsXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICBdLFxuICAgICAgICB9KSxcbiAgICAgIF0pLFxuICAgICAgLi4ucHJvcHMub3ZlcnJpZGVzPy5hd3NDdXN0b21SZXNvdXJjZVByb3BzLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { Function as LambdaFunction } from "aws-cdk-lib/aws-lambda";
|
|
2
|
+
import { SqsEventSourceProps } from "aws-cdk-lib/aws-lambda-event-sources";
|
|
3
|
+
import { Queue, QueueProps } from "aws-cdk-lib/aws-sqs";
|
|
4
|
+
import { Construct } from "constructs";
|
|
5
|
+
import { OptionalFunctionProps } from "./generated-structs/OptionalFunctionProps";
|
|
6
|
+
export interface NextjsRevalidationOverrides {
|
|
7
|
+
readonly queueProps?: QueueProps;
|
|
8
|
+
readonly functionProps?: OptionalFunctionProps;
|
|
9
|
+
readonly sqsEventSourceProps?: SqsEventSourceProps;
|
|
10
|
+
}
|
|
11
|
+
export interface NextjsRevalidationProps {
|
|
12
|
+
readonly fn: LambdaFunction;
|
|
13
|
+
readonly overrides?: NextjsRevalidationOverrides;
|
|
14
|
+
readonly previewModeId: string;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* [On-Demand Revalidation](https://nextjs.org/docs/app/building-your-application/caching#on-demand-revalidation)
|
|
18
|
+
* (i.e. `revalidateTag`, `revlidatePath`) doesn't work by default in Lambda
|
|
19
|
+
* environment because it tries to run every request completes when Lambda
|
|
20
|
+
* spins down. Therefore, we use a SQS Queue and Lambda function to run
|
|
21
|
+
* revalidation async.
|
|
22
|
+
*/
|
|
23
|
+
export declare class NextjsRevalidation extends Construct {
|
|
24
|
+
queue: Queue;
|
|
25
|
+
fn: LambdaFunction;
|
|
26
|
+
private props;
|
|
27
|
+
constructor(scope: Construct, id: string, props: NextjsRevalidationProps);
|
|
28
|
+
private createQueue;
|
|
29
|
+
private createFunction;
|
|
30
|
+
}
|