cdk-nuxt 0.4.1 → 0.4.2
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 +2 -2
- package/lib/stack/nuxt-app-stack.d.ts +17 -24
- package/lib/stack/nuxt-app-stack.js +15 -25
- package/lib/stack/nuxt-app-stack.ts +25 -36
- package/lib/templates/stack-index.ts +10 -2
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -28,7 +28,7 @@ Easily deploy a dynamic universal Nuxt application via CDK on AWS including the
|
|
|
28
28
|
```bash
|
|
29
29
|
yarn add cdk-nuxt --dev # The package itself
|
|
30
30
|
yarn add ts-node typescript --dev # To compile the CDK stacks via typescript
|
|
31
|
-
yarn add aws-cdk@2.
|
|
31
|
+
yarn add aws-cdk@2.15.0 --dev # CDK cli with this exact version for the deployment
|
|
32
32
|
yarn add nuxt-aws-lambda nuxt-start # To make the Nuxt app renderable via AWS Lambda
|
|
33
33
|
```
|
|
34
34
|
|
|
@@ -42,7 +42,7 @@ After the installation steps the `package.json` file should look something like
|
|
|
42
42
|
{
|
|
43
43
|
"name": "nuxt-app",
|
|
44
44
|
"devDependencies": {
|
|
45
|
-
"aws-cdk": "2.
|
|
45
|
+
"aws-cdk": "2.15.0",
|
|
46
46
|
"cdk-nuxt": "^X.X.X",
|
|
47
47
|
"nuxt": "^X.X.X",
|
|
48
48
|
"ts-node": "^X.X.X",
|
|
@@ -19,11 +19,17 @@ export interface NuxtAppStackProps extends AppStackProps {
|
|
|
19
19
|
*/
|
|
20
20
|
readonly hostedZoneId: string;
|
|
21
21
|
/**
|
|
22
|
-
* The ARN of the certificate to use for the Nuxt app to make it accessible via HTTPS.
|
|
22
|
+
* The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.
|
|
23
23
|
* The certificate must be issued for the specified domain in us-east-1 (global) regardless of the
|
|
24
|
-
* region
|
|
24
|
+
* region specified via 'env.region' as CloudFront only works globally.
|
|
25
25
|
*/
|
|
26
26
|
readonly globalTlsCertificateArn: string;
|
|
27
|
+
/**
|
|
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.
|
|
30
|
+
* The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
|
|
31
|
+
*/
|
|
32
|
+
readonly regionalTlsCertificateArn: string;
|
|
27
33
|
/**
|
|
28
34
|
* The nuxt.config.js of the Nuxt app.
|
|
29
35
|
*/
|
|
@@ -46,12 +52,6 @@ export declare class NuxtAppStack extends Stack {
|
|
|
46
52
|
* @private
|
|
47
53
|
*/
|
|
48
54
|
private readonly deploymentRevision;
|
|
49
|
-
/**
|
|
50
|
-
* The certificate to use for the Nuxt app to make it accessible via HTTPS.
|
|
51
|
-
*
|
|
52
|
-
* @private
|
|
53
|
-
*/
|
|
54
|
-
private readonly tlsCertificate;
|
|
55
55
|
/**
|
|
56
56
|
* The identity to use for accessing the deployment assets on S3.
|
|
57
57
|
*
|
|
@@ -81,7 +81,7 @@ export declare class NuxtAppStack extends Stack {
|
|
|
81
81
|
*/
|
|
82
82
|
private staticAssetConfigs;
|
|
83
83
|
/**
|
|
84
|
-
* The
|
|
84
|
+
* The CloudFront distribution to route incoming requests to the Nuxt lambda function (via the API gateway)
|
|
85
85
|
* or the S3 assets folder (with caching).
|
|
86
86
|
*
|
|
87
87
|
* @private
|
|
@@ -89,14 +89,7 @@ export declare class NuxtAppStack extends Stack {
|
|
|
89
89
|
private readonly cdn;
|
|
90
90
|
constructor(scope: Construct, id: string, props: NuxtAppStackProps);
|
|
91
91
|
/**
|
|
92
|
-
*
|
|
93
|
-
*
|
|
94
|
-
* @param props
|
|
95
|
-
* @private
|
|
96
|
-
*/
|
|
97
|
-
private findTlsCertificate;
|
|
98
|
-
/**
|
|
99
|
-
* Creates the identity to access our S3 deployment asset files via the cloudfront distribution.
|
|
92
|
+
* Creates the identity to access the S3 deployment asset files via the CloudFront distribution.
|
|
100
93
|
*
|
|
101
94
|
* @private
|
|
102
95
|
*/
|
|
@@ -126,7 +119,7 @@ export declare class NuxtAppStack extends Stack {
|
|
|
126
119
|
*/
|
|
127
120
|
private createApiGateway;
|
|
128
121
|
/**
|
|
129
|
-
* Creates the
|
|
122
|
+
* Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
|
|
130
123
|
* or the S3 assets folder (with caching).
|
|
131
124
|
*
|
|
132
125
|
* @param props
|
|
@@ -134,21 +127,21 @@ export declare class NuxtAppStack extends Stack {
|
|
|
134
127
|
*/
|
|
135
128
|
private createCloudFrontDistribution;
|
|
136
129
|
/**
|
|
137
|
-
* Creates a behavior for the
|
|
130
|
+
* Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
|
|
138
131
|
* Additionally, this automatically redirects HTTP requests to HTTPS.
|
|
139
132
|
*
|
|
140
133
|
* @private
|
|
141
134
|
*/
|
|
142
135
|
private createNuxtAppRouteBehavior;
|
|
143
136
|
/**
|
|
144
|
-
* Creates a cache policy for the Nuxt app route behavior of
|
|
137
|
+
* Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.
|
|
145
138
|
* Eventhough we don't want to cache SSR requests, we still have to create this cache policy in order to
|
|
146
139
|
* forward required cookies, query params and headers. This doesn't make any sense, because if nothing
|
|
147
140
|
* is cached, one would expect, that anything would/could be forwarded, but anyway...
|
|
148
141
|
*/
|
|
149
142
|
private createSsrCachePolicy;
|
|
150
143
|
/**
|
|
151
|
-
* Creates a behavior for the
|
|
144
|
+
* Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets
|
|
152
145
|
* to the S3 bucket that holds these static assets.
|
|
153
146
|
*
|
|
154
147
|
* @private
|
|
@@ -162,21 +155,21 @@ export declare class NuxtAppStack extends Stack {
|
|
|
162
155
|
*/
|
|
163
156
|
private configureDeployments;
|
|
164
157
|
/**
|
|
165
|
-
* Resolves the hosted zone at which the DNS records shall be created to access
|
|
158
|
+
* Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.
|
|
166
159
|
*
|
|
167
160
|
* @param props
|
|
168
161
|
* @private
|
|
169
162
|
*/
|
|
170
163
|
private findHostedZone;
|
|
171
164
|
/**
|
|
172
|
-
* Creates the DNS records to access
|
|
165
|
+
* Creates the DNS records to access the Nuxt app on the internet via the custom domain.
|
|
173
166
|
*
|
|
174
167
|
* @param props
|
|
175
168
|
* @private
|
|
176
169
|
*/
|
|
177
170
|
private createDnsRecords;
|
|
178
171
|
/**
|
|
179
|
-
* Creates a scheduled rule to ping
|
|
172
|
+
* Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
|
|
180
173
|
* and speed up initial SSR requests.
|
|
181
174
|
*
|
|
182
175
|
* @private
|
|
@@ -27,7 +27,6 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
27
27
|
this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;
|
|
28
28
|
this.deploymentRevision = new Date().toISOString();
|
|
29
29
|
this.staticAssetConfigs = (0, nuxt_app_static_assets_1.getNuxtAppStaticAssetConfigs)(props.nuxtConfig);
|
|
30
|
-
this.tlsCertificate = this.findTlsCertificate(props);
|
|
31
30
|
this.cdnAccessIdentity = this.createCdnAccessIdentity();
|
|
32
31
|
this.staticAssetsBucket = this.createStaticAssetsBucket();
|
|
33
32
|
this.lambdaFunction = this.createLambdaFunction();
|
|
@@ -38,16 +37,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
38
37
|
this.createPingRule();
|
|
39
38
|
}
|
|
40
39
|
/**
|
|
41
|
-
*
|
|
42
|
-
*
|
|
43
|
-
* @param props
|
|
44
|
-
* @private
|
|
45
|
-
*/
|
|
46
|
-
findTlsCertificate(props) {
|
|
47
|
-
return aws_certificatemanager_1.Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-tls-certificate`, props.globalTlsCertificateArn);
|
|
48
|
-
}
|
|
49
|
-
/**
|
|
50
|
-
* Creates the identity to access our S3 deployment asset files via the cloudfront distribution.
|
|
40
|
+
* Creates the identity to access the S3 deployment asset files via the CloudFront distribution.
|
|
51
41
|
*
|
|
52
42
|
* @private
|
|
53
43
|
*/
|
|
@@ -119,17 +109,17 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
119
109
|
const apiName = `${this.resourceIdPrefix}-api`;
|
|
120
110
|
const lambdaIntegration = new aws_apigatewayv2_integrations_alpha_1.HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.lambdaFunction);
|
|
121
111
|
// We want the API gateway to be accessible by the custom domain name.
|
|
122
|
-
// Even though we access the gateway via
|
|
123
|
-
// to be able to redirect the original 'Host' header to
|
|
112
|
+
// Even though we access the gateway via CloudFront (for auto http to https redirects), this is required
|
|
113
|
+
// to be able to redirect the original 'Host' header to the Nuxt application, if requested.
|
|
124
114
|
const domainName = new aws_apigatewayv2_alpha_1.DomainName(this, `${this.resourceIdPrefix}-api-domain`, {
|
|
125
115
|
domainName: props.domain,
|
|
126
|
-
certificate: this.
|
|
116
|
+
certificate: aws_certificatemanager_1.Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),
|
|
127
117
|
endpointType: aws_apigatewayv2_alpha_1.EndpointType.REGIONAL,
|
|
128
118
|
securityPolicy: aws_apigatewayv2_alpha_1.SecurityPolicy.TLS_1_2
|
|
129
119
|
});
|
|
130
120
|
const apiGateway = new aws_apigatewayv2_alpha_1.HttpApi(this, apiName, {
|
|
131
121
|
apiName,
|
|
132
|
-
description: `Connects the ${this.resourceIdPrefix}
|
|
122
|
+
description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} lambda function to make it publicly available.`,
|
|
133
123
|
// The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere
|
|
134
124
|
corsPreflight: undefined,
|
|
135
125
|
defaultIntegration: lambdaIntegration,
|
|
@@ -145,7 +135,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
145
135
|
return apiGateway;
|
|
146
136
|
}
|
|
147
137
|
/**
|
|
148
|
-
* Creates the
|
|
138
|
+
* Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
|
|
149
139
|
* or the S3 assets folder (with caching).
|
|
150
140
|
*
|
|
151
141
|
* @param props
|
|
@@ -157,14 +147,14 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
157
147
|
domainNames: [props.domain],
|
|
158
148
|
comment: `${this.resourceIdPrefix}-redirect`,
|
|
159
149
|
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2018,
|
|
160
|
-
certificate: this.
|
|
150
|
+
certificate: aws_certificatemanager_1.Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),
|
|
161
151
|
defaultBehavior: this.createNuxtAppRouteBehavior(),
|
|
162
152
|
additionalBehaviors: this.createStaticAssetsRouteBehavior(),
|
|
163
153
|
priceClass: aws_cloudfront_1.PriceClass.PRICE_CLASS_100, // Use only North America and Europe
|
|
164
154
|
});
|
|
165
155
|
}
|
|
166
156
|
/**
|
|
167
|
-
* Creates a behavior for the
|
|
157
|
+
* Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
|
|
168
158
|
* Additionally, this automatically redirects HTTP requests to HTTPS.
|
|
169
159
|
*
|
|
170
160
|
* @private
|
|
@@ -185,13 +175,13 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
185
175
|
};
|
|
186
176
|
}
|
|
187
177
|
/**
|
|
188
|
-
* Creates a cache policy for the Nuxt app route behavior of
|
|
178
|
+
* Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.
|
|
189
179
|
* Eventhough we don't want to cache SSR requests, we still have to create this cache policy in order to
|
|
190
180
|
* forward required cookies, query params and headers. This doesn't make any sense, because if nothing
|
|
191
181
|
* is cached, one would expect, that anything would/could be forwarded, but anyway...
|
|
192
182
|
*/
|
|
193
183
|
createSsrCachePolicy() {
|
|
194
|
-
// The headers to make accessible in
|
|
184
|
+
// The headers to make accessible in the Nuxt app code.
|
|
195
185
|
// There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.
|
|
196
186
|
const headers = [
|
|
197
187
|
'User-Agent',
|
|
@@ -212,7 +202,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
212
202
|
});
|
|
213
203
|
}
|
|
214
204
|
/**
|
|
215
|
-
* Creates a behavior for the
|
|
205
|
+
* Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets
|
|
216
206
|
* to the S3 bucket that holds these static assets.
|
|
217
207
|
*
|
|
218
208
|
* @private
|
|
@@ -268,7 +258,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
268
258
|
});
|
|
269
259
|
}
|
|
270
260
|
/**
|
|
271
|
-
* Resolves the hosted zone at which the DNS records shall be created to access
|
|
261
|
+
* Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.
|
|
272
262
|
*
|
|
273
263
|
* @param props
|
|
274
264
|
* @private
|
|
@@ -281,7 +271,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
281
271
|
});
|
|
282
272
|
}
|
|
283
273
|
/**
|
|
284
|
-
* Creates the DNS records to access
|
|
274
|
+
* Creates the DNS records to access the Nuxt app on the internet via the custom domain.
|
|
285
275
|
*
|
|
286
276
|
* @param props
|
|
287
277
|
* @private
|
|
@@ -303,7 +293,7 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
303
293
|
});
|
|
304
294
|
}
|
|
305
295
|
/**
|
|
306
|
-
* Creates a scheduled rule to ping
|
|
296
|
+
* Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
|
|
307
297
|
* and speed up initial SSR requests.
|
|
308
298
|
*
|
|
309
299
|
* @private
|
|
@@ -319,4 +309,4 @@ class NuxtAppStack extends aws_cdk_lib_1.Stack {
|
|
|
319
309
|
}
|
|
320
310
|
}
|
|
321
311
|
exports.NuxtAppStack = NuxtAppStack;
|
|
322
|
-
//# 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, ICertificate} 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 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 used for the Nuxt app itself.\n     */\n    readonly globalTlsCertificateArn: 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 certificate to use for the Nuxt app to make it accessible via HTTPS.\n     *\n     * @private\n     */\n    private readonly tlsCertificate: ICertificate;\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.tlsCertificate = this.findTlsCertificate(props);\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     * Finds the certificate to use for providing HTTPS requests to our Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private findTlsCertificate(props: NuxtAppStackProps): ICertificate {\n        return Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-tls-certificate`, props.globalTlsCertificateArn);\n    }\n\n    /**\n     * Creates the identity to access our 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 our Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: this.tlsCertificate,\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: this.tlsCertificate,\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 our 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 our 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 our 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 our 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 our Nuxt app on the internet via our 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 our 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"]}
|
|
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"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';
|
|
2
2
|
import {Construct} from 'constructs';
|
|
3
|
-
import {Certificate
|
|
3
|
+
import {Certificate} from "aws-cdk-lib/aws-certificatemanager";
|
|
4
4
|
import {
|
|
5
5
|
AllowedMethods,
|
|
6
6
|
BehaviorOptions,
|
|
@@ -53,12 +53,19 @@ export interface NuxtAppStackProps extends AppStackProps {
|
|
|
53
53
|
readonly hostedZoneId: string;
|
|
54
54
|
|
|
55
55
|
/**
|
|
56
|
-
* The ARN of the certificate to use for the Nuxt app to make it accessible via HTTPS.
|
|
56
|
+
* The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.
|
|
57
57
|
* The certificate must be issued for the specified domain in us-east-1 (global) regardless of the
|
|
58
|
-
* region
|
|
58
|
+
* region specified via 'env.region' as CloudFront only works globally.
|
|
59
59
|
*/
|
|
60
60
|
readonly globalTlsCertificateArn: string;
|
|
61
61
|
|
|
62
|
+
/**
|
|
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.
|
|
65
|
+
* The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
|
|
66
|
+
*/
|
|
67
|
+
readonly regionalTlsCertificateArn: string;
|
|
68
|
+
|
|
62
69
|
/**
|
|
63
70
|
* The nuxt.config.js of the Nuxt app.
|
|
64
71
|
*/
|
|
@@ -85,13 +92,6 @@ export class NuxtAppStack extends Stack {
|
|
|
85
92
|
*/
|
|
86
93
|
private readonly deploymentRevision: string;
|
|
87
94
|
|
|
88
|
-
/**
|
|
89
|
-
* The certificate to use for the Nuxt app to make it accessible via HTTPS.
|
|
90
|
-
*
|
|
91
|
-
* @private
|
|
92
|
-
*/
|
|
93
|
-
private readonly tlsCertificate: ICertificate;
|
|
94
|
-
|
|
95
95
|
/**
|
|
96
96
|
* The identity to use for accessing the deployment assets on S3.
|
|
97
97
|
*
|
|
@@ -126,7 +126,7 @@ export class NuxtAppStack extends Stack {
|
|
|
126
126
|
private staticAssetConfigs: StaticAssetConfig[];
|
|
127
127
|
|
|
128
128
|
/**
|
|
129
|
-
* The
|
|
129
|
+
* The CloudFront distribution to route incoming requests to the Nuxt lambda function (via the API gateway)
|
|
130
130
|
* or the S3 assets folder (with caching).
|
|
131
131
|
*
|
|
132
132
|
* @private
|
|
@@ -139,7 +139,6 @@ export class NuxtAppStack extends Stack {
|
|
|
139
139
|
this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;
|
|
140
140
|
this.deploymentRevision = new Date().toISOString();
|
|
141
141
|
this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);
|
|
142
|
-
this.tlsCertificate = this.findTlsCertificate(props);
|
|
143
142
|
this.cdnAccessIdentity = this.createCdnAccessIdentity();
|
|
144
143
|
this.staticAssetsBucket = this.createStaticAssetsBucket();
|
|
145
144
|
this.lambdaFunction = this.createLambdaFunction();
|
|
@@ -151,17 +150,7 @@ export class NuxtAppStack extends Stack {
|
|
|
151
150
|
}
|
|
152
151
|
|
|
153
152
|
/**
|
|
154
|
-
*
|
|
155
|
-
*
|
|
156
|
-
* @param props
|
|
157
|
-
* @private
|
|
158
|
-
*/
|
|
159
|
-
private findTlsCertificate(props: NuxtAppStackProps): ICertificate {
|
|
160
|
-
return Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-tls-certificate`, props.globalTlsCertificateArn);
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Creates the identity to access our S3 deployment asset files via the cloudfront distribution.
|
|
153
|
+
* Creates the identity to access the S3 deployment asset files via the CloudFront distribution.
|
|
165
154
|
*
|
|
166
155
|
* @private
|
|
167
156
|
*/
|
|
@@ -241,18 +230,18 @@ export class NuxtAppStack extends Stack {
|
|
|
241
230
|
const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.lambdaFunction);
|
|
242
231
|
|
|
243
232
|
// We want the API gateway to be accessible by the custom domain name.
|
|
244
|
-
// Even though we access the gateway via
|
|
245
|
-
// to be able to redirect the original 'Host' header to
|
|
233
|
+
// Even though we access the gateway via CloudFront (for auto http to https redirects), this is required
|
|
234
|
+
// to be able to redirect the original 'Host' header to the Nuxt application, if requested.
|
|
246
235
|
const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {
|
|
247
236
|
domainName: props.domain,
|
|
248
|
-
certificate: this.
|
|
237
|
+
certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),
|
|
249
238
|
endpointType: EndpointType.REGIONAL,
|
|
250
239
|
securityPolicy: SecurityPolicy.TLS_1_2
|
|
251
240
|
});
|
|
252
241
|
|
|
253
242
|
const apiGateway = new HttpApi(this, apiName, {
|
|
254
243
|
apiName,
|
|
255
|
-
description: `Connects the ${this.resourceIdPrefix}
|
|
244
|
+
description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} lambda function to make it publicly available.`,
|
|
256
245
|
// The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere
|
|
257
246
|
corsPreflight: undefined,
|
|
258
247
|
defaultIntegration: lambdaIntegration,
|
|
@@ -271,7 +260,7 @@ export class NuxtAppStack extends Stack {
|
|
|
271
260
|
}
|
|
272
261
|
|
|
273
262
|
/**
|
|
274
|
-
* Creates the
|
|
263
|
+
* Creates the CloudFront distribution that routes incoming requests to the Nuxt lambda function (via the API gateway)
|
|
275
264
|
* or the S3 assets folder (with caching).
|
|
276
265
|
*
|
|
277
266
|
* @param props
|
|
@@ -284,7 +273,7 @@ export class NuxtAppStack extends Stack {
|
|
|
284
273
|
domainNames: [props.domain],
|
|
285
274
|
comment: `${this.resourceIdPrefix}-redirect`,
|
|
286
275
|
minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,
|
|
287
|
-
certificate: this.
|
|
276
|
+
certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),
|
|
288
277
|
defaultBehavior: this.createNuxtAppRouteBehavior(),
|
|
289
278
|
additionalBehaviors: this.createStaticAssetsRouteBehavior(),
|
|
290
279
|
priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe
|
|
@@ -292,7 +281,7 @@ export class NuxtAppStack extends Stack {
|
|
|
292
281
|
}
|
|
293
282
|
|
|
294
283
|
/**
|
|
295
|
-
* Creates a behavior for the
|
|
284
|
+
* Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render lambda function (via API gateway).
|
|
296
285
|
* Additionally, this automatically redirects HTTP requests to HTTPS.
|
|
297
286
|
*
|
|
298
287
|
* @private
|
|
@@ -314,14 +303,14 @@ export class NuxtAppStack extends Stack {
|
|
|
314
303
|
}
|
|
315
304
|
|
|
316
305
|
/**
|
|
317
|
-
* Creates a cache policy for the Nuxt app route behavior of
|
|
306
|
+
* Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.
|
|
318
307
|
* Eventhough we don't want to cache SSR requests, we still have to create this cache policy in order to
|
|
319
308
|
* forward required cookies, query params and headers. This doesn't make any sense, because if nothing
|
|
320
309
|
* is cached, one would expect, that anything would/could be forwarded, but anyway...
|
|
321
310
|
*/
|
|
322
311
|
private createSsrCachePolicy(): ICachePolicy {
|
|
323
312
|
|
|
324
|
-
// The headers to make accessible in
|
|
313
|
+
// The headers to make accessible in the Nuxt app code.
|
|
325
314
|
// There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.
|
|
326
315
|
const headers = [
|
|
327
316
|
'User-Agent', // Required to distinguish between mobile and desktop template
|
|
@@ -344,7 +333,7 @@ export class NuxtAppStack extends Stack {
|
|
|
344
333
|
}
|
|
345
334
|
|
|
346
335
|
/**
|
|
347
|
-
* Creates a behavior for the
|
|
336
|
+
* Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets
|
|
348
337
|
* to the S3 bucket that holds these static assets.
|
|
349
338
|
*
|
|
350
339
|
* @private
|
|
@@ -404,7 +393,7 @@ export class NuxtAppStack extends Stack {
|
|
|
404
393
|
}
|
|
405
394
|
|
|
406
395
|
/**
|
|
407
|
-
* Resolves the hosted zone at which the DNS records shall be created to access
|
|
396
|
+
* Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.
|
|
408
397
|
*
|
|
409
398
|
* @param props
|
|
410
399
|
* @private
|
|
@@ -419,7 +408,7 @@ export class NuxtAppStack extends Stack {
|
|
|
419
408
|
}
|
|
420
409
|
|
|
421
410
|
/**
|
|
422
|
-
* Creates the DNS records to access
|
|
411
|
+
* Creates the DNS records to access the Nuxt app on the internet via the custom domain.
|
|
423
412
|
*
|
|
424
413
|
* @param props
|
|
425
414
|
* @private
|
|
@@ -444,7 +433,7 @@ export class NuxtAppStack extends Stack {
|
|
|
444
433
|
}
|
|
445
434
|
|
|
446
435
|
/**
|
|
447
|
-
* Creates a scheduled rule to ping
|
|
436
|
+
* Creates a scheduled rule to ping the Nuxt app lambda function every 5 minutes in order to keep it warm
|
|
448
437
|
* and speed up initial SSR requests.
|
|
449
438
|
*
|
|
450
439
|
* @private
|
|
@@ -30,9 +30,17 @@ const appStackProps: NuxtAppStackProps = {
|
|
|
30
30
|
...commonProps,
|
|
31
31
|
// The domain (without the protocol) at which the Nuxt app shall be publicly available.
|
|
32
32
|
domain: 'example.com',
|
|
33
|
-
|
|
34
|
-
// The
|
|
33
|
+
|
|
34
|
+
// The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.
|
|
35
|
+
// The certificate must be issued for the specified domain in us-east-1 (global) regardless of the
|
|
36
|
+
// region specified via 'env.region' as CloudFront only works globally.
|
|
35
37
|
globalTlsCertificateArn: 'arn:aws:acm:us-east-1:XXXXXXXXXX:certificate/XXXXXXXXXXXXXXXXX',
|
|
38
|
+
|
|
39
|
+
// The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain
|
|
40
|
+
// and to provide the custom domain to the Nuxt app on server side rendering.
|
|
41
|
+
// The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
|
|
42
|
+
regionalTlsCertificateArn: 'arn:aws:acm:eu-central-1:XXXXXXXXXX:certificate/XXXXXXXXXXXXXXXXX',
|
|
43
|
+
|
|
36
44
|
// The id of the hosted zone to create a DNS record for the specified domain.
|
|
37
45
|
hostedZoneId: 'XXXXXXXXXXXXX',
|
|
38
46
|
|