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