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.
Files changed (132) hide show
  1. package/.jsii +13834 -0
  2. package/.prettierrc +0 -0
  3. package/API.md +9655 -0
  4. package/LICENSE +202 -0
  5. package/README.md +164 -2
  6. package/THIRD-PARTY-LICENSES.md +31 -0
  7. package/assets/lambdas/assets-deployment/assets-deployment.lambda/assets-deployment.Dockerfile +18 -0
  8. package/assets/lambdas/assets-deployment/assets-deployment.lambda/index.js +8833 -0
  9. package/assets/lambdas/revalidate/revalidate.lambda/index.js +67 -0
  10. package/assets/lambdas/sign-fn-url/sign-fn-url.edge-lambda/index.js +2024 -0
  11. package/docs/breaking-changes.md +8 -0
  12. package/docs/cdk-nextjs-NextjsGlobalContainers.png +0 -0
  13. package/docs/cdk-nextjs-NextjsGlobalFunctions.png +0 -0
  14. package/docs/cdk-nextjs-NextjsRegionalContainers.png +0 -0
  15. package/docs/cdk-nextjs.drawio +163 -0
  16. package/docs/notes.md +26 -0
  17. package/examples/README.md +15 -0
  18. package/lib/common.d.ts +23 -0
  19. package/lib/common.js +28 -0
  20. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.d.ts +106 -0
  21. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.js +3 -0
  22. package/lib/generated-structs/OptionalCloudFrontFunctionProps.d.ts +43 -0
  23. package/lib/generated-structs/OptionalCloudFrontFunctionProps.js +3 -0
  24. package/lib/generated-structs/OptionalClusterProps.d.ts +62 -0
  25. package/lib/generated-structs/OptionalClusterProps.js +3 -0
  26. package/lib/generated-structs/OptionalDistributionProps.d.ts +155 -0
  27. package/lib/generated-structs/OptionalDistributionProps.js +3 -0
  28. package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +124 -0
  29. package/lib/generated-structs/OptionalDockerImageAssetProps.js +3 -0
  30. package/lib/generated-structs/OptionalDockerImageFunctionProps.d.ts +399 -0
  31. package/lib/generated-structs/OptionalDockerImageFunctionProps.js +3 -0
  32. package/lib/generated-structs/OptionalEdgeFunctionProps.d.ts +428 -0
  33. package/lib/generated-structs/OptionalEdgeFunctionProps.js +3 -0
  34. package/lib/generated-structs/OptionalFunctionProps.d.ts +422 -0
  35. package/lib/generated-structs/OptionalFunctionProps.js +3 -0
  36. package/lib/generated-structs/OptionalFunctionUrlProps.d.ts +30 -0
  37. package/lib/generated-structs/OptionalFunctionUrlProps.js +3 -0
  38. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +40 -0
  39. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +3 -0
  40. package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +26 -0
  41. package/lib/generated-structs/OptionalNextjsBuildProps.js +3 -0
  42. package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +39 -0
  43. package/lib/generated-structs/OptionalNextjsContainersProps.js +3 -0
  44. package/lib/generated-structs/OptionalNextjsDistributionProps.d.ts +45 -0
  45. package/lib/generated-structs/OptionalNextjsDistributionProps.js +3 -0
  46. package/lib/generated-structs/OptionalNextjsFileSystemProps.d.ts +10 -0
  47. package/lib/generated-structs/OptionalNextjsFileSystemProps.js +3 -0
  48. package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +11 -0
  49. package/lib/generated-structs/OptionalNextjsInvalidationProps.js +3 -0
  50. package/lib/generated-structs/OptionalNextjsVpcProps.d.ts +16 -0
  51. package/lib/generated-structs/OptionalNextjsVpcProps.js +3 -0
  52. package/lib/generated-structs/OptionalS3OriginBucketWithOACProps.d.ts +70 -0
  53. package/lib/generated-structs/OptionalS3OriginBucketWithOACProps.js +3 -0
  54. package/lib/generated-structs/OptionalVpcProps.d.ts +224 -0
  55. package/lib/generated-structs/OptionalVpcProps.js +3 -0
  56. package/lib/index.d.ts +35 -0
  57. package/lib/index.js +32 -0
  58. package/lib/lambdas/assets-deployment/assets-deployment-function.d.ts +13 -0
  59. package/lib/lambdas/assets-deployment/assets-deployment-function.js +22 -0
  60. package/lib/lambdas/assets-deployment/assets-deployment.lambda.d.ts +2 -0
  61. package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +63 -0
  62. package/lib/lambdas/assets-deployment/common.d.ts +8 -0
  63. package/lib/lambdas/assets-deployment/common.js +32 -0
  64. package/lib/lambdas/assets-deployment/fs-to-fs.d.ts +2 -0
  65. package/lib/lambdas/assets-deployment/fs-to-fs.js +9 -0
  66. package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +2 -0
  67. package/lib/lambdas/assets-deployment/fs-to-s3.js +45 -0
  68. package/lib/lambdas/assets-deployment/prune-s3.d.ts +15 -0
  69. package/lib/lambdas/assets-deployment/prune-s3.js +42 -0
  70. package/lib/lambdas/assets-deployment/s3.d.ts +2 -0
  71. package/lib/lambdas/assets-deployment/s3.js +7 -0
  72. package/lib/lambdas/assets-deployment/utils.d.ts +18 -0
  73. package/lib/lambdas/assets-deployment/utils.js +35 -0
  74. package/lib/lambdas/revalidate/revalidate-function.d.ts +13 -0
  75. package/lib/lambdas/revalidate/revalidate-function.js +22 -0
  76. package/lib/lambdas/revalidate/revalidate.lambda.d.ts +2 -0
  77. package/lib/lambdas/revalidate/revalidate.lambda.js +53 -0
  78. package/lib/lambdas/sign-fn-url/sign-fn-url-function.d.ts +13 -0
  79. package/lib/lambdas/sign-fn-url/sign-fn-url-function.js +23 -0
  80. package/lib/lambdas/sign-fn-url/sign-fn-url.edge-lambda.d.ts +9 -0
  81. package/lib/lambdas/sign-fn-url/sign-fn-url.edge-lambda.js +35 -0
  82. package/lib/lambdas/sign-fn-url/sign-request.d.ts +28 -0
  83. package/lib/lambdas/sign-fn-url/sign-request.js +119 -0
  84. package/lib/lambdas/sign-fn-url/sign-request.test.d.ts +1 -0
  85. package/lib/lambdas/sign-fn-url/sign-request.test.js +129 -0
  86. package/lib/nextjs-assets-deployment.d.ts +116 -0
  87. package/lib/nextjs-assets-deployment.js +93 -0
  88. package/lib/nextjs-build/add-cache-handler.d.ts +1 -0
  89. package/lib/nextjs-build/add-cache-handler.js +23 -0
  90. package/lib/nextjs-build/add-cache-handler.mjs +18 -0
  91. package/lib/nextjs-build/builder.Dockerfile +25 -0
  92. package/lib/nextjs-build/cache-handler.cjs +21080 -0
  93. package/lib/nextjs-build/cache-handler.d.ts +6 -0
  94. package/lib/nextjs-build/cache-handler.js +22 -0
  95. package/lib/nextjs-build/global-containers.Dockerfile +45 -0
  96. package/lib/nextjs-build/global-functions.Dockerfile +46 -0
  97. package/lib/nextjs-build/nextjs-build.d.ts +150 -0
  98. package/lib/nextjs-build/nextjs-build.js +220 -0
  99. package/lib/nextjs-build/regional-containers.Dockerfile +45 -0
  100. package/lib/nextjs-build/symlink-full-route-cache.d.ts +1 -0
  101. package/lib/nextjs-build/symlink-full-route-cache.js +37 -0
  102. package/lib/nextjs-build/symlink-full-route-cache.mjs +23 -0
  103. package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +8 -0
  104. package/lib/nextjs-compute/nextjs-compute-base-props.js +3 -0
  105. package/lib/nextjs-compute/nextjs-containers.d.ts +43 -0
  106. package/lib/nextjs-compute/nextjs-containers.js +149 -0
  107. package/lib/nextjs-compute/nextjs-functions.d.ts +23 -0
  108. package/lib/nextjs-compute/nextjs-functions.js +57 -0
  109. package/lib/nextjs-distribution.d.ts +107 -0
  110. package/lib/nextjs-distribution.js +331 -0
  111. package/lib/nextjs-file-system.d.ts +42 -0
  112. package/lib/nextjs-file-system.js +74 -0
  113. package/lib/nextjs-invalidation.d.ts +19 -0
  114. package/lib/nextjs-invalidation.js +52 -0
  115. package/lib/nextjs-revalidation.d.ts +30 -0
  116. package/lib/nextjs-revalidation.js +65 -0
  117. package/lib/nextjs-static-assets.d.ts +21 -0
  118. package/lib/nextjs-static-assets.js +33 -0
  119. package/lib/nextjs-vpc.d.ts +42 -0
  120. package/lib/nextjs-vpc.js +64 -0
  121. package/lib/root-constructs/nextjs-base-overrides.d.ts +26 -0
  122. package/lib/root-constructs/nextjs-base-overrides.js +3 -0
  123. package/lib/root-constructs/nextjs-base-props.d.ts +56 -0
  124. package/lib/root-constructs/nextjs-base-props.js +3 -0
  125. package/lib/root-constructs/nextjs-global-containers.d.ts +74 -0
  126. package/lib/root-constructs/nextjs-global-containers.js +125 -0
  127. package/lib/root-constructs/nextjs-global-functions.d.ts +76 -0
  128. package/lib/root-constructs/nextjs-global-functions.js +131 -0
  129. package/lib/root-constructs/nextjs-regional-containers.d.ts +43 -0
  130. package/lib/root-constructs/nextjs-regional-containers.js +92 -0
  131. package/package.json +163 -2
  132. 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":"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","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
+ }