cdk-nuxt 0.4.2 → 0.4.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -65,10 +65,11 @@ After the installation steps the `package.json` file should look something like
65
65
  ...
66
66
  }
67
67
  ```
68
- 2. [Create an AWS account](https://aws.amazon.com/premiumsupport/knowledge-center/create-and-activate-aws-account/?nc1=h_ls), if you don't have one yet. Then login into the AWS console and note the `Account ID`. You will need it in step 5.
69
- 3. [Create a hosted zone in Route53](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) for the desired domain, if you don't have one yet.<br/>This is required to create DNS records for the domain to make the Nuxt app publicly available on that domain.<br/>On the hosted zone details you should see the `Hosted zone ID` of the hosted zone. You will need it in step 5.
70
- 4. [Request a public certificate in the AWS Certificate Manager (ACM)](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html) for the desired domain in `us-east-1` (global) and validate it, if you don't have one yet.<br/>This is required to provide the Nuxt app via HTTPS on the public internet.<br/>Take note of the displayed `ARN` for the certificate. You will need it in step 5.<br/>**Important: The certificate must be issued in us-east-1 (global) regardless of the region used for the Nuxt app itself as it will be attached to the Cloudfront distribution which works globally.**
71
- 5. Run the following command to automatically create the required CDK stack entrypoint at `stack/index.ts`. This file defines the config how the Nuxt app will be deployed via CDK. You should adapt the file to the project's needs, especially the props `env.account` (setup step 2), `hostedZoneId` (setup step 3) and `globalTlsCertificateArn` (setup step 4).
68
+ 2. [Create an AWS account](https://aws.amazon.com/premiumsupport/knowledge-center/create-and-activate-aws-account/?nc1=h_ls), if you don't have one yet. Then login into the AWS console and note the `Account ID`. You will need it in step 6.
69
+ 3. [Create a hosted zone in Route53](https://docs.aws.amazon.com/Route53/latest/DeveloperGuide/AboutHZWorkingWith.html) for the desired domain, if you don't have one yet.<br/>This is required to create DNS records for the domain to make the Nuxt app publicly available on that domain.<br/>On the hosted zone details you should see the `Hosted zone ID` of the hosted zone. You will need it in step 6.
70
+ 4. [Request a public **regional** certificate in the AWS Certificate Manager (ACM)](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html) for the desired domain in your desired region, e.g., `eu-central-1`, and validate it, if you don't have one yet.<br/>This is required to make the Nuxt app accessible via the custom domain and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.<br/>Take note of the displayed `ARN` for the certificate. You will need it in step 6.
71
+ 5. [Request a public **global** certificate in the AWS Certificate Manager (ACM)](https://docs.aws.amazon.com/acm/latest/userguide/gs-acm-request-public.html) for the desired domain in `us-east-1` (**global**) and validate it, if you don't have one yet.<br/>This is required to provide the Nuxt app via HTTPS on the public internet.<br/>Take note of the displayed `ARN` for the certificate. You will need it in step 6.<br/>**Important: The certificate must be issued in us-east-1 (global) regardless of the region used for the Nuxt app itself as it will be attached to the Cloudfront distribution which works globally.**
72
+ 6. Run the following command to automatically create the required CDK stack entrypoint at `stack/index.ts`. This file defines the config how the Nuxt app will be deployed via CDK. You should adapt the file to the project's needs, especially the props `env.account` (setup step 2), `hostedZoneId` (setup step 3), `regionalTlsCertificateArn` (setup step 4) and `globalTlsCertificateArn` (setup step 5).
72
73
 
73
74
  ```bash
74
75
  node_modules/.bin/cdk-nuxt-init
@@ -26,7 +26,7 @@ export interface NuxtAppStackProps extends AppStackProps {
26
26
  readonly globalTlsCertificateArn: string;
27
27
  /**
28
28
  * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain
29
- * and to provide the custom domain to the Nuxt app on server side rendering.
29
+ * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.
30
30
  * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
31
31
  */
32
32
  readonly regionalTlsCertificateArn: string;
@@ -34,9 +34,14 @@ export interface NuxtAppStackProps extends AppStackProps {
34
34
  * The nuxt.config.js of the Nuxt app.
35
35
  */
36
36
  readonly nuxtConfig: NuxtConfig;
37
+ /**
38
+ * The memory size to apply to the Nuxt app's Lambda.
39
+ * Defaults to 512MB.
40
+ */
41
+ readonly memorySize?: number;
37
42
  }
38
43
  /**
39
- * Creates a lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
44
+ * Creates a Lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
40
45
  */
41
46
  export declare class NuxtAppStack extends Stack {
42
47
  /**
@@ -63,13 +68,13 @@ export declare class NuxtAppStack extends Stack {
63
68
  */
64
69
  staticAssetsBucket: IBucket;
65
70
  /**
66
- * The lambda function to render the Nuxt app on the server side.
71
+ * The Lambda function to render the Nuxt app on the server side.
67
72
  *
68
73
  * @private
69
74
  */
70
75
  private readonly lambdaFunction;
71
76
  /**
72
- * The API gateway to make the lambda function to render the Nuxt app publicly available.
77
+ * The API gateway to make the Lambda function to render the Nuxt app publicly available.
73
78
  *
74
79
  * @private
75
80
  */
@@ -81,7 +86,7 @@ export declare class NuxtAppStack extends Stack {
81
86
  */
82
87
  private staticAssetConfigs;
83
88
  /**
84
- * The CloudFront distribution to route incoming requests to the Nuxt lambda function (via the API gateway)
89
+ * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)
85
90
  * or the S3 assets folder (with caching).
86
91
  *
87
92
  * @private
@@ -101,25 +106,25 @@ export declare class NuxtAppStack extends Stack {
101
106
  */
102
107
  private createStaticAssetsBucket;
103
108
  /**
104
- * Creates a lambda layer with the node_modules required to render the Nuxt app on the server side.
109
+ * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.
105
110
  *
106
111
  * @private
107
112
  */
108
113
  private createSsrLambdaLayer;
109
114
  /**
110
- * Creates the lambda function to render the Nuxt app.
115
+ * Creates the Lambda function to render the Nuxt app.
111
116
  *
112
117
  * @private
113
118
  */
114
119
  private createLambdaFunction;
115
120
  /**
116
- * Creates the API gateway to make the Nuxt app render lambda function publicly available.
121
+ * Creates the API gateway to make the Nuxt app render Lambda function publicly available.
117
122
  *
118
123
  * @private
119
124
  */
120
125
  private createApiGateway;
121
126
  /**
122
- * Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
127
+ * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)
123
128
  * or the S3 assets folder (with caching).
124
129
  *
125
130
  * @param props
@@ -127,7 +132,7 @@ export declare class NuxtAppStack extends Stack {
127
132
  */
128
133
  private createCloudFrontDistribution;
129
134
  /**
130
- * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
135
+ * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).
131
136
  * Additionally, this automatically redirects HTTP requests to HTTPS.
132
137
  *
133
138
  * @private
@@ -151,7 +156,7 @@ export declare class NuxtAppStack extends Stack {
151
156
  * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.
152
157
  * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.
153
158
  * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up
154
- * after a specified period of time via the lambda function in the {@see NuxtAppAssetsCleanupStack}.
159
+ * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.
155
160
  */
156
161
  private configureDeployments;
157
162
  /**
@@ -169,7 +174,7 @@ export declare class NuxtAppStack extends Stack {
169
174
  */
170
175
  private createDnsRecords;
171
176
  /**
172
- * Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
177
+ * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm
173
178
  * and speed up initial SSR requests.
174
179
  *
175
180
  * @private
@@ -19,7 +19,7 @@ const fs = require("fs");
19
19
  const aws_events_1 = require("aws-cdk-lib/aws-events");
20
20
  const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets");
21
21
  /**
22
- * Creates a lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
22
+ * Creates a Lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
23
23
  */
24
24
  class NuxtAppStack extends aws_cdk_lib_1.Stack {
25
25
  constructor(scope, id, props) {
@@ -29,7 +29,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
29
29
  this.staticAssetConfigs = (0, nuxt_app_static_assets_1.getNuxtAppStaticAssetConfigs)(props.nuxtConfig);
30
30
  this.cdnAccessIdentity = this.createCdnAccessIdentity();
31
31
  this.staticAssetsBucket = this.createStaticAssetsBucket();
32
- this.lambdaFunction = this.createLambdaFunction();
32
+ this.lambdaFunction = this.createLambdaFunction(props);
33
33
  this.apiGateway = this.createApiGateway(props);
34
34
  this.cdn = this.createCloudFrontDistribution(props);
35
35
  this.configureDeployments();
@@ -64,7 +64,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
64
64
  return bucket;
65
65
  }
66
66
  /**
67
- * Creates a lambda layer with the node_modules required to render the Nuxt app on the server side.
67
+ * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.
68
68
  *
69
69
  * @private
70
70
  */
@@ -78,11 +78,12 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
78
78
  });
79
79
  }
80
80
  /**
81
- * Creates the lambda function to render the Nuxt app.
81
+ * Creates the Lambda function to render the Nuxt app.
82
82
  *
83
83
  * @private
84
84
  */
85
- createLambdaFunction() {
85
+ createLambdaFunction(props) {
86
+ var _a;
86
87
  const funcName = `${this.resourceIdPrefix}-function`;
87
88
  return new aws_lambda_1.Function(this, funcName, {
88
89
  functionName: funcName,
@@ -95,13 +96,13 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
95
96
  exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],
96
97
  }),
97
98
  timeout: aws_cdk_lib_1.Duration.seconds(10),
98
- memorySize: 512,
99
+ memorySize: (_a = props.memorySize) !== null && _a !== void 0 ? _a : 512,
99
100
  logRetention: aws_logs_1.RetentionDays.ONE_MONTH,
100
101
  allowPublicSubnet: false
101
102
  });
102
103
  }
103
104
  /**
104
- * Creates the API gateway to make the Nuxt app render lambda function publicly available.
105
+ * Creates the API gateway to make the Nuxt app render Lambda function publicly available.
105
106
  *
106
107
  * @private
107
108
  */
@@ -119,7 +120,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
119
120
  });
120
121
  const apiGateway = new aws_apigatewayv2_alpha_1.HttpApi(this, apiName, {
121
122
  apiName,
122
- description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} lambda function to make it publicly available.`,
123
+ description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,
123
124
  // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere
124
125
  corsPreflight: undefined,
125
126
  defaultIntegration: lambdaIntegration,
@@ -135,7 +136,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
135
136
  return apiGateway;
136
137
  }
137
138
  /**
138
- * Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
139
+ * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)
139
140
  * or the S3 assets folder (with caching).
140
141
  *
141
142
  * @param props
@@ -154,7 +155,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
154
155
  });
155
156
  }
156
157
  /**
157
- * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
158
+ * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).
158
159
  * Additionally, this automatically redirects HTTP requests to HTTPS.
159
160
  *
160
161
  * @private
@@ -231,7 +232,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
231
232
  * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.
232
233
  * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.
233
234
  * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up
234
- * after a specified period of time via the lambda function in the {@see NuxtAppAssetsCleanupStack}.
235
+ * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.
235
236
  */
236
237
  configureDeployments() {
237
238
  const defaultCacheConfig = [
@@ -293,7 +294,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
293
294
  });
294
295
  }
295
296
  /**
296
- * Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
297
+ * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm
297
298
  * and speed up initial SSR requests.
298
299
  *
299
300
  * @private
@@ -301,7 +302,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
301
302
  createPingRule() {
302
303
  new aws_events_1.Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {
303
304
  ruleName: `${this.resourceIdPrefix}-pinger`,
304
- description: `Pings the lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,
305
+ description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,
305
306
  enabled: true,
306
307
  schedule: aws_events_1.Schedule.rate(aws_cdk_lib_1.Duration.minutes(5)),
307
308
  targets: [new aws_events_targets_1.LambdaFunction(this.lambdaFunction)],
@@ -309,4 +310,4 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
309
310
  }
310
311
  }
311
312
  exports.NuxtAppStack = NuxtAppStack;
312
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-app-stack.js","sourceRoot":"","sources":["nuxt-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution,\n    ICachePolicy,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, LayerVersion, Runtime} from \"aws-cdk-lib/aws-lambda\";\nimport {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {HttpMethod} from \"aws-cdk-lib/aws-stepfunctions-tasks\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {HttpLambdaIntegration} from '@aws-cdk/aws-apigatewayv2-integrations-alpha';\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"@aws-cdk/aws-apigatewayv2-alpha\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"./nuxt-app-static-assets\";\nimport {AppStackProps} from \"./app-stack-props\";\nimport * as fs from \"fs\";\nimport {Rule, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport {NuxtConfig} from \"./nuxt-config\";\n\n/**\n * Defines the props required for the {@see NuxtAppStack}.\n */\nexport interface NuxtAppStackProps extends AppStackProps {\n    /**\n     * The domain (without the protocol) at which the Nuxt app shall be publicly available.\n     * A DNS record will be automatically created in Route53 for the domain.\n     * This also supports subdomains.\n     * Examples: \"example.com\", \"sub.example.com\"\n     */\n    readonly domain: string;\n\n    /**\n     * The id of the hosted zone to create a DNS record for the specified domain.\n     */\n    readonly hostedZoneId: string;\n\n    /**\n     * The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.\n     * The certificate must be issued for the specified domain in us-east-1 (global) regardless of the\n     * region specified via 'env.region' as CloudFront only works globally.\n     */\n    readonly globalTlsCertificateArn: string;\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app on server side rendering.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The nuxt.config.js of the Nuxt app.\n     */\n    readonly nuxtConfig: NuxtConfig;\n}\n\n/**\n * Creates a lambda function that renders the Nuxt app and is publicly reachable via a specified domain.\n */\nexport class NuxtAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used as S3 folder name\n     * to store the static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly lambdaFunction: Function;\n\n    /**\n     * The API gateway to make the lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.deploymentRevision = new Date().toISOString();\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.lambdaFunction = this.createLambdaFunction();\n        this.apiGateway = this.createApiGateway(props);\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createPingRule();\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            accessControl: BucketAccessControl.AUTHENTICATED_READ,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates a lambda layer with the node_modules required to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private createSsrLambdaLayer(): LayerVersion {\n        const layerName = `${this.resourceIdPrefix}-ssr-layer`;\n        return new LayerVersion(this, layerName, {\n            layerVersionName: layerName,\n            code: Code.fromAsset('.nuxt/cdk-deployment/layer'),\n            compatibleRuntimes: [Runtime.NODEJS_12_X],\n            description: `Provides the node_modules required for SSR of ${this.resourceIdPrefix}.`,\n        });\n    }\n\n    /**\n     * Creates the lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createLambdaFunction(): Function {\n        const funcName = `${this.resourceIdPrefix}-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_12_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.createSsrLambdaLayer()],\n            handler: 'index.handler',\n            code: Code.fromAsset('.nuxt/cdk-deployment/src', {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: 512,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false\n        });\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.lambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [HttpMethod.GET, HttpMethod.HEAD],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: `${this.resourceIdPrefix}-redirect`,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.createStaticAssetsRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(2),\n                readTimeout: Duration.seconds(10),\n                protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n            }),\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.createSsrCachePolicy(),\n        };\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     * Eventhough we don't want to cache SSR requests, we still have to create this cache policy in order to\n     * forward required cookies, query params and headers. This doesn't make any sense, because if nothing\n     * is cached, one would expect, that anything would/could be forwarded, but anyway...\n     */\n    private createSsrCachePolicy(): ICachePolicy {\n\n        // The headers to make accessible in the Nuxt app code.\n        // There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.\n        const headers = [\n            'User-Agent', // Required to distinguish between mobile and desktop template\n            'Authorization', // For authorization\n            'Host' // To access the domain name on SSR requests\n        ];\n\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Passes all required request data to the ${this.resourceIdPrefix} origin.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.seconds(1), // The max TTL must not be 0 for a cache policy\n            queryStringBehavior: CacheQueryStringBehavior.all(),\n            headerBehavior: CacheHeaderBehavior.allowList(...headers),\n            cookieBehavior: CacheCookieBehavior.all(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n                originPath: this.deploymentRevision,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the lambda function in the {@see NuxtAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: this.deploymentRevision + asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createPingRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.lambdaFunction)],\n        });\n    }\n}\n"]}
313
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-app-stack.js","sourceRoot":"","sources":["nuxt-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution,\n    ICachePolicy,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, LayerVersion, Runtime} from \"aws-cdk-lib/aws-lambda\";\nimport {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {HttpMethod} from \"aws-cdk-lib/aws-stepfunctions-tasks\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {HttpLambdaIntegration} from '@aws-cdk/aws-apigatewayv2-integrations-alpha';\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"@aws-cdk/aws-apigatewayv2-alpha\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"./nuxt-app-static-assets\";\nimport {AppStackProps} from \"./app-stack-props\";\nimport * as fs from \"fs\";\nimport {Rule, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport {NuxtConfig} from \"./nuxt-config\";\n\n/**\n * Defines the props required for the {@see NuxtAppStack}.\n */\nexport interface NuxtAppStackProps extends AppStackProps {\n    /**\n     * The domain (without the protocol) at which the Nuxt app shall be publicly available.\n     * A DNS record will be automatically created in Route53 for the domain.\n     * This also supports subdomains.\n     * Examples: \"example.com\", \"sub.example.com\"\n     */\n    readonly domain: string;\n\n    /**\n     * The id of the hosted zone to create a DNS record for the specified domain.\n     */\n    readonly hostedZoneId: string;\n\n    /**\n     * The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.\n     * The certificate must be issued for the specified domain in us-east-1 (global) regardless of the\n     * region specified via 'env.region' as CloudFront only works globally.\n     */\n    readonly globalTlsCertificateArn: string;\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The nuxt.config.js of the Nuxt app.\n     */\n    readonly nuxtConfig: NuxtConfig;\n\n    /**\n     * The memory size to apply to the Nuxt app's Lambda.\n     * Defaults to 512MB.\n     */\n    readonly memorySize?: number;\n}\n\n/**\n * Creates a Lambda function that renders the Nuxt app and is publicly reachable via a specified domain.\n */\nexport class NuxtAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used as S3 folder name\n     * to store the static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly lambdaFunction: Function;\n\n    /**\n     * The API gateway to make the Lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.deploymentRevision = new Date().toISOString();\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.lambdaFunction = this.createLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createPingRule();\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            accessControl: BucketAccessControl.AUTHENTICATED_READ,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private createSsrLambdaLayer(): LayerVersion {\n        const layerName = `${this.resourceIdPrefix}-ssr-layer`;\n        return new LayerVersion(this, layerName, {\n            layerVersionName: layerName,\n            code: Code.fromAsset('.nuxt/cdk-deployment/layer'),\n            compatibleRuntimes: [Runtime.NODEJS_12_X],\n            description: `Provides the node_modules required for SSR of ${this.resourceIdPrefix}.`,\n        });\n    }\n\n    /**\n     * Creates the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createLambdaFunction(props: NuxtAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_12_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.createSsrLambdaLayer()],\n            handler: 'index.handler',\n            code: Code.fromAsset('.nuxt/cdk-deployment/src', {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: props.memorySize ?? 512,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false\n        });\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render Lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.lambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [HttpMethod.GET, HttpMethod.HEAD],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: `${this.resourceIdPrefix}-redirect`,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.createStaticAssetsRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(2),\n                readTimeout: Duration.seconds(10),\n                protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n            }),\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.createSsrCachePolicy(),\n        };\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     * Eventhough we don't want to cache SSR requests, we still have to create this cache policy in order to\n     * forward required cookies, query params and headers. This doesn't make any sense, because if nothing\n     * is cached, one would expect, that anything would/could be forwarded, but anyway...\n     */\n    private createSsrCachePolicy(): ICachePolicy {\n\n        // The headers to make accessible in the Nuxt app code.\n        // There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.\n        const headers = [\n            'User-Agent', // Required to distinguish between mobile and desktop template\n            'Authorization', // For authorization\n            'Host' // To access the domain name on SSR requests\n        ];\n\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Passes all required request data to the ${this.resourceIdPrefix} origin.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.seconds(1), // The max TTL must not be 0 for a cache policy\n            queryStringBehavior: CacheQueryStringBehavior.all(),\n            headerBehavior: CacheHeaderBehavior.allowList(...headers),\n            cookieBehavior: CacheCookieBehavior.all(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n                originPath: this.deploymentRevision,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: this.deploymentRevision + asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createPingRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.lambdaFunction)],\n        });\n    }\n}\n"]}
@@ -61,7 +61,7 @@ export interface NuxtAppStackProps extends AppStackProps {
61
61
 
62
62
  /**
63
63
  * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain
64
- * and to provide the custom domain to the Nuxt app on server side rendering.
64
+ * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.
65
65
  * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
66
66
  */
67
67
  readonly regionalTlsCertificateArn: string;
@@ -70,10 +70,16 @@ export interface NuxtAppStackProps extends AppStackProps {
70
70
  * The nuxt.config.js of the Nuxt app.
71
71
  */
72
72
  readonly nuxtConfig: NuxtConfig;
73
+
74
+ /**
75
+ * The memory size to apply to the Nuxt app's Lambda.
76
+ * Defaults to 512MB.
77
+ */
78
+ readonly memorySize?: number;
73
79
  }
74
80
 
75
81
  /**
76
- * Creates a lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
82
+ * Creates a Lambda function that renders the Nuxt app and is publicly reachable via a specified domain.
77
83
  */
78
84
  export class NuxtAppStack extends Stack {
79
85
 
@@ -105,14 +111,14 @@ export class NuxtAppStack extends Stack {
105
111
  public staticAssetsBucket: IBucket;
106
112
 
107
113
  /**
108
- * The lambda function to render the Nuxt app on the server side.
114
+ * The Lambda function to render the Nuxt app on the server side.
109
115
  *
110
116
  * @private
111
117
  */
112
118
  private readonly lambdaFunction: Function;
113
119
 
114
120
  /**
115
- * The API gateway to make the lambda function to render the Nuxt app publicly available.
121
+ * The API gateway to make the Lambda function to render the Nuxt app publicly available.
116
122
  *
117
123
  * @private
118
124
  */
@@ -126,7 +132,7 @@ export class NuxtAppStack extends Stack {
126
132
  private staticAssetConfigs: StaticAssetConfig[];
127
133
 
128
134
  /**
129
- * The CloudFront distribution to route incoming requests to the Nuxt lambda function (via the API gateway)
135
+ * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)
130
136
  * or the S3 assets folder (with caching).
131
137
  *
132
138
  * @private
@@ -141,7 +147,7 @@ export class NuxtAppStack extends Stack {
141
147
  this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);
142
148
  this.cdnAccessIdentity = this.createCdnAccessIdentity();
143
149
  this.staticAssetsBucket = this.createStaticAssetsBucket();
144
- this.lambdaFunction = this.createLambdaFunction();
150
+ this.lambdaFunction = this.createLambdaFunction(props);
145
151
  this.apiGateway = this.createApiGateway(props);
146
152
  this.cdn = this.createCloudFrontDistribution(props);
147
153
  this.configureDeployments();
@@ -181,7 +187,7 @@ export class NuxtAppStack extends Stack {
181
187
  }
182
188
 
183
189
  /**
184
- * Creates a lambda layer with the node_modules required to render the Nuxt app on the server side.
190
+ * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.
185
191
  *
186
192
  * @private
187
193
  */
@@ -196,11 +202,11 @@ export class NuxtAppStack extends Stack {
196
202
  }
197
203
 
198
204
  /**
199
- * Creates the lambda function to render the Nuxt app.
205
+ * Creates the Lambda function to render the Nuxt app.
200
206
  *
201
207
  * @private
202
208
  */
203
- private createLambdaFunction(): Function {
209
+ private createLambdaFunction(props: NuxtAppStackProps): Function {
204
210
  const funcName = `${this.resourceIdPrefix}-function`;
205
211
 
206
212
  return new Function(this, funcName, {
@@ -214,14 +220,14 @@ export class NuxtAppStack extends Stack {
214
220
  exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],
215
221
  }),
216
222
  timeout: Duration.seconds(10),
217
- memorySize: 512,
223
+ memorySize: props.memorySize ?? 512,
218
224
  logRetention: RetentionDays.ONE_MONTH,
219
225
  allowPublicSubnet: false
220
226
  });
221
227
  }
222
228
 
223
229
  /**
224
- * Creates the API gateway to make the Nuxt app render lambda function publicly available.
230
+ * Creates the API gateway to make the Nuxt app render Lambda function publicly available.
225
231
  *
226
232
  * @private
227
233
  */
@@ -241,7 +247,7 @@ export class NuxtAppStack extends Stack {
241
247
 
242
248
  const apiGateway = new HttpApi(this, apiName, {
243
249
  apiName,
244
- description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} lambda function to make it publicly available.`,
250
+ description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,
245
251
  // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere
246
252
  corsPreflight: undefined,
247
253
  defaultIntegration: lambdaIntegration,
@@ -260,7 +266,7 @@ export class NuxtAppStack extends Stack {
260
266
  }
261
267
 
262
268
  /**
263
- * Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
269
+ * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)
264
270
  * or the S3 assets folder (with caching).
265
271
  *
266
272
  * @param props
@@ -281,7 +287,7 @@ export class NuxtAppStack extends Stack {
281
287
  }
282
288
 
283
289
  /**
284
- * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
290
+ * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).
285
291
  * Additionally, this automatically redirects HTTP requests to HTTPS.
286
292
  *
287
293
  * @private
@@ -365,7 +371,7 @@ export class NuxtAppStack extends Stack {
365
371
  * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.
366
372
  * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.
367
373
  * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up
368
- * after a specified period of time via the lambda function in the {@see NuxtAppAssetsCleanupStack}.
374
+ * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.
369
375
  */
370
376
  private configureDeployments(): BucketDeployment[] {
371
377
  const defaultCacheConfig = [
@@ -433,7 +439,7 @@ export class NuxtAppStack extends Stack {
433
439
  }
434
440
 
435
441
  /**
436
- * Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
442
+ * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm
437
443
  * and speed up initial SSR requests.
438
444
  *
439
445
  * @private
@@ -441,7 +447,7 @@ export class NuxtAppStack extends Stack {
441
447
  private createPingRule(): void {
442
448
  new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {
443
449
  ruleName: `${this.resourceIdPrefix}-pinger`,
444
- description: `Pings the lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,
450
+ description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,
445
451
  enabled: true,
446
452
  schedule: Schedule.rate(Duration.minutes(5)),
447
453
  targets: [new LambdaFunction(this.lambdaFunction)],
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cdk-nuxt",
3
- "version": "0.4.2",
3
+ "version": "0.4.3",
4
4
  "license": "MIT",
5
5
  "files": [
6
6
  "lib",