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,23 @@
1
+ // src/nextjs-build/symlink-full-route-cache.ts
2
+ import { mkdirSync, readdirSync, rmSync, symlinkSync } from "node:fs";
3
+ import { extname, join } from "node:path";
4
+ function createSymlinks(srcPath, destPath2) {
5
+ mkdirSync(destPath2, { recursive: true });
6
+ const entries = readdirSync(srcPath, { withFileTypes: true });
7
+ for (const entry of entries) {
8
+ const entryPath = join(srcPath, entry.name);
9
+ if (entry.isDirectory()) {
10
+ createSymlinks(entryPath, join(destPath2, entry.name));
11
+ } else if (entry.isFile()) {
12
+ const ext = extname(entryPath).slice(1);
13
+ if (["html", "rsc", "meta"].includes(ext)) {
14
+ const symlink = entryPath;
15
+ const target = join(destPath2, entry.name);
16
+ rmSync(symlink);
17
+ symlinkSync(target, symlink, "file");
18
+ }
19
+ }
20
+ }
21
+ }
22
+ var [sourcePath, destPath] = process.argv.slice(2);
23
+ createSymlinks(sourcePath, destPath);
@@ -0,0 +1,8 @@
1
+ import { IVpc } from "aws-cdk-lib/aws-ec2";
2
+ import { AccessPoint } from "aws-cdk-lib/aws-efs";
3
+ export interface NextjsComputeBaseProps {
4
+ readonly accessPoint: AccessPoint;
5
+ readonly containerMountPathForEfs: string;
6
+ readonly healthCheckPath: string;
7
+ readonly vpc: IVpc;
8
+ }
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV4dGpzLWNvbXB1dGUtYmFzZS1wcm9wcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uZXh0anMtY29tcHV0ZS9uZXh0anMtY29tcHV0ZS1iYXNlLXByb3BzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJVnBjIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzJcIjtcbmltcG9ydCB7IEFjY2Vzc1BvaW50IH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1lZnNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBOZXh0anNDb21wdXRlQmFzZVByb3BzIHtcbiAgcmVhZG9ubHkgYWNjZXNzUG9pbnQ6IEFjY2Vzc1BvaW50O1xuICByZWFkb25seSBjb250YWluZXJNb3VudFBhdGhGb3JFZnM6IHN0cmluZztcbiAgcmVhZG9ubHkgaGVhbHRoQ2hlY2tQYXRoOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHZwYzogSVZwYztcbn1cbiJdfQ==
@@ -0,0 +1,43 @@
1
+ import { DockerImageAsset } from "aws-cdk-lib/aws-ecr-assets";
2
+ import { Cluster } from "aws-cdk-lib/aws-ecs";
3
+ import { ApplicationLoadBalancedFargateService, ApplicationLoadBalancedFargateServiceProps } from "aws-cdk-lib/aws-ecs-patterns";
4
+ import { FileSystem } from "aws-cdk-lib/aws-efs";
5
+ import { Construct } from "constructs";
6
+ import { NextjsComputeBaseProps } from "./nextjs-compute-base-props";
7
+ import { NextjsType } from "../common";
8
+ import { OptionalApplicationLoadBalancedTaskImageOptions } from "../generated-structs/OptionalApplicationLoadBalancedTaskImageOptions";
9
+ import { OptionalClusterProps } from "../generated-structs/OptionalClusterProps";
10
+ export interface NextjsContainersOverrides {
11
+ readonly ecsClusterProps?: OptionalClusterProps;
12
+ readonly albFargateServiceProps?: ApplicationLoadBalancedFargateServiceProps;
13
+ readonly taskImageOptions?: OptionalApplicationLoadBalancedTaskImageOptions;
14
+ }
15
+ export interface NextjsContainersProps extends NextjsComputeBaseProps {
16
+ readonly dockerImageAsset: DockerImageAsset;
17
+ readonly fileSystem: FileSystem;
18
+ readonly nextjsType: NextjsType;
19
+ readonly overrides?: NextjsContainersOverrides;
20
+ readonly relativeEntrypointPath: string;
21
+ }
22
+ /**
23
+ * Next.js load balanced via Application Load Balancer with containers via AWS
24
+ * Fargate.
25
+ */
26
+ export declare class NextjsContainers extends Construct {
27
+ albFargateService: ApplicationLoadBalancedFargateService;
28
+ ecsCluster: Cluster;
29
+ url: string;
30
+ private props;
31
+ constructor(scope: Construct, id: string, props: NextjsContainersProps);
32
+ private createEcsCluster;
33
+ private createAlbFargateSevice;
34
+ /**
35
+ * Configure health checks for containers at ALB and ECS level. This ensures
36
+ * unhealthy containers are removed. Both of these health checks can be
37
+ * overwritten by user by accessing `albFargateService` property, so no need
38
+ * for `overrides`.
39
+ */
40
+ private configureHealthCheck;
41
+ private attachFileSystem;
42
+ private getUrl;
43
+ }
@@ -0,0 +1,149 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.NextjsContainers = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_ecs_1 = require("aws-cdk-lib/aws-ecs");
8
+ const aws_ecs_patterns_1 = require("aws-cdk-lib/aws-ecs-patterns");
9
+ const aws_elasticloadbalancingv2_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2");
10
+ const constructs_1 = require("constructs");
11
+ /**
12
+ * Next.js load balanced via Application Load Balancer with containers via AWS
13
+ * Fargate.
14
+ */
15
+ class NextjsContainers extends constructs_1.Construct {
16
+ constructor(scope, id, props) {
17
+ super(scope, id);
18
+ this.props = props;
19
+ this.ecsCluster = this.createEcsCluster();
20
+ this.albFargateService = this.createAlbFargateSevice();
21
+ this.configureHealthCheck();
22
+ this.attachFileSystem();
23
+ this.url = this.getUrl();
24
+ }
25
+ createEcsCluster() {
26
+ const cluster = new aws_ecs_1.Cluster(this, "EcsCluster", {
27
+ enableFargateCapacityProviders: true,
28
+ containerInsights: true,
29
+ vpc: this.props.vpc,
30
+ ...this.props.overrides?.ecsClusterProps,
31
+ });
32
+ return cluster;
33
+ }
34
+ createAlbFargateSevice() {
35
+ let cpuArchitecture = undefined;
36
+ if (process.arch === "x64") {
37
+ cpuArchitecture = aws_ecs_1.CpuArchitecture.X86_64;
38
+ }
39
+ else if (process.arch === "arm64") {
40
+ cpuArchitecture = aws_ecs_1.CpuArchitecture.ARM64;
41
+ }
42
+ const albFargateService = new aws_ecs_patterns_1.ApplicationLoadBalancedFargateService(this, "AlbFargateService", {
43
+ circuitBreaker: { rollback: true, enable: true },
44
+ cluster: this.ecsCluster,
45
+ cpu: 1024,
46
+ healthCheckGracePeriod: aws_cdk_lib_1.Duration.seconds(10),
47
+ maxHealthyPercent: 200,
48
+ memoryLimitMiB: 2048,
49
+ minHealthyPercent: 100, // maintain service availability during deployment
50
+ /*
51
+ This protocol version is for the target group (Fargate), not the ALB
52
+ Listener. From docs, "Application Load Balancers provide native support
53
+ for HTTP/2 with HTTPS listeners". See https://docs.aws.amazon.com/elasticloadbalancing/latest/application/load-balancer-listeners.html
54
+ Next.js default server does not support HTTP/2 as it's recommended
55
+ to proxy your Next.js server which we're doing with ALB.
56
+ Also note, CloudFront only supports HTTP/1.1 origins.
57
+ https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/RequestAndResponseBehaviorCustomOrigin.html#RequestCustomHTTPVersion
58
+ */
59
+ protocolVersion: aws_elasticloadbalancingv2_1.ApplicationProtocolVersion.HTTP1,
60
+ runtimePlatform: {
61
+ cpuArchitecture,
62
+ },
63
+ taskImageOptions: {
64
+ command: ["node", this.props.relativeEntrypointPath],
65
+ containerName: "nextjs",
66
+ containerPort: 3000,
67
+ image: aws_ecs_1.ContainerImage.fromDockerImageAsset(this.props.dockerImageAsset),
68
+ logDriver: aws_ecs_1.LogDrivers.awsLogs({
69
+ streamPrefix: "nextjs",
70
+ mode: aws_ecs_1.AwsLogDriverMode.NON_BLOCKING,
71
+ }),
72
+ ...this.props.overrides?.taskImageOptions,
73
+ },
74
+ ...this.props.overrides?.albFargateServiceProps,
75
+ });
76
+ // required or health checks fail
77
+ albFargateService.taskDefinition.defaultContainer?.addEnvironment("HOSTNAME", "0.0.0.0");
78
+ // security best practice to enforce readonly for root filesystem
79
+ // hopefully https://github.com/aws/aws-cdk/issues/23935 will result in
80
+ // easier way to set this property
81
+ const cfnTaskDef = albFargateService.taskDefinition.node
82
+ .defaultChild;
83
+ cfnTaskDef.addPropertyOverride("ContainerDefinitions.0.ReadonlyRootFilesystem", true);
84
+ // speed up deployments by shortening deregistration delay
85
+ // https://docs.aws.amazon.com/AmazonECS/latest/bestpracticesguide/load-balancer-connection-draining.html
86
+ // TODO: document that this should be increased if long lived connections are expected
87
+ albFargateService.targetGroup.setAttribute("deregistration_delay.timeout_seconds", "30");
88
+ // best practice to enable cross zone load balancing
89
+ // @see https://docs.aws.amazon.com/elasticloadbalancing/latest/application/disable-cross-zone.html
90
+ albFargateService.loadBalancer.setAttribute("load_balancing.cross_zone.enabled", "true");
91
+ return albFargateService;
92
+ }
93
+ /**
94
+ * Configure health checks for containers at ALB and ECS level. This ensures
95
+ * unhealthy containers are removed. Both of these health checks can be
96
+ * overwritten by user by accessing `albFargateService` property, so no need
97
+ * for `overrides`.
98
+ */
99
+ configureHealthCheck() {
100
+ // speed up deployments by shortening health checks
101
+ // see https://docs.aws.amazon.com/AmazonECS/latest/bestpracticesguide/load-balancer-healthcheck.html
102
+ this.albFargateService.targetGroup.configureHealthCheck({
103
+ path: this.props.healthCheckPath,
104
+ healthyThresholdCount: 2,
105
+ interval: aws_cdk_lib_1.Duration.seconds(10), // too frequent? but enables faster rollback...
106
+ timeout: aws_cdk_lib_1.Duration.seconds(5), // must be less than interval
107
+ });
108
+ const healthCheck = {
109
+ command: [
110
+ "CMD-SHELL",
111
+ // curl isn't available in alpine linux
112
+ `wget --quiet --tries=1 --spider http://localhost:3000${this.props.healthCheckPath} || exit 1`,
113
+ ],
114
+ };
115
+ const defaultContainer = this.albFargateService.taskDefinition.defaultContainer;
116
+ if (defaultContainer) {
117
+ // @ts-expect-error must use internal "props" attribute b/c no other way to add health check
118
+ defaultContainer.props.healthCheck = healthCheck;
119
+ }
120
+ }
121
+ attachFileSystem() {
122
+ const container = this.albFargateService.taskDefinition.defaultContainer;
123
+ const volumeName = "cdk-nextjs-volume";
124
+ this.albFargateService.taskDefinition.addVolume({
125
+ name: volumeName,
126
+ efsVolumeConfiguration: {
127
+ fileSystemId: this.props.fileSystem.fileSystemId,
128
+ transitEncryption: "ENABLED",
129
+ authorizationConfig: {
130
+ accessPointId: this.props.accessPoint.accessPointId,
131
+ iam: "ENABLED",
132
+ },
133
+ },
134
+ });
135
+ container?.addMountPoints({
136
+ sourceVolume: volumeName,
137
+ containerPath: this.props.containerMountPathForEfs,
138
+ readOnly: false,
139
+ });
140
+ }
141
+ getUrl() {
142
+ const protocol = this.albFargateService.certificate ? "https" : "http";
143
+ return `${protocol}://${this.albFargateService.loadBalancer.loadBalancerDnsName}`;
144
+ }
145
+ }
146
+ exports.NextjsContainers = NextjsContainers;
147
+ _a = JSII_RTTI_SYMBOL_1;
148
+ NextjsContainers[_a] = { fqn: "cdk-nextjs.NextjsContainers", version: "0.0.0" };
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nextjs-containers.js","sourceRoot":"","sources":["../../src/nextjs-compute/nextjs-containers.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from \"aws-cdk-lib\";\nimport { DockerImageAsset } from \"aws-cdk-lib/aws-ecr-assets\";\nimport {\n  AwsLogDriverMode,\n  CfnTaskDefinition,\n  Cluster,\n  ContainerImage,\n  CpuArchitecture,\n  HealthCheck,\n  LogDrivers,\n} from \"aws-cdk-lib/aws-ecs\";\nimport {\n  ApplicationLoadBalancedFargateService,\n  ApplicationLoadBalancedFargateServiceProps,\n} from \"aws-cdk-lib/aws-ecs-patterns\";\nimport { FileSystem } from \"aws-cdk-lib/aws-efs\";\nimport { ApplicationProtocolVersion } from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { Construct } from \"constructs\";\nimport { NextjsComputeBaseProps } from \"./nextjs-compute-base-props\";\nimport { NextjsType } from \"../common\";\nimport { OptionalApplicationLoadBalancedTaskImageOptions } from \"../generated-structs/OptionalApplicationLoadBalancedTaskImageOptions\";\nimport { OptionalClusterProps } from \"../generated-structs/OptionalClusterProps\";\n\nexport interface NextjsContainersOverrides {\n  readonly ecsClusterProps?: OptionalClusterProps;\n  readonly albFargateServiceProps?: ApplicationLoadBalancedFargateServiceProps;\n  readonly taskImageOptions?: OptionalApplicationLoadBalancedTaskImageOptions;\n}\n\nexport interface NextjsContainersProps extends NextjsComputeBaseProps {\n  readonly dockerImageAsset: DockerImageAsset;\n  readonly fileSystem: FileSystem;\n  readonly nextjsType: NextjsType;\n  readonly overrides?: NextjsContainersOverrides;\n  readonly relativeEntrypointPath: string;\n}\n\n/**\n * Next.js load balanced via Application Load Balancer with containers via AWS\n * Fargate.\n */\nexport class NextjsContainers extends Construct {\n  albFargateService: ApplicationLoadBalancedFargateService;\n  ecsCluster: Cluster;\n  url: string;\n\n  private props: NextjsContainersProps;\n\n  constructor(scope: Construct, id: string, props: NextjsContainersProps) {\n    super(scope, id);\n    this.props = props;\n    this.ecsCluster = this.createEcsCluster();\n    this.albFargateService = this.createAlbFargateSevice();\n    this.configureHealthCheck();\n    this.attachFileSystem();\n    this.url = this.getUrl();\n  }\n\n  private createEcsCluster(): Cluster {\n    const cluster = new Cluster(this, \"EcsCluster\", {\n      enableFargateCapacityProviders: true,\n      containerInsights: true,\n      vpc: this.props.vpc,\n      ...this.props.overrides?.ecsClusterProps,\n    });\n    return cluster;\n  }\n  private createAlbFargateSevice(): ApplicationLoadBalancedFargateService {\n    let cpuArchitecture: CpuArchitecture | undefined = undefined;\n    if (process.arch === \"x64\") {\n      cpuArchitecture = CpuArchitecture.X86_64;\n    } else if (process.arch === \"arm64\") {\n      cpuArchitecture = CpuArchitecture.ARM64;\n    }\n    const albFargateService = new ApplicationLoadBalancedFargateService(\n      this,\n      \"AlbFargateService\",\n      {\n        circuitBreaker: { rollback: true, enable: true },\n        cluster: this.ecsCluster,\n        cpu: 1024,\n        healthCheckGracePeriod: Duration.seconds(10),\n        maxHealthyPercent: 200,\n        memoryLimitMiB: 2048,\n        minHealthyPercent: 100, // maintain service availability during deployment\n        /*\n          This protocol version is for the target group (Fargate), not the ALB\n          Listener. From docs, \"Application Load Balancers provide native support\n          for HTTP/2 with HTTPS listeners\". See https://docs.aws.amazon.com/elasticloadbalancing/latest/application/load-balancer-listeners.html\n          Next.js default server does not support HTTP/2 as it's recommended\n          to proxy your Next.js server which we're doing with ALB.\n          Also note, CloudFront only supports HTTP/1.1 origins.\n          https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/RequestAndResponseBehaviorCustomOrigin.html#RequestCustomHTTPVersion\n        */\n        protocolVersion: ApplicationProtocolVersion.HTTP1,\n        runtimePlatform: {\n          cpuArchitecture,\n        },\n        taskImageOptions: {\n          command: [\"node\", this.props.relativeEntrypointPath],\n          containerName: \"nextjs\",\n          containerPort: 3000,\n          image: ContainerImage.fromDockerImageAsset(\n            this.props.dockerImageAsset,\n          ),\n          logDriver: LogDrivers.awsLogs({\n            streamPrefix: \"nextjs\",\n            mode: AwsLogDriverMode.NON_BLOCKING,\n          }),\n          ...this.props.overrides?.taskImageOptions,\n        },\n        ...this.props.overrides?.albFargateServiceProps,\n      },\n    );\n    // required or health checks fail\n    albFargateService.taskDefinition.defaultContainer?.addEnvironment(\n      \"HOSTNAME\",\n      \"0.0.0.0\",\n    );\n    // security best practice to enforce readonly for root filesystem\n    // hopefully https://github.com/aws/aws-cdk/issues/23935 will result in\n    // easier way to set this property\n    const cfnTaskDef = albFargateService.taskDefinition.node\n      .defaultChild as CfnTaskDefinition;\n    cfnTaskDef.addPropertyOverride(\n      \"ContainerDefinitions.0.ReadonlyRootFilesystem\",\n      true, // TODO: figure why symlink not being created between full route cache\n    );\n    // speed up deployments by shortening deregistration delay\n    // https://docs.aws.amazon.com/AmazonECS/latest/bestpracticesguide/load-balancer-connection-draining.html\n    // TODO: document that this should be increased if long lived connections are expected\n    albFargateService.targetGroup.setAttribute(\n      \"deregistration_delay.timeout_seconds\",\n      \"30\",\n    );\n    // best practice to enable cross zone load balancing\n    // @see https://docs.aws.amazon.com/elasticloadbalancing/latest/application/disable-cross-zone.html\n    albFargateService.loadBalancer.setAttribute(\n      \"load_balancing.cross_zone.enabled\",\n      \"true\",\n    );\n    return albFargateService;\n  }\n  /**\n   * Configure health checks for containers at ALB and ECS level. This ensures\n   * unhealthy containers are removed. Both of these health checks can be\n   * overwritten by user by accessing `albFargateService` property, so no need\n   * for `overrides`.\n   */\n  private configureHealthCheck() {\n    // speed up deployments by shortening health checks\n    // see https://docs.aws.amazon.com/AmazonECS/latest/bestpracticesguide/load-balancer-healthcheck.html\n    this.albFargateService.targetGroup.configureHealthCheck({\n      path: this.props.healthCheckPath,\n      healthyThresholdCount: 2,\n      interval: Duration.seconds(10), // too frequent? but enables faster rollback...\n      timeout: Duration.seconds(5), // must be less than interval\n    });\n    const healthCheck: HealthCheck = {\n      command: [\n        \"CMD-SHELL\",\n        // curl isn't available in alpine linux\n        `wget --quiet --tries=1 --spider http://localhost:3000${this.props.healthCheckPath} || exit 1`,\n      ],\n    };\n    const defaultContainer =\n      this.albFargateService.taskDefinition.defaultContainer;\n    if (defaultContainer) {\n      // @ts-expect-error must use internal \"props\" attribute b/c no other way to add health check\n      defaultContainer.props.healthCheck = healthCheck;\n    }\n  }\n  private attachFileSystem() {\n    const container = this.albFargateService.taskDefinition.defaultContainer;\n    const volumeName = \"cdk-nextjs-volume\";\n    this.albFargateService.taskDefinition.addVolume({\n      name: volumeName,\n      efsVolumeConfiguration: {\n        fileSystemId: this.props.fileSystem.fileSystemId,\n        transitEncryption: \"ENABLED\",\n        authorizationConfig: {\n          accessPointId: this.props.accessPoint.accessPointId,\n          iam: \"ENABLED\",\n        },\n      },\n    });\n    container?.addMountPoints({\n      sourceVolume: volumeName,\n      containerPath: this.props.containerMountPathForEfs,\n      readOnly: false,\n    });\n  }\n  private getUrl(): string {\n    const protocol = this.albFargateService.certificate ? \"https\" : \"http\";\n    return `${protocol}://${this.albFargateService.loadBalancer.loadBalancerDnsName}`;\n  }\n}\n"]}
@@ -0,0 +1,23 @@
1
+ import { DockerImageCode, DockerImageFunction, FunctionUrl } from "aws-cdk-lib/aws-lambda";
2
+ import { Construct } from "constructs";
3
+ import { NextjsComputeBaseProps } from "./nextjs-compute-base-props";
4
+ import { OptionalDockerImageFunctionProps } from "../generated-structs/OptionalDockerImageFunctionProps";
5
+ import { OptionalFunctionUrlProps } from "../generated-structs/OptionalFunctionUrlProps";
6
+ export interface NextjsFunctionsOverrides {
7
+ readonly dockerImageFunctionProps?: OptionalDockerImageFunctionProps;
8
+ readonly functionUrlProps?: OptionalFunctionUrlProps;
9
+ }
10
+ export interface NextjsFunctionsProps extends NextjsComputeBaseProps {
11
+ readonly dockerImageCode: DockerImageCode;
12
+ readonly overrides?: NextjsFunctionsOverrides;
13
+ }
14
+ /**
15
+ * Run Next.js in functions on AWS with AWS Lambda.
16
+ */
17
+ export declare class NextjsFunctions extends Construct {
18
+ function: DockerImageFunction;
19
+ functionUrl: FunctionUrl;
20
+ private props;
21
+ constructor(scope: Construct, id: string, props: NextjsFunctionsProps);
22
+ private createFunction;
23
+ }
@@ -0,0 +1,57 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.NextjsFunctions = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
8
+ const constructs_1 = require("constructs");
9
+ /**
10
+ * Run Next.js in functions on AWS with AWS Lambda.
11
+ */
12
+ class NextjsFunctions extends constructs_1.Construct {
13
+ constructor(scope, id, props) {
14
+ super(scope, id);
15
+ this.props = props;
16
+ this.function = this.createFunction();
17
+ this.functionUrl = this.function.addFunctionUrl({
18
+ authType: aws_lambda_1.FunctionUrlAuthType.AWS_IAM,
19
+ invokeMode: aws_lambda_1.InvokeMode.RESPONSE_STREAM,
20
+ ...this.props.overrides?.functionUrlProps,
21
+ });
22
+ if (!this.function.role) {
23
+ throw new Error("Function role is undefined");
24
+ }
25
+ }
26
+ createFunction() {
27
+ let architecture = undefined;
28
+ if (process.arch === "x64") {
29
+ architecture = aws_lambda_1.Architecture.X86_64;
30
+ }
31
+ else if (process.arch === "arm64") {
32
+ architecture = aws_lambda_1.Architecture.ARM_64;
33
+ }
34
+ const fn = new aws_lambda_1.DockerImageFunction(this, "Functions", {
35
+ architecture,
36
+ code: this.props.dockerImageCode,
37
+ filesystem: aws_lambda_1.FileSystem.fromEfsAccessPoint(this.props.accessPoint, this.props.containerMountPathForEfs),
38
+ memorySize: 2048,
39
+ timeout: aws_cdk_lib_1.Duration.seconds(30),
40
+ vpc: this.props.vpc,
41
+ ...this.props.overrides?.dockerImageFunctionProps,
42
+ environment: {
43
+ AWS_LWA_ENABLE_COMPRESSION: "true",
44
+ AWS_LWA_INVOKE_MODE: "response_stream",
45
+ AWS_LWA_READINESS_CHECK_PATH: this.props.healthCheckPath,
46
+ AWS_LWA_READINESS_CHECK_PORT: "3000",
47
+ READINESS_CHECK_PATH: `http://127.0.0.1:3000${this.props.healthCheckPath}`,
48
+ ...this.props.overrides?.dockerImageFunctionProps?.environment,
49
+ },
50
+ });
51
+ return fn;
52
+ }
53
+ }
54
+ exports.NextjsFunctions = NextjsFunctions;
55
+ _a = JSII_RTTI_SYMBOL_1;
56
+ NextjsFunctions[_a] = { fqn: "cdk-nextjs.NextjsFunctions", version: "0.0.0" };
57
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV4dGpzLWZ1bmN0aW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uZXh0anMtY29tcHV0ZS9uZXh0anMtZnVuY3Rpb25zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkNBQXVDO0FBQ3ZDLHVEQVFnQztBQUNoQywyQ0FBdUM7QUFldkM7O0dBRUc7QUFDSCxNQUFhLGVBQWdCLFNBQVEsc0JBQVM7SUFNNUMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUEyQjtRQUNuRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ3RDLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxjQUFjLENBQUM7WUFDOUMsUUFBUSxFQUFFLGdDQUFtQixDQUFDLE9BQU87WUFDckMsVUFBVSxFQUFFLHVCQUFVLENBQUMsZUFBZTtZQUN0QyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxFQUFFLGdCQUFnQjtTQUMxQyxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUN4QixNQUFNLElBQUksS0FBSyxDQUFDLDRCQUE0QixDQUFDLENBQUM7UUFDaEQsQ0FBQztJQUNILENBQUM7SUFFTyxjQUFjO1FBQ3BCLElBQUksWUFBWSxHQUE2QixTQUFTLENBQUM7UUFDdkQsSUFBSSxPQUFPLENBQUMsSUFBSSxLQUFLLEtBQUssRUFBRSxDQUFDO1lBQzNCLFlBQVksR0FBRyx5QkFBWSxDQUFDLE1BQU0sQ0FBQztRQUNyQyxDQUFDO2FBQU0sSUFBSSxPQUFPLENBQUMsSUFBSSxLQUFLLE9BQU8sRUFBRSxDQUFDO1lBQ3BDLFlBQVksR0FBRyx5QkFBWSxDQUFDLE1BQU0sQ0FBQztRQUNyQyxDQUFDO1FBQ0QsTUFBTSxFQUFFLEdBQUcsSUFBSSxnQ0FBbUIsQ0FBQyxJQUFJLEVBQUUsV0FBVyxFQUFFO1lBQ3BELFlBQVk7WUFDWixJQUFJLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxlQUFlO1lBQ2hDLFVBQVUsRUFBRSx1QkFBVSxDQUFDLGtCQUFrQixDQUN2QyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFDdEIsSUFBSSxDQUFDLEtBQUssQ0FBQyx3QkFBd0IsQ0FDcEM7WUFDRCxVQUFVLEVBQUUsSUFBSTtZQUNoQixPQUFPLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQzdCLEdBQUcsRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUc7WUFDbkIsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSx3QkFBd0I7WUFDakQsV0FBVyxFQUFFO2dCQUNYLDBCQUEwQixFQUFFLE1BQU07Z0JBQ2xDLG1CQUFtQixFQUFFLGlCQUFpQjtnQkFDdEMsNEJBQTRCLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxlQUFlO2dCQUN4RCw0QkFBNEIsRUFBRSxNQUFNO2dCQUNwQyxvQkFBb0IsRUFBRSx3QkFBd0IsSUFBSSxDQUFDLEtBQUssQ0FBQyxlQUFlLEVBQUU7Z0JBQzFFLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLEVBQUUsd0JBQXdCLEVBQUUsV0FBVzthQUMvRDtTQUNGLENBQUMsQ0FBQztRQUNILE9BQU8sRUFBRSxDQUFDO0lBQ1osQ0FBQzs7QUFoREgsMENBaURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRHVyYXRpb24gfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7XG4gIEFyY2hpdGVjdHVyZSxcbiAgRG9ja2VySW1hZ2VDb2RlLFxuICBEb2NrZXJJbWFnZUZ1bmN0aW9uLFxuICBGaWxlU3lzdGVtLFxuICBGdW5jdGlvblVybCxcbiAgRnVuY3Rpb25VcmxBdXRoVHlwZSxcbiAgSW52b2tlTW9kZSxcbn0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5pbXBvcnQgeyBOZXh0anNDb21wdXRlQmFzZVByb3BzIH0gZnJvbSBcIi4vbmV4dGpzLWNvbXB1dGUtYmFzZS1wcm9wc1wiO1xuaW1wb3J0IHsgT3B0aW9uYWxEb2NrZXJJbWFnZUZ1bmN0aW9uUHJvcHMgfSBmcm9tIFwiLi4vZ2VuZXJhdGVkLXN0cnVjdHMvT3B0aW9uYWxEb2NrZXJJbWFnZUZ1bmN0aW9uUHJvcHNcIjtcbmltcG9ydCB7IE9wdGlvbmFsRnVuY3Rpb25VcmxQcm9wcyB9IGZyb20gXCIuLi9nZW5lcmF0ZWQtc3RydWN0cy9PcHRpb25hbEZ1bmN0aW9uVXJsUHJvcHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBOZXh0anNGdW5jdGlvbnNPdmVycmlkZXMge1xuICByZWFkb25seSBkb2NrZXJJbWFnZUZ1bmN0aW9uUHJvcHM/OiBPcHRpb25hbERvY2tlckltYWdlRnVuY3Rpb25Qcm9wcztcbiAgcmVhZG9ubHkgZnVuY3Rpb25VcmxQcm9wcz86IE9wdGlvbmFsRnVuY3Rpb25VcmxQcm9wcztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBOZXh0anNGdW5jdGlvbnNQcm9wcyBleHRlbmRzIE5leHRqc0NvbXB1dGVCYXNlUHJvcHMge1xuICByZWFkb25seSBkb2NrZXJJbWFnZUNvZGU6IERvY2tlckltYWdlQ29kZTtcbiAgcmVhZG9ubHkgb3ZlcnJpZGVzPzogTmV4dGpzRnVuY3Rpb25zT3ZlcnJpZGVzO1xufVxuXG4vKipcbiAqIFJ1biBOZXh0LmpzIGluIGZ1bmN0aW9ucyBvbiBBV1Mgd2l0aCBBV1MgTGFtYmRhLlxuICovXG5leHBvcnQgY2xhc3MgTmV4dGpzRnVuY3Rpb25zIGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgZnVuY3Rpb246IERvY2tlckltYWdlRnVuY3Rpb247XG4gIGZ1bmN0aW9uVXJsOiBGdW5jdGlvblVybDtcblxuICBwcml2YXRlIHByb3BzOiBOZXh0anNGdW5jdGlvbnNQcm9wcztcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogTmV4dGpzRnVuY3Rpb25zUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuICAgIHRoaXMucHJvcHMgPSBwcm9wcztcbiAgICB0aGlzLmZ1bmN0aW9uID0gdGhpcy5jcmVhdGVGdW5jdGlvbigpO1xuICAgIHRoaXMuZnVuY3Rpb25VcmwgPSB0aGlzLmZ1bmN0aW9uLmFkZEZ1bmN0aW9uVXJsKHtcbiAgICAgIGF1dGhUeXBlOiBGdW5jdGlvblVybEF1dGhUeXBlLkFXU19JQU0sXG4gICAgICBpbnZva2VNb2RlOiBJbnZva2VNb2RlLlJFU1BPTlNFX1NUUkVBTSxcbiAgICAgIC4uLnRoaXMucHJvcHMub3ZlcnJpZGVzPy5mdW5jdGlvblVybFByb3BzLFxuICAgIH0pO1xuICAgIGlmICghdGhpcy5mdW5jdGlvbi5yb2xlKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJGdW5jdGlvbiByb2xlIGlzIHVuZGVmaW5lZFwiKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGNyZWF0ZUZ1bmN0aW9uKCkge1xuICAgIGxldCBhcmNoaXRlY3R1cmU6IEFyY2hpdGVjdHVyZSB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZDtcbiAgICBpZiAocHJvY2Vzcy5hcmNoID09PSBcIng2NFwiKSB7XG4gICAgICBhcmNoaXRlY3R1cmUgPSBBcmNoaXRlY3R1cmUuWDg2XzY0O1xuICAgIH0gZWxzZSBpZiAocHJvY2Vzcy5hcmNoID09PSBcImFybTY0XCIpIHtcbiAgICAgIGFyY2hpdGVjdHVyZSA9IEFyY2hpdGVjdHVyZS5BUk1fNjQ7XG4gICAgfVxuICAgIGNvbnN0IGZuID0gbmV3IERvY2tlckltYWdlRnVuY3Rpb24odGhpcywgXCJGdW5jdGlvbnNcIiwge1xuICAgICAgYXJjaGl0ZWN0dXJlLFxuICAgICAgY29kZTogdGhpcy5wcm9wcy5kb2NrZXJJbWFnZUNvZGUsXG4gICAgICBmaWxlc3lzdGVtOiBGaWxlU3lzdGVtLmZyb21FZnNBY2Nlc3NQb2ludChcbiAgICAgICAgdGhpcy5wcm9wcy5hY2Nlc3NQb2ludCxcbiAgICAgICAgdGhpcy5wcm9wcy5jb250YWluZXJNb3VudFBhdGhGb3JFZnMsXG4gICAgICApLFxuICAgICAgbWVtb3J5U2l6ZTogMjA0OCxcbiAgICAgIHRpbWVvdXQ6IER1cmF0aW9uLnNlY29uZHMoMzApLFxuICAgICAgdnBjOiB0aGlzLnByb3BzLnZwYyxcbiAgICAgIC4uLnRoaXMucHJvcHMub3ZlcnJpZGVzPy5kb2NrZXJJbWFnZUZ1bmN0aW9uUHJvcHMsXG4gICAgICBlbnZpcm9ubWVudDoge1xuICAgICAgICBBV1NfTFdBX0VOQUJMRV9DT01QUkVTU0lPTjogXCJ0cnVlXCIsXG4gICAgICAgIEFXU19MV0FfSU5WT0tFX01PREU6IFwicmVzcG9uc2Vfc3RyZWFtXCIsXG4gICAgICAgIEFXU19MV0FfUkVBRElORVNTX0NIRUNLX1BBVEg6IHRoaXMucHJvcHMuaGVhbHRoQ2hlY2tQYXRoLFxuICAgICAgICBBV1NfTFdBX1JFQURJTkVTU19DSEVDS19QT1JUOiBcIjMwMDBcIixcbiAgICAgICAgUkVBRElORVNTX0NIRUNLX1BBVEg6IGBodHRwOi8vMTI3LjAuMC4xOjMwMDAke3RoaXMucHJvcHMuaGVhbHRoQ2hlY2tQYXRofWAsXG4gICAgICAgIC4uLnRoaXMucHJvcHMub3ZlcnJpZGVzPy5kb2NrZXJJbWFnZUZ1bmN0aW9uUHJvcHM/LmVudmlyb25tZW50LFxuICAgICAgfSxcbiAgICB9KTtcbiAgICByZXR1cm4gZm47XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,120 @@
1
+ import { ICertificate } from "aws-cdk-lib/aws-certificatemanager";
2
+ import { AddBehaviorOptions, CachePolicyProps, Distribution, ResponseHeadersPolicyProps } from "aws-cdk-lib/aws-cloudfront";
3
+ import { HttpOriginProps } from "aws-cdk-lib/aws-cloudfront-origins";
4
+ import { IBucket } from "aws-cdk-lib/aws-s3";
5
+ import { Construct } from "constructs";
6
+ import { NextjsType } from "./common";
7
+ import { OptionalDistributionProps } from "./generated-structs/OptionalDistributionProps";
8
+ import { OptionalFunctionProps } from "./generated-structs/OptionalFunctionProps";
9
+ import { OptionalS3OriginProps } from "./generated-structs/OptionalS3OriginProps";
10
+ import { PublicDirEntry } from "./nextjs-build/nextjs-build";
11
+ export interface NextjsDistributionOverrides {
12
+ readonly edgeFunctionProps?: OptionalFunctionProps;
13
+ readonly distributionProps?: OptionalDistributionProps;
14
+ readonly imageBehaviorOptions?: AddBehaviorOptions;
15
+ readonly imageCachePolicyProps?: CachePolicyProps;
16
+ readonly imageResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;
17
+ readonly dynamicBehaviorOptions?: AddBehaviorOptions;
18
+ readonly dynamicCachePolicyProps?: CachePolicyProps;
19
+ readonly dynamicResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;
20
+ readonly dynamicHttpOriginProps?: HttpOriginProps;
21
+ readonly staticBehaviorOptions?: AddBehaviorOptions;
22
+ readonly staticResponseHeadersPolicyProps?: ResponseHeadersPolicyProps;
23
+ readonly s3OriginProps?: OptionalS3OriginProps;
24
+ }
25
+ export interface NextjsDistributionProps {
26
+ /**
27
+ * Bucket containing static assets.
28
+ * Must be provided if you want to serve static files.
29
+ */
30
+ readonly assetsBucket: IBucket;
31
+ readonly basePath?: string;
32
+ /**
33
+ * Optional but only applicable for `NextjsType.GLOBAL_CONTAINERS`
34
+ */
35
+ readonly certificate?: ICertificate;
36
+ readonly distribution?: Distribution;
37
+ /**
38
+ * Dynamic (Next.js server) URL to add behavior to distribution
39
+ */
40
+ readonly dynamicUrl: string;
41
+ /**
42
+ * Required if `NextjsType.GLOBAL_FUNCTIONS`
43
+ */
44
+ readonly functionArn?: string;
45
+ readonly nextjsType: NextjsType;
46
+ /**
47
+ * Override props for every construct.
48
+ */
49
+ readonly overrides?: NextjsDistributionOverrides;
50
+ /**
51
+ * Path to directory of Next.js app's public directory. Used to add static
52
+ * behaviors to distribution.
53
+ */
54
+ readonly publicDirEntries: PublicDirEntry[];
55
+ }
56
+ export declare class NextjsDistribution extends Construct {
57
+ distribution: Distribution;
58
+ private props;
59
+ /**
60
+ * Common security headers applied by default to all origins
61
+ * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-response-headers-policies.html#managed-response-headers-policies-security
62
+ */
63
+ private commonSecurityHeadersBehavior;
64
+ private staticOrigin;
65
+ private dynamicOrigin;
66
+ private dynamicOriginResponsePolicy;
67
+ private dynamicCloudFrontFunctionAssociations;
68
+ private edgeLambdas?;
69
+ private isFunctionCompute;
70
+ private staticBehaviorOptions;
71
+ private dynamicBehaviorOptions;
72
+ private imageBehaviorOptions;
73
+ /**
74
+ * Given stack id: "arn:aws:cloudformation:us-east-1:905418358903:stack/lh-stickb-idp/4bf74be0-e880-11ee-aea9-0affc6185b25",
75
+ * returns "4bf74be0"
76
+ */
77
+ private uniqueStackIdPart;
78
+ constructor(scope: Construct, id: string, props: NextjsDistributionProps);
79
+ private createStaticOrigin;
80
+ private createDynamicOrigin;
81
+ /**
82
+ * Lambda Function URLs "expect the `Host` header to contain the origin domain
83
+ * name, not the domain name of the CloudFront distribution."
84
+ * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/using-managed-origin-request-policies.html#managed-origin-request-policy-all-viewer-except-host-header
85
+ */
86
+ private createDynamicOriginRequestPolicy;
87
+ /**
88
+ * Ensures Next.js `request.url` will be correct domain instead of URL of
89
+ * compute option (App Runner, Fargate, or Lambda)
90
+ * @see https://open-next.js.org/advanced/workaround#workaround-set-x-forwarded-host-header-aws-specific
91
+ */
92
+ private createDynamicCloudFrontFunctionAssociations;
93
+ /**
94
+ * Required to sign requests so that we can use IAM_AUTH for Lambda Function URL
95
+ * to prevent public access. Once CloudFront Lambda OAC is released, we can
96
+ * use infra configuration for this instead of custom code.
97
+ */
98
+ private createEdgeLambdas;
99
+ private createStaticBehaviorOptions;
100
+ private createDynamicBehaviorOptions;
101
+ private createImageBehaviorOptions;
102
+ /**
103
+ * Creates or uses user specified CloudFront Distribution
104
+ */
105
+ private getDistribution;
106
+ private addDynamicBehaviors;
107
+ private addStaticBehaviors;
108
+ /**
109
+ * Optionally prepends base path to given path pattern.
110
+ */
111
+ private getPathPattern;
112
+ /**
113
+ * Add Origin Access Control (OAC) to CloudFront Distribution which is preferred
114
+ * way to secure access from Distribution to S3. Remove legacy OAI.
115
+ *
116
+ * When CDK releases L2 support for this, please remove this code.
117
+ * @see https://github.com/aws/aws-cdk/issues/21771#issuecomment-1567647338
118
+ */
119
+ private addS3OacAndRemoveOai;
120
+ }