cdk-nextjs 0.3.11 → 0.4.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 (106) hide show
  1. package/.jsii +1248 -748
  2. package/API.md +1033 -531
  3. package/README.md +1 -2
  4. package/assets/lambdas/assets-deployment/assets-deployment.lambda/assets-deployment.Dockerfile +12 -8
  5. package/assets/lambdas/assets-deployment/assets-deployment.lambda/index.js +67 -56
  6. package/assets/lambdas/post-deploy/post-deploy.lambda/index.js +246 -0
  7. package/docs/breaking-changes.md +9 -0
  8. package/docs/notes.md +170 -20
  9. package/lib/constants.d.ts +32 -0
  10. package/lib/constants.js +38 -0
  11. package/lib/generated-structs/OptionalCustomResourceProps.d.ts +104 -0
  12. package/lib/generated-structs/OptionalCustomResourceProps.js +3 -0
  13. package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +21 -0
  14. package/lib/generated-structs/OptionalDockerImageAssetProps.js +1 -1
  15. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +4 -3
  16. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +1 -1
  17. package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +1 -1
  18. package/lib/generated-structs/OptionalNextjsBuildProps.js +1 -1
  19. package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +3 -3
  20. package/lib/generated-structs/OptionalNextjsContainersProps.js +1 -1
  21. package/lib/generated-structs/OptionalNextjsPostDeployProps.d.ts +42 -0
  22. package/lib/generated-structs/OptionalNextjsPostDeployProps.js +3 -0
  23. package/lib/generated-structs/OptionalPostDeployCustomResourceProperties.d.ts +41 -0
  24. package/lib/generated-structs/OptionalPostDeployCustomResourceProperties.js +4 -0
  25. package/lib/index.d.ts +6 -5
  26. package/lib/index.js +6 -8
  27. package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +5 -23
  28. package/lib/lambdas/assets-deployment/fs-to-fs.js +18 -3
  29. package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +8 -2
  30. package/lib/lambdas/assets-deployment/fs-to-s3.js +14 -8
  31. package/lib/lambdas/post-deploy/create-invalidation.d.ts +2 -0
  32. package/lib/lambdas/post-deploy/create-invalidation.js +19 -0
  33. package/lib/lambdas/post-deploy/post-deploy-function.d.ts +13 -0
  34. package/lib/lambdas/post-deploy/post-deploy-function.js +22 -0
  35. package/lib/lambdas/post-deploy/post-deploy.lambda.d.ts +9 -0
  36. package/lib/lambdas/post-deploy/post-deploy.lambda.js +56 -0
  37. package/lib/lambdas/post-deploy/prune-fs.d.ts +10 -0
  38. package/lib/lambdas/post-deploy/prune-fs.js +33 -0
  39. package/lib/lambdas/post-deploy/prune-s3.d.ts +15 -0
  40. package/lib/lambdas/post-deploy/prune-s3.js +101 -0
  41. package/lib/lambdas/utils.js +35 -0
  42. package/lib/nextjs-assets-deployment.d.ts +15 -51
  43. package/lib/nextjs-assets-deployment.js +27 -25
  44. package/lib/nextjs-build/builder.Dockerfile +12 -8
  45. package/lib/nextjs-build/cdk-nextjs-cache-handler.cjs +63 -0
  46. package/lib/nextjs-build/cdk-nextjs-cache-handler.d.ts +21 -0
  47. package/lib/nextjs-build/cdk-nextjs-cache-handler.js +49 -0
  48. package/lib/nextjs-build/global-containers.Dockerfile +29 -19
  49. package/lib/nextjs-build/global-functions.Dockerfile +30 -24
  50. package/lib/nextjs-build/nextjs-build.d.ts +51 -26
  51. package/lib/nextjs-build/nextjs-build.js +108 -88
  52. package/lib/nextjs-build/regional-containers.Dockerfile +30 -20
  53. package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +0 -1
  54. package/lib/nextjs-compute/nextjs-compute-base-props.js +1 -1
  55. package/lib/nextjs-compute/nextjs-containers.d.ts +2 -1
  56. package/lib/nextjs-compute/nextjs-containers.js +7 -5
  57. package/lib/nextjs-compute/nextjs-functions.d.ts +1 -0
  58. package/lib/nextjs-compute/nextjs-functions.js +8 -11
  59. package/lib/nextjs-distribution.d.ts +1 -1
  60. package/lib/nextjs-distribution.js +4 -4
  61. package/lib/nextjs-file-system.js +1 -1
  62. package/lib/nextjs-post-deploy.d.ts +101 -0
  63. package/lib/nextjs-post-deploy.js +78 -0
  64. package/lib/nextjs-static-assets.js +1 -1
  65. package/lib/nextjs-vpc.d.ts +1 -1
  66. package/lib/nextjs-vpc.js +4 -4
  67. package/lib/root-constructs/nextjs-base-overrides.d.ts +4 -0
  68. package/lib/root-constructs/nextjs-base-overrides.js +1 -1
  69. package/lib/root-constructs/nextjs-base-props.d.ts +15 -7
  70. package/lib/root-constructs/nextjs-base-props.js +1 -1
  71. package/lib/root-constructs/nextjs-global-containers.d.ts +8 -8
  72. package/lib/root-constructs/nextjs-global-containers.js +25 -15
  73. package/lib/root-constructs/nextjs-global-functions.d.ts +8 -11
  74. package/lib/root-constructs/nextjs-global-functions.js +25 -25
  75. package/lib/root-constructs/nextjs-regional-containers.d.ts +8 -0
  76. package/lib/root-constructs/nextjs-regional-containers.js +26 -9
  77. package/lib/utils/get-architecture.d.ts +2 -0
  78. package/lib/utils/get-architecture.js +8 -0
  79. package/lib/utils/handle-deprecated-properties.d.ts +7 -0
  80. package/lib/utils/handle-deprecated-properties.js +14 -0
  81. package/package.json +17 -19
  82. package/assets/lambdas/revalidate/revalidate.lambda/index.js +0 -67
  83. package/lib/common.d.ts +0 -25
  84. package/lib/common.js +0 -30
  85. package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +0 -11
  86. package/lib/generated-structs/OptionalNextjsInvalidationProps.js +0 -3
  87. package/lib/lambdas/assets-deployment/prune-s3.d.ts +0 -15
  88. package/lib/lambdas/assets-deployment/prune-s3.js +0 -42
  89. package/lib/lambdas/assets-deployment/s3.d.ts +0 -2
  90. package/lib/lambdas/assets-deployment/s3.js +0 -7
  91. package/lib/lambdas/assets-deployment/utils.js +0 -35
  92. package/lib/lambdas/revalidate/revalidate-function.d.ts +0 -13
  93. package/lib/lambdas/revalidate/revalidate-function.js +0 -22
  94. package/lib/lambdas/revalidate/revalidate.lambda.d.ts +0 -2
  95. package/lib/lambdas/revalidate/revalidate.lambda.js +0 -53
  96. package/lib/nextjs-build/add-cache-handler.d.ts +0 -1
  97. package/lib/nextjs-build/add-cache-handler.js +0 -23
  98. package/lib/nextjs-build/add-cache-handler.mjs +0 -18
  99. package/lib/nextjs-build/cache-handler.cjs +0 -21878
  100. package/lib/nextjs-build/cache-handler.d.ts +0 -6
  101. package/lib/nextjs-build/cache-handler.js +0 -26
  102. package/lib/nextjs-invalidation.d.ts +0 -19
  103. package/lib/nextjs-invalidation.js +0 -52
  104. package/lib/nextjs-revalidation.d.ts +0 -30
  105. package/lib/nextjs-revalidation.js +0 -65
  106. /package/lib/lambdas/{assets-deployment/utils.d.ts → utils.d.ts} +0 -0
@@ -0,0 +1,101 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.pruneS3 = pruneS3;
4
+ // eslint-disable-next-line import/no-extraneous-dependencies
5
+ const client_s3_1 = require("@aws-sdk/client-s3");
6
+ const utils_1 = require("../utils");
7
+ const s3Client = new client_s3_1.S3Client();
8
+ // Maximum number of concurrent operations
9
+ const MAX_CONCURRENT_OPERATIONS = 50;
10
+ /**
11
+ * Given `bucketName`, `currentBuildId`, and `msTtl`, list all objects
12
+ * in the bucket and delete any that 1/ do not have a metadata key of "next-build-id"
13
+ * and value of `currentBuildId` and 2/ were created more than `msTtl` ago
14
+ */
15
+ async function pruneS3(props) {
16
+ const { bucketName, currentBuildId, msTtl } = props;
17
+ const cutoffDate = new Date(Date.now() - msTtl);
18
+ const objectsToDelete = [];
19
+ let continuationToken = undefined;
20
+ let listObjectsCount = 0;
21
+ do {
22
+ // List objects in the bucket
23
+ const listObjectsV2Input = {
24
+ Bucket: bucketName,
25
+ ContinuationToken: continuationToken,
26
+ };
27
+ const listResponse = await s3Client.send(new client_s3_1.ListObjectsV2Command(listObjectsV2Input));
28
+ if (!listResponse.Contents || listResponse.Contents.length === 0) {
29
+ break;
30
+ }
31
+ // Filter out objects without keys
32
+ const oldObjects = listResponse.Contents.filter((obj) => {
33
+ const lastModified = obj.LastModified || new Date();
34
+ return obj.Key && lastModified < cutoffDate;
35
+ });
36
+ (0, utils_1.debug)(`Checking old objects metadata to determine pruning: ${oldObjects.map((o) => o.Key)}`);
37
+ // Process objects in parallel with controlled concurrency
38
+ const checkResults = await processBatch(oldObjects, MAX_CONCURRENT_OPERATIONS, async (object) => {
39
+ if (!object.Key)
40
+ return null;
41
+ try {
42
+ const headResponse = await s3Client.send(new client_s3_1.HeadObjectCommand({
43
+ Bucket: bucketName,
44
+ Key: object.Key,
45
+ }));
46
+ const objectBuildId = headResponse.Metadata?.["next-build-id"];
47
+ // Return the key if it should be deleted
48
+ if (objectBuildId !== currentBuildId) {
49
+ return { Key: object.Key };
50
+ }
51
+ }
52
+ catch (error) {
53
+ console.error(`Error checking object ${object.Key}:`, error);
54
+ }
55
+ return null;
56
+ });
57
+ // Add valid objects to delete list
58
+ objectsToDelete.push(...checkResults.filter(Boolean));
59
+ if (listResponse.NextContinuationToken) {
60
+ continuationToken = listResponse.NextContinuationToken;
61
+ }
62
+ listObjectsCount++;
63
+ // assume less than 100K objects (100 * 1K objects per ListObjectsV2Command = 100K)
64
+ } while (continuationToken && listObjectsCount <= 100);
65
+ // Delete objects in parallel batches (respecting S3's 1000 objects per request limit)
66
+ if (objectsToDelete.length > 0) {
67
+ const deleteBatches = [];
68
+ for (let i = 0; i < objectsToDelete.length; i += 1000) {
69
+ const batch = objectsToDelete.slice(i, i + 1000);
70
+ deleteBatches.push(batch);
71
+ }
72
+ await processBatch(deleteBatches, 5, // Process up to 5 delete batches in parallel
73
+ async (batch) => {
74
+ try {
75
+ (0, utils_1.debug)(`Deleting objects: ${batch.map((b) => b.Key)} from ${bucketName}`);
76
+ await s3Client.send(new client_s3_1.DeleteObjectsCommand({
77
+ Bucket: bucketName,
78
+ Delete: { Objects: batch },
79
+ }));
80
+ (0, utils_1.debug)(`Deleted ${batch.length} objects from ${bucketName}`);
81
+ }
82
+ catch (error) {
83
+ console.error("Error deleting objects:", error);
84
+ }
85
+ });
86
+ }
87
+ (0, utils_1.debug)(`Pruning complete. Deleted ${objectsToDelete.length} objects from ${bucketName}`);
88
+ }
89
+ /**
90
+ * Process objects in batches to avoid overwhelming the system
91
+ */
92
+ async function processBatch(items, batchSize, processFn) {
93
+ const results = [];
94
+ for (let i = 0; i < items.length; i += batchSize) {
95
+ const batch = items.slice(i, i + batchSize);
96
+ const batchResults = await Promise.all(batch.map(processFn));
97
+ results.push(...batchResults);
98
+ }
99
+ return results;
100
+ }
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prune-s3.js","sourceRoot":"","sources":["../../../src/lambdas/post-deploy/prune-s3.ts"],"names":[],"mappings":"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","sourcesContent":["// eslint-disable-next-line import/no-extraneous-dependencies\nimport {\n  S3Client,\n  ListObjectsV2Command,\n  DeleteObjectsCommand,\n  HeadObjectCommand,\n  ListObjectsV2CommandInput,\n} from \"@aws-sdk/client-s3\";\nimport { debug } from \"../utils\";\n\nconst s3Client = new S3Client();\n// Maximum number of concurrent operations\nconst MAX_CONCURRENT_OPERATIONS = 50;\n\ninterface PruneS3Props {\n  bucketName: string;\n  currentBuildId: string;\n  /**\n   * Time to live in milliseconds.\n   */\n  msTtl: number;\n}\n\n/**\n * Given `bucketName`, `currentBuildId`, and `msTtl`, list all objects\n * in the bucket and delete any that 1/ do not have a metadata key of \"next-build-id\"\n * and value of `currentBuildId` and 2/ were created more than `msTtl` ago\n */\nexport async function pruneS3(props: PruneS3Props) {\n  const { bucketName, currentBuildId, msTtl } = props;\n\n  const cutoffDate = new Date(Date.now() - msTtl);\n  const objectsToDelete: { Key: string }[] = [];\n\n  let continuationToken: string | undefined = undefined;\n  let listObjectsCount = 0;\n\n  do {\n    // List objects in the bucket\n    const listObjectsV2Input: ListObjectsV2CommandInput = {\n      Bucket: bucketName,\n      ContinuationToken: continuationToken,\n    };\n    const listResponse = await s3Client.send(\n      new ListObjectsV2Command(listObjectsV2Input),\n    );\n\n    if (!listResponse.Contents || listResponse.Contents.length === 0) {\n      break;\n    }\n\n    // Filter out objects without keys\n    const oldObjects = listResponse.Contents.filter((obj) => {\n      const lastModified = obj.LastModified || new Date();\n      return obj.Key && lastModified < cutoffDate;\n    });\n    debug(\n      `Checking old objects metadata to determine pruning: ${oldObjects.map((o) => o.Key)}`,\n    );\n\n    // Process objects in parallel with controlled concurrency\n    const checkResults = await processBatch(\n      oldObjects,\n      MAX_CONCURRENT_OPERATIONS,\n      async (object) => {\n        if (!object.Key) return null;\n\n        try {\n          const headResponse = await s3Client.send(\n            new HeadObjectCommand({\n              Bucket: bucketName,\n              Key: object.Key,\n            }),\n          );\n\n          const objectBuildId = headResponse.Metadata?.[\"next-build-id\"];\n\n          // Return the key if it should be deleted\n          if (objectBuildId !== currentBuildId) {\n            return { Key: object.Key };\n          }\n        } catch (error) {\n          console.error(`Error checking object ${object.Key}:`, error);\n        }\n\n        return null;\n      },\n    );\n\n    // Add valid objects to delete list\n    objectsToDelete.push(\n      ...(checkResults.filter(Boolean) as { Key: string }[]),\n    );\n\n    if (listResponse.NextContinuationToken) {\n      continuationToken = listResponse.NextContinuationToken;\n    }\n    listObjectsCount++;\n    // assume less than 100K objects (100 * 1K objects per ListObjectsV2Command = 100K)\n  } while (continuationToken && listObjectsCount <= 100);\n\n  // Delete objects in parallel batches (respecting S3's 1000 objects per request limit)\n  if (objectsToDelete.length > 0) {\n    const deleteBatches = [];\n\n    for (let i = 0; i < objectsToDelete.length; i += 1000) {\n      const batch = objectsToDelete.slice(i, i + 1000);\n      deleteBatches.push(batch);\n    }\n\n    await processBatch(\n      deleteBatches,\n      5, // Process up to 5 delete batches in parallel\n      async (batch) => {\n        try {\n          debug(\n            `Deleting objects: ${batch.map((b) => b.Key)} from ${bucketName}`,\n          );\n          await s3Client.send(\n            new DeleteObjectsCommand({\n              Bucket: bucketName,\n              Delete: { Objects: batch },\n            }),\n          );\n          debug(`Deleted ${batch.length} objects from ${bucketName}`);\n        } catch (error) {\n          console.error(\"Error deleting objects:\", error);\n        }\n      },\n    );\n  }\n\n  debug(\n    `Pruning complete. Deleted ${objectsToDelete.length} objects from ${bucketName}`,\n  );\n}\n\n/**\n * Process objects in batches to avoid overwhelming the system\n */\nasync function processBatch<T, R>(\n  items: T[],\n  batchSize: number,\n  processFn: (item: T) => Promise<R>,\n): Promise<R[]> {\n  const results: R[] = [];\n\n  for (let i = 0; i < items.length; i += batchSize) {\n    const batch = items.slice(i, i + batchSize);\n    const batchResults = await Promise.all(batch.map(processFn));\n    results.push(...batchResults);\n  }\n\n  return results;\n}\n"]}
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CfnResponseStatus = void 0;
4
+ exports.cfnResponse = cfnResponse;
5
+ exports.debug = debug;
6
+ var CfnResponseStatus;
7
+ (function (CfnResponseStatus) {
8
+ CfnResponseStatus["Success"] = "SUCCESS";
9
+ CfnResponseStatus["Failed"] = "FAILED";
10
+ })(CfnResponseStatus || (exports.CfnResponseStatus = CfnResponseStatus = {}));
11
+ /**
12
+ * Inspired by: https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/cfn-lambda-function-code-cfnresponsemodule.html
13
+ */
14
+ function cfnResponse(props) {
15
+ const body = JSON.stringify({
16
+ Status: props.responseStatus,
17
+ Reason: "See the details in CloudWatch Log Stream: " +
18
+ props.context.logStreamName,
19
+ PhysicalResourceId: props.physicalResourceId || props.context.logStreamName,
20
+ StackId: props.event.StackId,
21
+ RequestId: props.event.RequestId,
22
+ LogicalResourceId: props.event.LogicalResourceId,
23
+ Data: props.responseData,
24
+ });
25
+ return fetch(props.event.ResponseURL, {
26
+ method: "PUT",
27
+ body,
28
+ headers: { "content-type": "", "content-length": body.length.toString() },
29
+ });
30
+ }
31
+ function debug(value) {
32
+ if (process.env.DEBUG)
33
+ console.log(JSON.stringify(value, null, 2));
34
+ }
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbGFtYmRhcy91dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFrQkEsa0NBaUJDO0FBRUQsc0JBRUM7QUFyQ0QsSUFBWSxpQkFHWDtBQUhELFdBQVksaUJBQWlCO0lBQzNCLHdDQUFtQixDQUFBO0lBQ25CLHNDQUFpQixDQUFBO0FBQ25CLENBQUMsRUFIVyxpQkFBaUIsaUNBQWpCLGlCQUFpQixRQUc1QjtBQVVEOztHQUVHO0FBQ0gsU0FBZ0IsV0FBVyxDQUFDLEtBQXVCO0lBQ2pELE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUM7UUFDMUIsTUFBTSxFQUFFLEtBQUssQ0FBQyxjQUFjO1FBQzVCLE1BQU0sRUFDSiw0Q0FBNEM7WUFDNUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxhQUFhO1FBQzdCLGtCQUFrQixFQUFFLEtBQUssQ0FBQyxrQkFBa0IsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLGFBQWE7UUFDM0UsT0FBTyxFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUMsT0FBTztRQUM1QixTQUFTLEVBQUUsS0FBSyxDQUFDLEtBQUssQ0FBQyxTQUFTO1FBQ2hDLGlCQUFpQixFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUMsaUJBQWlCO1FBQ2hELElBQUksRUFBRSxLQUFLLENBQUMsWUFBWTtLQUN6QixDQUFDLENBQUM7SUFDSCxPQUFPLEtBQUssQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRTtRQUNwQyxNQUFNLEVBQUUsS0FBSztRQUNiLElBQUk7UUFDSixPQUFPLEVBQUUsRUFBRSxjQUFjLEVBQUUsRUFBRSxFQUFFLGdCQUFnQixFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLEVBQUU7S0FDMUUsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVELFNBQWdCLEtBQUssQ0FBQyxLQUFjO0lBQ2xDLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxLQUFLO1FBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNyRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBDbG91ZEZvcm1hdGlvbkN1c3RvbVJlc291cmNlSGFuZGxlciB9IGZyb20gXCJhd3MtbGFtYmRhXCI7XG5cbmV4cG9ydCBlbnVtIENmblJlc3BvbnNlU3RhdHVzIHtcbiAgU3VjY2VzcyA9IFwiU1VDQ0VTU1wiLFxuICBGYWlsZWQgPSBcIkZBSUxFRFwiLFxufVxuXG5pbnRlcmZhY2UgQ2ZuUmVzcG9uc2VQcm9wcyB7XG4gIGV2ZW50OiBQYXJhbWV0ZXJzPENsb3VkRm9ybWF0aW9uQ3VzdG9tUmVzb3VyY2VIYW5kbGVyPlswXTtcbiAgY29udGV4dDogUGFyYW1ldGVyczxDbG91ZEZvcm1hdGlvbkN1c3RvbVJlc291cmNlSGFuZGxlcj5bMV07XG4gIHJlc3BvbnNlU3RhdHVzOiBDZm5SZXNwb25zZVN0YXR1cztcbiAgcmVzcG9uc2VEYXRhPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgcGh5c2ljYWxSZXNvdXJjZUlkPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIEluc3BpcmVkIGJ5OiBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vQVdTQ2xvdWRGb3JtYXRpb24vbGF0ZXN0L1VzZXJHdWlkZS9jZm4tbGFtYmRhLWZ1bmN0aW9uLWNvZGUtY2ZucmVzcG9uc2Vtb2R1bGUuaHRtbFxuICovXG5leHBvcnQgZnVuY3Rpb24gY2ZuUmVzcG9uc2UocHJvcHM6IENmblJlc3BvbnNlUHJvcHMpIHtcbiAgY29uc3QgYm9keSA9IEpTT04uc3RyaW5naWZ5KHtcbiAgICBTdGF0dXM6IHByb3BzLnJlc3BvbnNlU3RhdHVzLFxuICAgIFJlYXNvbjpcbiAgICAgIFwiU2VlIHRoZSBkZXRhaWxzIGluIENsb3VkV2F0Y2ggTG9nIFN0cmVhbTogXCIgK1xuICAgICAgcHJvcHMuY29udGV4dC5sb2dTdHJlYW1OYW1lLFxuICAgIFBoeXNpY2FsUmVzb3VyY2VJZDogcHJvcHMucGh5c2ljYWxSZXNvdXJjZUlkIHx8IHByb3BzLmNvbnRleHQubG9nU3RyZWFtTmFtZSxcbiAgICBTdGFja0lkOiBwcm9wcy5ldmVudC5TdGFja0lkLFxuICAgIFJlcXVlc3RJZDogcHJvcHMuZXZlbnQuUmVxdWVzdElkLFxuICAgIExvZ2ljYWxSZXNvdXJjZUlkOiBwcm9wcy5ldmVudC5Mb2dpY2FsUmVzb3VyY2VJZCxcbiAgICBEYXRhOiBwcm9wcy5yZXNwb25zZURhdGEsXG4gIH0pO1xuICByZXR1cm4gZmV0Y2gocHJvcHMuZXZlbnQuUmVzcG9uc2VVUkwsIHtcbiAgICBtZXRob2Q6IFwiUFVUXCIsXG4gICAgYm9keSxcbiAgICBoZWFkZXJzOiB7IFwiY29udGVudC10eXBlXCI6IFwiXCIsIFwiY29udGVudC1sZW5ndGhcIjogYm9keS5sZW5ndGgudG9TdHJpbmcoKSB9LFxuICB9KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGRlYnVnKHZhbHVlOiB1bmtub3duKSB7XG4gIGlmIChwcm9jZXNzLmVudi5ERUJVRykgY29uc29sZS5sb2coSlNPTi5zdHJpbmdpZnkodmFsdWUsIG51bGwsIDIpKTtcbn1cbiJdfQ==
@@ -4,10 +4,14 @@ import { AccessPoint } from "aws-cdk-lib/aws-efs";
4
4
  import { DockerImageCode, DockerImageFunction } from "aws-cdk-lib/aws-lambda";
5
5
  import { Bucket } from "aws-cdk-lib/aws-s3";
6
6
  import { Construct } from "constructs";
7
- import { NextjsType } from "./common";
7
+ import { NextjsType } from "./constants";
8
+ import { OptionalCustomResourceProps } from "./generated-structs/OptionalCustomResourceProps";
8
9
  import { OptionalDockerImageFunctionProps } from "./generated-structs/OptionalDockerImageFunctionProps";
9
- import { NextjsBuild } from "./nextjs-build/nextjs-build";
10
10
  export interface NextjsAssetDeploymentOverrides {
11
+ /**
12
+ * Props that define the custom resource
13
+ */
14
+ readonly customResourceProps?: OptionalCustomResourceProps;
11
15
  readonly dockerImageFunctionProps?: OptionalDockerImageFunctionProps;
12
16
  }
13
17
  export interface NextjsAssetsDeploymentProps {
@@ -17,15 +21,13 @@ export interface NextjsAssetsDeploymentProps {
17
21
  * @example "/my-base-path"
18
22
  */
19
23
  readonly basePath?: string;
24
+ readonly buildId: string;
20
25
  /**
21
26
  * @see {@link NextjsBuild.buildImageDigest}
22
27
  */
23
28
  readonly buildImageDigest: string;
24
29
  /**
25
- * @see {@link NextjsBuild.containerMountPathForEfs}
26
- */
27
- readonly containerMountPathForEfs: NextjsBuild["containerMountPathForEfs"];
28
- /**
30
+ * If true, logs details in custom resource lambda
29
31
  * @default true
30
32
  */
31
33
  readonly debug?: boolean;
@@ -33,9 +35,9 @@ export interface NextjsAssetsDeploymentProps {
33
35
  readonly nextjsType: NextjsType;
34
36
  readonly overrides?: NextjsAssetDeploymentOverrides;
35
37
  /**
36
- * @see {@link NextjsBaseProps.relativePathToWorkspace}
38
+ * @see {@link NextjsBaseProps.relativePathToPackage}
37
39
  */
38
- readonly relativePathToWorkspace?: string;
40
+ readonly relativePathToPackage?: string;
39
41
  /**
40
42
  * Required for `NextjsType.GlobalFunctions` and `NextjsType.GlobalContainers`
41
43
  */
@@ -58,53 +60,19 @@ export interface FsToFsAction {
58
60
  type: "fs-to-fs";
59
61
  sourcePath: string;
60
62
  destinationPath: string;
63
+ includeExtensions?: string[];
61
64
  }
62
65
  /**
63
66
  * @internal
64
67
  */
65
- export interface PruneS3Action {
66
- type: "prune-s3";
67
- /**
68
- * The minimum previous deployment count to prune
69
- * @default 3
70
- */
71
- minPreviousDeployCountToPrune: number;
72
- /**
73
- * The minimum previous deployment date to prune
74
- * @default new Date(new Date().setMonth(new Date().getMonth() - 1))
75
- */
76
- minPreviousDeployDateToPrune: string;
77
- bucketName: string;
78
- bucketPrefix?: string;
79
- }
80
- /**
81
- * @internal
82
- */
83
- export interface PruneFsAction {
84
- type: "prune-fs";
68
+ export interface StaticAssetsCustomResourceProperties {
69
+ actions: (FsToS3Action | FsToFsAction)[];
70
+ buildId: string;
85
71
  /**
86
- * The minimum previous deployment count to prune
87
- * @default 3
88
- */
89
- minPreviousDeployCountToPrune: number;
90
- /**
91
- * The minimum previous deployment date to prune
92
- * @default new Date(new Date().setMonth(new Date().getMonth() - 1))
93
- */
94
- minPreviousDeployDateToPrune: string;
95
- directory: string;
96
- }
97
- /**
98
- * @internal
99
- */
100
- export interface CustomResourceProperties {
101
- actions: (FsToS3Action | FsToFsAction | PruneS3Action | PruneFsAction)[];
102
- /**
103
- * {@link NextjsAssetDeploymentProps.builderImageDigest}
72
+ * @see {@link NextjsBuild.buildImageDigest}
104
73
  */
105
74
  buildImageDigest: string;
106
75
  imageCachePath: string;
107
- prerenderManifestPath: string;
108
76
  nextjsType: NextjsType;
109
77
  }
110
78
  /**
@@ -113,10 +81,6 @@ export interface CustomResourceProperties {
113
81
  export declare class NextjsAssetsDeployment extends Construct {
114
82
  customResource: CustomResource;
115
83
  dockerImageFunction: DockerImageFunction;
116
- /**
117
- * Only used for `NextjsGlobalFunctions`
118
- */
119
- previewModeId: string;
120
84
  private props;
121
85
  constructor(scope: Construct, id: string, props: NextjsAssetsDeploymentProps);
122
86
  private createFunction;
@@ -7,7 +7,8 @@ const posix_1 = require("path/posix");
7
7
  const aws_cdk_lib_1 = require("aws-cdk-lib");
8
8
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
9
9
  const constructs_1 = require("constructs");
10
- const common_1 = require("./common");
10
+ const constants_1 = require("./constants");
11
+ const get_architecture_1 = require("./utils/get-architecture");
11
12
  /**
12
13
  * Deploys static assets to S3 and cache assets to EFS in Lambda Custom Resource.
13
14
  */
@@ -17,21 +18,13 @@ class NextjsAssetsDeployment extends constructs_1.Construct {
17
18
  this.props = props;
18
19
  this.dockerImageFunction = this.createFunction();
19
20
  this.customResource = this.createCustomResource();
20
- this.previewModeId = this.customResource.getAttString("previewModeId");
21
21
  }
22
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
23
  const fn = new aws_lambda_1.DockerImageFunction(this, "Fn", {
31
- architecture,
24
+ architecture: (0, get_architecture_1.getLambdaArchitecture)(),
32
25
  code: this.props.dockerImageCode,
33
26
  memorySize: 2048,
34
- filesystem: aws_lambda_1.FileSystem.fromEfsAccessPoint(this.props.accessPoint, this.props.containerMountPathForEfs),
27
+ filesystem: aws_lambda_1.FileSystem.fromEfsAccessPoint(this.props.accessPoint, constants_1.MOUNT_PATH),
35
28
  vpc: this.props.vpc,
36
29
  timeout: aws_cdk_lib_1.Duration.minutes(5),
37
30
  ...this.props.overrides?.dockerImageFunctionProps,
@@ -56,51 +49,60 @@ class NextjsAssetsDeployment extends constructs_1.Construct {
56
49
  // static files
57
50
  {
58
51
  type: "fs-to-s3",
59
- sourcePath: (0, posix_1.join)(root, ".next", "static"),
52
+ sourcePath: (0, posix_1.join)(root, constants_1.STATIC_PATH),
60
53
  destinationBucketName: this.props.staticAssetsBucket.bucketName,
61
54
  destinationKeyPrefix: staticKeyPrefix,
62
55
  },
63
56
  // public directory to s3 for CloudFront -> S3
64
57
  {
65
58
  type: "fs-to-s3",
66
- sourcePath: (0, posix_1.join)(root, "public"),
59
+ sourcePath: (0, posix_1.join)(root, constants_1.PUBLIC_PATH),
67
60
  destinationBucketName: this.props.staticAssetsBucket.bucketName,
68
61
  });
69
62
  }
70
63
  actions.push(
71
- // data cache - https://nextjs.org/docs/app/building-your-application/caching#data-cache
64
+ // full route cache - https://nextjs.org/docs/app/building-your-application/caching#full-route-cache
72
65
  {
73
66
  type: "fs-to-fs",
74
- sourcePath: (0, posix_1.join)(root, ".next", "cache", "fetch-cache"),
75
- destinationPath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.DATA_CACHE_DIR),
67
+ sourcePath: (0, posix_1.join)(root, ".next", "standalone", this.props.relativePathToPackage || "", constants_1.FULL_ROUTE_CACHE_PATH),
68
+ destinationPath: (0, posix_1.join)(constants_1.MOUNT_PATH, this.props.buildId, constants_1.FULL_ROUTE_CACHE_PATH),
69
+ // only files with these extensions are needed in EFS FileSystem cache
70
+ // since these files are the only ones updated by Next.js during runtime.
71
+ // other files like .js and .nft.json files are static per deployment
72
+ includeExtensions: ["body", "html", "rsc", "meta"],
76
73
  },
77
- // full route cache - https://nextjs.org/docs/app/building-your-application/caching#full-route-cache
74
+ // after `next build` data cache https://nextjs.org/docs/app/building-your-application/caching#data-cache
75
+ // exists at top level .next/cache so we need to copy into relativePathToPackage
76
+ // normalized path
78
77
  {
79
78
  type: "fs-to-fs",
80
- sourcePath: (0, posix_1.join)(root, ".next", "standalone", this.props.relativePathToWorkspace || "", ".next", "server", "app"),
81
- destinationPath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.FULL_ROUTE_CACHE_DIR),
79
+ sourcePath: (0, posix_1.join)(root, constants_1.DATA_CACHE_PATH),
80
+ destinationPath: (0, posix_1.join)(constants_1.MOUNT_PATH, this.props.buildId, constants_1.DATA_CACHE_PATH),
82
81
  },
83
82
  // public directory to EFS for needed optimizing images in public directory
83
+ // we don't load these into compute to save on image size
84
84
  {
85
85
  type: "fs-to-fs",
86
- sourcePath: (0, posix_1.join)(root, "public"),
87
- destinationPath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.PUBLIC_DIR),
86
+ sourcePath: (0, posix_1.join)(root, constants_1.PUBLIC_PATH),
87
+ destinationPath: (0, posix_1.join)(constants_1.MOUNT_PATH, this.props.buildId, constants_1.PUBLIC_PATH),
88
88
  });
89
89
  const properties = {
90
90
  actions,
91
+ buildId: this.props.buildId,
92
+ // ensures this CR runs each time new builder image
91
93
  buildImageDigest: this.props.buildImageDigest,
92
- imageCachePath: (0, posix_1.join)(this.props.containerMountPathForEfs, common_1.IMAGE_CACHE_DIR),
94
+ imageCachePath: (0, posix_1.join)(constants_1.MOUNT_PATH, this.props.buildId, constants_1.IMAGE_CACHE_PATH),
93
95
  nextjsType: this.props.nextjsType,
94
- prerenderManifestPath: (0, posix_1.join)(root, ".next", "prerender-manifest.json"),
95
96
  };
96
97
  return new aws_cdk_lib_1.CustomResource(this, "CustomResource", {
97
98
  properties,
98
99
  resourceType: "Custom::NextjsAssetsDeployment",
99
100
  serviceToken: this.dockerImageFunction.functionArn,
101
+ ...this.props.overrides?.customResourceProps,
100
102
  });
101
103
  }
102
104
  }
103
105
  exports.NextjsAssetsDeployment = NextjsAssetsDeployment;
104
106
  _a = JSII_RTTI_SYMBOL_1;
105
- NextjsAssetsDeployment[_a] = { fqn: "cdk-nextjs.NextjsAssetsDeployment", version: "0.3.11" };
106
- //# 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  NextjsType,\n  PUBLIC_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   * Prefix to the URI path the app will be served at.\n   * @example \"/my-base-path\"\n   */\n  readonly basePath?: string;\n  /**\n   * @see {@link NextjsBuild.buildImageDigest}\n   */\n  readonly buildImageDigest: string;\n  /**\n   * @see {@link NextjsBuild.containerMountPathForEfs}\n   */\n  readonly containerMountPathForEfs: NextjsBuild[\"containerMountPathForEfs\"];\n  /**\n   * @default true\n   */\n  readonly debug?: boolean;\n  readonly dockerImageCode: DockerImageCode; // TODO: remove and build from common builder base?\n  readonly nextjsType: NextjsType;\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  nextjsType: NextjsType;\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      // Prepare the destination key prefix with basePath when available\n      const staticKeyPrefix = this.props.basePath\n        ? `${this.props.basePath.replace(/^\\//, \"\")}/_next/static`\n        : \"_next/static\";\n\n      actions.push(\n        // static files\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, \".next\", \"static\"),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\n          destinationKeyPrefix: staticKeyPrefix,\n        },\n        // public directory to s3 for CloudFront -> S3\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, \"public\"),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\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      // public directory to EFS for needed optimizing images in public directory\n      {\n        type: \"fs-to-fs\",\n        sourcePath: join(root, \"public\"),\n        destinationPath: join(this.props.containerMountPathForEfs, PUBLIC_DIR),\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      nextjsType: this.props.nextjsType,\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"]}
107
+ NextjsAssetsDeployment[_a] = { fqn: "cdk-nextjs.NextjsAssetsDeployment", version: "0.4.1" };
108
+ //# 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  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_PATH,\n  FULL_ROUTE_CACHE_PATH,\n  IMAGE_CACHE_PATH,\n  MOUNT_PATH,\n  NextjsType,\n  PUBLIC_PATH,\n  STATIC_PATH,\n} from \"./constants\";\nimport { OptionalCustomResourceProps } from \"./generated-structs/OptionalCustomResourceProps\";\nimport { OptionalDockerImageFunctionProps } from \"./generated-structs/OptionalDockerImageFunctionProps\";\nimport { NextjsBuild } from \"./nextjs-build/nextjs-build\";\nimport { NextjsBaseProps } from \"./root-constructs/nextjs-base-props\";\nimport { getLambdaArchitecture } from \"./utils/get-architecture\";\n\nexport interface NextjsAssetDeploymentOverrides {\n  /**\n   * Props that define the custom resource\n   */\n  readonly customResourceProps?: OptionalCustomResourceProps;\n  // NOTE: we don't offer overriding `customResourceProperties` since jsii doesn't support union types\n  // readonly customResourceProperties?: OptionalStaticAssetsCustomResourceProperties\n  readonly dockerImageFunctionProps?: OptionalDockerImageFunctionProps;\n}\n\nexport interface NextjsAssetsDeploymentProps {\n  readonly accessPoint: AccessPoint;\n  /**\n   * Prefix to the URI path the app will be served at.\n   * @example \"/my-base-path\"\n   */\n  readonly basePath?: string;\n  readonly buildId: string;\n  /**\n   * @see {@link NextjsBuild.buildImageDigest}\n   */\n  readonly buildImageDigest: string;\n  /**\n   * If true, logs details in custom resource lambda\n   * @default true\n   */\n  readonly debug?: boolean;\n  readonly dockerImageCode: DockerImageCode; // TODO: remove and build from common builder base?\n  readonly nextjsType: NextjsType;\n  readonly overrides?: NextjsAssetDeploymentOverrides;\n  /**\n   * @see {@link NextjsBaseProps.relativePathToPackage}\n   */\n  readonly relativePathToPackage?: 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  includeExtensions?: string[];\n}\n\n/**\n * @internal\n */\nexport interface StaticAssetsCustomResourceProperties {\n  actions: (FsToS3Action | FsToFsAction)[];\n  buildId: string;\n  /**\n   * @see {@link NextjsBuild.buildImageDigest}\n   */\n  buildImageDigest: string;\n  imageCachePath: string;\n  nextjsType: NextjsType;\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  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  }\n\n  private createFunction() {\n    const fn = new DockerImageFunction(this, \"Fn\", {\n      architecture: getLambdaArchitecture(),\n      code: this.props.dockerImageCode,\n      memorySize: 2048,\n      filesystem: FileSystem.fromEfsAccessPoint(\n        this.props.accessPoint,\n        MOUNT_PATH,\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: StaticAssetsCustomResourceProperties[\"actions\"] = [];\n    if (this.props.staticAssetsBucket?.bucketName) {\n      // Prepare the destination key prefix with basePath when available\n      const staticKeyPrefix = this.props.basePath\n        ? `${this.props.basePath.replace(/^\\//, \"\")}/_next/static`\n        : \"_next/static\";\n\n      actions.push(\n        // static files\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, STATIC_PATH),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\n          destinationKeyPrefix: staticKeyPrefix,\n        },\n        // public directory to s3 for CloudFront -> S3\n        {\n          type: \"fs-to-s3\",\n          sourcePath: join(root, PUBLIC_PATH),\n          destinationBucketName: this.props.staticAssetsBucket.bucketName,\n        },\n      );\n    }\n    actions.push(\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.relativePathToPackage || \"\",\n          FULL_ROUTE_CACHE_PATH,\n        ),\n        destinationPath: join(\n          MOUNT_PATH,\n          this.props.buildId,\n          FULL_ROUTE_CACHE_PATH,\n        ),\n        // only files with these extensions are needed in EFS FileSystem cache\n        // since these files are the only ones updated by Next.js during runtime.\n        // other files like .js and .nft.json files are static per deployment\n        includeExtensions: [\"body\", \"html\", \"rsc\", \"meta\"],\n      },\n      // after `next build` data cache https://nextjs.org/docs/app/building-your-application/caching#data-cache\n      // exists at top level .next/cache so we need to copy into relativePathToPackage\n      // normalized path\n      {\n        type: \"fs-to-fs\",\n        sourcePath: join(root, DATA_CACHE_PATH),\n        destinationPath: join(MOUNT_PATH, this.props.buildId, DATA_CACHE_PATH),\n      },\n      // public directory to EFS for needed optimizing images in public directory\n      // we don't load these into compute to save on image size\n      {\n        type: \"fs-to-fs\",\n        sourcePath: join(root, PUBLIC_PATH),\n        destinationPath: join(MOUNT_PATH, this.props.buildId, PUBLIC_PATH),\n      },\n      // images are optimized at runtime so nothing to deploy\n    );\n    const properties: StaticAssetsCustomResourceProperties = {\n      actions,\n      buildId: this.props.buildId,\n      // ensures this CR runs each time new builder image\n      buildImageDigest: this.props.buildImageDigest,\n      imageCachePath: join(MOUNT_PATH, this.props.buildId, IMAGE_CACHE_PATH),\n      nextjsType: this.props.nextjsType,\n    };\n    return new CustomResource(this, \"CustomResource\", {\n      properties,\n      resourceType: \"Custom::NextjsAssetsDeployment\",\n      serviceToken: this.dockerImageFunction.functionArn,\n      ...this.props.overrides?.customResourceProps,\n    });\n  }\n}\n"]}
@@ -6,9 +6,7 @@ RUN apk add --no-cache libc6-compat
6
6
  WORKDIR /app
7
7
  # It's preferrable to only copy package.json's and lockfiles, install, and then
8
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.nextjs[GlobalFunctions].nextjsBuildProps.builderImageProps.srcDockerfilePath`
11
- # to optimize for their setup.
9
+ # here because it's highly customized based on a projects setup. See cdk-nextjs/examples/turbo.
12
10
  COPY . .
13
11
  # Install dependencies based on the preferred package manager
14
12
  RUN \
@@ -18,8 +16,14 @@ RUN \
18
16
  else echo "Lockfile not found." && exit 1; \
19
17
  fi
20
18
  ARG BUILD_COMMAND
21
- ARG RELATIVE_PATH_TO_WORKSPACE
22
- RUN if [ -f ./cdk-nextjs-load-env-vars.sh ]; then chmod u+x ./cdk-nextjs-load-env-vars.sh && . ./cdk-nextjs-load-env-vars.sh; fi
23
- RUN cd $RELATIVE_PATH_TO_WORKSPACE && $BUILD_COMMAND
24
- # after building, node_modules aren't needed anymore. this reduces image size by over 50mb
25
- RUN rm -rf node_modules
19
+ ARG RELATIVE_PATH_TO_PACKAGE
20
+ RUN cd $RELATIVE_PATH_TO_PACKAGE && {{INJECT_CDK_NEXTJS_BUILD_ENV_VARS}} $BUILD_COMMAND
21
+ RUN rm -rf node_modules && \
22
+ # remove pnpm cache if exists
23
+ command -v pnpm >/dev/null 2>&1 && rm -rf $(pnpm store path) || echo "pnpm not installed, skipping cache cleanup" && \
24
+ # .next/cache/webpack used to speed up builds on subsequent `next build`
25
+ rm -rf $RELATIVE_PATH_TO_PACKAGE/.next/cache/webpack && \
26
+ # .next/trace used for debugging purposes
27
+ rm $RELATIVE_PATH_TO_PACKAGE/.next/trace && \
28
+ # not needed since we use .next/server in standalone
29
+ rm -rf $RELATIVE_PATH_TO_PACKAGE/.next/server
@@ -0,0 +1,63 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+
30
+ // src/nextjs-build/cdk-nextjs-cache-handler.ts
31
+ var cdk_nextjs_cache_handler_exports = {};
32
+ __export(cdk_nextjs_cache_handler_exports, {
33
+ default: () => CdkNextjsCacheHandler
34
+ });
35
+ module.exports = __toCommonJS(cdk_nextjs_cache_handler_exports);
36
+ var import_file_system_cache = __toESM(require("next/dist/server/lib/incremental-cache/file-system-cache"));
37
+
38
+ // src/constants.ts
39
+ var CACHE_PATH = ".next/cache";
40
+ var DATA_CACHE_PATH = `${CACHE_PATH}/fetch-cache`;
41
+ var IMAGE_CACHE_PATH = `${CACHE_PATH}/images`;
42
+ var SERVER_DIST_PATH = ".next/server";
43
+ var FULL_ROUTE_CACHE_PATH = `${SERVER_DIST_PATH}/app`;
44
+ var CDK_NEXTJS_SERVER_DIST_DIR_ENV_VAR_NAME = "CDK_NEXTJS_SERVER_DIST_DIR";
45
+
46
+ // src/nextjs-build/cdk-nextjs-cache-handler.ts
47
+ var CdkNextjsCacheHandler = class extends import_file_system_cache.default {
48
+ constructor(options) {
49
+ const cdkNextjsServerDistDir = process.env[CDK_NEXTJS_SERVER_DIST_DIR_ENV_VAR_NAME];
50
+ if (!cdkNextjsServerDistDir) {
51
+ throw new Error(
52
+ `${CDK_NEXTJS_SERVER_DIST_DIR_ENV_VAR_NAME} environment variable not set`
53
+ );
54
+ }
55
+ super({ ...options, serverDistDir: cdkNextjsServerDistDir });
56
+ }
57
+ get(cacheKey, ctx) {
58
+ return super.get(cacheKey, ctx);
59
+ }
60
+ set(pathname, data, ctx) {
61
+ return super.set(pathname, data, ctx);
62
+ }
63
+ };
@@ -0,0 +1,21 @@
1
+ import { CacheHandlerValue } from "next/dist/server/lib/incremental-cache";
2
+ import FileSystemCache from "next/dist/server/lib/incremental-cache/file-system-cache";
3
+ import { IncrementalCacheKindHint, IncrementalCacheValue } from "next/dist/server/response-cache";
4
+ export default class CdkNextjsCacheHandler extends FileSystemCache {
5
+ constructor(options: ConstructorParameters<typeof FileSystemCache>[0]);
6
+ get(cacheKey: string, ctx?: {
7
+ kindHint?: IncrementalCacheKindHint;
8
+ revalidate?: number | false;
9
+ fetchUrl?: string;
10
+ fetchIdx?: number;
11
+ tags?: string[];
12
+ softTags?: string[];
13
+ } | undefined): Promise<CacheHandlerValue | null>;
14
+ set(pathname: string, data: IncrementalCacheValue | null, ctx: {
15
+ revalidate?: number | false;
16
+ fetchCache?: boolean;
17
+ fetchUrl?: string;
18
+ fetchIdx?: number;
19
+ tags?: string[];
20
+ }): Promise<void>;
21
+ }
@@ -0,0 +1,49 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ // eslint-disable-next-line import/no-extraneous-dependencies
4
+ const file_system_cache_1 = require("next/dist/server/lib/incremental-cache/file-system-cache");
5
+ const constants_1 = require("../constants");
6
+ class CdkNextjsCacheHandler extends file_system_cache_1.default {
7
+ constructor(options) {
8
+ /**
9
+ * @example `/mnt/cdk-nextjs/{relativePathToPackage}/.next/server`
10
+ * This allows us to tell Next.js `FileSystemCache` to write .next/cache/fetch-cache
11
+ * data and .next/server/app/*.{html,rsc,meta,body} data to our EFS shared
12
+ * file system. Note, we cannot control where optimized images are cached
13
+ * through public Next.js APIs so we must symlink.
14
+ *
15
+ * Note, the `serverDistDir` defaults to current directory of the Next.js
16
+ * app but we cannot use this because 1/ we want to share cache between
17
+ * mutiple functions or containers via EFS mount 2/ lambda cannot write
18
+ * to disk (unless /tmp or EFS mount)
19
+ */
20
+ const cdkNextjsServerDistDir = process.env[constants_1.CDK_NEXTJS_SERVER_DIST_DIR_ENV_VAR_NAME];
21
+ if (!cdkNextjsServerDistDir) {
22
+ throw new Error(`${constants_1.CDK_NEXTJS_SERVER_DIST_DIR_ENV_VAR_NAME} environment variable not set`);
23
+ }
24
+ super({ ...options, serverDistDir: cdkNextjsServerDistDir });
25
+ }
26
+ get(cacheKey, ctx) {
27
+ // console.log(JSON.stringify({ cacheKey, ctx }, null, 2));
28
+ return super.get(cacheKey, ctx);
29
+ }
30
+ set(pathname, data, ctx) {
31
+ // console.log(
32
+ // JSON.stringify(
33
+ // {
34
+ // pathname,
35
+ // data:
36
+ // data?.kind === "PAGE"
37
+ // ? { ...data, pageData: "...", html: "..." }
38
+ // : data,
39
+ // ctx,
40
+ // },
41
+ // null,
42
+ // 2,
43
+ // ),
44
+ // );
45
+ return super.set(pathname, data, ctx);
46
+ }
47
+ }
48
+ exports.default = CdkNextjsCacheHandler;
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2RrLW5leHRqcy1jYWNoZS1oYW5kbGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL25leHRqcy1idWlsZC9jZGstbmV4dGpzLWNhY2hlLWhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFNQSw2REFBNkQ7QUFDN0QsZ0dBQXVGO0FBTXZGLDRDQUF1RTtBQUV2RSxNQUFxQixxQkFBc0IsU0FBUSwyQkFBZTtJQUNoRSxZQUFZLE9BQXlEO1FBQ25FOzs7Ozs7Ozs7OztXQVdHO1FBQ0gsTUFBTSxzQkFBc0IsR0FDMUIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxtREFBdUMsQ0FBQyxDQUFDO1FBQ3ZELElBQUksQ0FBQyxzQkFBc0IsRUFBRSxDQUFDO1lBQzVCLE1BQU0sSUFBSSxLQUFLLENBQ2IsR0FBRyxtREFBdUMsK0JBQStCLENBQzFFLENBQUM7UUFDSixDQUFDO1FBQ0QsS0FBSyxDQUFDLEVBQUUsR0FBRyxPQUFPLEVBQUUsYUFBYSxFQUFFLHNCQUFzQixFQUFFLENBQUMsQ0FBQztJQUMvRCxDQUFDO0lBRUQsR0FBRyxDQUNELFFBQWdCLEVBQ2hCLEdBU2E7UUFFYiwyREFBMkQ7UUFDM0QsT0FBTyxLQUFLLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUNsQyxDQUFDO0lBRUQsR0FBRyxDQUNELFFBQWdCLEVBQ2hCLElBQWtDLEVBQ2xDLEdBTUM7UUFFRCxlQUFlO1FBQ2Ysb0JBQW9CO1FBQ3BCLFFBQVE7UUFDUixrQkFBa0I7UUFDbEIsY0FBYztRQUNkLGdDQUFnQztRQUNoQyx3REFBd0Q7UUFDeEQsb0JBQW9CO1FBQ3BCLGFBQWE7UUFDYixTQUFTO1FBQ1QsWUFBWTtRQUNaLFNBQVM7UUFDVCxPQUFPO1FBQ1AsS0FBSztRQUNMLE9BQU8sS0FBSyxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUUsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3hDLENBQUM7Q0FDRjtBQXBFRCx3Q0FvRUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICBUaGlzIGZpbGUgaXMgY29tcGlsZWQgYXMgZGVmaW5lZCBpbiAucHJvamVucmMudHMgdG8gYmUgdXNlZCBhcyBOZXh0LmpzXG4gIEN1c3RvbSBDYWNoZSBIYW5kbGVyLiBTZWU6IGh0dHBzOi8vbmV4dGpzLm9yZy9kb2NzL2FwcC9hcGktcmVmZXJlbmNlL25leHQtY29uZmlnLWpzL2luY3JlbWVudGFsQ2FjaGVIYW5kbGVyUGF0aFxuKi9cbi8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBpbXBvcnQvbm8tZXh0cmFuZW91cy1kZXBlbmRlbmNpZXNcbmltcG9ydCB7IENhY2hlSGFuZGxlclZhbHVlIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvbGliL2luY3JlbWVudGFsLWNhY2hlXCI7XG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzXG5pbXBvcnQgRmlsZVN5c3RlbUNhY2hlIGZyb20gXCJuZXh0L2Rpc3Qvc2VydmVyL2xpYi9pbmNyZW1lbnRhbC1jYWNoZS9maWxlLXN5c3RlbS1jYWNoZVwiO1xuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGltcG9ydC9uby1leHRyYW5lb3VzLWRlcGVuZGVuY2llc1xuaW1wb3J0IHtcbiAgSW5jcmVtZW50YWxDYWNoZUtpbmRIaW50LFxuICBJbmNyZW1lbnRhbENhY2hlVmFsdWUsXG59IGZyb20gXCJuZXh0L2Rpc3Qvc2VydmVyL3Jlc3BvbnNlLWNhY2hlXCI7XG5pbXBvcnQgeyBDREtfTkVYVEpTX1NFUlZFUl9ESVNUX0RJUl9FTlZfVkFSX05BTUUgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIENka05leHRqc0NhY2hlSGFuZGxlciBleHRlbmRzIEZpbGVTeXN0ZW1DYWNoZSB7XG4gIGNvbnN0cnVjdG9yKG9wdGlvbnM6IENvbnN0cnVjdG9yUGFyYW1ldGVyczx0eXBlb2YgRmlsZVN5c3RlbUNhY2hlPlswXSkge1xuICAgIC8qKlxuICAgICAqIEBleGFtcGxlIGAvbW50L2Nkay1uZXh0anMve3JlbGF0aXZlUGF0aFRvUGFja2FnZX0vLm5leHQvc2VydmVyYFxuICAgICAqIFRoaXMgYWxsb3dzIHVzIHRvIHRlbGwgTmV4dC5qcyBgRmlsZVN5c3RlbUNhY2hlYCB0byB3cml0ZSAubmV4dC9jYWNoZS9mZXRjaC1jYWNoZVxuICAgICAqIGRhdGEgYW5kIC5uZXh0L3NlcnZlci9hcHAvKi57aHRtbCxyc2MsbWV0YSxib2R5fSBkYXRhIHRvIG91ciBFRlMgc2hhcmVkXG4gICAgICogZmlsZSBzeXN0ZW0uIE5vdGUsIHdlIGNhbm5vdCBjb250cm9sIHdoZXJlIG9wdGltaXplZCBpbWFnZXMgYXJlIGNhY2hlZFxuICAgICAqIHRocm91Z2ggcHVibGljIE5leHQuanMgQVBJcyBzbyB3ZSBtdXN0IHN5bWxpbmsuXG4gICAgICpcbiAgICAgKiBOb3RlLCB0aGUgYHNlcnZlckRpc3REaXJgIGRlZmF1bHRzIHRvIGN1cnJlbnQgZGlyZWN0b3J5IG9mIHRoZSBOZXh0LmpzXG4gICAgICogYXBwIGJ1dCB3ZSBjYW5ub3QgdXNlIHRoaXMgYmVjYXVzZSAxLyB3ZSB3YW50IHRvIHNoYXJlIGNhY2hlIGJldHdlZW5cbiAgICAgKiBtdXRpcGxlIGZ1bmN0aW9ucyBvciBjb250YWluZXJzIHZpYSBFRlMgbW91bnQgMi8gbGFtYmRhIGNhbm5vdCB3cml0ZVxuICAgICAqIHRvIGRpc2sgKHVubGVzcyAvdG1wIG9yIEVGUyBtb3VudClcbiAgICAgKi9cbiAgICBjb25zdCBjZGtOZXh0anNTZXJ2ZXJEaXN0RGlyID1cbiAgICAgIHByb2Nlc3MuZW52W0NES19ORVhUSlNfU0VSVkVSX0RJU1RfRElSX0VOVl9WQVJfTkFNRV07XG4gICAgaWYgKCFjZGtOZXh0anNTZXJ2ZXJEaXN0RGlyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgIGAke0NES19ORVhUSlNfU0VSVkVSX0RJU1RfRElSX0VOVl9WQVJfTkFNRX0gZW52aXJvbm1lbnQgdmFyaWFibGUgbm90IHNldGAsXG4gICAgICApO1xuICAgIH1cbiAgICBzdXBlcih7IC4uLm9wdGlvbnMsIHNlcnZlckRpc3REaXI6IGNka05leHRqc1NlcnZlckRpc3REaXIgfSk7XG4gIH1cblxuICBnZXQoXG4gICAgY2FjaGVLZXk6IHN0cmluZyxcbiAgICBjdHg/OlxuICAgICAgfCB7XG4gICAgICAgICAga2luZEhpbnQ/OiBJbmNyZW1lbnRhbENhY2hlS2luZEhpbnQ7XG4gICAgICAgICAgcmV2YWxpZGF0ZT86IG51bWJlciB8IGZhbHNlO1xuICAgICAgICAgIGZldGNoVXJsPzogc3RyaW5nO1xuICAgICAgICAgIGZldGNoSWR4PzogbnVtYmVyO1xuICAgICAgICAgIHRhZ3M/OiBzdHJpbmdbXTtcbiAgICAgICAgICBzb2Z0VGFncz86IHN0cmluZ1tdO1xuICAgICAgICB9XG4gICAgICB8IHVuZGVmaW5lZCxcbiAgKTogUHJvbWlzZTxDYWNoZUhhbmRsZXJWYWx1ZSB8IG51bGw+IHtcbiAgICAvLyBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeSh7IGNhY2hlS2V5LCBjdHggfSwgbnVsbCwgMikpO1xuICAgIHJldHVybiBzdXBlci5nZXQoY2FjaGVLZXksIGN0eCk7XG4gIH1cblxuICBzZXQoXG4gICAgcGF0aG5hbWU6IHN0cmluZyxcbiAgICBkYXRhOiBJbmNyZW1lbnRhbENhY2hlVmFsdWUgfCBudWxsLFxuICAgIGN0eDoge1xuICAgICAgcmV2YWxpZGF0ZT86IG51bWJlciB8IGZhbHNlO1xuICAgICAgZmV0Y2hDYWNoZT86IGJvb2xlYW47XG4gICAgICBmZXRjaFVybD86IHN0cmluZztcbiAgICAgIGZldGNoSWR4PzogbnVtYmVyO1xuICAgICAgdGFncz86IHN0cmluZ1tdO1xuICAgIH0sXG4gICk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIGNvbnNvbGUubG9nKFxuICAgIC8vICAgSlNPTi5zdHJpbmdpZnkoXG4gICAgLy8gICAgIHtcbiAgICAvLyAgICAgICBwYXRobmFtZSxcbiAgICAvLyAgICAgICBkYXRhOlxuICAgIC8vICAgICAgICAgZGF0YT8ua2luZCA9PT0gXCJQQUdFXCJcbiAgICAvLyAgICAgICAgICAgPyB7IC4uLmRhdGEsIHBhZ2VEYXRhOiBcIi4uLlwiLCBodG1sOiBcIi4uLlwiIH1cbiAgICAvLyAgICAgICAgICAgOiBkYXRhLFxuICAgIC8vICAgICAgIGN0eCxcbiAgICAvLyAgICAgfSxcbiAgICAvLyAgICAgbnVsbCxcbiAgICAvLyAgICAgMixcbiAgICAvLyAgICksXG4gICAgLy8gKTtcbiAgICByZXR1cm4gc3VwZXIuc2V0KHBhdGhuYW1lLCBkYXRhLCBjdHgpO1xuICB9XG59XG4iXX0=