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.
- package/.jsii +1248 -748
- package/API.md +1033 -531
- package/README.md +1 -2
- package/assets/lambdas/assets-deployment/assets-deployment.lambda/assets-deployment.Dockerfile +12 -8
- package/assets/lambdas/assets-deployment/assets-deployment.lambda/index.js +67 -56
- package/assets/lambdas/post-deploy/post-deploy.lambda/index.js +246 -0
- package/docs/breaking-changes.md +9 -0
- package/docs/notes.md +170 -20
- package/lib/constants.d.ts +32 -0
- package/lib/constants.js +38 -0
- package/lib/generated-structs/OptionalCustomResourceProps.d.ts +104 -0
- package/lib/generated-structs/OptionalCustomResourceProps.js +3 -0
- package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +21 -0
- package/lib/generated-structs/OptionalDockerImageAssetProps.js +1 -1
- package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +4 -3
- package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +1 -1
- package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +1 -1
- package/lib/generated-structs/OptionalNextjsBuildProps.js +1 -1
- package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +3 -3
- package/lib/generated-structs/OptionalNextjsContainersProps.js +1 -1
- package/lib/generated-structs/OptionalNextjsPostDeployProps.d.ts +42 -0
- package/lib/generated-structs/OptionalNextjsPostDeployProps.js +3 -0
- package/lib/generated-structs/OptionalPostDeployCustomResourceProperties.d.ts +41 -0
- package/lib/generated-structs/OptionalPostDeployCustomResourceProperties.js +4 -0
- package/lib/index.d.ts +6 -5
- package/lib/index.js +6 -8
- package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +5 -23
- package/lib/lambdas/assets-deployment/fs-to-fs.js +18 -3
- package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +8 -2
- package/lib/lambdas/assets-deployment/fs-to-s3.js +14 -8
- package/lib/lambdas/post-deploy/create-invalidation.d.ts +2 -0
- package/lib/lambdas/post-deploy/create-invalidation.js +19 -0
- package/lib/lambdas/post-deploy/post-deploy-function.d.ts +13 -0
- package/lib/lambdas/post-deploy/post-deploy-function.js +22 -0
- package/lib/lambdas/post-deploy/post-deploy.lambda.d.ts +9 -0
- package/lib/lambdas/post-deploy/post-deploy.lambda.js +56 -0
- package/lib/lambdas/post-deploy/prune-fs.d.ts +10 -0
- package/lib/lambdas/post-deploy/prune-fs.js +33 -0
- package/lib/lambdas/post-deploy/prune-s3.d.ts +15 -0
- package/lib/lambdas/post-deploy/prune-s3.js +101 -0
- package/lib/lambdas/utils.js +35 -0
- package/lib/nextjs-assets-deployment.d.ts +15 -51
- package/lib/nextjs-assets-deployment.js +27 -25
- package/lib/nextjs-build/builder.Dockerfile +12 -8
- package/lib/nextjs-build/cdk-nextjs-cache-handler.cjs +63 -0
- package/lib/nextjs-build/cdk-nextjs-cache-handler.d.ts +21 -0
- package/lib/nextjs-build/cdk-nextjs-cache-handler.js +49 -0
- package/lib/nextjs-build/global-containers.Dockerfile +29 -19
- package/lib/nextjs-build/global-functions.Dockerfile +30 -24
- package/lib/nextjs-build/nextjs-build.d.ts +51 -26
- package/lib/nextjs-build/nextjs-build.js +108 -88
- package/lib/nextjs-build/regional-containers.Dockerfile +30 -20
- package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +0 -1
- package/lib/nextjs-compute/nextjs-compute-base-props.js +1 -1
- package/lib/nextjs-compute/nextjs-containers.d.ts +2 -1
- package/lib/nextjs-compute/nextjs-containers.js +7 -5
- package/lib/nextjs-compute/nextjs-functions.d.ts +1 -0
- package/lib/nextjs-compute/nextjs-functions.js +8 -11
- package/lib/nextjs-distribution.d.ts +1 -1
- package/lib/nextjs-distribution.js +4 -4
- package/lib/nextjs-file-system.js +1 -1
- package/lib/nextjs-post-deploy.d.ts +101 -0
- package/lib/nextjs-post-deploy.js +78 -0
- package/lib/nextjs-static-assets.js +1 -1
- package/lib/nextjs-vpc.d.ts +1 -1
- package/lib/nextjs-vpc.js +4 -4
- package/lib/root-constructs/nextjs-base-overrides.d.ts +4 -0
- package/lib/root-constructs/nextjs-base-overrides.js +1 -1
- package/lib/root-constructs/nextjs-base-props.d.ts +15 -7
- package/lib/root-constructs/nextjs-base-props.js +1 -1
- package/lib/root-constructs/nextjs-global-containers.d.ts +8 -8
- package/lib/root-constructs/nextjs-global-containers.js +25 -15
- package/lib/root-constructs/nextjs-global-functions.d.ts +8 -11
- package/lib/root-constructs/nextjs-global-functions.js +25 -25
- package/lib/root-constructs/nextjs-regional-containers.d.ts +8 -0
- package/lib/root-constructs/nextjs-regional-containers.js +26 -9
- package/lib/utils/get-architecture.d.ts +2 -0
- package/lib/utils/get-architecture.js +8 -0
- package/lib/utils/handle-deprecated-properties.d.ts +7 -0
- package/lib/utils/handle-deprecated-properties.js +14 -0
- package/package.json +17 -19
- package/assets/lambdas/revalidate/revalidate.lambda/index.js +0 -67
- package/lib/common.d.ts +0 -25
- package/lib/common.js +0 -30
- package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +0 -11
- package/lib/generated-structs/OptionalNextjsInvalidationProps.js +0 -3
- package/lib/lambdas/assets-deployment/prune-s3.d.ts +0 -15
- package/lib/lambdas/assets-deployment/prune-s3.js +0 -42
- package/lib/lambdas/assets-deployment/s3.d.ts +0 -2
- package/lib/lambdas/assets-deployment/s3.js +0 -7
- package/lib/lambdas/assets-deployment/utils.js +0 -35
- package/lib/lambdas/revalidate/revalidate-function.d.ts +0 -13
- package/lib/lambdas/revalidate/revalidate-function.js +0 -22
- package/lib/lambdas/revalidate/revalidate.lambda.d.ts +0 -2
- package/lib/lambdas/revalidate/revalidate.lambda.js +0 -53
- package/lib/nextjs-build/add-cache-handler.d.ts +0 -1
- package/lib/nextjs-build/add-cache-handler.js +0 -23
- package/lib/nextjs-build/add-cache-handler.mjs +0 -18
- package/lib/nextjs-build/cache-handler.cjs +0 -21878
- package/lib/nextjs-build/cache-handler.d.ts +0 -6
- package/lib/nextjs-build/cache-handler.js +0 -26
- package/lib/nextjs-invalidation.d.ts +0 -19
- package/lib/nextjs-invalidation.js +0 -52
- package/lib/nextjs-revalidation.d.ts +0 -30
- package/lib/nextjs-revalidation.js +0 -65
- /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 "./
|
|
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
|
-
*
|
|
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.
|
|
38
|
+
* @see {@link NextjsBaseProps.relativePathToPackage}
|
|
37
39
|
*/
|
|
38
|
-
readonly
|
|
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
|
|
66
|
-
|
|
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
|
-
*
|
|
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
|
|
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,
|
|
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,
|
|
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,
|
|
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
|
-
//
|
|
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", "
|
|
75
|
-
destinationPath: (0, posix_1.join)(this.props.
|
|
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
|
-
//
|
|
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,
|
|
81
|
-
destinationPath: (0, posix_1.join)(this.props.
|
|
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,
|
|
87
|
-
destinationPath: (0, posix_1.join)(this.props.
|
|
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.
|
|
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.
|
|
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.
|
|
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
|
|
22
|
-
RUN
|
|
23
|
-
RUN
|
|
24
|
-
#
|
|
25
|
-
|
|
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=
|