cdk-nextjs 0.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (125) hide show
  1. package/.jsii +13811 -0
  2. package/.prettierrc +0 -0
  3. package/API.md +9694 -0
  4. package/LICENSE +202 -0
  5. package/README.md +137 -0
  6. package/THIRD-PARTY-LICENSES.md +31 -0
  7. package/assets/lambdas/assets-deployment/assets-deployment.lambda/assets-deployment.Dockerfile +18 -0
  8. package/assets/lambdas/assets-deployment/assets-deployment.lambda/index.js +8831 -0
  9. package/assets/lambdas/revalidate/revalidate.lambda/index.js +67 -0
  10. package/assets/lambdas/sign-fn-url/sign-fn-url.lambda/index.js +2002 -0
  11. package/examples/README.md +14 -0
  12. package/lib/common.d.ts +23 -0
  13. package/lib/common.js +28 -0
  14. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.d.ts +106 -0
  15. package/lib/generated-structs/OptionalApplicationLoadBalancedTaskImageOptions.js +3 -0
  16. package/lib/generated-structs/OptionalCloudFrontFunctionProps.d.ts +43 -0
  17. package/lib/generated-structs/OptionalCloudFrontFunctionProps.js +3 -0
  18. package/lib/generated-structs/OptionalClusterProps.d.ts +49 -0
  19. package/lib/generated-structs/OptionalClusterProps.js +3 -0
  20. package/lib/generated-structs/OptionalDistributionProps.d.ts +155 -0
  21. package/lib/generated-structs/OptionalDistributionProps.js +3 -0
  22. package/lib/generated-structs/OptionalDockerImageAssetProps.d.ts +124 -0
  23. package/lib/generated-structs/OptionalDockerImageAssetProps.js +3 -0
  24. package/lib/generated-structs/OptionalDockerImageFunctionProps.d.ts +399 -0
  25. package/lib/generated-structs/OptionalDockerImageFunctionProps.js +3 -0
  26. package/lib/generated-structs/OptionalEdgeFunctionProps.d.ts +428 -0
  27. package/lib/generated-structs/OptionalEdgeFunctionProps.js +3 -0
  28. package/lib/generated-structs/OptionalFunctionProps.d.ts +422 -0
  29. package/lib/generated-structs/OptionalFunctionProps.js +3 -0
  30. package/lib/generated-structs/OptionalFunctionUrlProps.d.ts +30 -0
  31. package/lib/generated-structs/OptionalFunctionUrlProps.js +3 -0
  32. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.d.ts +45 -0
  33. package/lib/generated-structs/OptionalNextjsAssetsDeploymentProps.js +3 -0
  34. package/lib/generated-structs/OptionalNextjsBuildProps.d.ts +30 -0
  35. package/lib/generated-structs/OptionalNextjsBuildProps.js +3 -0
  36. package/lib/generated-structs/OptionalNextjsContainersProps.d.ts +43 -0
  37. package/lib/generated-structs/OptionalNextjsContainersProps.js +3 -0
  38. package/lib/generated-structs/OptionalNextjsDistributionProps.d.ts +50 -0
  39. package/lib/generated-structs/OptionalNextjsDistributionProps.js +3 -0
  40. package/lib/generated-structs/OptionalNextjsFileSystemProps.d.ts +15 -0
  41. package/lib/generated-structs/OptionalNextjsFileSystemProps.js +3 -0
  42. package/lib/generated-structs/OptionalNextjsInvalidationProps.d.ts +17 -0
  43. package/lib/generated-structs/OptionalNextjsInvalidationProps.js +3 -0
  44. package/lib/generated-structs/OptionalNextjsVpcProps.d.ts +21 -0
  45. package/lib/generated-structs/OptionalNextjsVpcProps.js +3 -0
  46. package/lib/generated-structs/OptionalS3OriginProps.d.ts +64 -0
  47. package/lib/generated-structs/OptionalS3OriginProps.js +3 -0
  48. package/lib/generated-structs/OptionalVpcProps.d.ts +224 -0
  49. package/lib/generated-structs/OptionalVpcProps.js +3 -0
  50. package/lib/index.d.ts +35 -0
  51. package/lib/index.js +32 -0
  52. package/lib/lambdas/assets-deployment/assets-deployment-function.d.ts +13 -0
  53. package/lib/lambdas/assets-deployment/assets-deployment-function.js +23 -0
  54. package/lib/lambdas/assets-deployment/assets-deployment.lambda.d.ts +2 -0
  55. package/lib/lambdas/assets-deployment/assets-deployment.lambda.js +62 -0
  56. package/lib/lambdas/assets-deployment/common.d.ts +8 -0
  57. package/lib/lambdas/assets-deployment/common.js +32 -0
  58. package/lib/lambdas/assets-deployment/fs-to-fs.d.ts +2 -0
  59. package/lib/lambdas/assets-deployment/fs-to-fs.js +9 -0
  60. package/lib/lambdas/assets-deployment/fs-to-s3.d.ts +2 -0
  61. package/lib/lambdas/assets-deployment/fs-to-s3.js +45 -0
  62. package/lib/lambdas/assets-deployment/prune-s3.d.ts +15 -0
  63. package/lib/lambdas/assets-deployment/prune-s3.js +42 -0
  64. package/lib/lambdas/assets-deployment/s3.d.ts +2 -0
  65. package/lib/lambdas/assets-deployment/s3.js +7 -0
  66. package/lib/lambdas/assets-deployment/utils.d.ts +18 -0
  67. package/lib/lambdas/assets-deployment/utils.js +35 -0
  68. package/lib/lambdas/revalidate/revalidate-function.d.ts +13 -0
  69. package/lib/lambdas/revalidate/revalidate-function.js +23 -0
  70. package/lib/lambdas/revalidate/revalidate.lambda.d.ts +2 -0
  71. package/lib/lambdas/revalidate/revalidate.lambda.js +53 -0
  72. package/lib/lambdas/sign-fn-url/sign-fn-url-function.d.ts +13 -0
  73. package/lib/lambdas/sign-fn-url/sign-fn-url-function.js +23 -0
  74. package/lib/lambdas/sign-fn-url/sign-fn-url.lambda.d.ts +9 -0
  75. package/lib/lambdas/sign-fn-url/sign-fn-url.lambda.js +35 -0
  76. package/lib/lambdas/sign-fn-url/sign-request.d.ts +28 -0
  77. package/lib/lambdas/sign-fn-url/sign-request.js +119 -0
  78. package/lib/lambdas/sign-fn-url/sign-request.test.d.ts +1 -0
  79. package/lib/lambdas/sign-fn-url/sign-request.test.js +129 -0
  80. package/lib/nextjs-assets-deployment.d.ts +116 -0
  81. package/lib/nextjs-assets-deployment.js +93 -0
  82. package/lib/nextjs-build/add-cache-handler.d.ts +1 -0
  83. package/lib/nextjs-build/add-cache-handler.js +23 -0
  84. package/lib/nextjs-build/add-cache-handler.mjs +18 -0
  85. package/lib/nextjs-build/builder.Dockerfile +29 -0
  86. package/lib/nextjs-build/cache-handler.cjs +21513 -0
  87. package/lib/nextjs-build/cache-handler.d.ts +6 -0
  88. package/lib/nextjs-build/cache-handler.js +22 -0
  89. package/lib/nextjs-build/global-containers.Dockerfile +45 -0
  90. package/lib/nextjs-build/global-functions.Dockerfile +46 -0
  91. package/lib/nextjs-build/nextjs-build.d.ts +150 -0
  92. package/lib/nextjs-build/nextjs-build.js +220 -0
  93. package/lib/nextjs-build/regional-containers.Dockerfile +45 -0
  94. package/lib/nextjs-build/symlink-full-route-cache.d.ts +1 -0
  95. package/lib/nextjs-build/symlink-full-route-cache.js +37 -0
  96. package/lib/nextjs-build/symlink-full-route-cache.mjs +23 -0
  97. package/lib/nextjs-compute/nextjs-compute-base-props.d.ts +8 -0
  98. package/lib/nextjs-compute/nextjs-compute-base-props.js +3 -0
  99. package/lib/nextjs-compute/nextjs-containers.d.ts +43 -0
  100. package/lib/nextjs-compute/nextjs-containers.js +149 -0
  101. package/lib/nextjs-compute/nextjs-functions.d.ts +23 -0
  102. package/lib/nextjs-compute/nextjs-functions.js +57 -0
  103. package/lib/nextjs-distribution.d.ts +120 -0
  104. package/lib/nextjs-distribution.js +362 -0
  105. package/lib/nextjs-file-system.d.ts +42 -0
  106. package/lib/nextjs-file-system.js +74 -0
  107. package/lib/nextjs-invalidation.d.ts +19 -0
  108. package/lib/nextjs-invalidation.js +52 -0
  109. package/lib/nextjs-revalidation.d.ts +30 -0
  110. package/lib/nextjs-revalidation.js +65 -0
  111. package/lib/nextjs-static-assets.d.ts +21 -0
  112. package/lib/nextjs-static-assets.js +32 -0
  113. package/lib/nextjs-vpc.d.ts +42 -0
  114. package/lib/nextjs-vpc.js +64 -0
  115. package/lib/root-constructs/nextjs-base-overrides.d.ts +26 -0
  116. package/lib/root-constructs/nextjs-base-overrides.js +3 -0
  117. package/lib/root-constructs/nextjs-base-props.d.ts +56 -0
  118. package/lib/root-constructs/nextjs-base-props.js +3 -0
  119. package/lib/root-constructs/nextjs-global-containers.d.ts +74 -0
  120. package/lib/root-constructs/nextjs-global-containers.js +125 -0
  121. package/lib/root-constructs/nextjs-global-functions.d.ts +76 -0
  122. package/lib/root-constructs/nextjs-global-functions.js +131 -0
  123. package/lib/root-constructs/nextjs-regional-containers.d.ts +43 -0
  124. package/lib/root-constructs/nextjs-regional-containers.js +92 -0
  125. package/package.json +165 -0
@@ -0,0 +1,6 @@
1
+ import FileSystemCache from "next/dist/server/lib/incremental-cache/file-system-cache";
2
+ type RevalidateTagResponse = ReturnType<FileSystemCache["revalidateTag"]>;
3
+ export default class CacheHandler extends FileSystemCache {
4
+ revalidateTag(tag: string): RevalidateTagResponse;
5
+ }
6
+ export {};
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ // eslint-disable-next-line import/no-extraneous-dependencies
4
+ const client_sqs_1 = require("@aws-sdk/client-sqs");
5
+ // eslint-disable-next-line import/no-extraneous-dependencies
6
+ const file_system_cache_1 = require("next/dist/server/lib/incremental-cache/file-system-cache");
7
+ const sqsClient = new client_sqs_1.SQSClient();
8
+ class CacheHandler extends file_system_cache_1.default {
9
+ async revalidateTag(tag) {
10
+ console.log({ tag });
11
+ console.log({ this: this });
12
+ await sqsClient.send(new client_sqs_1.SendMessageCommand({
13
+ QueueUrl: process.env.CDK_NEXTJS_QUEUE_URL,
14
+ MessageBody: tag,
15
+ // MessageGroupId: "", // do i need this?
16
+ }));
17
+ // put message in sqs queue if for full route cache
18
+ return super.revalidateTag(tag);
19
+ }
20
+ }
21
+ exports.default = CacheHandler;
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FjaGUtaGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uZXh0anMtYnVpbGQvY2FjaGUtaGFuZGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLDZEQUE2RDtBQUM3RCxvREFBb0U7QUFDcEUsNkRBQTZEO0FBQzdELGdHQUF1RjtBQUl2RixNQUFNLFNBQVMsR0FBRyxJQUFJLHNCQUFTLEVBQUUsQ0FBQztBQUVsQyxNQUFxQixZQUFhLFNBQVEsMkJBQWU7SUFDdkQsS0FBSyxDQUFDLGFBQWEsQ0FBQyxHQUFXO1FBQzdCLE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDO1FBQ3JCLE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUM1QixNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQ2xCLElBQUksK0JBQWtCLENBQUM7WUFDckIsUUFBUSxFQUFFLE9BQU8sQ0FBQyxHQUFHLENBQUMsb0JBQW9CO1lBQzFDLFdBQVcsRUFBRSxHQUFHO1lBQ2hCLHlDQUF5QztTQUMxQyxDQUFDLENBQ0gsQ0FBQztRQUNGLG1EQUFtRDtRQUNuRCxPQUFPLEtBQUssQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDbEMsQ0FBQztDQUNGO0FBZEQsK0JBY0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzXG5pbXBvcnQgeyBTUVNDbGllbnQsIFNlbmRNZXNzYWdlQ29tbWFuZCB9IGZyb20gXCJAYXdzLXNkay9jbGllbnQtc3FzXCI7XG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L25vLWV4dHJhbmVvdXMtZGVwZW5kZW5jaWVzXG5pbXBvcnQgRmlsZVN5c3RlbUNhY2hlIGZyb20gXCJuZXh0L2Rpc3Qvc2VydmVyL2xpYi9pbmNyZW1lbnRhbC1jYWNoZS9maWxlLXN5c3RlbS1jYWNoZVwiO1xuXG50eXBlIFJldmFsaWRhdGVUYWdSZXNwb25zZSA9IFJldHVyblR5cGU8RmlsZVN5c3RlbUNhY2hlW1wicmV2YWxpZGF0ZVRhZ1wiXT47XG5cbmNvbnN0IHNxc0NsaWVudCA9IG5ldyBTUVNDbGllbnQoKTtcblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgQ2FjaGVIYW5kbGVyIGV4dGVuZHMgRmlsZVN5c3RlbUNhY2hlIHtcbiAgYXN5bmMgcmV2YWxpZGF0ZVRhZyh0YWc6IHN0cmluZyk6IFJldmFsaWRhdGVUYWdSZXNwb25zZSB7XG4gICAgY29uc29sZS5sb2coeyB0YWcgfSk7XG4gICAgY29uc29sZS5sb2coeyB0aGlzOiB0aGlzIH0pO1xuICAgIGF3YWl0IHNxc0NsaWVudC5zZW5kKFxuICAgICAgbmV3IFNlbmRNZXNzYWdlQ29tbWFuZCh7XG4gICAgICAgIFF1ZXVlVXJsOiBwcm9jZXNzLmVudi5DREtfTkVYVEpTX1FVRVVFX1VSTCxcbiAgICAgICAgTWVzc2FnZUJvZHk6IHRhZyxcbiAgICAgICAgLy8gTWVzc2FnZUdyb3VwSWQ6IFwiXCIsIC8vIGRvIGkgbmVlZCB0aGlzP1xuICAgICAgfSksXG4gICAgKTtcbiAgICAvLyBwdXQgbWVzc2FnZSBpbiBzcXMgcXVldWUgaWYgZm9yIGZ1bGwgcm91dGUgY2FjaGVcbiAgICByZXR1cm4gc3VwZXIucmV2YWxpZGF0ZVRhZyh0YWcpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,45 @@
1
+ #checkov:skip=CKV_DOCKER_2: healthcheck run by ALB and ECS
2
+ #checkov:skip=CKV_DOCKER_7: latest tag is ok to use for local builder container
3
+ # Modified from: https://github.com/vercel/next.js/blob/canary/examples/with-docker/Dockerfile
4
+ ARG BUILDER_IMAGE_TAG
5
+ FROM $BUILDER_IMAGE_TAG as builder
6
+ # Production image, copy all the files and run next
7
+ FROM public.ecr.aws/docker/library/node:20-alpine as runner
8
+ WORKDIR /app
9
+
10
+ ENV NODE_ENV production
11
+ # Uncomment the following line in case you want to disable telemetry during runtime.
12
+ # ENV NEXT_TELEMETRY_DISABLED 1
13
+
14
+ RUN addgroup --system --gid 1001 nodejs
15
+ RUN adduser --system --uid 1001 nextjs
16
+
17
+ ARG RELATIVE_PATH_TO_WORKSPACE
18
+ # even though public is served through cloudfront, we need public for image optimization
19
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/public ./$RELATIVE_PATH_TO_WORKSPACE/public
20
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/standalone ./
21
+ COPY --chown=nextjs:nodejs ./symlink-full-route-cache.mjs ./
22
+ ARG MOUNT_PATH
23
+ ARG DATA_CACHE_DIR
24
+ ARG FULL_ROUTE_CACHE_DIR
25
+ ARG IMAGE_CACHE_DIR
26
+ RUN mkdir -p $MOUNT_PATH/$DATA_CACHE_DIR && \
27
+ mkdir -p $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
28
+ mkdir -p $MOUNT_PATH/$IMAGE_CACHE_DIR && \
29
+ chmod -R u+rw $MOUNT_PATH && \
30
+ mkdir -p ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache && \
31
+ ln -s $MOUNT_PATH/$DATA_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/fetch-cache && \
32
+ ln -s $MOUNT_PATH/$IMAGE_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/images && \
33
+ node symlink-full-route-cache.mjs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/server/app $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
34
+ rm symlink-full-route-cache.mjs
35
+
36
+ USER nextjs
37
+
38
+ EXPOSE 3000
39
+
40
+ ENV PORT 3000
41
+
42
+ # server.js is created by next build from the standalone output
43
+ # https://nextjs.org/docs/pages/api-reference/next-config-js/output
44
+ # CMD will be overwritten by ECS Task Definition
45
+ CMD HOSTNAME="0.0.0.0" node server.js
@@ -0,0 +1,46 @@
1
+ #checkov:skip=CKV_DOCKER_2: healthcheck run by AWS Lambda Web Adapter
2
+ #checkov:skip=CKV_DOCKER_7: latest tag is ok to use for local builder container
3
+ # Modified from: https://github.com/vercel/next.js/blob/canary/examples/with-docker/Dockerfile
4
+ ARG BUILDER_IMAGE_TAG
5
+ FROM $BUILDER_IMAGE_TAG as builder
6
+ # Production image, copy all the files and run next
7
+ FROM public.ecr.aws/docker/library/node:20-alpine as runner
8
+ ARG MOUNT_PATH
9
+ COPY --from=public.ecr.aws/awsguru/aws-lambda-adapter:0.8.3 /lambda-adapter /opt/extensions/lambda-adapter
10
+ WORKDIR /app
11
+
12
+ ENV NODE_ENV production
13
+ # Uncomment the following line in case you want to disable telemetry during runtime.
14
+ # ENV NEXT_TELEMETRY_DISABLED 1
15
+
16
+ RUN addgroup --system --gid 1001 nodejs
17
+ RUN adduser --system --uid 1001 nextjs
18
+
19
+ ARG RELATIVE_PATH_TO_WORKSPACE
20
+ # even though public is served through cloudfront, we need public for image optimization
21
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/public ./$RELATIVE_PATH_TO_WORKSPACE/public
22
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/standalone ./
23
+ COPY --chown=nextjs:nodejs ./add-cache-handler.mjs ./cache-handler.cjs ./symlink-full-route-cache.mjs ./
24
+ ARG MOUNT_PATH
25
+ ARG DATA_CACHE_DIR
26
+ ARG FULL_ROUTE_CACHE_DIR
27
+ ARG IMAGE_CACHE_DIR
28
+ RUN node add-cache-handler.mjs ./$RELATIVE_PATH_TO_WORKSPACE/.next/required-server-files.json && \
29
+ rm add-cache-handler.mjs && \
30
+ mkdir -p $MOUNT_PATH/$DATA_CACHE_DIR && \
31
+ mkdir -p $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
32
+ mkdir -p $MOUNT_PATH/$IMAGE_CACHE_DIR && \
33
+ chmod -R u+rw $MOUNT_PATH && \
34
+ mkdir -p ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache && \
35
+ ln -s $MOUNT_PATH/$DATA_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/fetch-cache && \
36
+ ln -s $MOUNT_PATH/$IMAGE_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/images && \
37
+ node symlink-full-route-cache.mjs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/server/app $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
38
+ rm symlink-full-route-cache.mjs
39
+
40
+ USER nextjs
41
+
42
+ EXPOSE 3000
43
+
44
+ ENV PORT 3000
45
+
46
+ # CMD will be overwritten by Lambda IaC
@@ -0,0 +1,150 @@
1
+ import { DockerImageAsset, Platform } from "aws-cdk-lib/aws-ecr-assets";
2
+ import { AssetImageCodeProps, DockerImageCode } from "aws-cdk-lib/aws-lambda";
3
+ import { Construct } from "constructs";
4
+ import { NextjsType } from "../common";
5
+ import { OptionalDockerImageAssetProps } from "../generated-structs/OptionalDockerImageAssetProps";
6
+ import { NextjsBaseProps } from "../root-constructs/nextjs-base-props";
7
+ export interface BuilderImageProps {
8
+ /**
9
+ * Build Args to be passed to `docker build` command.
10
+ */
11
+ readonly buildArgs?: Record<string, string>;
12
+ /**
13
+ * `docker build ...` command to run in {@link NextBaseProps.buildContext}.
14
+ * Default interpolates other props. If you override, other props will have
15
+ * no effect on command.
16
+ */
17
+ readonly command?: string;
18
+ /**
19
+ * Environment variables names to pass from host to container during build process.
20
+ *
21
+ * Note, a shell script, cdk-nextjs-load-env-vars.sh is created within the
22
+ * {@link NextBaseProps.buildContext} directory, which will contain all the
23
+ * environment variables defined in this prop. If you've created your own
24
+ * custom Dockerfile (passed in via {@link BuilderImageProps.customDockerfilePath})
25
+ * then you need to make sure you're copying it into the image.
26
+ *
27
+ * @example ["MY_API_KEY"]
28
+ */
29
+ readonly envVarNames?: string[];
30
+ /**
31
+ * Lines in .dockerignore file which will be created in your {@link NextBaseProps.buildContext}
32
+ * @default ["node_modules", ".git", ".gitignore", ".md"]
33
+ */
34
+ readonly exclude?: string[];
35
+ /**
36
+ * Name of Dockerfile
37
+ * @default "builder.Dockerfile"
38
+ */
39
+ readonly file?: string;
40
+ readonly platform?: Platform;
41
+ /**
42
+ * Skip building the builder image.
43
+ * @default false
44
+ */
45
+ readonly skipBuild?: boolean;
46
+ /**
47
+ * Path to your custom builder.Dockerfile which will be copied into {@link NextBaseProps.buildContext}.
48
+ * It is recommended to override this prop to optimize build caching for your setup.
49
+ */
50
+ readonly customDockerfilePath?: string;
51
+ }
52
+ export interface NextjsBuildOverrides {
53
+ readonly nextjsContainersDockerImageAssetProps?: OptionalDockerImageAssetProps;
54
+ readonly nextjsFunctionsAssetImageCodeProps?: AssetImageCodeProps;
55
+ readonly nextjsAssetDeploymentAssetImageCodeProps?: AssetImageCodeProps;
56
+ }
57
+ export interface NextjsBuildProps {
58
+ /**
59
+ * @see {@link NextjsBaseProps["buildCommand"]}
60
+ */
61
+ readonly buildCommand: NextjsBaseProps["buildCommand"];
62
+ /**
63
+ * @see {@link NextjsBaseProps["buildContext"]}
64
+ */
65
+ readonly buildContext: NextjsBaseProps["buildContext"];
66
+ /**
67
+ *
68
+ */
69
+ readonly builderImageProps?: BuilderImageProps;
70
+ /**
71
+ * @see {@link NextjsBaseProps.relativePathToWorkspace}
72
+ */
73
+ readonly relativePathToWorkspace?: NextjsBaseProps["relativePathToWorkspace"];
74
+ readonly nextjsType: NextjsType;
75
+ readonly overrides?: NextjsBuildOverrides;
76
+ }
77
+ export interface PublicDirEntry {
78
+ readonly name: string;
79
+ readonly isDirectory: boolean;
80
+ }
81
+ /**
82
+ * Builds Next.js assets.
83
+ * @link https://nextjs.org/docs/pages/api-reference/next-config-js/output
84
+ */
85
+ export declare class NextjsBuild extends Construct {
86
+ /**
87
+ * Hash of builder image which will change whenever the image changes. Useful
88
+ * for passing to properties of custom resources that depend upon the builder
89
+ * image to re-run when build image changes.
90
+ */
91
+ buildImageDigest: string;
92
+ /**
93
+ * Mount path in container for EFS. Next.js image optimization, data, and full
94
+ * route cache will be symlinked to this location.
95
+ *
96
+ * Must comply with pattern: ^/mnt/[a-zA-Z0-9-_.]+$
97
+ * @see https://docs.aws.amazon.com/lambda/latest/api/API_FileSystemConfig.html
98
+ */
99
+ containerMountPathForEfs: string;
100
+ /**
101
+ * Docker image built if using Fargate.
102
+ */
103
+ imageForNextjsContainers?: DockerImageAsset;
104
+ /**
105
+ * Docker image built if using Lambda.
106
+ */
107
+ imageForNextjsFunctions?: DockerImageCode;
108
+ /**
109
+ * Docker image built for `NextjsAssetsDeployment`
110
+ */
111
+ imageForNextjsAssetsDeployment: DockerImageCode;
112
+ /**
113
+ * Absolute path to public. Use by CloudFront/ALB to create behaviors/rules
114
+ * @example "/Users/john/myapp/public"
115
+ */
116
+ publicDirEntries: PublicDirEntry[];
117
+ /**
118
+ * The entrypoint JavaScript file used as an argument for Node.js to run the
119
+ * Next.js standalone server relative to the standalone directory.
120
+ * @example "./server.js"
121
+ * @example "./packages/ui/server.js" (monorepo)
122
+ */
123
+ relativePathToEntrypoint: string;
124
+ /**
125
+ * Tag of builder image Next.js app which is built for other images to be
126
+ * built `FROM`. This image isn't built with CDK Assets construct b/c it
127
+ * doesn't need to be uploaded to ECR. We still need to include slice of
128
+ * `node.addr` in tag in case multiple cdk-nextjs constructs are used.
129
+ */
130
+ private builderImageTag;
131
+ private containerRuntime;
132
+ private props;
133
+ private relativePathToWorkspace;
134
+ constructor(scope: Construct, id: string, props: NextjsBuildProps);
135
+ private getRelativeEntrypointPath;
136
+ /**
137
+ * A builder or base image needs to be created so that the same image can be
138
+ * built `FROM` for `NextjsFunctions` or `NextjsContainers` and `NextjsAssetsDeployment`.
139
+ * This image doesn't need to be uploaded to ECR so we're "manually" creating
140
+ * it with `execSync` and other images will be built `FROM` it.
141
+ */
142
+ private createBuilderImage;
143
+ private createBuildArgStr;
144
+ private createLoadEnvVarsScript;
145
+ private getBuilderImageDigest;
146
+ private getPublicDirEntries;
147
+ private createImageForNextjsContainers;
148
+ private createImageForNextjsFunctions;
149
+ private createImageForNextjsAssetsDeployment;
150
+ }
@@ -0,0 +1,220 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.NextjsBuild = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const node_child_process_1 = require("node:child_process");
7
+ const node_fs_1 = require("node:fs");
8
+ const node_path_1 = require("node:path");
9
+ const posix_1 = require("node:path/posix");
10
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
11
+ const aws_ecr_assets_1 = require("aws-cdk-lib/aws-ecr-assets");
12
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
13
+ const constructs_1 = require("constructs");
14
+ const common_1 = require("../common");
15
+ /**
16
+ * Builds Next.js assets.
17
+ * @link https://nextjs.org/docs/pages/api-reference/next-config-js/output
18
+ */
19
+ class NextjsBuild extends constructs_1.Construct {
20
+ constructor(scope, id, props) {
21
+ super(scope, id);
22
+ /**
23
+ * Mount path in container for EFS. Next.js image optimization, data, and full
24
+ * route cache will be symlinked to this location.
25
+ *
26
+ * Must comply with pattern: ^/mnt/[a-zA-Z0-9-_.]+$
27
+ * @see https://docs.aws.amazon.com/lambda/latest/api/API_FileSystemConfig.html
28
+ */
29
+ this.containerMountPathForEfs = "/mnt/cdk-nextjs-cache";
30
+ this.containerRuntime = process.env.CDK_DOCKER || "docker";
31
+ this.builderImageTag = `cdk-nextjs/builder-${this.node.addr.slice(30)}`;
32
+ this.relativePathToWorkspace = props.relativePathToWorkspace || ".";
33
+ this.props = props;
34
+ this.relativePathToEntrypoint = this.getRelativeEntrypointPath();
35
+ this.buildImageDigest = this.createBuilderImage();
36
+ this.publicDirEntries = this.getPublicDirEntries();
37
+ if (props.nextjsType === common_1.NextjsType.GLOBAL_CONTAINERS ||
38
+ props.nextjsType === common_1.NextjsType.REGIONAL_CONTAINERS) {
39
+ this.imageForNextjsContainers = this.createImageForNextjsContainers();
40
+ }
41
+ else {
42
+ this.imageForNextjsFunctions = this.createImageForNextjsFunctions();
43
+ }
44
+ this.imageForNextjsAssetsDeployment =
45
+ this.createImageForNextjsAssetsDeployment();
46
+ }
47
+ getRelativeEntrypointPath() {
48
+ // joinPosix b/c this will be used in linux container
49
+ return (0, posix_1.join)(this.props.relativePathToWorkspace || "", "server.js");
50
+ }
51
+ /**
52
+ * A builder or base image needs to be created so that the same image can be
53
+ * built `FROM` for `NextjsFunctions` or `NextjsContainers` and `NextjsAssetsDeployment`.
54
+ * This image doesn't need to be uploaded to ECR so we're "manually" creating
55
+ * it with `execSync` and other images will be built `FROM` it.
56
+ */
57
+ createBuilderImage() {
58
+ const buildCommand = this.props.buildCommand || "npm run build";
59
+ const { buildArgs = {
60
+ BUILD_COMMAND: buildCommand,
61
+ RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,
62
+ ...this.props.builderImageProps?.buildArgs,
63
+ }, envVarNames = [], exclude = [
64
+ "**/node_modules",
65
+ ".git",
66
+ "**/cdk.out",
67
+ "**/.next",
68
+ ".gitignore",
69
+ "*.md",
70
+ ], file = "builder.Dockerfile", platform, skipBuild = false, } = this.props.builderImageProps || {};
71
+ const srcDockerfilePath = this.props.builderImageProps?.customDockerfilePath ||
72
+ (0, node_path_1.join)(__dirname, file);
73
+ const destDockerfilePath = (0, node_path_1.join)(this.props.buildContext, file);
74
+ (0, node_fs_1.cpSync)(srcDockerfilePath, destDockerfilePath);
75
+ const excludeFileStr = exclude?.join("\n");
76
+ const dockerignoreFilePath = (0, node_path_1.join)(this.props.buildContext, ".dockerignore");
77
+ (0, node_fs_1.writeFileSync)(dockerignoreFilePath, excludeFileStr);
78
+ const buildArgsStr = this.createBuildArgStr(buildArgs);
79
+ const loadEnvVarsScriptPath = (0, node_path_1.join)(this.props.buildContext, "cdk-nextjs-load-env-vars.sh");
80
+ if (envVarNames.length) {
81
+ this.createLoadEnvVarsScript(envVarNames, loadEnvVarsScriptPath);
82
+ }
83
+ const command = this.props.builderImageProps?.command ||
84
+ `${this.containerRuntime} build ${platform ? `--platform ${platform.platform}` : ""} --file ${file} --tag ${this.builderImageTag} ${buildArgsStr} .`;
85
+ let error;
86
+ try {
87
+ if (!skipBuild) {
88
+ console.log(`Building image with command: ${command} in directory: ${this.props.buildContext}`);
89
+ (0, node_child_process_1.execSync)(command, {
90
+ stdio: "inherit",
91
+ cwd: this.props.buildContext,
92
+ env: process.env,
93
+ });
94
+ }
95
+ }
96
+ catch (err) {
97
+ error = err;
98
+ }
99
+ finally {
100
+ (0, node_fs_1.rmSync)(destDockerfilePath);
101
+ (0, node_fs_1.rmSync)(dockerignoreFilePath);
102
+ if (envVarNames.length) {
103
+ (0, node_fs_1.rmSync)(loadEnvVarsScriptPath);
104
+ }
105
+ }
106
+ if (error)
107
+ throw error;
108
+ return this.getBuilderImageDigest();
109
+ }
110
+ createBuildArgStr(buildArgs) {
111
+ return Object.entries(buildArgs).reduce((acc, [key, value]) => {
112
+ return `${acc} --build-arg ${key}="${value}"`;
113
+ }, "");
114
+ }
115
+ createLoadEnvVarsScript(envVarNames = [], loadEnvVarsScriptPath) {
116
+ let loadEnvVarsScript = "# Dynamically generated script to load env vars into build container";
117
+ for (const envVarName of envVarNames) {
118
+ loadEnvVarsScript += `\nexport ${envVarName}=${process.env[envVarName] || '""'}`;
119
+ }
120
+ (0, node_fs_1.writeFileSync)(loadEnvVarsScriptPath, loadEnvVarsScript);
121
+ }
122
+ getBuilderImageDigest() {
123
+ const digest = (0, node_child_process_1.execSync)(`${this.containerRuntime} images --no-trunc --quiet ${this.builderImageTag}`, { encoding: "utf-8" });
124
+ return digest.slice(0, -1); // remove trailing \n
125
+ }
126
+ getPublicDirEntries() {
127
+ const publicDirPath = (0, posix_1.join)("/app", this.props.relativePathToWorkspace || "", "public");
128
+ const publicDirEntriesString = (0, node_child_process_1.execSync)(`${this.containerRuntime} run ${this.builderImageTag} node -e "console.log(JSON.stringify(fs.readdirSync('${publicDirPath}', { withFileTypes: true }).map((e) => ({ name: e.name, isDir: e.isDirectory()}))))"`, { encoding: "utf-8" });
129
+ return JSON.parse(publicDirEntriesString);
130
+ }
131
+ createImageForNextjsContainers() {
132
+ const dockerfileNamePrefix = this.props.nextjsType === common_1.NextjsType.GLOBAL_CONTAINERS
133
+ ? "global"
134
+ : "regional";
135
+ const dockerfileName = `${dockerfileNamePrefix}-containers.Dockerfile`;
136
+ // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll
137
+ // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile
138
+ // and symlink-full-route-cache scripts which are in lib/nextjs-build folder.
139
+ const buildContext = (0, node_path_1.join)(__dirname, "..", "..", "lib", "nextjs-build");
140
+ const dockerImageAsset = new aws_ecr_assets_1.DockerImageAsset(this, "Image", {
141
+ buildArgs: {
142
+ BUILDER_IMAGE_TAG: this.builderImageTag,
143
+ DATA_CACHE_DIR: common_1.DATA_CACHE_DIR,
144
+ FULL_ROUTE_CACHE_DIR: common_1.FULL_ROUTE_CACHE_DIR,
145
+ IMAGE_CACHE_DIR: common_1.IMAGE_CACHE_DIR,
146
+ MOUNT_PATH: this.containerMountPathForEfs,
147
+ RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,
148
+ },
149
+ directory: buildContext,
150
+ exclude: [
151
+ "*",
152
+ `!${dockerfileName}`,
153
+ "!symlink-full-route-cache.mjs",
154
+ "!add-cache-handler.mjs",
155
+ ],
156
+ extraHash: this.buildImageDigest, // rebuild when builder hash changes
157
+ file: dockerfileName,
158
+ ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER,
159
+ ...this.props.overrides?.nextjsContainersDockerImageAssetProps,
160
+ });
161
+ return dockerImageAsset;
162
+ }
163
+ createImageForNextjsFunctions() {
164
+ const dockerfileName = "global-functions.Dockerfile";
165
+ // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll
166
+ // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile
167
+ // and symlink-full-route-cache scripts which are in lib/nextjs-build folder.
168
+ const buildContext = (0, node_path_1.join)(__dirname, "..", "..", "lib", "nextjs-build");
169
+ const dockerImageCode = aws_lambda_1.DockerImageCode.fromImageAsset(buildContext, {
170
+ buildArgs: {
171
+ BUILDER_IMAGE_TAG: this.builderImageTag,
172
+ DATA_CACHE_DIR: common_1.DATA_CACHE_DIR,
173
+ FULL_ROUTE_CACHE_DIR: common_1.FULL_ROUTE_CACHE_DIR,
174
+ IMAGE_CACHE_DIR: common_1.IMAGE_CACHE_DIR,
175
+ MOUNT_PATH: this.containerMountPathForEfs,
176
+ RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,
177
+ },
178
+ cmd: ["node", this.relativePathToEntrypoint],
179
+ exclude: [
180
+ "*",
181
+ `!${dockerfileName}`,
182
+ "!symlink-full-route-cache.mjs",
183
+ "!add-cache-handler.mjs",
184
+ "!cache-handler.cjs",
185
+ ],
186
+ extraHash: this.buildImageDigest, // rebuild when builder hash changes
187
+ file: dockerfileName,
188
+ ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER,
189
+ ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps,
190
+ });
191
+ // TODO: how to clean up temp dir?
192
+ // rmSync(tempDir, { recursive: true });
193
+ return dockerImageCode;
194
+ }
195
+ createImageForNextjsAssetsDeployment() {
196
+ const dockerfileName = "assets-deployment.Dockerfile";
197
+ /**
198
+ * Path to bundled custom resource code
199
+ */
200
+ const lambdaBuildContext = (0, node_path_1.join)(__dirname, "..", "..", "assets", "lambdas", "assets-deployment", "assets-deployment.lambda");
201
+ // cdk-nextjs/builder-{hash} already contains Next.js built code which
202
+ // we'll copy into final image. But we also need lambda code to run
203
+ // asset deployment tasks
204
+ const dockerImageCode = aws_lambda_1.DockerImageCode.fromImageAsset(lambdaBuildContext, {
205
+ buildArgs: {
206
+ BUILDER_IMAGE_TAG: this.builderImageTag,
207
+ RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,
208
+ },
209
+ extraHash: this.buildImageDigest, // rebuild when builder hash changes
210
+ file: dockerfileName,
211
+ ignoreMode: aws_cdk_lib_1.IgnoreMode.DOCKER,
212
+ ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps,
213
+ });
214
+ return dockerImageCode;
215
+ }
216
+ }
217
+ exports.NextjsBuild = NextjsBuild;
218
+ _a = JSII_RTTI_SYMBOL_1;
219
+ NextjsBuild[_a] = { fqn: "cdk-nextjs.NextjsBuild", version: "0.0.0" };
220
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-build.js","sourceRoot":"","sources":["../../src/nextjs-build/nextjs-build.ts"],"names":[],"mappings":";;;;;AAAA,2DAA8C;AAC9C,qCAAwD;AACxD,yCAAiC;AACjC,2CAAoD;AACpD,6CAAyC;AACzC,+DAAwE;AACxE,uDAA8E;AAC9E,2CAAuC;AACvC,sCAKmB;AAkFnB;;;GAGG;AACH,MAAa,WAAY,SAAQ,sBAAS;IAmDxC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAuB;QAC/D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QA7CnB;;;;;;WAMG;QACH,6BAAwB,GAAG,uBAAuB,CAAC;QAiC3C,qBAAgB,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,QAAQ,CAAC;QAM5D,IAAI,CAAC,eAAe,GAAG,sBAAsB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC;QACxE,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAC,uBAAuB,IAAI,GAAG,CAAC;QACpE,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,yBAAyB,EAAE,CAAC;QACjE,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAClD,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACnD,IACE,KAAK,CAAC,UAAU,KAAK,mBAAU,CAAC,iBAAiB;YACjD,KAAK,CAAC,UAAU,KAAK,mBAAU,CAAC,mBAAmB,EACnD,CAAC;YACD,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,8BAA8B,EAAE,CAAC;QACxE,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,6BAA6B,EAAE,CAAC;QACtE,CAAC;QACD,IAAI,CAAC,8BAA8B;YACjC,IAAI,CAAC,oCAAoC,EAAE,CAAC;IAChD,CAAC;IAEO,yBAAyB;QAC/B,qDAAqD;QACrD,OAAO,IAAA,YAAS,EAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,IAAI,EAAE,EAAE,WAAW,CAAC,CAAC;IAC1E,CAAC;IACD;;;;;OAKG;IACK,kBAAkB;QACxB,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,eAAe,CAAC;QAChE,MAAM,EACJ,SAAS,GAAG;YACV,aAAa,EAAE,YAAY;YAC3B,0BAA0B,EAAE,IAAI,CAAC,uBAAuB;YACxD,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,EAAE,SAAS;SAC3C,EACD,WAAW,GAAG,EAAE,EAChB,OAAO,GAAG;YACR,iBAAiB;YACjB,MAAM;YACN,YAAY;YACZ,UAAU;YACV,YAAY;YACZ,MAAM;SACP,EACD,IAAI,GAAG,oBAAoB,EAC3B,QAAQ,EACR,SAAS,GAAG,KAAK,GAClB,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,EAAE,CAAC;QACvC,MAAM,iBAAiB,GACrB,IAAI,CAAC,KAAK,CAAC,iBAAiB,EAAE,oBAAoB;YAClD,IAAA,gBAAI,EAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QACxB,MAAM,kBAAkB,GAAG,IAAA,gBAAI,EAAC,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;QAC/D,IAAA,gBAAM,EAAC,iBAAiB,EAAE,kBAAkB,CAAC,CAAC;QAC9C,MAAM,cAAc,GAAG,OAAO,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3C,MAAM,oBAAoB,GAAG,IAAA,gBAAI,EAAC,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,eAAe,CAAC,CAAC;QAC5E,IAAA,uBAAa,EAAC,oBAAoB,EAAE,cAAc,CAAC,CAAC;QACpD,MAAM,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC;QACvD,MAAM,qBAAqB,GAAG,IAAA,gBAAI,EAChC,IAAI,CAAC,KAAK,CAAC,YAAY,EACvB,6BAA6B,CAC9B,CAAC;QACF,IAAI,WAAW,CAAC,MAAM,EAAE,CAAC;YACvB,IAAI,CAAC,uBAAuB,CAAC,WAAW,EAAE,qBAAqB,CAAC,CAAC;QACnE,CAAC;QACD,MAAM,OAAO,GACX,IAAI,CAAC,KAAK,CAAC,iBAAiB,EAAE,OAAO;YACrC,GAAG,IAAI,CAAC,gBAAgB,UAAU,QAAQ,CAAC,CAAC,CAAC,cAAc,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,EAAE,WAAW,IAAI,UAAU,IAAI,CAAC,eAAe,IAAI,YAAY,IAAI,CAAC;QACvJ,IAAI,KAAc,CAAC;QACnB,IAAI,CAAC;YACH,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,OAAO,CAAC,GAAG,CACT,gCAAgC,OAAO,kBAAkB,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CACnF,CAAC;gBACF,IAAA,6BAAQ,EAAC,OAAO,EAAE;oBAChB,KAAK,EAAE,SAAS;oBAChB,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,YAAY;oBAC5B,GAAG,EAAE,OAAO,CAAC,GAAG;iBACjB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,KAAK,GAAG,GAAG,CAAC;QACd,CAAC;gBAAS,CAAC;YACT,IAAA,gBAAM,EAAC,kBAAkB,CAAC,CAAC;YAC3B,IAAA,gBAAM,EAAC,oBAAoB,CAAC,CAAC;YAC7B,IAAI,WAAW,CAAC,MAAM,EAAE,CAAC;gBACvB,IAAA,gBAAM,EAAC,qBAAqB,CAAC,CAAC;YAChC,CAAC;QACH,CAAC;QACD,IAAI,KAAK;YAAE,MAAM,KAAK,CAAC;QACvB,OAAO,IAAI,CAAC,qBAAqB,EAAE,CAAC;IACtC,CAAC;IACO,iBAAiB,CACvB,SAAmD;QAEnD,OAAO,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE;YAC5D,OAAO,GAAG,GAAG,gBAAgB,GAAG,KAAK,KAAK,GAAG,CAAC;QAChD,CAAC,EAAE,EAAE,CAAC,CAAC;IACT,CAAC;IACO,uBAAuB,CAC7B,cAA0D,EAAE,EAC5D,qBAA6B;QAE7B,IAAI,iBAAiB,GACnB,sEAAsE,CAAC;QACzE,KAAK,MAAM,UAAU,IAAI,WAAW,EAAE,CAAC;YACrC,iBAAiB,IAAI,YAAY,UAAU,IAAI,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,IAAI,EAAE,CAAC;QACnF,CAAC;QACD,IAAA,uBAAa,EAAC,qBAAqB,EAAE,iBAAiB,CAAC,CAAC;IAC1D,CAAC;IACO,qBAAqB;QAC3B,MAAM,MAAM,GAAG,IAAA,6BAAQ,EACrB,GAAG,IAAI,CAAC,gBAAgB,8BAA8B,IAAI,CAAC,eAAe,EAAE,EAC5E,EAAE,QAAQ,EAAE,OAAO,EAAE,CACtB,CAAC;QACF,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,qBAAqB;IACnD,CAAC;IACO,mBAAmB;QACzB,MAAM,aAAa,GAAG,IAAA,YAAS,EAC7B,MAAM,EACN,IAAI,CAAC,KAAK,CAAC,uBAAuB,IAAI,EAAE,EACxC,QAAQ,CACT,CAAC;QACF,MAAM,sBAAsB,GAAG,IAAA,6BAAQ,EACrC,GAAG,IAAI,CAAC,gBAAgB,QAAQ,IAAI,CAAC,eAAe,wDAAwD,aAAa,sFAAsF,EAC/M,EAAE,QAAQ,EAAE,OAAO,EAAE,CACtB,CAAC;QACF,OAAO,IAAI,CAAC,KAAK,CAAC,sBAAsB,CAAqB,CAAC;IAChE,CAAC;IACO,8BAA8B;QACpC,MAAM,oBAAoB,GACxB,IAAI,CAAC,KAAK,CAAC,UAAU,KAAK,mBAAU,CAAC,iBAAiB;YACpD,CAAC,CAAC,QAAQ;YACV,CAAC,CAAC,UAAU,CAAC;QACjB,MAAM,cAAc,GAAG,GAAG,oBAAoB,wBAAwB,CAAC;QACvE,0EAA0E;QAC1E,yEAAyE;QACzE,6EAA6E;QAC7E,MAAM,YAAY,GAAG,IAAA,gBAAI,EAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC;QACxE,MAAM,gBAAgB,GAAG,IAAI,iCAAgB,CAAC,IAAI,EAAE,OAAO,EAAE;YAC3D,SAAS,EAAE;gBACT,iBAAiB,EAAE,IAAI,CAAC,eAAe;gBACvC,cAAc,EAAd,uBAAc;gBACd,oBAAoB,EAApB,6BAAoB;gBACpB,eAAe,EAAf,wBAAe;gBACf,UAAU,EAAE,IAAI,CAAC,wBAAwB;gBACzC,0BAA0B,EAAE,IAAI,CAAC,uBAAuB;aACzD;YACD,SAAS,EAAE,YAAY;YACvB,OAAO,EAAE;gBACP,GAAG;gBACH,IAAI,cAAc,EAAE;gBACpB,+BAA+B;gBAC/B,wBAAwB;aACzB;YACD,SAAS,EAAE,IAAI,CAAC,gBAAgB,EAAE,oCAAoC;YACtE,IAAI,EAAE,cAAc;YACpB,UAAU,EAAE,wBAAU,CAAC,MAAM;YAC7B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,qCAAqC;SAC/D,CAAC,CAAC;QACH,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAEO,6BAA6B;QACnC,MAAM,cAAc,GAAG,6BAA6B,CAAC;QACrD,0EAA0E;QAC1E,yEAAyE;QACzE,6EAA6E;QAC7E,MAAM,YAAY,GAAG,IAAA,gBAAI,EAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC;QACxE,MAAM,eAAe,GAAG,4BAAe,CAAC,cAAc,CAAC,YAAY,EAAE;YACnE,SAAS,EAAE;gBACT,iBAAiB,EAAE,IAAI,CAAC,eAAe;gBACvC,cAAc,EAAd,uBAAc;gBACd,oBAAoB,EAApB,6BAAoB;gBACpB,eAAe,EAAf,wBAAe;gBACf,UAAU,EAAE,IAAI,CAAC,wBAAwB;gBACzC,0BAA0B,EAAE,IAAI,CAAC,uBAAuB;aACzD;YACD,GAAG,EAAE,CAAC,MAAM,EAAE,IAAI,CAAC,wBAAwB,CAAC;YAC5C,OAAO,EAAE;gBACP,GAAG;gBACH,IAAI,cAAc,EAAE;gBACpB,+BAA+B;gBAC/B,wBAAwB;gBACxB,oBAAoB;aACrB;YACD,SAAS,EAAE,IAAI,CAAC,gBAAgB,EAAE,oCAAoC;YACtE,IAAI,EAAE,cAAc;YACpB,UAAU,EAAE,wBAAU,CAAC,MAAM;YAC7B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,kCAAkC;SAC5D,CAAC,CAAC;QACH,kCAAkC;QAClC,wCAAwC;QACxC,OAAO,eAAe,CAAC;IACzB,CAAC;IAEO,oCAAoC;QAC1C,MAAM,cAAc,GAAG,8BAA8B,CAAC;QACtD;;WAEG;QACH,MAAM,kBAAkB,GAAG,IAAA,gBAAI,EAC7B,SAAS,EACT,IAAI,EACJ,IAAI,EACJ,QAAQ,EACR,SAAS,EACT,mBAAmB,EACnB,0BAA0B,CAC3B,CAAC;QACF,sEAAsE;QACtE,mEAAmE;QACnE,yBAAyB;QACzB,MAAM,eAAe,GAAG,4BAAe,CAAC,cAAc,CAAC,kBAAkB,EAAE;YACzE,SAAS,EAAE;gBACT,iBAAiB,EAAE,IAAI,CAAC,eAAe;gBACvC,0BAA0B,EAAE,IAAI,CAAC,uBAAuB;aACzD;YACD,SAAS,EAAE,IAAI,CAAC,gBAAgB,EAAE,oCAAoC;YACtE,IAAI,EAAE,cAAc;YACpB,UAAU,EAAE,wBAAU,CAAC,MAAM;YAC7B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,wCAAwC;SAClE,CAAC,CAAC;QACH,OAAO,eAAe,CAAC;IACzB,CAAC;;AArRH,kCAsRC","sourcesContent":["import { execSync } from \"node:child_process\";\nimport { cpSync, rmSync, writeFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { join as joinPosix } from \"node:path/posix\";\nimport { IgnoreMode } from \"aws-cdk-lib\";\nimport { DockerImageAsset, Platform } from \"aws-cdk-lib/aws-ecr-assets\";\nimport { AssetImageCodeProps, DockerImageCode } from \"aws-cdk-lib/aws-lambda\";\nimport { Construct } from \"constructs\";\nimport {\n  DATA_CACHE_DIR,\n  FULL_ROUTE_CACHE_DIR,\n  IMAGE_CACHE_DIR,\n  NextjsType,\n} from \"../common\";\nimport { OptionalDockerImageAssetProps } from \"../generated-structs/OptionalDockerImageAssetProps\";\nimport { NextjsBaseProps } from \"../root-constructs/nextjs-base-props\";\n\nexport interface BuilderImageProps {\n  /**\n   * Build Args to be passed to `docker build` command.\n   */\n  readonly buildArgs?: Record<string, string>;\n  /**\n   * `docker build ...` command to run in {@link NextBaseProps.buildContext}.\n   * Default interpolates other props. If you override, other props will have\n   * no effect on command.\n   */\n  readonly command?: string;\n  /**\n   * Environment variables names to pass from host to container during build process.\n   *\n   * Note, a shell script, cdk-nextjs-load-env-vars.sh is created within the\n   * {@link NextBaseProps.buildContext} directory, which will contain all the\n   * environment variables defined in this prop. If you've created your own\n   * custom Dockerfile (passed in via {@link BuilderImageProps.customDockerfilePath})\n   * then you need to make sure you're copying it into the image.\n   *\n   * @example [\"MY_API_KEY\"]\n   */\n  readonly envVarNames?: string[];\n  /**\n   * Lines in .dockerignore file which will be created in your {@link NextBaseProps.buildContext}\n   * @default [\"node_modules\", \".git\", \".gitignore\", \".md\"]\n   */\n  readonly exclude?: string[];\n  /**\n   * Name of Dockerfile\n   * @default \"builder.Dockerfile\"\n   */\n  readonly file?: string;\n  readonly platform?: Platform;\n  /**\n   * Skip building the builder image.\n   * @default false\n   */\n  readonly skipBuild?: boolean;\n  /**\n   * Path to your custom builder.Dockerfile which will be copied into {@link NextBaseProps.buildContext}.\n   * It is recommended to override this prop to optimize build caching for your setup.\n   */\n  readonly customDockerfilePath?: string;\n}\n\nexport interface NextjsBuildOverrides {\n  readonly nextjsContainersDockerImageAssetProps?: OptionalDockerImageAssetProps;\n  readonly nextjsFunctionsAssetImageCodeProps?: AssetImageCodeProps;\n  readonly nextjsAssetDeploymentAssetImageCodeProps?: AssetImageCodeProps;\n}\n\nexport interface NextjsBuildProps {\n  /**\n   * @see {@link NextjsBaseProps[\"buildCommand\"]}\n   */\n  readonly buildCommand: NextjsBaseProps[\"buildCommand\"];\n  /**\n   * @see {@link NextjsBaseProps[\"buildContext\"]}\n   */\n  readonly buildContext: NextjsBaseProps[\"buildContext\"];\n  /**\n   *\n   */\n  readonly builderImageProps?: BuilderImageProps;\n  /**\n   * @see {@link NextjsBaseProps.relativePathToWorkspace}\n   */\n  readonly relativePathToWorkspace?: NextjsBaseProps[\"relativePathToWorkspace\"];\n  readonly nextjsType: NextjsType;\n  readonly overrides?: NextjsBuildOverrides;\n}\n\nexport interface PublicDirEntry {\n  readonly name: string;\n  readonly isDirectory: boolean;\n}\n\n/**\n * Builds Next.js assets.\n * @link https://nextjs.org/docs/pages/api-reference/next-config-js/output\n */\nexport class NextjsBuild extends Construct {\n  /**\n   * Hash of builder image which will change whenever the image changes. Useful\n   * for passing to properties of custom resources that depend upon the builder\n   * image to re-run when build image changes.\n   */\n  buildImageDigest: string;\n  /**\n   * Mount path in container for EFS. Next.js image optimization, data, and full\n   * route cache will be symlinked to this location.\n   *\n   * Must comply with pattern: ^/mnt/[a-zA-Z0-9-_.]+$\n   * @see https://docs.aws.amazon.com/lambda/latest/api/API_FileSystemConfig.html\n   */\n  containerMountPathForEfs = \"/mnt/cdk-nextjs-cache\";\n  /**\n   * Docker image built if using Fargate.\n   */\n  imageForNextjsContainers?: DockerImageAsset;\n  /**\n   * Docker image built if using Lambda.\n   */\n  imageForNextjsFunctions?: DockerImageCode;\n  /**\n   * Docker image built for `NextjsAssetsDeployment`\n   */\n  imageForNextjsAssetsDeployment: DockerImageCode;\n  /**\n   * Absolute path to public. Use by CloudFront/ALB to create behaviors/rules\n   * @example \"/Users/john/myapp/public\"\n   */\n  publicDirEntries: PublicDirEntry[];\n  /**\n   * The entrypoint JavaScript file used as an argument for Node.js to run the\n   * Next.js standalone server relative to the standalone directory.\n   * @example \"./server.js\"\n   * @example \"./packages/ui/server.js\" (monorepo)\n   */\n  relativePathToEntrypoint: string;\n\n  /**\n   * Tag of builder image Next.js app which is built for other images to be\n   * built `FROM`. This image isn't built with CDK Assets construct b/c it\n   * doesn't need to be uploaded to ECR. We still need to include slice of\n   * `node.addr` in tag in case multiple cdk-nextjs constructs are used.\n   */\n  private builderImageTag: string;\n  private containerRuntime = process.env.CDK_DOCKER || \"docker\";\n  private props: NextjsBuildProps;\n  private relativePathToWorkspace: string;\n\n  constructor(scope: Construct, id: string, props: NextjsBuildProps) {\n    super(scope, id);\n    this.builderImageTag = `cdk-nextjs/builder-${this.node.addr.slice(30)}`;\n    this.relativePathToWorkspace = props.relativePathToWorkspace || \".\";\n    this.props = props;\n    this.relativePathToEntrypoint = this.getRelativeEntrypointPath();\n    this.buildImageDigest = this.createBuilderImage();\n    this.publicDirEntries = this.getPublicDirEntries();\n    if (\n      props.nextjsType === NextjsType.GLOBAL_CONTAINERS ||\n      props.nextjsType === NextjsType.REGIONAL_CONTAINERS\n    ) {\n      this.imageForNextjsContainers = this.createImageForNextjsContainers();\n    } else {\n      this.imageForNextjsFunctions = this.createImageForNextjsFunctions();\n    }\n    this.imageForNextjsAssetsDeployment =\n      this.createImageForNextjsAssetsDeployment();\n  }\n\n  private getRelativeEntrypointPath() {\n    // joinPosix b/c this will be used in linux container\n    return joinPosix(this.props.relativePathToWorkspace || \"\", \"server.js\");\n  }\n  /**\n   * A builder or base image needs to be created so that the same image can be\n   * built `FROM` for `NextjsFunctions` or `NextjsContainers` and `NextjsAssetsDeployment`.\n   * This image doesn't need to be uploaded to ECR so we're \"manually\" creating\n   * it with `execSync` and other images will be built `FROM` it.\n   */\n  private createBuilderImage() {\n    const buildCommand = this.props.buildCommand || \"npm run build\";\n    const {\n      buildArgs = {\n        BUILD_COMMAND: buildCommand,\n        RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,\n        ...this.props.builderImageProps?.buildArgs,\n      },\n      envVarNames = [],\n      exclude = [\n        \"**/node_modules\",\n        \".git\",\n        \"**/cdk.out\",\n        \"**/.next\",\n        \".gitignore\",\n        \"*.md\",\n      ],\n      file = \"builder.Dockerfile\",\n      platform,\n      skipBuild = false,\n    } = this.props.builderImageProps || {};\n    const srcDockerfilePath =\n      this.props.builderImageProps?.customDockerfilePath ||\n      join(__dirname, file);\n    const destDockerfilePath = join(this.props.buildContext, file);\n    cpSync(srcDockerfilePath, destDockerfilePath);\n    const excludeFileStr = exclude?.join(\"\\n\");\n    const dockerignoreFilePath = join(this.props.buildContext, \".dockerignore\");\n    writeFileSync(dockerignoreFilePath, excludeFileStr);\n    const buildArgsStr = this.createBuildArgStr(buildArgs);\n    const loadEnvVarsScriptPath = join(\n      this.props.buildContext,\n      \"cdk-nextjs-load-env-vars.sh\",\n    );\n    if (envVarNames.length) {\n      this.createLoadEnvVarsScript(envVarNames, loadEnvVarsScriptPath);\n    }\n    const command =\n      this.props.builderImageProps?.command ||\n      `${this.containerRuntime} build ${platform ? `--platform ${platform.platform}` : \"\"} --file ${file} --tag ${this.builderImageTag} ${buildArgsStr} .`;\n    let error: unknown;\n    try {\n      if (!skipBuild) {\n        console.log(\n          `Building image with command: ${command} in directory: ${this.props.buildContext}`,\n        );\n        execSync(command, {\n          stdio: \"inherit\",\n          cwd: this.props.buildContext,\n          env: process.env,\n        });\n      }\n    } catch (err) {\n      error = err;\n    } finally {\n      rmSync(destDockerfilePath);\n      rmSync(dockerignoreFilePath);\n      if (envVarNames.length) {\n        rmSync(loadEnvVarsScriptPath);\n      }\n    }\n    if (error) throw error;\n    return this.getBuilderImageDigest();\n  }\n  private createBuildArgStr(\n    buildArgs: Required<BuilderImageProps>[\"buildArgs\"],\n  ) {\n    return Object.entries(buildArgs).reduce((acc, [key, value]) => {\n      return `${acc} --build-arg ${key}=\"${value}\"`;\n    }, \"\");\n  }\n  private createLoadEnvVarsScript(\n    envVarNames: Required<BuilderImageProps>[\"envVarNames\"] = [],\n    loadEnvVarsScriptPath: string,\n  ) {\n    let loadEnvVarsScript =\n      \"# Dynamically generated script to load env vars into build container\";\n    for (const envVarName of envVarNames) {\n      loadEnvVarsScript += `\\nexport ${envVarName}=${process.env[envVarName] || '\"\"'}`;\n    }\n    writeFileSync(loadEnvVarsScriptPath, loadEnvVarsScript);\n  }\n  private getBuilderImageDigest() {\n    const digest = execSync(\n      `${this.containerRuntime} images --no-trunc --quiet ${this.builderImageTag}`,\n      { encoding: \"utf-8\" },\n    );\n    return digest.slice(0, -1); // remove trailing \\n\n  }\n  private getPublicDirEntries(): PublicDirEntry[] {\n    const publicDirPath = joinPosix(\n      \"/app\",\n      this.props.relativePathToWorkspace || \"\",\n      \"public\",\n    );\n    const publicDirEntriesString = execSync(\n      `${this.containerRuntime} run ${this.builderImageTag} node -e \"console.log(JSON.stringify(fs.readdirSync('${publicDirPath}', { withFileTypes: true }).map((e) => ({ name: e.name, isDir: e.isDirectory()}))))\"`,\n      { encoding: \"utf-8\" },\n    );\n    return JSON.parse(publicDirEntriesString) as PublicDirEntry[];\n  }\n  private createImageForNextjsContainers() {\n    const dockerfileNamePrefix =\n      this.props.nextjsType === NextjsType.GLOBAL_CONTAINERS\n        ? \"global\"\n        : \"regional\";\n    const dockerfileName = `${dockerfileNamePrefix}-containers.Dockerfile`;\n    // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll\n    // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile\n    // and symlink-full-route-cache scripts which are in lib/nextjs-build folder.\n    const buildContext = join(__dirname, \"..\", \"..\", \"lib\", \"nextjs-build\");\n    const dockerImageAsset = new DockerImageAsset(this, \"Image\", {\n      buildArgs: {\n        BUILDER_IMAGE_TAG: this.builderImageTag,\n        DATA_CACHE_DIR,\n        FULL_ROUTE_CACHE_DIR,\n        IMAGE_CACHE_DIR,\n        MOUNT_PATH: this.containerMountPathForEfs,\n        RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,\n      },\n      directory: buildContext,\n      exclude: [\n        \"*\",\n        `!${dockerfileName}`,\n        \"!symlink-full-route-cache.mjs\",\n        \"!add-cache-handler.mjs\",\n      ],\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsContainersDockerImageAssetProps,\n    });\n    return dockerImageAsset;\n  }\n\n  private createImageForNextjsFunctions() {\n    const dockerfileName = \"global-functions.Dockerfile\";\n    // cdk-nextjs/builder-{hash} already contains built nextjs app which we'll\n    // `COPY --from=cdk-nextjs/builder-{hash}` so we just need the Dockerfile\n    // and symlink-full-route-cache scripts which are in lib/nextjs-build folder.\n    const buildContext = join(__dirname, \"..\", \"..\", \"lib\", \"nextjs-build\");\n    const dockerImageCode = DockerImageCode.fromImageAsset(buildContext, {\n      buildArgs: {\n        BUILDER_IMAGE_TAG: this.builderImageTag,\n        DATA_CACHE_DIR,\n        FULL_ROUTE_CACHE_DIR,\n        IMAGE_CACHE_DIR,\n        MOUNT_PATH: this.containerMountPathForEfs,\n        RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,\n      },\n      cmd: [\"node\", this.relativePathToEntrypoint],\n      exclude: [\n        \"*\",\n        `!${dockerfileName}`,\n        \"!symlink-full-route-cache.mjs\",\n        \"!add-cache-handler.mjs\",\n        \"!cache-handler.cjs\",\n      ],\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsFunctionsAssetImageCodeProps,\n    });\n    // TODO: how to clean up temp dir?\n    // rmSync(tempDir, { recursive: true });\n    return dockerImageCode;\n  }\n\n  private createImageForNextjsAssetsDeployment() {\n    const dockerfileName = \"assets-deployment.Dockerfile\";\n    /**\n     * Path to bundled custom resource code\n     */\n    const lambdaBuildContext = join(\n      __dirname,\n      \"..\",\n      \"..\",\n      \"assets\",\n      \"lambdas\",\n      \"assets-deployment\",\n      \"assets-deployment.lambda\",\n    );\n    // cdk-nextjs/builder-{hash} already contains Next.js built code which\n    // we'll copy into final image. But we also need lambda code to run\n    // asset deployment tasks\n    const dockerImageCode = DockerImageCode.fromImageAsset(lambdaBuildContext, {\n      buildArgs: {\n        BUILDER_IMAGE_TAG: this.builderImageTag,\n        RELATIVE_PATH_TO_WORKSPACE: this.relativePathToWorkspace,\n      },\n      extraHash: this.buildImageDigest, // rebuild when builder hash changes\n      file: dockerfileName,\n      ignoreMode: IgnoreMode.DOCKER,\n      ...this.props.overrides?.nextjsAssetDeploymentAssetImageCodeProps,\n    });\n    return dockerImageCode;\n  }\n}\n"]}
@@ -0,0 +1,45 @@
1
+ #checkov:skip=CKV_DOCKER_2: healthcheck run by ALB and ECS
2
+ #checkov:skip=CKV_DOCKER_7: latest tag is ok to use for local builder container
3
+ # Modified from: https://github.com/vercel/next.js/blob/canary/examples/with-docker/Dockerfile
4
+ ARG BUILDER_IMAGE_TAG
5
+ FROM $BUILDER_IMAGE_TAG as builder
6
+ # Production image, copy all the files and run next
7
+ FROM public.ecr.aws/docker/library/node:20-alpine as runner
8
+ WORKDIR /app
9
+
10
+ ENV NODE_ENV production
11
+ # Uncomment the following line in case you want to disable telemetry during runtime.
12
+ # ENV NEXT_TELEMETRY_DISABLED 1
13
+
14
+ RUN addgroup --system --gid 1001 nodejs
15
+ RUN adduser --system --uid 1001 nextjs
16
+
17
+ ARG RELATIVE_PATH_TO_WORKSPACE
18
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/standalone ./
19
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/static ./$RELATIVE_PATH_TO_WORKSPACE/.next/static
20
+ COPY --from=builder --chown=nextjs:nodejs /app/$RELATIVE_PATH_TO_WORKSPACE/public ./$RELATIVE_PATH_TO_WORKSPACE/public
21
+ COPY --chown=nextjs:nodejs ./symlink-full-route-cache.mjs ./
22
+ ARG MOUNT_PATH
23
+ ARG DATA_CACHE_DIR
24
+ ARG FULL_ROUTE_CACHE_DIR
25
+ ARG IMAGE_CACHE_DIR
26
+ RUN mkdir -p $MOUNT_PATH/$DATA_CACHE_DIR && \
27
+ mkdir -p $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
28
+ mkdir -p $MOUNT_PATH/$IMAGE_CACHE_DIR && \
29
+ chmod -R u+rw $MOUNT_PATH && \
30
+ mkdir -p ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache && \
31
+ ln -s $MOUNT_PATH/$DATA_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/fetch-cache && \
32
+ ln -s $MOUNT_PATH/$IMAGE_CACHE_DIR ./$RELATIVE_PATH_TO_WORKSPACE/.next/cache/images && \
33
+ node symlink-full-route-cache.mjs /app/$RELATIVE_PATH_TO_WORKSPACE/.next/server/app $MOUNT_PATH/$FULL_ROUTE_CACHE_DIR && \
34
+ rm symlink-full-route-cache.mjs
35
+
36
+ USER nextjs
37
+
38
+ EXPOSE 3000
39
+
40
+ ENV PORT 3000
41
+
42
+ # server.js is created by next build from the standalone output
43
+ # https://nextjs.org/docs/pages/api-reference/next-config-js/output
44
+ # CMD will be overwritten by ECS Task Definition
45
+ CMD HOSTNAME="0.0.0.0" node server.js
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,37 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const node_fs_1 = require("node:fs");
4
+ const node_path_1 = require("node:path");
5
+ /**
6
+ * Creates symbolic links recursively from `srcPath` to `destPath`.
7
+ *
8
+ * Used to symlink .next/server/app -> /mnt/cdk-nextjs-cache/full-route-cache.
9
+ * See docs/dev.md for details.
10
+ */
11
+ function createSymlinks(srcPath, destPath) {
12
+ // Create the destination directory if it doesn't exist
13
+ (0, node_fs_1.mkdirSync)(destPath, { recursive: true });
14
+ const entries = (0, node_fs_1.readdirSync)(srcPath, { withFileTypes: true });
15
+ for (const entry of entries) {
16
+ const entryPath = (0, node_path_1.join)(srcPath, entry.name);
17
+ if (entry.isDirectory()) {
18
+ // If the entry is a directory, recurse into it
19
+ createSymlinks(entryPath, (0, node_path_1.join)(destPath, entry.name));
20
+ }
21
+ else if (entry.isFile()) {
22
+ const ext = (0, node_path_1.extname)(entryPath).slice(1);
23
+ // Check if the file extension is html, rsc, or meta
24
+ if (["html", "rsc", "meta"].includes(ext)) {
25
+ // Create a symbolic link for html, rsc, or meta files
26
+ const symlink = entryPath;
27
+ const target = (0, node_path_1.join)(destPath, entry.name);
28
+ // console.log(`linking ${symlink} -> ${target}`);
29
+ (0, node_fs_1.rmSync)(symlink); // symlink path must not exist in order to create
30
+ (0, node_fs_1.symlinkSync)(target, symlink, "file");
31
+ }
32
+ }
33
+ }
34
+ }
35
+ const [sourcePath, destPath] = process.argv.slice(2);
36
+ createSymlinks(sourcePath, destPath);
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3ltbGluay1mdWxsLXJvdXRlLWNhY2hlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL25leHRqcy1idWlsZC9zeW1saW5rLWZ1bGwtcm91dGUtY2FjaGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQSxxQ0FBc0U7QUFDdEUseUNBQTBDO0FBRTFDOzs7OztHQUtHO0FBQ0gsU0FBUyxjQUFjLENBQUMsT0FBZSxFQUFFLFFBQWdCO0lBQ3ZELHVEQUF1RDtJQUN2RCxJQUFBLG1CQUFTLEVBQUMsUUFBUSxFQUFFLEVBQUUsU0FBUyxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7SUFFekMsTUFBTSxPQUFPLEdBQUcsSUFBQSxxQkFBVyxFQUFDLE9BQU8sRUFBRSxFQUFFLGFBQWEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQzlELEtBQUssTUFBTSxLQUFLLElBQUksT0FBTyxFQUFFLENBQUM7UUFDNUIsTUFBTSxTQUFTLEdBQUcsSUFBQSxnQkFBSSxFQUFDLE9BQU8sRUFBRSxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDNUMsSUFBSSxLQUFLLENBQUMsV0FBVyxFQUFFLEVBQUUsQ0FBQztZQUN4QiwrQ0FBK0M7WUFDL0MsY0FBYyxDQUFDLFNBQVMsRUFBRSxJQUFBLGdCQUFJLEVBQUMsUUFBUSxFQUFFLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1FBQ3hELENBQUM7YUFBTSxJQUFJLEtBQUssQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDO1lBQzFCLE1BQU0sR0FBRyxHQUFHLElBQUEsbUJBQU8sRUFBQyxTQUFTLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFFeEMsb0RBQW9EO1lBQ3BELElBQUksQ0FBQyxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sQ0FBQyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDO2dCQUMxQyxzREFBc0Q7Z0JBQ3RELE1BQU0sT0FBTyxHQUFHLFNBQVMsQ0FBQztnQkFDMUIsTUFBTSxNQUFNLEdBQUcsSUFBQSxnQkFBSSxFQUFDLFFBQVEsRUFBRSxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7Z0JBQzFDLGtEQUFrRDtnQkFDbEQsSUFBQSxnQkFBTSxFQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsaURBQWlEO2dCQUNsRSxJQUFBLHFCQUFXLEVBQUMsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsQ0FBQztZQUN2QyxDQUFDO1FBQ0gsQ0FBQztJQUNILENBQUM7QUFDSCxDQUFDO0FBRUQsTUFBTSxDQUFDLFVBQVUsRUFBRSxRQUFRLENBQUMsR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNyRCxjQUFjLENBQUMsVUFBVSxFQUFFLFFBQVEsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgbWtkaXJTeW5jLCByZWFkZGlyU3luYywgcm1TeW5jLCBzeW1saW5rU3luYyB9IGZyb20gXCJub2RlOmZzXCI7XG5pbXBvcnQgeyBleHRuYW1lLCBqb2luIH0gZnJvbSBcIm5vZGU6cGF0aFwiO1xuXG4vKipcbiAqIENyZWF0ZXMgc3ltYm9saWMgbGlua3MgcmVjdXJzaXZlbHkgZnJvbSBgc3JjUGF0aGAgdG8gYGRlc3RQYXRoYC5cbiAqXG4gKiBVc2VkIHRvIHN5bWxpbmsgLm5leHQvc2VydmVyL2FwcCAtPiAvbW50L2Nkay1uZXh0anMtY2FjaGUvZnVsbC1yb3V0ZS1jYWNoZS5cbiAqIFNlZSBkb2NzL2Rldi5tZCBmb3IgZGV0YWlscy5cbiAqL1xuZnVuY3Rpb24gY3JlYXRlU3ltbGlua3Moc3JjUGF0aDogc3RyaW5nLCBkZXN0UGF0aDogc3RyaW5nKTogdm9pZCB7XG4gIC8vIENyZWF0ZSB0aGUgZGVzdGluYXRpb24gZGlyZWN0b3J5IGlmIGl0IGRvZXNuJ3QgZXhpc3RcbiAgbWtkaXJTeW5jKGRlc3RQYXRoLCB7IHJlY3Vyc2l2ZTogdHJ1ZSB9KTtcblxuICBjb25zdCBlbnRyaWVzID0gcmVhZGRpclN5bmMoc3JjUGF0aCwgeyB3aXRoRmlsZVR5cGVzOiB0cnVlIH0pO1xuICBmb3IgKGNvbnN0IGVudHJ5IG9mIGVudHJpZXMpIHtcbiAgICBjb25zdCBlbnRyeVBhdGggPSBqb2luKHNyY1BhdGgsIGVudHJ5Lm5hbWUpO1xuICAgIGlmIChlbnRyeS5pc0RpcmVjdG9yeSgpKSB7XG4gICAgICAvLyBJZiB0aGUgZW50cnkgaXMgYSBkaXJlY3RvcnksIHJlY3Vyc2UgaW50byBpdFxuICAgICAgY3JlYXRlU3ltbGlua3MoZW50cnlQYXRoLCBqb2luKGRlc3RQYXRoLCBlbnRyeS5uYW1lKSk7XG4gICAgfSBlbHNlIGlmIChlbnRyeS5pc0ZpbGUoKSkge1xuICAgICAgY29uc3QgZXh0ID0gZXh0bmFtZShlbnRyeVBhdGgpLnNsaWNlKDEpO1xuXG4gICAgICAvLyBDaGVjayBpZiB0aGUgZmlsZSBleHRlbnNpb24gaXMgaHRtbCwgcnNjLCBvciBtZXRhXG4gICAgICBpZiAoW1wiaHRtbFwiLCBcInJzY1wiLCBcIm1ldGFcIl0uaW5jbHVkZXMoZXh0KSkge1xuICAgICAgICAvLyBDcmVhdGUgYSBzeW1ib2xpYyBsaW5rIGZvciBodG1sLCByc2MsIG9yIG1ldGEgZmlsZXNcbiAgICAgICAgY29uc3Qgc3ltbGluayA9IGVudHJ5UGF0aDtcbiAgICAgICAgY29uc3QgdGFyZ2V0ID0gam9pbihkZXN0UGF0aCwgZW50cnkubmFtZSk7XG4gICAgICAgIC8vIGNvbnNvbGUubG9nKGBsaW5raW5nICR7c3ltbGlua30gLT4gJHt0YXJnZXR9YCk7XG4gICAgICAgIHJtU3luYyhzeW1saW5rKTsgLy8gc3ltbGluayBwYXRoIG11c3Qgbm90IGV4aXN0IGluIG9yZGVyIHRvIGNyZWF0ZVxuICAgICAgICBzeW1saW5rU3luYyh0YXJnZXQsIHN5bWxpbmssIFwiZmlsZVwiKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cblxuY29uc3QgW3NvdXJjZVBhdGgsIGRlc3RQYXRoXSA9IHByb2Nlc3MuYXJndi5zbGljZSgyKTtcbmNyZWF0ZVN5bWxpbmtzKHNvdXJjZVBhdGgsIGRlc3RQYXRoKTtcbiJdfQ==