cdk-nextjs 0.0.0

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 (125) hide show
  1. package/.jsii +13811 -0
  2. package/.prettierrc +0 -0
  3. package/API.md +9694 -0
  4. package/LICENSE +202 -0
  5. package/README.md +137 -0
  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 +8831 -0
  9. package/assets/lambdas/revalidate/revalidate.lambda/index.js +67 -0
  10. package/assets/lambdas/sign-fn-url/sign-fn-url.lambda/index.js +2002 -0
  11. package/examples/README.md +14 -0
  12. package/lib/common.d.ts +23 -0
  13. package/lib/common.js +28 -0
  14. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.d.ts +106 -0
  15. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.js +3 -0
  16. package/lib/generated-structs/OptionalCloudFrontFunctionProps.d.ts +43 -0
  17. package/lib/generated-structs/OptionalCloudFrontFunctionProps.js +3 -0
  18. package/lib/generated-structs/OptionalClusterProps.d.ts +49 -0
  19. package/lib/generated-structs/OptionalClusterProps.js +3 -0
  20. package/lib/generated-structs/OptionalDistributionProps.d.ts +155 -0
  21. package/lib/generated-structs/OptionalDistributionProps.js +3 -0
  22. package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +124 -0
  23. package/lib/generated-structs/OptionalDockerImageAssetProps.js +3 -0
  24. package/lib/generated-structs/OptionalDockerImageFunctionProps.d.ts +399 -0
  25. package/lib/generated-structs/OptionalDockerImageFunctionProps.js +3 -0
  26. package/lib/generated-structs/OptionalEdgeFunctionProps.d.ts +428 -0
  27. package/lib/generated-structs/OptionalEdgeFunctionProps.js +3 -0
  28. package/lib/generated-structs/OptionalFunctionProps.d.ts +422 -0
  29. package/lib/generated-structs/OptionalFunctionProps.js +3 -0
  30. package/lib/generated-structs/OptionalFunctionUrlProps.d.ts +30 -0
  31. package/lib/generated-structs/OptionalFunctionUrlProps.js +3 -0
  32. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +45 -0
  33. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +3 -0
  34. package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +30 -0
  35. package/lib/generated-structs/OptionalNextjsBuildProps.js +3 -0
  36. package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +43 -0
  37. package/lib/generated-structs/OptionalNextjsContainersProps.js +3 -0
  38. package/lib/generated-structs/OptionalNextjsDistributionProps.d.ts +50 -0
  39. package/lib/generated-structs/OptionalNextjsDistributionProps.js +3 -0
  40. package/lib/generated-structs/OptionalNextjsFileSystemProps.d.ts +15 -0
  41. package/lib/generated-structs/OptionalNextjsFileSystemProps.js +3 -0
  42. package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +17 -0
  43. package/lib/generated-structs/OptionalNextjsInvalidationProps.js +3 -0
  44. package/lib/generated-structs/OptionalNextjsVpcProps.d.ts +21 -0
  45. package/lib/generated-structs/OptionalNextjsVpcProps.js +3 -0
  46. package/lib/generated-structs/OptionalS3OriginProps.d.ts +64 -0
  47. package/lib/generated-structs/OptionalS3OriginProps.js +3 -0
  48. package/lib/generated-structs/OptionalVpcProps.d.ts +224 -0
  49. package/lib/generated-structs/OptionalVpcProps.js +3 -0
  50. package/lib/index.d.ts +35 -0
  51. package/lib/index.js +32 -0
  52. package/lib/lambdas/assets-deployment/assets-deployment-function.d.ts +13 -0
  53. package/lib/lambdas/assets-deployment/assets-deployment-function.js +23 -0
  54. package/lib/lambdas/assets-deployment/assets-deployment.lambda.d.ts +2 -0
  55. package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +62 -0
  56. package/lib/lambdas/assets-deployment/common.d.ts +8 -0
  57. package/lib/lambdas/assets-deployment/common.js +32 -0
  58. package/lib/lambdas/assets-deployment/fs-to-fs.d.ts +2 -0
  59. package/lib/lambdas/assets-deployment/fs-to-fs.js +9 -0
  60. package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +2 -0
  61. package/lib/lambdas/assets-deployment/fs-to-s3.js +45 -0
  62. package/lib/lambdas/assets-deployment/prune-s3.d.ts +15 -0
  63. package/lib/lambdas/assets-deployment/prune-s3.js +42 -0
  64. package/lib/lambdas/assets-deployment/s3.d.ts +2 -0
  65. package/lib/lambdas/assets-deployment/s3.js +7 -0
  66. package/lib/lambdas/assets-deployment/utils.d.ts +18 -0
  67. package/lib/lambdas/assets-deployment/utils.js +35 -0
  68. package/lib/lambdas/revalidate/revalidate-function.d.ts +13 -0
  69. package/lib/lambdas/revalidate/revalidate-function.js +23 -0
  70. package/lib/lambdas/revalidate/revalidate.lambda.d.ts +2 -0
  71. package/lib/lambdas/revalidate/revalidate.lambda.js +53 -0
  72. package/lib/lambdas/sign-fn-url/sign-fn-url-function.d.ts +13 -0
  73. package/lib/lambdas/sign-fn-url/sign-fn-url-function.js +23 -0
  74. package/lib/lambdas/sign-fn-url/sign-fn-url.lambda.d.ts +9 -0
  75. package/lib/lambdas/sign-fn-url/sign-fn-url.lambda.js +35 -0
  76. package/lib/lambdas/sign-fn-url/sign-request.d.ts +28 -0
  77. package/lib/lambdas/sign-fn-url/sign-request.js +119 -0
  78. package/lib/lambdas/sign-fn-url/sign-request.test.d.ts +1 -0
  79. package/lib/lambdas/sign-fn-url/sign-request.test.js +129 -0
  80. package/lib/nextjs-assets-deployment.d.ts +116 -0
  81. package/lib/nextjs-assets-deployment.js +93 -0
  82. package/lib/nextjs-build/add-cache-handler.d.ts +1 -0
  83. package/lib/nextjs-build/add-cache-handler.js +23 -0
  84. package/lib/nextjs-build/add-cache-handler.mjs +18 -0
  85. package/lib/nextjs-build/builder.Dockerfile +29 -0
  86. package/lib/nextjs-build/cache-handler.cjs +21513 -0
  87. package/lib/nextjs-build/cache-handler.d.ts +6 -0
  88. package/lib/nextjs-build/cache-handler.js +22 -0
  89. package/lib/nextjs-build/global-containers.Dockerfile +45 -0
  90. package/lib/nextjs-build/global-functions.Dockerfile +46 -0
  91. package/lib/nextjs-build/nextjs-build.d.ts +150 -0
  92. package/lib/nextjs-build/nextjs-build.js +220 -0
  93. package/lib/nextjs-build/regional-containers.Dockerfile +45 -0
  94. package/lib/nextjs-build/symlink-full-route-cache.d.ts +1 -0
  95. package/lib/nextjs-build/symlink-full-route-cache.js +37 -0
  96. package/lib/nextjs-build/symlink-full-route-cache.mjs +23 -0
  97. package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +8 -0
  98. package/lib/nextjs-compute/nextjs-compute-base-props.js +3 -0
  99. package/lib/nextjs-compute/nextjs-containers.d.ts +43 -0
  100. package/lib/nextjs-compute/nextjs-containers.js +149 -0
  101. package/lib/nextjs-compute/nextjs-functions.d.ts +23 -0
  102. package/lib/nextjs-compute/nextjs-functions.js +57 -0
  103. package/lib/nextjs-distribution.d.ts +120 -0
  104. package/lib/nextjs-distribution.js +362 -0
  105. package/lib/nextjs-file-system.d.ts +42 -0
  106. package/lib/nextjs-file-system.js +74 -0
  107. package/lib/nextjs-invalidation.d.ts +19 -0
  108. package/lib/nextjs-invalidation.js +52 -0
  109. package/lib/nextjs-revalidation.d.ts +30 -0
  110. package/lib/nextjs-revalidation.js +65 -0
  111. package/lib/nextjs-static-assets.d.ts +21 -0
  112. package/lib/nextjs-static-assets.js +32 -0
  113. package/lib/nextjs-vpc.d.ts +42 -0
  114. package/lib/nextjs-vpc.js +64 -0
  115. package/lib/root-constructs/nextjs-base-overrides.d.ts +26 -0
  116. package/lib/root-constructs/nextjs-base-overrides.js +3 -0
  117. package/lib/root-constructs/nextjs-base-props.d.ts +56 -0
  118. package/lib/root-constructs/nextjs-base-props.js +3 -0
  119. package/lib/root-constructs/nextjs-global-containers.d.ts +74 -0
  120. package/lib/root-constructs/nextjs-global-containers.js +125 -0
  121. package/lib/root-constructs/nextjs-global-functions.d.ts +76 -0
  122. package/lib/root-constructs/nextjs-global-functions.js +131 -0
  123. package/lib/root-constructs/nextjs-regional-containers.d.ts +43 -0
  124. package/lib/root-constructs/nextjs-regional-containers.js +92 -0
  125. package/package.json +165 -0
@@ -0,0 +1,119 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.signRequest = signRequest;
4
+ exports.getRegionFromLambdaUrl = getRegionFromLambdaUrl;
5
+ exports.cfHeadersToHeaderBag = cfHeadersToHeaderBag;
6
+ exports.headerBagToCfHeaders = headerBagToCfHeaders;
7
+ exports.queryStringToQueryParamBag = queryStringToQueryParamBag;
8
+ // we bundle lambdas during build, so below dependencies actually don't need to be included in "dependencies"
9
+ // eslint-disable-next-line import/no-extraneous-dependencies
10
+ const sha256_js_1 = require("@aws-crypto/sha256-js");
11
+ // eslint-disable-next-line import/no-extraneous-dependencies
12
+ const signature_v4_1 = require("@smithy/signature-v4");
13
+ let sigv4;
14
+ /**
15
+ * Enables use of IAM_AUTH on Lambda Function URL
16
+ * @link https://medium.com/@dario_26152/restrict-access-to-lambda-functionurl-to-cloudfront-using-aws-iam-988583834705
17
+ */
18
+ async function signRequest(request) {
19
+ if (!sigv4) {
20
+ const region = getRegionFromLambdaUrl(request.origin?.custom?.domainName || "");
21
+ sigv4 = getSigV4(region);
22
+ }
23
+ // remove x-forwarded-for b/c it changes from hop to hop
24
+ delete request.headers["x-forwarded-for"];
25
+ const headerBag = cfHeadersToHeaderBag(request.headers);
26
+ const hostname = headerBag.host;
27
+ if (!hostname)
28
+ throw new Error("host header missing");
29
+ let body;
30
+ if (request.body?.data) {
31
+ body = Buffer.from(request.body.data, "base64").toString();
32
+ }
33
+ const params = queryStringToQueryParamBag(request.querystring);
34
+ const signed = await sigv4.sign({
35
+ method: request.method,
36
+ headers: headerBag,
37
+ hostname,
38
+ path: request.uri,
39
+ body,
40
+ query: params,
41
+ protocol: "https",
42
+ });
43
+ request.headers = headerBagToCfHeaders(signed.headers);
44
+ }
45
+ function getSigV4(region) {
46
+ const accessKeyId = process.env.AWS_ACCESS_KEY_ID;
47
+ const secretAccessKey = process.env.AWS_SECRET_ACCESS_KEY;
48
+ const sessionToken = process.env.AWS_SESSION_TOKEN;
49
+ if (!region)
50
+ throw new Error("AWS_REGION missing");
51
+ if (!accessKeyId)
52
+ throw new Error("AWS_ACCESS_KEY_ID missing");
53
+ if (!secretAccessKey)
54
+ throw new Error("AWS_SECRET_ACCESS_KEY missing");
55
+ if (!sessionToken)
56
+ throw new Error("AWS_SESSION_TOKEN missing");
57
+ return new signature_v4_1.SignatureV4({
58
+ service: "lambda",
59
+ region,
60
+ credentials: {
61
+ accessKeyId,
62
+ secretAccessKey,
63
+ sessionToken,
64
+ },
65
+ sha256: sha256_js_1.Sha256,
66
+ });
67
+ }
68
+ function getRegionFromLambdaUrl(url) {
69
+ const region = url.split(".").at(2);
70
+ if (!region)
71
+ throw new Error("Region couldn't be extracted from Lambda Function URL");
72
+ return region;
73
+ }
74
+ /**
75
+ * Converts CloudFront headers (can have array of header values) to simple
76
+ * header bag (object) required by `sigv4.sign`
77
+ *
78
+ * NOTE: only includes headers allowed by origin policy to prevent signature
79
+ * mismatch
80
+ */
81
+ function cfHeadersToHeaderBag(headers) {
82
+ const headerBag = {};
83
+ // assume first header value is the best match
84
+ // headerKey is case insensitive whereas key (adjacent property value that is
85
+ // not destructured) is case sensitive. we arbitrarily use case insensitive key
86
+ for (const [headerKey, [headerObj]] of Object.entries(headers)) {
87
+ const value = headerObj?.value;
88
+ if (value) {
89
+ headerBag[headerKey] = value;
90
+ }
91
+ }
92
+ return headerBag;
93
+ }
94
+ /**
95
+ * Converts simple header bag (object) to CloudFront headers
96
+ */
97
+ function headerBagToCfHeaders(headerBag) {
98
+ const cfHeaders = {};
99
+ for (const [headerKey, value] of Object.entries(headerBag)) {
100
+ /*
101
+ When your Lambda function adds or modifies request headers and you don't include the header key field, Lambda@Edge automatically inserts a header key using the header name that you provide. Regardless of how you've formatted the header name, the header key that's inserted automatically is formatted with initial capitalization for each part, separated by hyphens (-).
102
+ See: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/lambda-event-structure.html
103
+ */
104
+ cfHeaders[headerKey] = [{ value }];
105
+ }
106
+ return cfHeaders;
107
+ }
108
+ /**
109
+ * Converts CloudFront querystring to QueryParamaterBag for IAM Sig V4
110
+ */
111
+ function queryStringToQueryParamBag(querystring) {
112
+ const oldParams = new URLSearchParams(querystring);
113
+ const newParams = {};
114
+ for (const [k, v] of oldParams) {
115
+ newParams[k] = v;
116
+ }
117
+ return newParams;
118
+ }
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign-request.js","sourceRoot":"","sources":["../../../src/lambdas/sign-fn-url/sign-request.ts"],"names":[],"mappings":";;AAaA,kCA2BC;AAsBD,wDAKC;AAaD,oDAYC;AAKD,oDAUC;AAKD,gEAOC;AAvHD,6GAA6G;AAC7G,6DAA6D;AAC7D,qDAA+C;AAC/C,6DAA6D;AAC7D,uDAAmD;AAGnD,IAAI,KAAkB,CAAC;AAEvB;;;GAGG;AACI,KAAK,UAAU,WAAW,CAAC,OAA0B;IAC1D,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,MAAM,GAAG,sBAAsB,CACnC,OAAO,CAAC,MAAM,EAAE,MAAM,EAAE,UAAU,IAAI,EAAE,CACzC,CAAC;QACF,KAAK,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC;IAC3B,CAAC;IACD,wDAAwD;IACxD,OAAO,OAAO,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;IAC1C,MAAM,SAAS,GAAG,oBAAoB,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IACxD,MAAM,QAAQ,GAAG,SAAS,CAAC,IAAI,CAAC;IAChC,IAAI,CAAC,QAAQ;QAAE,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;IACtD,IAAI,IAAwB,CAAC;IAC7B,IAAI,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC;QACvB,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC7D,CAAC;IACD,MAAM,MAAM,GAAG,0BAA0B,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;IAC/D,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,IAAI,CAAC;QAC9B,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,OAAO,EAAE,SAAS;QAClB,QAAQ;QACR,IAAI,EAAE,OAAO,CAAC,GAAG;QACjB,IAAI;QACJ,KAAK,EAAE,MAAM;QACb,QAAQ,EAAE,OAAO;KAClB,CAAC,CAAC;IACH,OAAO,CAAC,OAAO,GAAG,oBAAoB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;AACzD,CAAC;AAED,SAAS,QAAQ,CAAC,MAAc;IAC9B,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC;IAClD,MAAM,eAAe,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC;IAC1D,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC;IACnD,IAAI,CAAC,MAAM;QAAE,MAAM,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC;IACnD,IAAI,CAAC,WAAW;QAAE,MAAM,IAAI,KAAK,CAAC,2BAA2B,CAAC,CAAC;IAC/D,IAAI,CAAC,eAAe;QAAE,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;IACvE,IAAI,CAAC,YAAY;QAAE,MAAM,IAAI,KAAK,CAAC,2BAA2B,CAAC,CAAC;IAChE,OAAO,IAAI,0BAAW,CAAC;QACrB,OAAO,EAAE,QAAQ;QACjB,MAAM;QACN,WAAW,EAAE;YACX,WAAW;YACX,eAAe;YACf,YAAY;SACb;QACD,MAAM,EAAE,kBAAM;KACf,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,sBAAsB,CAAC,GAAW;IAChD,MAAM,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IACpC,IAAI,CAAC,MAAM;QACT,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;IAC3E,OAAO,MAAM,CAAC;AAChB,CAAC;AAMD;;;;;;GAMG;AACH,SAAgB,oBAAoB,CAAC,OAA0B;IAC7D,MAAM,SAAS,GAAQ,EAAE,CAAC;IAC1B,8CAA8C;IAC9C,6EAA6E;IAC7E,+EAA+E;IAC/E,KAAK,MAAM,CAAC,SAAS,EAAE,CAAC,SAAS,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QAC/D,MAAM,KAAK,GAAG,SAAS,EAAE,KAAK,CAAC;QAC/B,IAAI,KAAK,EAAE,CAAC;YACV,SAAS,CAAC,SAAS,CAAC,GAAG,KAAK,CAAC;QAC/B,CAAC;IACH,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;GAEG;AACH,SAAgB,oBAAoB,CAAC,SAAc;IACjD,MAAM,SAAS,GAAsB,EAAE,CAAC;IACxC,KAAK,MAAM,CAAC,SAAS,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;QAC3D;;;UAGE;QACF,SAAS,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACrC,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;GAEG;AACH,SAAgB,0BAA0B,CAAC,WAAmB;IAC5D,MAAM,SAAS,GAAG,IAAI,eAAe,CAAC,WAAW,CAAC,CAAC;IACnD,MAAM,SAAS,GAAQ,EAAE,CAAC;IAC1B,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,SAAS,EAAE,CAAC;QAC/B,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IACnB,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC","sourcesContent":["// we bundle lambdas during build, so below dependencies actually don't need to be included in \"dependencies\"\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { Sha256 } from \"@aws-crypto/sha256-js\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { SignatureV4 } from \"@smithy/signature-v4\";\nimport type { CloudFrontHeaders, CloudFrontRequest } from \"aws-lambda\";\n\nlet sigv4: SignatureV4;\n\n/**\n * Enables use of IAM_AUTH on Lambda Function URL\n * @link https://medium.com/@dario_26152/restrict-access-to-lambda-functionurl-to-cloudfront-using-aws-iam-988583834705\n */\nexport async function signRequest(request: CloudFrontRequest) {\n  if (!sigv4) {\n    const region = getRegionFromLambdaUrl(\n      request.origin?.custom?.domainName || \"\",\n    );\n    sigv4 = getSigV4(region);\n  }\n  // remove x-forwarded-for b/c it changes from hop to hop\n  delete request.headers[\"x-forwarded-for\"];\n  const headerBag = cfHeadersToHeaderBag(request.headers);\n  const hostname = headerBag.host;\n  if (!hostname) throw new Error(\"host header missing\");\n  let body: string | undefined;\n  if (request.body?.data) {\n    body = Buffer.from(request.body.data, \"base64\").toString();\n  }\n  const params = queryStringToQueryParamBag(request.querystring);\n  const signed = await sigv4.sign({\n    method: request.method,\n    headers: headerBag,\n    hostname,\n    path: request.uri,\n    body,\n    query: params,\n    protocol: \"https\",\n  });\n  request.headers = headerBagToCfHeaders(signed.headers);\n}\n\nfunction getSigV4(region: string): SignatureV4 {\n  const accessKeyId = process.env.AWS_ACCESS_KEY_ID;\n  const secretAccessKey = process.env.AWS_SECRET_ACCESS_KEY;\n  const sessionToken = process.env.AWS_SESSION_TOKEN;\n  if (!region) throw new Error(\"AWS_REGION missing\");\n  if (!accessKeyId) throw new Error(\"AWS_ACCESS_KEY_ID missing\");\n  if (!secretAccessKey) throw new Error(\"AWS_SECRET_ACCESS_KEY missing\");\n  if (!sessionToken) throw new Error(\"AWS_SESSION_TOKEN missing\");\n  return new SignatureV4({\n    service: \"lambda\",\n    region,\n    credentials: {\n      accessKeyId,\n      secretAccessKey,\n      sessionToken,\n    },\n    sha256: Sha256,\n  });\n}\n\nexport function getRegionFromLambdaUrl(url: string): string {\n  const region = url.split(\".\").at(2);\n  if (!region)\n    throw new Error(\"Region couldn't be extracted from Lambda Function URL\");\n  return region;\n}\n\n/**\n * Bag or Map used for HeaderBag or QueryStringParameterBag for `sigv4.sign()`\n */\ntype Bag = Record<string, string>;\n/**\n * Converts CloudFront headers (can have array of header values) to simple\n * header bag (object) required by `sigv4.sign`\n *\n * NOTE: only includes headers allowed by origin policy to prevent signature\n * mismatch\n */\nexport function cfHeadersToHeaderBag(headers: CloudFrontHeaders): Bag {\n  const headerBag: Bag = {};\n  // assume first header value is the best match\n  // headerKey is case insensitive whereas key (adjacent property value that is\n  // not destructured) is case sensitive. we arbitrarily use case insensitive key\n  for (const [headerKey, [headerObj]] of Object.entries(headers)) {\n    const value = headerObj?.value;\n    if (value) {\n      headerBag[headerKey] = value;\n    }\n  }\n  return headerBag;\n}\n\n/**\n * Converts simple header bag (object) to CloudFront headers\n */\nexport function headerBagToCfHeaders(headerBag: Bag): CloudFrontHeaders {\n  const cfHeaders: CloudFrontHeaders = {};\n  for (const [headerKey, value] of Object.entries(headerBag)) {\n    /*\n      When your Lambda function adds or modifies request headers and you don't include the header key field, Lambda@Edge automatically inserts a header key using the header name that you provide. Regardless of how you've formatted the header name, the header key that's inserted automatically is formatted with initial capitalization for each part, separated by hyphens (-).\n      See: https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/lambda-event-structure.html\n    */\n    cfHeaders[headerKey] = [{ value }];\n  }\n  return cfHeaders;\n}\n\n/**\n * Converts CloudFront querystring to QueryParamaterBag for IAM Sig V4\n */\nexport function queryStringToQueryParamBag(querystring: string): Bag {\n  const oldParams = new URLSearchParams(querystring);\n  const newParams: Bag = {};\n  for (const [k, v] of oldParams) {\n    newParams[k] = v;\n  }\n  return newParams;\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,129 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const sign_request_1 = require("./sign-request");
4
+ describe("LambdaOriginRequestIamAuth", () => {
5
+ test("signRequest should add x-amz headers", async () => {
6
+ // dummy AWS credentials
7
+ process.env = { ...process.env, ...getFakeAwsCreds() };
8
+ const event = getFakePageRequest();
9
+ const request = event.Records[0]?.cf.request;
10
+ if (!request)
11
+ throw new Error("Missing request");
12
+ await (0, sign_request_1.signRequest)(request);
13
+ const securityHeaders = [
14
+ "x-amz-date",
15
+ "x-amz-security-token",
16
+ "x-amz-content-sha256",
17
+ "authorization",
18
+ ];
19
+ const hasSignedHeaders = securityHeaders.every((h) => h in request.headers);
20
+ expect(hasSignedHeaders).toBe(true);
21
+ });
22
+ test("getRegionFromLambdaUrl should correctly get region", () => {
23
+ const event = getFakePageRequest();
24
+ const request = event.Records[0]?.cf.request;
25
+ if (!request)
26
+ throw new Error("Missing request");
27
+ const actual = (0, sign_request_1.getRegionFromLambdaUrl)(request.origin?.custom?.domainName || "");
28
+ expect(actual).toBe("us-east-1");
29
+ });
30
+ });
31
+ function getFakePageRequest() {
32
+ return {
33
+ Records: [
34
+ {
35
+ cf: {
36
+ config: {
37
+ distributionDomainName: "d6b8brjqfujeb.cloudfront.net",
38
+ distributionId: "EHX2SDUU61T7U",
39
+ eventType: "origin-request",
40
+ requestId: "",
41
+ },
42
+ request: {
43
+ clientIp: "1.1.1.1",
44
+ headers: {
45
+ host: [
46
+ {
47
+ key: "Host",
48
+ value: "d6b8brjqfujeb.cloudfront.net",
49
+ },
50
+ ],
51
+ "accept-language": [
52
+ {
53
+ key: "Accept-Language",
54
+ value: "en-US,en;q=0.9",
55
+ },
56
+ ],
57
+ referer: [
58
+ {
59
+ key: "Referer",
60
+ value: "https://d6b8brjqfujeb.cloudfront.net/some/path",
61
+ },
62
+ ],
63
+ "x-forwarded-for": [
64
+ {
65
+ key: "X-Forwarded-For",
66
+ value: "1.1.1.1",
67
+ },
68
+ ],
69
+ "user-agent": [
70
+ {
71
+ key: "User-Agent",
72
+ value: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36",
73
+ },
74
+ ],
75
+ via: [
76
+ {
77
+ key: "Via",
78
+ value: "2.0 8bf94e29f889f8d0076c4502ae008b58.cloudfront.net (CloudFront)",
79
+ },
80
+ ],
81
+ "accept-encoding": [
82
+ {
83
+ key: "Accept-Encoding",
84
+ value: "br,gzip",
85
+ },
86
+ ],
87
+ "sec-ch-ua": [
88
+ {
89
+ key: "sec-ch-ua",
90
+ value: '"Google Chrome";v="113", "Chromium";v="113", "Not-A.Brand";v="24"',
91
+ },
92
+ ],
93
+ },
94
+ method: "GET",
95
+ querystring: "",
96
+ uri: "/some/path",
97
+ origin: {
98
+ custom: {
99
+ customHeaders: {},
100
+ domainName: "kjtbbx7u533q7p7n5font6gpci0phrng.lambda-url.us-east-1.on.aws",
101
+ keepaliveTimeout: 5,
102
+ path: "",
103
+ port: 443,
104
+ protocol: "https",
105
+ readTimeout: 30,
106
+ sslProtocols: ["TLSv1.2"],
107
+ },
108
+ },
109
+ body: {
110
+ action: "read-only",
111
+ data: "",
112
+ encoding: "base64",
113
+ inputTruncated: false,
114
+ },
115
+ },
116
+ },
117
+ },
118
+ ],
119
+ };
120
+ }
121
+ function getFakeAwsCreds() {
122
+ return {
123
+ AWS_REGION: "us-east-1",
124
+ AWS_ACCESS_KEY_ID: "AKIAIOSFODNN7EXAMPLE",
125
+ AWS_SECRET_ACCESS_KEY: "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
126
+ AWS_SESSION_TOKEN: "ZQoJb3JpZ2luX2VjEFgaCXVzLWVhc3QtMSJGMEQCIHijzdTXh59aSe2hRfCWpFd2/jacPUC+8rCq3qBIiuG2AiAGX8jqld+p04nPYfuShi1lLN/Z1hEXG9QSNEmEFLTxGSqmAgiR//////////8BEAIaDDI2ODkxNDQ2NTIzMSIMrAMO5/GTvMgoG+chKvoB4f4V1TfkZiHOlmeMK6Ep58mav65A0WU3K9WPzdrJojnGqqTuS85zTlKhm3lfmMxCOtwS/OlOuiBQ1MZNlksK2je1FazgbXN46fNSi+iHiY9VfyRAd0wSLmXB8FFrCGsU92QOy/+deji0qIVadsjEyvBRxzQj5oIUI5sb74Yt7uNvka9fVZcT4s4IndYda0N7oZwIrApCuzzBMuoMAhabmgVrZTbiLmvOiFHS2XZWBySABdygqaIzfV7G4hjckvcXhtxpkw+HJUZTNzVUlspghzte1UG6VvIRV8ax3kWA3zqm8nA/1gHkl40DubJIXz1AJbg5Cps5moE1pjD7vNijBjqeAZh0Q/e0awIHnV4dXMfXUu5mWJ7Db9K1eUlSSL9FyiKeKd94HEdrbIrnPuIWVT/I/5RjNm7NgPYiqmpyx3fSpVcq9CKws0oEfBw6J9Hxk0IhV8yWFZYNMWIarUUZdmL9vVeJmFZmwyL4JjY1s/SZIU/oa8DtvkmP4RG4tTJfpyyhoKL0wJOevkYyoigNllBlLN59SZAT8CCADpN/B+sK",
127
+ };
128
+ }
129
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign-request.test.js","sourceRoot":"","sources":["../../../src/lambdas/sign-fn-url/sign-request.test.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CloudFrontRequestEvent } from \"aws-lambda\";\nimport { getRegionFromLambdaUrl, signRequest } from \"./sign-request\";\n\ndescribe(\"LambdaOriginRequestIamAuth\", () => {\n  test(\"signRequest should add x-amz headers\", async () => {\n    // dummy AWS credentials\n    process.env = { ...process.env, ...getFakeAwsCreds() };\n    const event = getFakePageRequest();\n    const request = event.Records[0]?.cf.request;\n    if (!request) throw new Error(\"Missing request\");\n    await signRequest(request);\n    const securityHeaders = [\n      \"x-amz-date\",\n      \"x-amz-security-token\",\n      \"x-amz-content-sha256\",\n      \"authorization\",\n    ];\n    const hasSignedHeaders = securityHeaders.every((h) => h in request.headers);\n    expect(hasSignedHeaders).toBe(true);\n  });\n\n  test(\"getRegionFromLambdaUrl should correctly get region\", () => {\n    const event = getFakePageRequest();\n    const request = event.Records[0]?.cf.request;\n    if (!request) throw new Error(\"Missing request\");\n    const actual = getRegionFromLambdaUrl(\n      request.origin?.custom?.domainName || \"\",\n    );\n    expect(actual).toBe(\"us-east-1\");\n  });\n});\n\nfunction getFakePageRequest(): CloudFrontRequestEvent {\n  return {\n    Records: [\n      {\n        cf: {\n          config: {\n            distributionDomainName: \"d6b8brjqfujeb.cloudfront.net\",\n            distributionId: \"EHX2SDUU61T7U\",\n            eventType: \"origin-request\",\n            requestId: \"\",\n          },\n          request: {\n            clientIp: \"1.1.1.1\",\n            headers: {\n              host: [\n                {\n                  key: \"Host\",\n                  value: \"d6b8brjqfujeb.cloudfront.net\",\n                },\n              ],\n              \"accept-language\": [\n                {\n                  key: \"Accept-Language\",\n                  value: \"en-US,en;q=0.9\",\n                },\n              ],\n              referer: [\n                {\n                  key: \"Referer\",\n                  value: \"https://d6b8brjqfujeb.cloudfront.net/some/path\",\n                },\n              ],\n              \"x-forwarded-for\": [\n                {\n                  key: \"X-Forwarded-For\",\n                  value: \"1.1.1.1\",\n                },\n              ],\n              \"user-agent\": [\n                {\n                  key: \"User-Agent\",\n                  value:\n                    \"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36\",\n                },\n              ],\n              via: [\n                {\n                  key: \"Via\",\n                  value:\n                    \"2.0 8bf94e29f889f8d0076c4502ae008b58.cloudfront.net (CloudFront)\",\n                },\n              ],\n              \"accept-encoding\": [\n                {\n                  key: \"Accept-Encoding\",\n                  value: \"br,gzip\",\n                },\n              ],\n              \"sec-ch-ua\": [\n                {\n                  key: \"sec-ch-ua\",\n                  value:\n                    '\"Google Chrome\";v=\"113\", \"Chromium\";v=\"113\", \"Not-A.Brand\";v=\"24\"',\n                },\n              ],\n            },\n            method: \"GET\",\n            querystring: \"\",\n            uri: \"/some/path\",\n            origin: {\n              custom: {\n                customHeaders: {},\n                domainName:\n                  \"kjtbbx7u533q7p7n5font6gpci0phrng.lambda-url.us-east-1.on.aws\",\n                keepaliveTimeout: 5,\n                path: \"\",\n                port: 443,\n                protocol: \"https\",\n                readTimeout: 30,\n                sslProtocols: [\"TLSv1.2\"],\n              },\n            },\n            body: {\n              action: \"read-only\",\n              data: \"\",\n              encoding: \"base64\",\n              inputTruncated: false,\n            },\n          },\n        },\n      },\n    ],\n  };\n}\n\nfunction getFakeAwsCreds() {\n  return {\n    AWS_REGION: \"us-east-1\",\n    AWS_ACCESS_KEY_ID: \"AKIAIOSFODNN7EXAMPLE\",\n    AWS_SECRET_ACCESS_KEY: \"wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY\",\n    AWS_SESSION_TOKEN:\n      \"ZQoJb3JpZ2luX2VjEFgaCXVzLWVhc3QtMSJGMEQCIHijzdTXh59aSe2hRfCWpFd2/jacPUC+8rCq3qBIiuG2AiAGX8jqld+p04nPYfuShi1lLN/Z1hEXG9QSNEmEFLTxGSqmAgiR//////////8BEAIaDDI2ODkxNDQ2NTIzMSIMrAMO5/GTvMgoG+chKvoB4f4V1TfkZiHOlmeMK6Ep58mav65A0WU3K9WPzdrJojnGqqTuS85zTlKhm3lfmMxCOtwS/OlOuiBQ1MZNlksK2je1FazgbXN46fNSi+iHiY9VfyRAd0wSLmXB8FFrCGsU92QOy/+deji0qIVadsjEyvBRxzQj5oIUI5sb74Yt7uNvka9fVZcT4s4IndYda0N7oZwIrApCuzzBMuoMAhabmgVrZTbiLmvOiFHS2XZWBySABdygqaIzfV7G4hjckvcXhtxpkw+HJUZTNzVUlspghzte1UG6VvIRV8ax3kWA3zqm8nA/1gHkl40DubJIXz1AJbg5Cps5moE1pjD7vNijBjqeAZh0Q/e0awIHnV4dXMfXUu5mWJ7Db9K1eUlSSL9FyiKeKd94HEdrbIrnPuIWVT/I/5RjNm7NgPYiqmpyx3fSpVcq9CKws0oEfBw6J9Hxk0IhV8yWFZYNMWIarUUZdmL9vVeJmFZmwyL4JjY1s/SZIU/oa8DtvkmP4RG4tTJfpyyhoKL0wJOevkYyoigNllBlLN59SZAT8CCADpN/B+sK\",\n  };\n}\n"]}
@@ -0,0 +1,116 @@
1
+ import { CustomResource } from "aws-cdk-lib";
2
+ import { IVpc } from "aws-cdk-lib/aws-ec2";
3
+ import { AccessPoint } from "aws-cdk-lib/aws-efs";
4
+ import { DockerImageCode, DockerImageFunction } from "aws-cdk-lib/aws-lambda";
5
+ import { Bucket } from "aws-cdk-lib/aws-s3";
6
+ import { Construct } from "constructs";
7
+ import { OptionalDockerImageFunctionProps } from "./generated-structs/OptionalDockerImageFunctionProps";
8
+ import { NextjsBuild } from "./nextjs-build/nextjs-build";
9
+ export interface NextjsAssetDeploymentOverrides {
10
+ readonly dockerImageFunctionProps?: OptionalDockerImageFunctionProps;
11
+ }
12
+ export interface NextjsAssetsDeploymentProps {
13
+ readonly accessPoint: AccessPoint;
14
+ /**
15
+ * @see {@link NextjsBuild.buildImageDigest}
16
+ */
17
+ readonly buildImageDigest: string;
18
+ /**
19
+ * @default true
20
+ */
21
+ readonly debug?: boolean;
22
+ readonly dockerImageCode: DockerImageCode;
23
+ /**
24
+ * @see {@link NextjsBuild.containerMountPathForEfs}
25
+ */
26
+ readonly containerMountPathForEfs: NextjsBuild["containerMountPathForEfs"];
27
+ readonly overrides?: NextjsAssetDeploymentOverrides;
28
+ /**
29
+ * @see {@link NextjsBaseProps.relativePathToWorkspace}
30
+ */
31
+ readonly relativePathToWorkspace?: string;
32
+ /**
33
+ * Required for `NextjsType.GlobalFunctions` and `NextjsType.GlobalContainers`
34
+ */
35
+ readonly staticAssetsBucket?: Bucket;
36
+ readonly vpc: IVpc;
37
+ }
38
+ /**
39
+ * @internal
40
+ */
41
+ export interface FsToS3Action {
42
+ type: "fs-to-s3";
43
+ destinationBucketName: string;
44
+ destinationKeyPrefix?: string;
45
+ sourcePath: string;
46
+ }
47
+ /**
48
+ * @internal
49
+ */
50
+ export interface FsToFsAction {
51
+ type: "fs-to-fs";
52
+ sourcePath: string;
53
+ destinationPath: string;
54
+ }
55
+ /**
56
+ * @internal
57
+ */
58
+ export interface PruneS3Action {
59
+ type: "prune-s3";
60
+ /**
61
+ * The minimum previous deployment count to prune
62
+ * @default 3
63
+ */
64
+ minPreviousDeployCountToPrune: number;
65
+ /**
66
+ * The minimum previous deployment date to prune
67
+ * @default new Date(new Date().setMonth(new Date().getMonth() - 1))
68
+ */
69
+ minPreviousDeployDateToPrune: string;
70
+ bucketName: string;
71
+ bucketPrefix?: string;
72
+ }
73
+ /**
74
+ * @internal
75
+ */
76
+ export interface PruneFsAction {
77
+ type: "prune-fs";
78
+ /**
79
+ * The minimum previous deployment count to prune
80
+ * @default 3
81
+ */
82
+ minPreviousDeployCountToPrune: number;
83
+ /**
84
+ * The minimum previous deployment date to prune
85
+ * @default new Date(new Date().setMonth(new Date().getMonth() - 1))
86
+ */
87
+ minPreviousDeployDateToPrune: string;
88
+ directory: string;
89
+ }
90
+ /**
91
+ * @internal
92
+ */
93
+ export interface CustomResourceProperties {
94
+ actions: (FsToS3Action | FsToFsAction | PruneS3Action | PruneFsAction)[];
95
+ /**
96
+ * {@link NextjsAssetDeploymentProps.builderImageDigest}
97
+ */
98
+ buildImageDigest: string;
99
+ imageCachePath: string;
100
+ prerenderManifestPath: string;
101
+ }
102
+ /**
103
+ * Deploys static assets to S3 and cache assets to EFS in Lambda Custom Resource.
104
+ */
105
+ export declare class NextjsAssetsDeployment extends Construct {
106
+ customResource: CustomResource;
107
+ dockerImageFunction: DockerImageFunction;
108
+ /**
109
+ * Only used for `NextjsGlobalFunctions`
110
+ */
111
+ previewModeId: string;
112
+ private props;
113
+ constructor(scope: Construct, id: string, props: NextjsAssetsDeploymentProps);
114
+ private createFunction;
115
+ private createCustomResource;
116
+ }
@@ -0,0 +1,93 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.NextjsAssetsDeployment = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const posix_1 = require("path/posix");
7
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
8
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
9
+ const constructs_1 = require("constructs");
10
+ const common_1 = require("./common");
11
+ /**
12
+ * Deploys static assets to S3 and cache assets to EFS in Lambda Custom Resource.
13
+ */
14
+ class NextjsAssetsDeployment extends constructs_1.Construct {
15
+ constructor(scope, id, props) {
16
+ super(scope, id);
17
+ this.props = props;
18
+ this.dockerImageFunction = this.createFunction();
19
+ this.customResource = this.createCustomResource();
20
+ this.previewModeId = this.customResource.getAttString("previewModeId");
21
+ }
22
+ createFunction() {
23
+ let architecture = undefined;
24
+ if (process.arch === "x64") {
25
+ architecture = aws_lambda_1.Architecture.X86_64;
26
+ }
27
+ else if (process.arch === "arm64") {
28
+ architecture = aws_lambda_1.Architecture.ARM_64;
29
+ }
30
+ const fn = new aws_lambda_1.DockerImageFunction(this, "Fn", {
31
+ architecture,
32
+ code: this.props.dockerImageCode,
33
+ memorySize: 2048,
34
+ filesystem: aws_lambda_1.FileSystem.fromEfsAccessPoint(this.props.accessPoint, this.props.containerMountPathForEfs),
35
+ vpc: this.props.vpc,
36
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
37
+ ...this.props.overrides?.dockerImageFunctionProps,
38
+ });
39
+ if (this.props.debug !== false) {
40
+ fn.addEnvironment("DEBUG", "1");
41
+ }
42
+ if (this.props.staticAssetsBucket) {
43
+ this.props.staticAssetsBucket.grantReadWrite(fn);
44
+ }
45
+ return fn;
46
+ }
47
+ createCustomResource() {
48
+ const root = "/app";
49
+ const actions = [];
50
+ if (this.props.staticAssetsBucket?.bucketName) {
51
+ actions.push({
52
+ type: "fs-to-s3",
53
+ sourcePath: (0, posix_1.join)(root, "public"),
54
+ destinationBucketName: this.props.staticAssetsBucket.bucketName,
55
+ },
56
+ // static files
57
+ {
58
+ type: "fs-to-s3",
59
+ sourcePath: (0, posix_1.join)(root, ".next", "static"),
60
+ destinationBucketName: this.props.staticAssetsBucket.bucketName,
61
+ destinationKeyPrefix: "_next/static",
62
+ });
63
+ }
64
+ actions.push(
65
+ // data cache - https://nextjs.org/docs/app/building-your-application/caching#data-cache
66
+ {
67
+ type: "fs-to-fs",
68
+ sourcePath: (0, posix_1.join)(root, ".next", "cache", "fetch-cache"),
69
+ destinationPath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.DATA_CACHE_DIR),
70
+ },
71
+ // full route cache - https://nextjs.org/docs/app/building-your-application/caching#full-route-cache
72
+ {
73
+ type: "fs-to-fs",
74
+ sourcePath: (0, posix_1.join)(root, ".next", "standalone", this.props.relativePathToWorkspace || "", ".next", "server", "app"),
75
+ destinationPath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.FULL_ROUTE_CACHE_DIR),
76
+ });
77
+ const properties = {
78
+ actions,
79
+ buildImageDigest: this.props.buildImageDigest,
80
+ imageCachePath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.IMAGE_CACHE_DIR),
81
+ prerenderManifestPath: (0, posix_1.join)(root, ".next", "prerender-manifest.json"),
82
+ };
83
+ return new aws_cdk_lib_1.CustomResource(this, "CustomResource", {
84
+ properties,
85
+ resourceType: "Custom::NextjsAssetsDeployment",
86
+ serviceToken: this.dockerImageFunction.functionArn,
87
+ });
88
+ }
89
+ }
90
+ exports.NextjsAssetsDeployment = NextjsAssetsDeployment;
91
+ _a = JSII_RTTI_SYMBOL_1;
92
+ NextjsAssetsDeployment[_a] = { fqn: "cdk-nextjs.NextjsAssetsDeployment", version: "0.0.0" };
93
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-assets-deployment.js","sourceRoot":"","sources":["../src/nextjs-assets-deployment.ts"],"names":[],"mappings":"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","sourcesContent":["import { join } from \"path/posix\";\nimport { CustomResource, Duration } from \"aws-cdk-lib\";\nimport { IVpc } from \"aws-cdk-lib/aws-ec2\";\nimport { AccessPoint } from \"aws-cdk-lib/aws-efs\";\nimport {\n  Architecture,\n  DockerImageCode,\n  DockerImageFunction,\n  FileSystem,\n} from \"aws-cdk-lib/aws-lambda\";\nimport { Bucket } from \"aws-cdk-lib/aws-s3\";\nimport { Construct } from \"constructs\";\nimport {\n  DATA_CACHE_DIR,\n  FULL_ROUTE_CACHE_DIR,\n  IMAGE_CACHE_DIR,\n} from \"./common\";\nimport { OptionalDockerImageFunctionProps } from \"./generated-structs/OptionalDockerImageFunctionProps\";\nimport { NextjsBuild } from \"./nextjs-build/nextjs-build\";\nimport { NextjsBaseProps } from \"./root-constructs/nextjs-base-props\";\n\nexport interface NextjsAssetDeploymentOverrides {\n  readonly dockerImageFunctionProps?: OptionalDockerImageFunctionProps;\n}\n\nexport interface NextjsAssetsDeploymentProps {\n  readonly accessPoint: AccessPoint;\n  /**\n   * @see {@link NextjsBuild.buildImageDigest}\n   */\n  readonly buildImageDigest: string;\n  /**\n   * @default true\n   */\n  readonly debug?: boolean;\n  readonly dockerImageCode: DockerImageCode; // TODO: remove and build from common builder base?\n  /**\n   * @see {@link NextjsBuild.containerMountPathForEfs}\n   */\n  readonly containerMountPathForEfs: NextjsBuild[\"containerMountPathForEfs\"];\n  readonly overrides?: NextjsAssetDeploymentOverrides;\n  /**\n   * @see {@link NextjsBaseProps.relativePathToWorkspace}\n   */\n  readonly relativePathToWorkspace?: string;\n  /**\n   * Required for `NextjsType.GlobalFunctions` and `NextjsType.GlobalContainers`\n   */\n  readonly staticAssetsBucket?: Bucket;\n  readonly vpc: IVpc;\n}\n\n/**\n * @internal\n */\nexport interface FsToS3Action {\n  type: \"fs-to-s3\";\n  destinationBucketName: string;\n  destinationKeyPrefix?: string;\n  sourcePath: string;\n}\n/**\n * @internal\n */\nexport interface FsToFsAction {\n  type: \"fs-to-fs\";\n  sourcePath: string;\n  destinationPath: string;\n}\n/**\n * @internal\n */\nexport interface PruneS3Action {\n  type: \"prune-s3\";\n  /**\n   * The minimum previous deployment count to prune\n   * @default 3\n   */\n  minPreviousDeployCountToPrune: number;\n  /**\n   * The minimum previous deployment date to prune\n   * @default new Date(new Date().setMonth(new Date().getMonth() - 1))\n   */\n  minPreviousDeployDateToPrune: string;\n  bucketName: string;\n  bucketPrefix?: string;\n}\n/**\n * @internal\n */\nexport interface PruneFsAction {\n  type: \"prune-fs\";\n  /**\n   * The minimum previous deployment count to prune\n   * @default 3\n   */\n  minPreviousDeployCountToPrune: number;\n  /**\n   * The minimum previous deployment date to prune\n   * @default new Date(new Date().setMonth(new Date().getMonth() - 1))\n   */\n  minPreviousDeployDateToPrune: string;\n  directory: string;\n}\n\n/**\n * @internal\n */\nexport interface CustomResourceProperties {\n  actions: (FsToS3Action | FsToFsAction | PruneS3Action | PruneFsAction)[];\n  /**\n   * {@link NextjsAssetDeploymentProps.builderImageDigest}\n   */\n  buildImageDigest: string;\n  imageCachePath: string;\n  prerenderManifestPath: string;\n}\n\n/**\n * Deploys static assets to S3 and cache assets to EFS in Lambda Custom Resource.\n */\nexport class NextjsAssetsDeployment extends Construct {\n  customResource: CustomResource;\n  dockerImageFunction: DockerImageFunction;\n  /**\n   * Only used for `NextjsGlobalFunctions`\n   */\n  previewModeId: string;\n\n  private props: NextjsAssetsDeploymentProps;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: NextjsAssetsDeploymentProps,\n  ) {\n    super(scope, id);\n    this.props = props;\n    this.dockerImageFunction = this.createFunction();\n    this.customResource = this.createCustomResource();\n    this.previewModeId = this.customResource.getAttString(\"previewModeId\");\n  }\n\n  private createFunction() {\n    let architecture: Architecture | undefined = undefined;\n    if (process.arch === \"x64\") {\n      architecture = Architecture.X86_64;\n    } else if (process.arch === \"arm64\") {\n      architecture = Architecture.ARM_64;\n    }\n    const fn = new DockerImageFunction(this, \"Fn\", {\n      architecture,\n      code: this.props.dockerImageCode,\n      memorySize: 2048,\n      filesystem: FileSystem.fromEfsAccessPoint(\n        this.props.accessPoint,\n        this.props.containerMountPathForEfs,\n      ),\n      vpc: this.props.vpc,\n      timeout: Duration.minutes(5),\n      ...this.props.overrides?.dockerImageFunctionProps,\n    });\n    if (this.props.debug !== false) {\n      fn.addEnvironment(\"DEBUG\", \"1\");\n    }\n    if (this.props.staticAssetsBucket) {\n      this.props.staticAssetsBucket.grantReadWrite(fn);\n    }\n    return fn;\n  }\n\n  private createCustomResource() {\n    const root = \"/app\";\n    const actions: CustomResourceProperties[\"actions\"] = [];\n    if (this.props.staticAssetsBucket?.bucketName) {\n      actions.push(\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, \"public\"),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\n        },\n        // static files\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, \".next\", \"static\"),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\n          destinationKeyPrefix: \"_next/static\",\n        },\n      );\n    }\n    actions.push(\n      // data cache - https://nextjs.org/docs/app/building-your-application/caching#data-cache\n      {\n        type: \"fs-to-fs\",\n        sourcePath: join(root, \".next\", \"cache\", \"fetch-cache\"),\n        destinationPath: join(\n          this.props.containerMountPathForEfs,\n          DATA_CACHE_DIR,\n        ),\n      },\n      // full route cache - https://nextjs.org/docs/app/building-your-application/caching#full-route-cache\n      {\n        type: \"fs-to-fs\",\n        sourcePath: join(\n          root,\n          \".next\",\n          \"standalone\",\n          this.props.relativePathToWorkspace || \"\",\n          \".next\",\n          \"server\",\n          \"app\",\n        ),\n        destinationPath: join(\n          this.props.containerMountPathForEfs,\n          FULL_ROUTE_CACHE_DIR,\n        ),\n      },\n      // images are optimized at runtime so nothing to deploy\n    );\n    const properties: CustomResourceProperties = {\n      actions,\n      buildImageDigest: this.props.buildImageDigest,\n      imageCachePath: join(\n        this.props.containerMountPathForEfs,\n        IMAGE_CACHE_DIR,\n      ),\n      prerenderManifestPath: join(root, \".next\", \"prerender-manifest.json\"),\n    };\n    return new CustomResource(this, \"CustomResource\", {\n      properties,\n      resourceType: \"Custom::NextjsAssetsDeployment\",\n      serviceToken: this.dockerImageFunction.functionArn,\n    });\n  }\n}\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const fs_1 = require("fs");
4
+ /**
5
+ * For `NextjsGlobalFunctions`, adds custom [Cache Handler](https://nextjs.org/docs/app/api-reference/next-config-js/incrementalCacheHandlerPath)
6
+ * so that time-based revalidation works. In serverless environment, functions
7
+ * spin down after request is complete but Next.js depends upon compute to still
8
+ * be running for time-based revalidation to work properly so we need to perform
9
+ * work async and use SQS Queue to do so.
10
+ */
11
+ function addCacheHandler(requiredServerFilesPath) {
12
+ if (!(0, fs_1.existsSync)(requiredServerFilesPath)) {
13
+ throw new Error(`Could not find required server files path: ${requiredServerFilesPath}`);
14
+ }
15
+ const requiredServerFiles = JSON.parse((0, fs_1.readFileSync)(requiredServerFilesPath, {
16
+ encoding: "utf-8",
17
+ }));
18
+ requiredServerFiles.config.cacheHandler = "../cache-handler.cjs";
19
+ (0, fs_1.writeFileSync)(requiredServerFilesPath, JSON.stringify(requiredServerFiles));
20
+ }
21
+ const [requiredServerFilesPath] = process.argv.slice(2);
22
+ addCacheHandler(requiredServerFilesPath);
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWRkLWNhY2hlLWhhbmRsZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbmV4dGpzLWJ1aWxkL2FkZC1jYWNoZS1oYW5kbGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsMkJBQTZEO0FBRTdEOzs7Ozs7R0FNRztBQUNILFNBQVMsZUFBZSxDQUFDLHVCQUErQjtJQUN0RCxJQUFJLENBQUMsSUFBQSxlQUFVLEVBQUMsdUJBQXVCLENBQUMsRUFBRSxDQUFDO1FBQ3pDLE1BQU0sSUFBSSxLQUFLLENBQ2IsOENBQThDLHVCQUF1QixFQUFFLENBQ3hFLENBQUM7SUFDSixDQUFDO0lBQ0QsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUNwQyxJQUFBLGlCQUFZLEVBQUMsdUJBQXVCLEVBQUU7UUFDcEMsUUFBUSxFQUFFLE9BQU87S0FDbEIsQ0FBQyxDQUNILENBQUM7SUFDRixtQkFBbUIsQ0FBQyxNQUFNLENBQUMsWUFBWSxHQUFHLHNCQUFzQixDQUFDO0lBQ2pFLElBQUEsa0JBQWEsRUFBQyx1QkFBdUIsRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLG1CQUFtQixDQUFDLENBQUMsQ0FBQztBQUM5RSxDQUFDO0FBRUQsTUFBTSxDQUFDLHVCQUF1QixDQUFDLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7QUFDeEQsZUFBZSxDQUFDLHVCQUF1QixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBleGlzdHNTeW5jLCByZWFkRmlsZVN5bmMsIHdyaXRlRmlsZVN5bmMgfSBmcm9tIFwiZnNcIjtcblxuLyoqXG4gKiBGb3IgYE5leHRqc0dsb2JhbEZ1bmN0aW9uc2AsIGFkZHMgY3VzdG9tIFtDYWNoZSBIYW5kbGVyXShodHRwczovL25leHRqcy5vcmcvZG9jcy9hcHAvYXBpLXJlZmVyZW5jZS9uZXh0LWNvbmZpZy1qcy9pbmNyZW1lbnRhbENhY2hlSGFuZGxlclBhdGgpXG4gKiBzbyB0aGF0IHRpbWUtYmFzZWQgcmV2YWxpZGF0aW9uIHdvcmtzLiBJbiBzZXJ2ZXJsZXNzIGVudmlyb25tZW50LCBmdW5jdGlvbnNcbiAqIHNwaW4gZG93biBhZnRlciByZXF1ZXN0IGlzIGNvbXBsZXRlIGJ1dCBOZXh0LmpzIGRlcGVuZHMgdXBvbiBjb21wdXRlIHRvIHN0aWxsXG4gKiBiZSBydW5uaW5nIGZvciB0aW1lLWJhc2VkIHJldmFsaWRhdGlvbiB0byB3b3JrIHByb3Blcmx5IHNvIHdlIG5lZWQgdG8gcGVyZm9ybVxuICogd29yayBhc3luYyBhbmQgdXNlIFNRUyBRdWV1ZSB0byBkbyBzby5cbiAqL1xuZnVuY3Rpb24gYWRkQ2FjaGVIYW5kbGVyKHJlcXVpcmVkU2VydmVyRmlsZXNQYXRoOiBzdHJpbmcpIHtcbiAgaWYgKCFleGlzdHNTeW5jKHJlcXVpcmVkU2VydmVyRmlsZXNQYXRoKSkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIGBDb3VsZCBub3QgZmluZCByZXF1aXJlZCBzZXJ2ZXIgZmlsZXMgcGF0aDogJHtyZXF1aXJlZFNlcnZlckZpbGVzUGF0aH1gLFxuICAgICk7XG4gIH1cbiAgY29uc3QgcmVxdWlyZWRTZXJ2ZXJGaWxlcyA9IEpTT04ucGFyc2UoXG4gICAgcmVhZEZpbGVTeW5jKHJlcXVpcmVkU2VydmVyRmlsZXNQYXRoLCB7XG4gICAgICBlbmNvZGluZzogXCJ1dGYtOFwiLFxuICAgIH0pLFxuICApO1xuICByZXF1aXJlZFNlcnZlckZpbGVzLmNvbmZpZy5jYWNoZUhhbmRsZXIgPSBcIi4uL2NhY2hlLWhhbmRsZXIuY2pzXCI7XG4gIHdyaXRlRmlsZVN5bmMocmVxdWlyZWRTZXJ2ZXJGaWxlc1BhdGgsIEpTT04uc3RyaW5naWZ5KHJlcXVpcmVkU2VydmVyRmlsZXMpKTtcbn1cblxuY29uc3QgW3JlcXVpcmVkU2VydmVyRmlsZXNQYXRoXSA9IHByb2Nlc3MuYXJndi5zbGljZSgyKTtcbmFkZENhY2hlSGFuZGxlcihyZXF1aXJlZFNlcnZlckZpbGVzUGF0aCk7XG4iXX0=
@@ -0,0 +1,18 @@
1
+ // src/nextjs-build/add-cache-handler.ts
2
+ import { existsSync, readFileSync, writeFileSync } from "fs";
3
+ function addCacheHandler(requiredServerFilesPath2) {
4
+ if (!existsSync(requiredServerFilesPath2)) {
5
+ throw new Error(
6
+ `Could not find required server files path: ${requiredServerFilesPath2}`
7
+ );
8
+ }
9
+ const requiredServerFiles = JSON.parse(
10
+ readFileSync(requiredServerFilesPath2, {
11
+ encoding: "utf-8"
12
+ })
13
+ );
14
+ requiredServerFiles.config.cacheHandler = "../cache-handler.cjs";
15
+ writeFileSync(requiredServerFilesPath2, JSON.stringify(requiredServerFiles));
16
+ }
17
+ var [requiredServerFilesPath] = process.argv.slice(2);
18
+ addCacheHandler(requiredServerFilesPath);
@@ -0,0 +1,29 @@
1
+ #checkov:skip=CKV_DOCKER_2: healthcheck not needed for local only container
2
+ #checkov:skip=CKV_DOCKER_3: user not required for local builder container
3
+ FROM public.ecr.aws/docker/library/node:20-alpine
4
+ # Check https://github.com/nodejs/docker-node/tree/b4117f9333da4138b03a546ec926ef50a31506c3#nodealpine to understand why libc6-compat might be needed.
5
+ RUN apk add --no-cache libc6-compat
6
+ WORKDIR /app
7
+ # It's preferrable to only copy package.json's and lockfiles, install, and then
8
+ # copy the rest of the code to efficiently utilize build cache. We don't do that
9
+ #$ here because it's highly customized based on a projects setup. We recommend
10
+ # developers use `overrides.nextjsXX.nextjsBuildProps.builderImageProps.srcDockerfilePath`
11
+ # to optimize for their setup.
12
+ COPY . .
13
+ # Install dependencies based on the preferred package manager
14
+ RUN \
15
+ if [ -f yarn.lock ]; then yarn --frozen-lockfile; \
16
+ elif [ -f package-lock.json ]; then npm ci; \
17
+ elif [ -f pnpm-lock.yaml ]; then corepack enable pnpm && pnpm i --frozen-lockfile; \
18
+ else echo "Lockfile not found." && exit 1; \
19
+ fi
20
+ ARG BUILD_COMMAND
21
+ ARG RELATIVE_PATH_TO_WORKSPACE
22
+ RUN \
23
+ if [ -f ./cdk-nextjs-load-env-vars.sh ]; then \
24
+ chmod u+x ./cdk-nextjs-load-env-vars.sh && \
25
+ . ./cdk-nextjs-load-env-vars.sh; \
26
+ fi && \
27
+ cd $RELATIVE_PATH_TO_WORKSPACE && \
28
+ $BUILD_COMMAND
29
+ # TODO: remove unnecessary node_modules: https://github.com/sst/open-next/pull/242