cdk-nuxt 2.3.0-4 → 2.3.0-5
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 +5 -3
- package/lib/cli/deploy-server.js +0 -11
- package/lib/stack/access-logs-analysis/AccessLogsAnalysis.js +31 -15
- package/lib/stack/access-logs-analysis/AccessLogsAnalysis.ts +32 -17
- package/lib/stack/server/NuxtServerAppStack.js +15 -8
- package/lib/stack/server/NuxtServerAppStack.ts +14 -9
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -177,8 +177,8 @@ See https://docs.aws.amazon.com/cdk/v2/guide/getting_started.html for details.
|
|
|
177
177
|
|
|
178
178
|
### 2. Build and Deploy
|
|
179
179
|
|
|
180
|
-
By running the following script, the
|
|
181
|
-
|
|
180
|
+
By running the following script, the Nuxt app will be built automatically via `yarn build`
|
|
181
|
+
and the CDK stack will be deployed to AWS.
|
|
182
182
|
|
|
183
183
|
```bash
|
|
184
184
|
node_modules/.bin/cdk-nuxt-deploy-server
|
|
@@ -187,7 +187,6 @@ node_modules/.bin/cdk-nuxt-deploy-server
|
|
|
187
187
|
Alternatively, you can run the following commands separately to customize the deployment process:
|
|
188
188
|
|
|
189
189
|
```bash
|
|
190
|
-
yarn install
|
|
191
190
|
yarn build
|
|
192
191
|
yarn cdk deploy --require-approval never --all --app="yarn ts-node stack/index.ts"
|
|
193
192
|
```
|
|
@@ -256,6 +255,9 @@ jobs:
|
|
|
256
255
|
node-version: '20'
|
|
257
256
|
cache: 'yarn'
|
|
258
257
|
|
|
258
|
+
- name: Install dependencies
|
|
259
|
+
run: yarn install --frozen-lockfile # or `yarn install --immutable` for Yarn >= 2
|
|
260
|
+
|
|
259
261
|
- name: Build and deploy to AWS
|
|
260
262
|
run: node_modules/.bin/cdk-nuxt-deploy-server # Or run a customized deployment, see 'Build and Deploy' section
|
|
261
263
|
env:
|
package/lib/cli/deploy-server.js
CHANGED
|
@@ -7,21 +7,10 @@ shell.echo(`${logPrefix}: Starting deployment for dynamic Nuxt app...`);
|
|
|
7
7
|
|
|
8
8
|
const deploymentSourceFolder = '.output/server';
|
|
9
9
|
|
|
10
|
-
const yarnVersion = shell.exec('yarn -v', {silent:true}).stdout
|
|
11
|
-
const isYarn1 = yarnVersion && yarnVersion.startsWith('1.');
|
|
12
|
-
|
|
13
10
|
// Refresh the cdk output folder to have a clean state and prevent persisting outdated outputs
|
|
14
11
|
shell.echo(`${logPrefix}: Deleting outdated CDK files...`);
|
|
15
12
|
shell.rm('-rf', 'cdk.out');
|
|
16
13
|
|
|
17
|
-
// Install the Nuxt app dependencies
|
|
18
|
-
shell.echo(`${logPrefix}: Installing app dependencies...`);
|
|
19
|
-
const yarnInstallCommand = isYarn1 ? 'yarn install --frozen-lockfile' : 'yarn install --immutable';
|
|
20
|
-
if (shell.exec(yarnInstallCommand).code !== 0) {
|
|
21
|
-
shell.echo(`${logPrefix} Error: Installation of app dependencies failed.`);
|
|
22
|
-
shell.exit(1);
|
|
23
|
-
}
|
|
24
|
-
|
|
25
14
|
// Build the Nuxt app
|
|
26
15
|
shell.echo(`${logPrefix}: Building app...`);
|
|
27
16
|
if (shell.exec('yarn build').code !== 0) {
|
|
@@ -93,24 +93,28 @@ class AccessLogsAnalysis extends constructs_1.Construct {
|
|
|
93
93
|
}
|
|
94
94
|
createGroupByDateLayer() {
|
|
95
95
|
const layerVersionName = `${this.resourceIdPrefix}-group-by-date-dependencies`;
|
|
96
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');
|
|
96
97
|
return new aws_lambda_1.LayerVersion(this, layerVersionName, {
|
|
97
98
|
layerVersionName,
|
|
98
99
|
compatibleArchitectures: [aws_lambda_1.Architecture.ARM_64, aws_lambda_1.Architecture.X86_64],
|
|
99
100
|
compatibleRuntimes: [aws_lambda_1.Runtime.NODEJS_18_X, aws_lambda_1.Runtime.NODEJS_20_X],
|
|
100
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
101
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
101
102
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
102
103
|
bundling: {
|
|
104
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
103
105
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
104
106
|
local: {
|
|
105
107
|
tryBundle(outputDir) {
|
|
106
108
|
try {
|
|
107
|
-
(0, node_child_process_1.execSync)('
|
|
109
|
+
(0, node_child_process_1.execSync)('yarn install', {
|
|
110
|
+
cwd: functionDirPath
|
|
111
|
+
});
|
|
108
112
|
}
|
|
109
113
|
catch (_a) {
|
|
110
114
|
return false;
|
|
111
115
|
}
|
|
112
|
-
fs.cpSync(
|
|
113
|
-
recursive: true
|
|
116
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
117
|
+
recursive: true,
|
|
114
118
|
});
|
|
115
119
|
return true;
|
|
116
120
|
},
|
|
@@ -122,24 +126,28 @@ class AccessLogsAnalysis extends constructs_1.Construct {
|
|
|
122
126
|
}
|
|
123
127
|
createPartitioningLayer() {
|
|
124
128
|
const layerVersionName = `${this.resourceIdPrefix}-partitioning-dependencies`;
|
|
129
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');
|
|
125
130
|
return new aws_lambda_1.LayerVersion(this, layerVersionName, {
|
|
126
131
|
layerVersionName,
|
|
127
132
|
compatibleArchitectures: [aws_lambda_1.Architecture.ARM_64, aws_lambda_1.Architecture.X86_64],
|
|
128
133
|
compatibleRuntimes: [aws_lambda_1.Runtime.NODEJS_18_X, aws_lambda_1.Runtime.NODEJS_20_X],
|
|
129
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
134
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
130
135
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
131
136
|
bundling: {
|
|
137
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
132
138
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
133
139
|
local: {
|
|
134
140
|
tryBundle(outputDir) {
|
|
135
141
|
try {
|
|
136
|
-
(0, node_child_process_1.execSync)('
|
|
142
|
+
(0, node_child_process_1.execSync)('yarn install', {
|
|
143
|
+
cwd: functionDirPath
|
|
144
|
+
});
|
|
137
145
|
}
|
|
138
146
|
catch (_a) {
|
|
139
147
|
return false;
|
|
140
148
|
}
|
|
141
|
-
fs.cpSync(
|
|
142
|
-
recursive: true
|
|
149
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
150
|
+
recursive: true,
|
|
143
151
|
});
|
|
144
152
|
return true;
|
|
145
153
|
},
|
|
@@ -155,23 +163,27 @@ class AccessLogsAnalysis extends constructs_1.Construct {
|
|
|
155
163
|
*/
|
|
156
164
|
createGroupByDateLambda() {
|
|
157
165
|
const functionName = `${this.resourceIdPrefix}-group-by-date`;
|
|
166
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');
|
|
158
167
|
const lambda = new aws_lambda_1.Function(this, functionName, {
|
|
159
168
|
functionName,
|
|
160
169
|
architecture: aws_lambda_1.Architecture.ARM_64,
|
|
161
170
|
runtime: aws_lambda_1.Runtime.NODEJS_20_X,
|
|
162
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
171
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
163
172
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
164
173
|
bundling: {
|
|
174
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
165
175
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
166
176
|
local: {
|
|
167
177
|
tryBundle(outputDir) {
|
|
168
178
|
try {
|
|
169
|
-
(0, node_child_process_1.execSync)('
|
|
179
|
+
(0, node_child_process_1.execSync)('yarn install && yarn build', {
|
|
180
|
+
cwd: functionDirPath
|
|
181
|
+
});
|
|
170
182
|
}
|
|
171
183
|
catch (_a) {
|
|
172
184
|
return false;
|
|
173
185
|
}
|
|
174
|
-
fs.cpSync(
|
|
186
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
175
187
|
recursive: true
|
|
176
188
|
});
|
|
177
189
|
return true;
|
|
@@ -206,24 +218,28 @@ class AccessLogsAnalysis extends constructs_1.Construct {
|
|
|
206
218
|
*/
|
|
207
219
|
createCreatePartitionLambda() {
|
|
208
220
|
const functionName = `${this.resourceIdPrefix}-create-part`;
|
|
221
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');
|
|
209
222
|
const lambda = new aws_lambda_1.Function(this, functionName, {
|
|
210
223
|
functionName,
|
|
211
224
|
architecture: aws_lambda_1.Architecture.ARM_64,
|
|
212
225
|
runtime: aws_lambda_1.Runtime.NODEJS_20_X,
|
|
213
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
226
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
214
227
|
exclude: ['transform-partition*', '*.d.ts'],
|
|
215
228
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
216
229
|
bundling: {
|
|
230
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
217
231
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
218
232
|
local: {
|
|
219
233
|
tryBundle(outputDir) {
|
|
220
234
|
try {
|
|
221
|
-
(0, node_child_process_1.execSync)('
|
|
235
|
+
(0, node_child_process_1.execSync)('yarn install && yarn build', {
|
|
236
|
+
cwd: functionDirPath
|
|
237
|
+
});
|
|
222
238
|
}
|
|
223
239
|
catch (_a) {
|
|
224
240
|
return false;
|
|
225
241
|
}
|
|
226
|
-
fs.cpSync(
|
|
242
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
227
243
|
recursive: true
|
|
228
244
|
});
|
|
229
245
|
return true;
|
|
@@ -363,4 +379,4 @@ AccessLogsAnalysis.ACCESS_LOGS_FOLDER_UNPROCESSED = 'unprocessed';
|
|
|
363
379
|
AccessLogsAnalysis.ACCESS_LOGS_FOLDER_GROUPED_BY_DATE = 'by-date';
|
|
364
380
|
AccessLogsAnalysis.ACCESS_LOGS_FOLDER_TRANSFORMED = 'transformed';
|
|
365
381
|
AccessLogsAnalysis.ACCESS_LOGS_FOLDER_ATHENA_RESULTS = 'athena-query-results';
|
|
366
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AccessLogsAnalysis.js","sourceRoot":"","sources":["AccessLogsAnalysis.ts"],"names":[],"mappings":"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","sourcesContent":["import {Bucket, EventType, NotificationKeyFilter} from 'aws-cdk-lib/aws-s3';\nimport {Construct} from 'constructs';\nimport {CfnWorkGroup} from 'aws-cdk-lib/aws-athena';\nimport {CloudFrontAccessLogsByDateTable} from './CloudFrontAccessLogsByDateTable';\nimport {AccessLogsParquetTable} from './AccessLogsParquetTable';\nimport {Architecture, Code, Function, LayerVersion, Runtime} from 'aws-cdk-lib/aws-lambda';\nimport {Rule, RuleTargetInput, Schedule} from 'aws-cdk-lib/aws-events';\nimport {AssetHashType, CfnTag, Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Column, Database} from '@aws-cdk/aws-glue-alpha';\nimport {S3EventSource} from 'aws-cdk-lib/aws-lambda-event-sources';\nimport {RetentionDays} from 'aws-cdk-lib/aws-logs';\nimport {Effect, PolicyStatement} from 'aws-cdk-lib/aws-iam';\nimport {LambdaFunction} from 'aws-cdk-lib/aws-events-targets';\nimport * as path from 'path';\nimport {AccessLogsAnalysisProps} from \"./AccessLogsAnalysisProps\";\nimport {ColumnTransformationRules} from \"../../functions/access-logs-analysis/partitioning/types\";\nimport {execSync} from \"node:child_process\";\nimport * as fs from \"fs\";\n\n/**\n * Provides the AWS resources to analyze access logs. This construct is derived from the official AWS sample\n * CloudFormation stack:\n * {@link https://github.com/aws-samples/amazon-cloudfront-access-logs-queries/blob/mainline/template.yaml}\n */\nexport abstract class AccessLogsAnalysis extends Construct {\n\n    protected static readonly ACCESS_LOGS_FOLDER_UNPROCESSED = 'unprocessed';\n    protected static readonly ACCESS_LOGS_FOLDER_GROUPED_BY_DATE = 'by-date';\n    protected static readonly ACCESS_LOGS_FOLDER_TRANSFORMED = 'transformed';\n    protected static readonly ACCESS_LOGS_FOLDER_ATHENA_RESULTS = 'athena-query-results';\n\n    protected readonly resourceIdPrefix: string;\n    protected readonly bucket: Bucket;\n    protected readonly workgroup: CfnWorkGroup;\n    protected readonly database: Database;\n    protected readonly accessLogsByDateTable: CloudFrontAccessLogsByDateTable;\n    protected readonly accessLogsParquetTable: AccessLogsParquetTable;\n    protected readonly groupByDateLayer: LayerVersion;\n    protected readonly groupByDateLambda: Function;\n    protected readonly partitioningLayer: LayerVersion;\n    protected readonly createPartitionLambda: Function;\n    protected readonly transformPartitionLambda: Function;\n    protected readonly createPartitionsScheduler: Rule;\n    protected readonly transformPartitionsScheduler: Rule;\n\n    protected constructor(scope: Construct, id: string, props: AccessLogsAnalysisProps) {\n        super(scope, id);\n        this.resourceIdPrefix = props.resourcePrefix;\n        this.bucket = props.bucket;\n        this.setupLifecycleRules(props);\n        this.workgroup = this.createWorkgroup();\n        this.database = this.createGlueDatabase();\n        this.accessLogsByDateTable = this.createAccessLogsByDateTable();\n        this.accessLogsParquetTable = this.createAccessLogsParquetTable();\n        this.groupByDateLayer = this.createGroupByDateLayer();\n        this.groupByDateLambda = this.createGroupByDateLambda();\n        this.partitioningLayer = this.createPartitioningLayer();\n        this.createPartitionLambda = this.createCreatePartitionLambda();\n        this.transformPartitionLambda = this.createTransformPartitionLambda();\n        this.createPartitionsScheduler = this.createCreatePartitionsScheduler();\n        this.transformPartitionsScheduler = this.createTransformPartitionsScheduler(props);\n    }\n\n    private setupLifecycleRules(props: AccessLogsAnalysisProps): void {\n        // cleanup raw and partitioned access logs after a configurable time range\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-unprocessed`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_UNPROCESSED}/*`,\n            expiration: props.expireRawLogsAfter ?? Duration.days(7),\n        });\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-grouped`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_GROUPED_BY_DATE}/*`,\n            expiration: props.expireIntermediateLogsAfter ?? Duration.days(7),\n        });\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-transformed`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_TRANSFORMED}/*`,\n            expiration: props.expireTransformedLogsAfter ?? Duration.days(180),\n        });\n\n        /* CHECKME: delete query results after 1 week\n        this.bucket.addLifecycleRule({\n          id: `${this.resourceIdPrefix}-cleanup-query-results`,\n          prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_ATHENA_RESULTS}/*`,\n          expiration: Duration.days(7),\n        });\n        */\n    }\n\n    private createWorkgroup(): CfnWorkGroup {\n        const workgroupName = `${this.resourceIdPrefix}-workgroup`;\n\n        // due to a current bug, the tags are not automatically assigned to the workgroup\n        // note, that the keys must be converted to lower case\n        const tags: CfnTag[] = Stack.of(this)\n            .tags.renderTags()\n            .map((tag: { Key: string; Value: string }) => ({key: tag.Key, value: tag.Value}));\n\n        return new CfnWorkGroup(this, workgroupName, {\n            name: workgroupName,\n            recursiveDeleteOption: true,\n            workGroupConfiguration: {\n                publishCloudWatchMetricsEnabled: false,\n                resultConfiguration: {\n                    outputLocation: `s3://${this.bucket.bucketName}/${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_ATHENA_RESULTS}`,\n                },\n                enforceWorkGroupConfiguration: true\n            },\n            tags,\n        });\n    }\n\n    private createGlueDatabase(): Database {\n        const databaseName = `${this.resourceIdPrefix}-database`;\n        return new Database(this, databaseName, {\n            // Athena doesn't support dashes in database/table names\n            databaseName: databaseName.replace(/-/g, '_'),\n        });\n    }\n\n    protected abstract createAccessLogsByDateTable(): CloudFrontAccessLogsByDateTable;\n\n    protected abstract createAccessLogsParquetTable(): CloudFrontAccessLogsByDateTable;\n\n    private createGroupByDateLayer(): LayerVersion {\n        const layerVersionName = `${this.resourceIdPrefix}-group-by-date-dependencies`;\n        return new LayerVersion(this, layerVersionName, {\n            layerVersionName,\n            compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],\n            compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/group-by-date/build/layer'), {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/group-by-date') + ' && yarn install');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/group-by-date/node_modules'), `${outputDir}/nodejs/node_modules`, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n    }\n\n    private createPartitioningLayer(): LayerVersion {\n        const layerVersionName = `${this.resourceIdPrefix}-partitioning-dependencies`;\n        return new LayerVersion(this, layerVersionName, {\n            layerVersionName,\n            compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],\n            compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/layer'), {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/partitioning') + ' && yarn install');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/node_modules'), `${outputDir}/nodejs/node_modules`, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n    }\n\n    /**\n     * A Lambda function to be triggered whenever a new access log is created.\n     * It moves the raw access logs into a sub folder hierarchy by year, month, day and hour.\n     */\n    private createGroupByDateLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-group-by-date`;\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/group-by-date/build/app'), {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/group-by-date') + ' && yarn install && yarn build');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/group-by-date/build/app'), outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 512,\n            timeout: Duration.seconds(20),\n            handler: 'index.handler',\n            layers: [this.groupByDateLayer],\n            events: [\n                new S3EventSource(this.bucket, {\n                    events: [EventType.OBJECT_CREATED],\n                    filters: [this.getUnprocessedObjectsFilter()],\n                }),\n            ],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                TARGET_FOLDER: AccessLogsAnalysis.ACCESS_LOGS_FOLDER_GROUPED_BY_DATE,\n                RAW_ACCESS_LOG_FILE_PATTERN: this.getRawAccessLogFilePattern().source,\n            },\n        });\n\n        this.bucket.grantReadWrite(lambda);\n        this.bucket.grantDelete(lambda, `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_UNPROCESSED}/*`);\n\n        return lambda;\n    }\n\n    protected abstract getUnprocessedObjectsFilter(): NotificationKeyFilter;\n\n    /** The pattern to identify unprocessed access logs depends on the producing source */\n    protected abstract getRawAccessLogFilePattern(): RegExp;\n\n    /**\n     * Creates a new partition for the upcoming hour in the access log database.\n     */\n    private createCreatePartitionLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-create-part`;\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), {\n                exclude: ['transform-partition*', '*.d.ts'],\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/partitioning') + ' && yarn install && yarn build');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 128,\n            timeout: Duration.seconds(20),\n            handler: 'create-partition.handler',\n            layers: [this.partitioningLayer],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                WORKGROUP: this.workgroup.name,\n                DATABASE: this.database.databaseName,\n                TABLE: this.accessLogsByDateTable.tableName,\n            },\n        });\n\n        // grant access to S3 bucket and to execute Athena queries which create new partitions\n        this.bucket.grantReadWrite(lambda);\n        lambda.addToRolePolicy(\n            new PolicyStatement({\n                effect: Effect.ALLOW,\n                actions: [\n                    'athena:StartQueryExecution',\n                    'athena:GetQueryExecution',\n                    'glue:CreateDatabase',\n                    'glue:CreatePartition',\n                    'glue:GetDatabase',\n                    'glue:GetTable',\n                    'glue:BatchCreatePartition',\n                ],\n                resources: ['*'],\n            })\n        );\n\n        return lambda;\n    }\n\n    /**\n     * Transforms partitions from the Hive format to Parquet.\n     */\n    private createTransformPartitionLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-transform-part`;\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), {\n                exclude: ['create-partition*', '*.d.ts'],\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/partitioning') + ' && yarn install && yarn build');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 128,\n            timeout: Duration.seconds(20),\n            handler: 'transform-partition.handler',\n            layers: [this.partitioningLayer],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                WORKGROUP: this.workgroup.name,\n                DATABASE: this.database.databaseName,\n                SOURCE_TABLE: this.accessLogsByDateTable.tableName,\n                TARGET_TABLE: this.accessLogsParquetTable.tableName,\n            },\n        });\n\n        // grant access to S3 bucket and to execute Athena queries which create new partitions\n        this.bucket.grantReadWrite(lambda);\n        lambda.addToRolePolicy(\n            new PolicyStatement({\n                effect: Effect.ALLOW,\n                actions: [\n                    'athena:StartQueryExecution',\n                    'athena:GetQueryExecution',\n                    'glue:CreateDatabase',\n                    'glue:CreatePartition',\n                    'glue:GetDatabase',\n                    'glue:GetTable',\n                    'glue:BatchCreatePartition',\n                    'glue:GetPartition',\n                    'glue:GetPartitions',\n                    'glue:CreateTable',\n                    'glue:DeleteTable',\n                    'glue:DeletePartition',\n                ],\n                resources: ['*'],\n            })\n        );\n\n        return lambda;\n    }\n\n    private createCreatePartitionsScheduler(): Rule {\n        const ruleName = `${this.resourceIdPrefix}-create-part-sched`;\n        return new Rule(this, ruleName, {\n            ruleName,\n            schedule: Schedule.cron({minute: '55'}),\n            targets: [new LambdaFunction(this.createPartitionLambda, {retryAttempts: 10})],\n        });\n    }\n\n    private createTransformPartitionsScheduler(props: AccessLogsAnalysisProps): Rule {\n        const ruleName = `${this.resourceIdPrefix}-transform-part-sched`;\n        return new Rule(this, ruleName, {\n            ruleName,\n            schedule: Schedule.cron({minute: '1'}),\n            targets: [\n                new LambdaFunction(this.transformPartitionLambda, {\n                    event: this.getTransformPartitionInvocationProps(props),\n                    retryAttempts: 10,\n                }),\n            ],\n        });\n    }\n\n    private getTransformPartitionInvocationProps(props: AccessLogsAnalysisProps): RuleTargetInput {\n        return RuleTargetInput.fromObject({\n            columnNames: [\n                ...this.accessLogsParquetTable.columns,\n                ...(this.accessLogsParquetTable.partitionKeys as Column[]),\n            ].map(column => column.name),\n            columnTransformations: this.getColumnTransformationRules(props),\n        });\n    }\n\n    /**\n     * Needs to return the log source specific column transformation rules\n     * (e.g. to anonymize IP addresses).\n     */\n    protected abstract getColumnTransformationRules(props: AccessLogsAnalysisProps): ColumnTransformationRules;\n}\n"]}
|
|
382
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AccessLogsAnalysis.js","sourceRoot":"","sources":["AccessLogsAnalysis.ts"],"names":[],"mappings":"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","sourcesContent":["import {Bucket, EventType, NotificationKeyFilter} from 'aws-cdk-lib/aws-s3';\nimport {Construct} from 'constructs';\nimport {CfnWorkGroup} from 'aws-cdk-lib/aws-athena';\nimport {CloudFrontAccessLogsByDateTable} from './CloudFrontAccessLogsByDateTable';\nimport {AccessLogsParquetTable} from './AccessLogsParquetTable';\nimport {Architecture, Code, Function, LayerVersion, Runtime} from 'aws-cdk-lib/aws-lambda';\nimport {Rule, RuleTargetInput, Schedule} from 'aws-cdk-lib/aws-events';\nimport {AssetHashType, CfnTag, Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Column, Database} from '@aws-cdk/aws-glue-alpha';\nimport {S3EventSource} from 'aws-cdk-lib/aws-lambda-event-sources';\nimport {RetentionDays} from 'aws-cdk-lib/aws-logs';\nimport {Effect, PolicyStatement} from 'aws-cdk-lib/aws-iam';\nimport {LambdaFunction} from 'aws-cdk-lib/aws-events-targets';\nimport * as path from 'path';\nimport {AccessLogsAnalysisProps} from \"./AccessLogsAnalysisProps\";\nimport {ColumnTransformationRules} from \"../../functions/access-logs-analysis/partitioning/types\";\nimport {execSync} from \"node:child_process\";\nimport * as fs from \"fs\";\n\n/**\n * Provides the AWS resources to analyze access logs. This construct is derived from the official AWS sample\n * CloudFormation stack:\n * {@link https://github.com/aws-samples/amazon-cloudfront-access-logs-queries/blob/mainline/template.yaml}\n */\nexport abstract class AccessLogsAnalysis extends Construct {\n\n    protected static readonly ACCESS_LOGS_FOLDER_UNPROCESSED = 'unprocessed';\n    protected static readonly ACCESS_LOGS_FOLDER_GROUPED_BY_DATE = 'by-date';\n    protected static readonly ACCESS_LOGS_FOLDER_TRANSFORMED = 'transformed';\n    protected static readonly ACCESS_LOGS_FOLDER_ATHENA_RESULTS = 'athena-query-results';\n\n    protected readonly resourceIdPrefix: string;\n    protected readonly bucket: Bucket;\n    protected readonly workgroup: CfnWorkGroup;\n    protected readonly database: Database;\n    protected readonly accessLogsByDateTable: CloudFrontAccessLogsByDateTable;\n    protected readonly accessLogsParquetTable: AccessLogsParquetTable;\n    protected readonly groupByDateLayer: LayerVersion;\n    protected readonly groupByDateLambda: Function;\n    protected readonly partitioningLayer: LayerVersion;\n    protected readonly createPartitionLambda: Function;\n    protected readonly transformPartitionLambda: Function;\n    protected readonly createPartitionsScheduler: Rule;\n    protected readonly transformPartitionsScheduler: Rule;\n\n    protected constructor(scope: Construct, id: string, props: AccessLogsAnalysisProps) {\n        super(scope, id);\n        this.resourceIdPrefix = props.resourcePrefix;\n        this.bucket = props.bucket;\n        this.setupLifecycleRules(props);\n        this.workgroup = this.createWorkgroup();\n        this.database = this.createGlueDatabase();\n        this.accessLogsByDateTable = this.createAccessLogsByDateTable();\n        this.accessLogsParquetTable = this.createAccessLogsParquetTable();\n        this.groupByDateLayer = this.createGroupByDateLayer();\n        this.groupByDateLambda = this.createGroupByDateLambda();\n        this.partitioningLayer = this.createPartitioningLayer();\n        this.createPartitionLambda = this.createCreatePartitionLambda();\n        this.transformPartitionLambda = this.createTransformPartitionLambda();\n        this.createPartitionsScheduler = this.createCreatePartitionsScheduler();\n        this.transformPartitionsScheduler = this.createTransformPartitionsScheduler(props);\n    }\n\n    private setupLifecycleRules(props: AccessLogsAnalysisProps): void {\n        // cleanup raw and partitioned access logs after a configurable time range\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-unprocessed`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_UNPROCESSED}/*`,\n            expiration: props.expireRawLogsAfter ?? Duration.days(7),\n        });\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-grouped`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_GROUPED_BY_DATE}/*`,\n            expiration: props.expireIntermediateLogsAfter ?? Duration.days(7),\n        });\n\n        this.bucket.addLifecycleRule({\n            id: `${this.resourceIdPrefix}-cleanup-transformed`,\n            prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_TRANSFORMED}/*`,\n            expiration: props.expireTransformedLogsAfter ?? Duration.days(180),\n        });\n\n        /* CHECKME: delete query results after 1 week\n        this.bucket.addLifecycleRule({\n          id: `${this.resourceIdPrefix}-cleanup-query-results`,\n          prefix: `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_ATHENA_RESULTS}/*`,\n          expiration: Duration.days(7),\n        });\n        */\n    }\n\n    private createWorkgroup(): CfnWorkGroup {\n        const workgroupName = `${this.resourceIdPrefix}-workgroup`;\n\n        // due to a current bug, the tags are not automatically assigned to the workgroup\n        // note, that the keys must be converted to lower case\n        const tags: CfnTag[] = Stack.of(this)\n            .tags.renderTags()\n            .map((tag: { Key: string; Value: string }) => ({key: tag.Key, value: tag.Value}));\n\n        return new CfnWorkGroup(this, workgroupName, {\n            name: workgroupName,\n            recursiveDeleteOption: true,\n            workGroupConfiguration: {\n                publishCloudWatchMetricsEnabled: false,\n                resultConfiguration: {\n                    outputLocation: `s3://${this.bucket.bucketName}/${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_ATHENA_RESULTS}`,\n                },\n                enforceWorkGroupConfiguration: true\n            },\n            tags,\n        });\n    }\n\n    private createGlueDatabase(): Database {\n        const databaseName = `${this.resourceIdPrefix}-database`;\n        return new Database(this, databaseName, {\n            // Athena doesn't support dashes in database/table names\n            databaseName: databaseName.replace(/-/g, '_'),\n        });\n    }\n\n    protected abstract createAccessLogsByDateTable(): CloudFrontAccessLogsByDateTable;\n\n    protected abstract createAccessLogsParquetTable(): CloudFrontAccessLogsByDateTable;\n\n    private createGroupByDateLayer(): LayerVersion {\n        const layerVersionName = `${this.resourceIdPrefix}-group-by-date-dependencies`;\n        const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');\n\n        return new LayerVersion(this, layerVersionName, {\n            layerVersionName,\n            compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],\n            compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],\n            code: Code.fromAsset(functionDirPath, {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('yarn install', {\n                                    cwd: functionDirPath\n                                });\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {\n                                recursive: true,\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n    }\n\n    private createPartitioningLayer(): LayerVersion {\n        const layerVersionName = `${this.resourceIdPrefix}-partitioning-dependencies`;\n        const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');\n\n        return new LayerVersion(this, layerVersionName, {\n            layerVersionName,\n            compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],\n            compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],\n            code: Code.fromAsset(functionDirPath, {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('yarn install', {\n                                    cwd: functionDirPath\n                                });\n                            } catch {\n                                return false;\n                            }\n                            fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {\n                                recursive: true,\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n    }\n\n    /**\n     * A Lambda function to be triggered whenever a new access log is created.\n     * It moves the raw access logs into a sub folder hierarchy by year, month, day and hour.\n     */\n    private createGroupByDateLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-group-by-date`;\n        const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(functionDirPath, {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('yarn install && yarn build', {\n                                    cwd: functionDirPath\n                                });\n                            } catch {\n                                return false;\n                            }\n                            fs.cpSync(`${functionDirPath}/build/app`, outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 512,\n            timeout: Duration.seconds(20),\n            handler: 'index.handler',\n            layers: [this.groupByDateLayer],\n            events: [\n                new S3EventSource(this.bucket, {\n                    events: [EventType.OBJECT_CREATED],\n                    filters: [this.getUnprocessedObjectsFilter()],\n                }),\n            ],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                TARGET_FOLDER: AccessLogsAnalysis.ACCESS_LOGS_FOLDER_GROUPED_BY_DATE,\n                RAW_ACCESS_LOG_FILE_PATTERN: this.getRawAccessLogFilePattern().source,\n            },\n        });\n\n        this.bucket.grantReadWrite(lambda);\n        this.bucket.grantDelete(lambda, `${AccessLogsAnalysis.ACCESS_LOGS_FOLDER_UNPROCESSED}/*`);\n\n        return lambda;\n    }\n\n    protected abstract getUnprocessedObjectsFilter(): NotificationKeyFilter;\n\n    /** The pattern to identify unprocessed access logs depends on the producing source */\n    protected abstract getRawAccessLogFilePattern(): RegExp;\n\n    /**\n     * Creates a new partition for the upcoming hour in the access log database.\n     */\n    private createCreatePartitionLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-create-part`;\n        const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(functionDirPath, {\n                exclude: ['transform-partition*', '*.d.ts'],\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('yarn install && yarn build', {\n                                    cwd: functionDirPath\n                                });\n                            } catch {\n                                return false;\n                            }\n                            fs.cpSync(`${functionDirPath}/build/app`, outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 128,\n            timeout: Duration.seconds(20),\n            handler: 'create-partition.handler',\n            layers: [this.partitioningLayer],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                WORKGROUP: this.workgroup.name,\n                DATABASE: this.database.databaseName,\n                TABLE: this.accessLogsByDateTable.tableName,\n            },\n        });\n\n        // grant access to S3 bucket and to execute Athena queries which create new partitions\n        this.bucket.grantReadWrite(lambda);\n        lambda.addToRolePolicy(\n            new PolicyStatement({\n                effect: Effect.ALLOW,\n                actions: [\n                    'athena:StartQueryExecution',\n                    'athena:GetQueryExecution',\n                    'glue:CreateDatabase',\n                    'glue:CreatePartition',\n                    'glue:GetDatabase',\n                    'glue:GetTable',\n                    'glue:BatchCreatePartition',\n                ],\n                resources: ['*'],\n            })\n        );\n\n        return lambda;\n    }\n\n    /**\n     * Transforms partitions from the Hive format to Parquet.\n     */\n    private createTransformPartitionLambda(): Function {\n        const functionName = `${this.resourceIdPrefix}-transform-part`;\n\n        const lambda = new Function(this, functionName, {\n            functionName,\n            architecture: Architecture.ARM_64,\n            runtime: Runtime.NODEJS_20_X,\n            code: Code.fromAsset(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), {\n                exclude: ['create-partition*', '*.d.ts'],\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/access-logs-analysis/partitioning') + ' && yarn install && yarn build');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            memorySize: 128,\n            timeout: Duration.seconds(20),\n            handler: 'transform-partition.handler',\n            layers: [this.partitioningLayer],\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n                WORKGROUP: this.workgroup.name,\n                DATABASE: this.database.databaseName,\n                SOURCE_TABLE: this.accessLogsByDateTable.tableName,\n                TARGET_TABLE: this.accessLogsParquetTable.tableName,\n            },\n        });\n\n        // grant access to S3 bucket and to execute Athena queries which create new partitions\n        this.bucket.grantReadWrite(lambda);\n        lambda.addToRolePolicy(\n            new PolicyStatement({\n                effect: Effect.ALLOW,\n                actions: [\n                    'athena:StartQueryExecution',\n                    'athena:GetQueryExecution',\n                    'glue:CreateDatabase',\n                    'glue:CreatePartition',\n                    'glue:GetDatabase',\n                    'glue:GetTable',\n                    'glue:BatchCreatePartition',\n                    'glue:GetPartition',\n                    'glue:GetPartitions',\n                    'glue:CreateTable',\n                    'glue:DeleteTable',\n                    'glue:DeletePartition',\n                ],\n                resources: ['*'],\n            })\n        );\n\n        return lambda;\n    }\n\n    private createCreatePartitionsScheduler(): Rule {\n        const ruleName = `${this.resourceIdPrefix}-create-part-sched`;\n        return new Rule(this, ruleName, {\n            ruleName,\n            schedule: Schedule.cron({minute: '55'}),\n            targets: [new LambdaFunction(this.createPartitionLambda, {retryAttempts: 10})],\n        });\n    }\n\n    private createTransformPartitionsScheduler(props: AccessLogsAnalysisProps): Rule {\n        const ruleName = `${this.resourceIdPrefix}-transform-part-sched`;\n        return new Rule(this, ruleName, {\n            ruleName,\n            schedule: Schedule.cron({minute: '1'}),\n            targets: [\n                new LambdaFunction(this.transformPartitionLambda, {\n                    event: this.getTransformPartitionInvocationProps(props),\n                    retryAttempts: 10,\n                }),\n            ],\n        });\n    }\n\n    private getTransformPartitionInvocationProps(props: AccessLogsAnalysisProps): RuleTargetInput {\n        return RuleTargetInput.fromObject({\n            columnNames: [\n                ...this.accessLogsParquetTable.columns,\n                ...(this.accessLogsParquetTable.partitionKeys as Column[]),\n            ].map(column => column.name),\n            columnTransformations: this.getColumnTransformationRules(props),\n        });\n    }\n\n    /**\n     * Needs to return the log source specific column transformation rules\n     * (e.g. to anonymize IP addresses).\n     */\n    protected abstract getColumnTransformationRules(props: AccessLogsAnalysisProps): ColumnTransformationRules;\n}\n"]}
|
|
@@ -128,24 +128,29 @@ export abstract class AccessLogsAnalysis extends Construct {
|
|
|
128
128
|
|
|
129
129
|
private createGroupByDateLayer(): LayerVersion {
|
|
130
130
|
const layerVersionName = `${this.resourceIdPrefix}-group-by-date-dependencies`;
|
|
131
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');
|
|
132
|
+
|
|
131
133
|
return new LayerVersion(this, layerVersionName, {
|
|
132
134
|
layerVersionName,
|
|
133
135
|
compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],
|
|
134
136
|
compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],
|
|
135
|
-
code: Code.fromAsset(
|
|
137
|
+
code: Code.fromAsset(functionDirPath, {
|
|
136
138
|
assetHashType: AssetHashType.OUTPUT,
|
|
137
139
|
bundling: {
|
|
140
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
138
141
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
139
142
|
local: {
|
|
140
143
|
tryBundle(outputDir: string): boolean {
|
|
141
144
|
try {
|
|
142
|
-
execSync('
|
|
145
|
+
execSync('yarn install', {
|
|
146
|
+
cwd: functionDirPath
|
|
147
|
+
});
|
|
143
148
|
} catch {
|
|
144
149
|
return false;
|
|
145
150
|
}
|
|
146
151
|
|
|
147
|
-
fs.cpSync(
|
|
148
|
-
recursive: true
|
|
152
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
153
|
+
recursive: true,
|
|
149
154
|
});
|
|
150
155
|
return true
|
|
151
156
|
},
|
|
@@ -158,24 +163,28 @@ export abstract class AccessLogsAnalysis extends Construct {
|
|
|
158
163
|
|
|
159
164
|
private createPartitioningLayer(): LayerVersion {
|
|
160
165
|
const layerVersionName = `${this.resourceIdPrefix}-partitioning-dependencies`;
|
|
166
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');
|
|
167
|
+
|
|
161
168
|
return new LayerVersion(this, layerVersionName, {
|
|
162
169
|
layerVersionName,
|
|
163
170
|
compatibleArchitectures: [Architecture.ARM_64, Architecture.X86_64],
|
|
164
171
|
compatibleRuntimes: [Runtime.NODEJS_18_X, Runtime.NODEJS_20_X],
|
|
165
|
-
code: Code.fromAsset(
|
|
172
|
+
code: Code.fromAsset(functionDirPath, {
|
|
166
173
|
assetHashType: AssetHashType.OUTPUT,
|
|
167
174
|
bundling: {
|
|
175
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
168
176
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
169
177
|
local: {
|
|
170
178
|
tryBundle(outputDir: string): boolean {
|
|
171
179
|
try {
|
|
172
|
-
execSync('
|
|
180
|
+
execSync('yarn install', {
|
|
181
|
+
cwd: functionDirPath
|
|
182
|
+
});
|
|
173
183
|
} catch {
|
|
174
184
|
return false;
|
|
175
185
|
}
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
recursive: true
|
|
186
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
187
|
+
recursive: true,
|
|
179
188
|
});
|
|
180
189
|
return true
|
|
181
190
|
},
|
|
@@ -192,24 +201,27 @@ export abstract class AccessLogsAnalysis extends Construct {
|
|
|
192
201
|
*/
|
|
193
202
|
private createGroupByDateLambda(): Function {
|
|
194
203
|
const functionName = `${this.resourceIdPrefix}-group-by-date`;
|
|
204
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/group-by-date');
|
|
195
205
|
|
|
196
206
|
const lambda = new Function(this, functionName, {
|
|
197
207
|
functionName,
|
|
198
208
|
architecture: Architecture.ARM_64,
|
|
199
209
|
runtime: Runtime.NODEJS_20_X,
|
|
200
|
-
code: Code.fromAsset(
|
|
210
|
+
code: Code.fromAsset(functionDirPath, {
|
|
201
211
|
assetHashType: AssetHashType.OUTPUT,
|
|
202
212
|
bundling: {
|
|
213
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
203
214
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
204
215
|
local: {
|
|
205
216
|
tryBundle(outputDir: string): boolean {
|
|
206
217
|
try {
|
|
207
|
-
execSync('
|
|
218
|
+
execSync('yarn install && yarn build', {
|
|
219
|
+
cwd: functionDirPath
|
|
220
|
+
});
|
|
208
221
|
} catch {
|
|
209
222
|
return false;
|
|
210
223
|
}
|
|
211
|
-
|
|
212
|
-
fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/group-by-date/build/app'), outputDir, {
|
|
224
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
213
225
|
recursive: true
|
|
214
226
|
});
|
|
215
227
|
return true
|
|
@@ -252,25 +264,28 @@ export abstract class AccessLogsAnalysis extends Construct {
|
|
|
252
264
|
*/
|
|
253
265
|
private createCreatePartitionLambda(): Function {
|
|
254
266
|
const functionName = `${this.resourceIdPrefix}-create-part`;
|
|
267
|
+
const functionDirPath = path.join(__dirname, '../../functions/access-logs-analysis/partitioning');
|
|
255
268
|
|
|
256
269
|
const lambda = new Function(this, functionName, {
|
|
257
270
|
functionName,
|
|
258
271
|
architecture: Architecture.ARM_64,
|
|
259
272
|
runtime: Runtime.NODEJS_20_X,
|
|
260
|
-
code: Code.fromAsset(
|
|
273
|
+
code: Code.fromAsset(functionDirPath, {
|
|
261
274
|
exclude: ['transform-partition*', '*.d.ts'],
|
|
262
275
|
assetHashType: AssetHashType.OUTPUT,
|
|
263
276
|
bundling: {
|
|
277
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
264
278
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
265
279
|
local: {
|
|
266
280
|
tryBundle(outputDir: string): boolean {
|
|
267
281
|
try {
|
|
268
|
-
execSync('
|
|
282
|
+
execSync('yarn install && yarn build', {
|
|
283
|
+
cwd: functionDirPath
|
|
284
|
+
});
|
|
269
285
|
} catch {
|
|
270
286
|
return false;
|
|
271
287
|
}
|
|
272
|
-
|
|
273
|
-
fs.cpSync(path.join(__dirname, '../../functions/access-logs-analysis/partitioning/build/app'), outputDir, {
|
|
288
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
274
289
|
recursive: true
|
|
275
290
|
});
|
|
276
291
|
return true
|
|
@@ -149,6 +149,7 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
149
149
|
createCleanupLambdaFunction(props) {
|
|
150
150
|
var _a;
|
|
151
151
|
const functionName = `${this.resourceIdPrefix}-cleanup-function`;
|
|
152
|
+
const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');
|
|
152
153
|
const result = new aws_lambda_1.Function(this, functionName, {
|
|
153
154
|
functionName: functionName,
|
|
154
155
|
description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,
|
|
@@ -156,20 +157,23 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
156
157
|
architecture: aws_lambda_1.Architecture.ARM_64,
|
|
157
158
|
layers: [new aws_lambda_1.LayerVersion(this, `${functionName}-layer`, {
|
|
158
159
|
layerVersionName: `${functionName}-layer`,
|
|
159
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
160
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
160
161
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
161
162
|
bundling: {
|
|
163
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
162
164
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
163
165
|
local: {
|
|
164
166
|
tryBundle(outputDir) {
|
|
165
167
|
try {
|
|
166
|
-
(0, node_child_process_1.execSync)('
|
|
168
|
+
(0, node_child_process_1.execSync)('yarn install', {
|
|
169
|
+
cwd: functionDirPath
|
|
170
|
+
});
|
|
167
171
|
}
|
|
168
172
|
catch (_a) {
|
|
169
173
|
return false;
|
|
170
174
|
}
|
|
171
|
-
fs.cpSync(
|
|
172
|
-
recursive: true
|
|
175
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
176
|
+
recursive: true,
|
|
173
177
|
});
|
|
174
178
|
return true;
|
|
175
179
|
},
|
|
@@ -180,19 +184,22 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
180
184
|
description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`
|
|
181
185
|
})],
|
|
182
186
|
handler: 'index.handler',
|
|
183
|
-
code: aws_lambda_1.Code.fromAsset(
|
|
187
|
+
code: aws_lambda_1.Code.fromAsset(functionDirPath, {
|
|
184
188
|
assetHashType: aws_cdk_lib_1.AssetHashType.OUTPUT,
|
|
185
189
|
bundling: {
|
|
190
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
186
191
|
image: aws_lambda_1.Runtime.NODEJS_20_X.bundlingImage,
|
|
187
192
|
local: {
|
|
188
193
|
tryBundle(outputDir) {
|
|
189
194
|
try {
|
|
190
|
-
(0, node_child_process_1.execSync)('
|
|
195
|
+
(0, node_child_process_1.execSync)('yarn install && yarn build', {
|
|
196
|
+
cwd: functionDirPath
|
|
197
|
+
});
|
|
191
198
|
}
|
|
192
199
|
catch (_a) {
|
|
193
200
|
return false;
|
|
194
201
|
}
|
|
195
|
-
fs.cpSync(
|
|
202
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
196
203
|
recursive: true
|
|
197
204
|
});
|
|
198
205
|
return true;
|
|
@@ -543,4 +550,4 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
543
550
|
}
|
|
544
551
|
}
|
|
545
552
|
exports.NuxtServerAppStack = NuxtServerAppStack;
|
|
546
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {AssetHashType, 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, HttpVersion,\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, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {\n    BlockPublicAccess,\n    Bucket,\n    BucketAccessControl,\n    BucketEncryption,\n    ObjectOwnership\n} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, 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 {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../NuxtAppStaticAssets\";\nimport * as fs from \"fs\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\nimport {writeFileSync} from \"fs\";\nimport {NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {CloudFrontAccessLogsAnalysis} from \"../access-logs-analysis/CloudFrontAccessLogsAnalysis\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport {execSync} from \"node:child_process\";\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack 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 to tag the static assets of the deployment\n     * to later be able to clean up outdated assets.\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: Bucket;\n\n    /**\n     * The S3 bucket where the access logs of the CloudFront distribution gets stored.\n     */\n    public accessLogsBucket: Bucket|undefined;\n\n    /**\n     * The S3 bucket where the sitemap assets gets stored.\n     */\n    public sitemapBucket: Bucket|undefined;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: 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 origin for the API gateway to route incoming requests to the Nuxt Lambda function.\n     */\n    private httpOrigin: HttpOrigin;\n\n    /**\n     * The cache policy for the Nuxt app route behaviors of the CloudFront distribution.\n     */\n    private appCachePolicy: CachePolicy;\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: NuxtServerAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = this.createDeploymentRevision(props);\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.rootDir ?? '.');\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n\n        if (props.enableAccessLogsAnalysis) {\n            this.accessLogsBucket = this.createAccessLogsBucket();\n            this.createAccessLogsAnalysis(props);\n        }\n\n        if (props.enableSitemap) {\n            this.sitemapBucket = this.createSitemapBucket();\n        }\n\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.httpOrigin = this.createNuxtAppHttpOrigin();\n        this.appCachePolicy = this.createNuxtAppCachePolicy(props)\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the current deployment revision file in the public folder of the Nuxt app to be accessible\n     * and returns the current revision.\n     */\n    private createDeploymentRevision(props: NuxtServerAppStackProps): string {\n        const revisionFilePath = `${props.rootDir ?? '.'}/.output/public/app-revision`;\n        const appRevision = new Date().toISOString();\n\n        writeFileSync(revisionFilePath, appRevision, {encoding: 'utf-8'});\n\n        return appRevision;\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(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\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            objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the bucket to store the sitemap assets of the Nuxt app.\n     *\n     * @private\n     */\n    private createSitemapBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-sitemap`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            accessControl: BucketAccessControl.PRIVATE,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            encryption: BucketEncryption.S3_MANAGED,\n            enforceSSL: true,\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 the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-app-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: `${props.entrypoint ?? 'index'}.handler`,\n            code: Code.fromAsset(`${props.rootDir ?? '.' }/.output/server`, {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            environment: {\n                NODE_OPTIONS: '--enable-source-maps',\n                ...JSON.parse(props.entrypointEnv ?? '{}'),\n            },\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup-function`;\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            layers: [new LayerVersion(this, `${functionName}-layer`, {\n                layerVersionName: `${functionName}-layer`,\n                code: Code.fromAsset(path.join(__dirname, '../../functions/assets-cleanup/build/layer'), {\n                    assetHashType: AssetHashType.OUTPUT,\n                    bundling: {\n                        image: Runtime.NODEJS_20_X.bundlingImage,\n                        local: {\n                            tryBundle(outputDir: string): boolean {\n                                try {\n                                    execSync('cd ' + path.join(__dirname, '../../functions/assets-cleanup') + ' && yarn install');\n                                } catch {\n                                    return false;\n                                }\n\n                                fs.cpSync(path.join(__dirname, '../../functions/assets-cleanup/node_modules'), `${outputDir}/nodejs/node_modules`, {\n                                    recursive: true\n                                });\n                                return true\n                            },\n                        }\n                    }\n                }),\n                compatibleRuntimes: [Runtime.NODEJS_20_X],\n                description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`\n            })],\n            handler: 'index.handler',\n            code: Code.fromAsset(path.join(__dirname, '../../functions/assets-cleanup/build/app'), {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('cd ' + path.join(__dirname, '../../functions/assets-cleanup') + ' && yarn install && yarn build');\n                            } catch {\n                                return false;\n                            }\n\n                            fs.cpSync(path.join(__dirname, '../../functions/assets-cleanup/build/app'), outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            timeout: Duration.minutes(5),\n            memorySize: 128,\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,\n                OUTDATED_ASSETS_RETENTION_DAYS: `${props.outdatedAssetsRetentionDays ?? 30}`,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\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: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\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: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            httpVersion: HttpVersion.HTTP2_AND_3,\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.setupCloudFrontRouting(props),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n            logBucket: this.accessLogsBucket,\n            logFilePrefix: props.enableAccessLogsAnalysis ? CloudFrontAccessLogsAnalysis.getLogFilePrefix() : undefined,\n            logIncludesCookies: true,\n        });\n    }\n\n    /**\n     * Creates the CloudFront distribution behavior origin to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     */\n    private createNuxtAppHttpOrigin(): HttpOrigin {\n        return 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    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {};\n\n        // Specific ones first\n        if (props.enableApi) {\n            routingBehaviours = {...routingBehaviours, ...this.createApiRouteBehavior()};\n        }\n        if (props.enableSitemap) {\n            routingBehaviours = {...routingBehaviours, ...this.createSitemapRouteBehavior()};\n        }\n\n        routingBehaviours = {...routingBehaviours, ...this.createStaticAssetsRouteBehavior()};\n\n        return routingBehaviours;\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     */\n    private createNuxtAppCachePolicy(props: NuxtServerAppStackProps): CachePolicy {\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin and how the cache key is calculated.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.days(365),\n            queryStringBehavior: props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()),\n            headerBehavior: props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none(),\n            cookieBehavior: props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching Nuxt app API requests to the API gateway.\n     */\n    private createApiRouteBehavior(): Record<string, BehaviorOptions> {\n        const apiBehavior: BehaviorOptions = {\n            origin: this.httpOrigin,\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_ALL,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: this.appCachePolicy,\n            viewerProtocolPolicy: ViewerProtocolPolicy.HTTPS_ONLY\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['/api/*'] = apiBehavior;\n\n        return rules;\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            }),\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     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the sitemap assets\n     * to the S3 bucket that holds these sitemap assets.\n     *\n     * @private\n     */\n    private createSitemapRouteBehavior(): Record<string, BehaviorOptions> {\n        if (!this.sitemapBucket) {\n            throw new Error(\"Sitemap bucket must exist before creating sitemap route behavior.\");\n        }\n\n        const sitemapCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.sitemapBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\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        rules['*sitemap.xml'] = sitemapCacheConfig;\n        rules['*sitemap-gone.xml'] = sitemapCacheConfig;\n        rules['/sitemaps/*'] = sitemapCacheConfig;\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 with minimal storage load,\n     * we deploy the static assets of every deployment into the same folder but mark them with a deployment revision.\n     * By doing so, the files of previous deployments are retained to allow clients to continue to work with an older revision\n     * but gets cleaned up after a specified period of time via the cleanup Lambda function.\n     */\n    private configureDeployments(): BucketDeployment[] {\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                    exclude: asset.exclude,\n                })],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target.replace(/^\\/+/g, ''), // Remove leading slash\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? [`/${asset.pattern}`] : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n\n                metadata: {\n                    // Store build revision on every asset to allow cleanup of outdated assets\n                    revision: this.deploymentRevision,\n                },\n\n                // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory.\n                // Additionally, a high memory limit might speed up deployments.\n                memoryLimit: 1792\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: NuxtServerAppStackProps): 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: NuxtServerAppStackProps): 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 createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\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.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n\n    /**\n     * Creates a S3 bucket to store the access logs of the CloudFront distribution.\n     */\n    private createAccessLogsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-access-logs`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_PREFERRED,\n            // When the stack is destroyed, we expect everything to be deleted\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): CloudFrontAccessLogsAnalysis {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n        return new CloudFrontAccessLogsAnalysis(this, `${this.resourceIdPrefix}-access-logs-analysis`, {\n            bucket: this.accessLogsBucket,\n            resourcePrefix: `${this.resourceIdPrefix}-access-logs`,\n            accessLogCookies: props.accessLogCookies,\n        });\n    }\n}\n"]}
|
|
553
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {AssetHashType, 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, HttpVersion,\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, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {\n    BlockPublicAccess,\n    Bucket,\n    BucketAccessControl,\n    BucketEncryption,\n    ObjectOwnership\n} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, 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 {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../NuxtAppStaticAssets\";\nimport * as fs from \"fs\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\nimport {writeFileSync} from \"fs\";\nimport {NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {CloudFrontAccessLogsAnalysis} from \"../access-logs-analysis/CloudFrontAccessLogsAnalysis\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport {execSync} from \"node:child_process\";\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack 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 to tag the static assets of the deployment\n     * to later be able to clean up outdated assets.\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: Bucket;\n\n    /**\n     * The S3 bucket where the access logs of the CloudFront distribution gets stored.\n     */\n    public accessLogsBucket: Bucket|undefined;\n\n    /**\n     * The S3 bucket where the sitemap assets gets stored.\n     */\n    public sitemapBucket: Bucket|undefined;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: 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 origin for the API gateway to route incoming requests to the Nuxt Lambda function.\n     */\n    private httpOrigin: HttpOrigin;\n\n    /**\n     * The cache policy for the Nuxt app route behaviors of the CloudFront distribution.\n     */\n    private appCachePolicy: CachePolicy;\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: NuxtServerAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = this.createDeploymentRevision(props);\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.rootDir ?? '.');\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n\n        if (props.enableAccessLogsAnalysis) {\n            this.accessLogsBucket = this.createAccessLogsBucket();\n            this.createAccessLogsAnalysis(props);\n        }\n\n        if (props.enableSitemap) {\n            this.sitemapBucket = this.createSitemapBucket();\n        }\n\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.httpOrigin = this.createNuxtAppHttpOrigin();\n        this.appCachePolicy = this.createNuxtAppCachePolicy(props)\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the current deployment revision file in the public folder of the Nuxt app to be accessible\n     * and returns the current revision.\n     */\n    private createDeploymentRevision(props: NuxtServerAppStackProps): string {\n        const revisionFilePath = `${props.rootDir ?? '.'}/.output/public/app-revision`;\n        const appRevision = new Date().toISOString();\n\n        writeFileSync(revisionFilePath, appRevision, {encoding: 'utf-8'});\n\n        return appRevision;\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(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\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            objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the bucket to store the sitemap assets of the Nuxt app.\n     *\n     * @private\n     */\n    private createSitemapBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-sitemap`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            accessControl: BucketAccessControl.PRIVATE,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            encryption: BucketEncryption.S3_MANAGED,\n            enforceSSL: true,\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 the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-app-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: `${props.entrypoint ?? 'index'}.handler`,\n            code: Code.fromAsset(`${props.rootDir ?? '.' }/.output/server`, {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            environment: {\n                NODE_OPTIONS: '--enable-source-maps',\n                ...JSON.parse(props.entrypointEnv ?? '{}'),\n            },\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup-function`;\n        const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            layers: [new LayerVersion(this, `${functionName}-layer`, {\n                layerVersionName: `${functionName}-layer`,\n                code: Code.fromAsset(functionDirPath, {\n                    assetHashType: AssetHashType.OUTPUT,\n                    bundling: {\n                        command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                        image: Runtime.NODEJS_20_X.bundlingImage,\n                        local: {\n                            tryBundle(outputDir: string): boolean {\n                                try {\n                                    execSync('yarn install', {\n                                        cwd: functionDirPath\n                                    });\n                                } catch {\n                                    return false;\n                                }\n                                fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {\n                                    recursive: true,\n                                });\n                                return true\n                            },\n                        }\n                    }\n                }),\n                compatibleRuntimes: [Runtime.NODEJS_20_X],\n                description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`\n            })],\n            handler: 'index.handler',\n            code: Code.fromAsset(functionDirPath, {\n                assetHashType: AssetHashType.OUTPUT,\n                bundling: {\n                    command: ['sh', '-c', 'echo \"Docker build not supported. Please install yarn.\"'],\n                    image: Runtime.NODEJS_20_X.bundlingImage,\n                    local: {\n                        tryBundle(outputDir: string): boolean {\n                            try {\n                                execSync('yarn install && yarn build', {\n                                    cwd: functionDirPath\n                                });\n                            } catch {\n                                return false;\n                            }\n                            fs.cpSync(`${functionDirPath}/build/app`, outputDir, {\n                                recursive: true\n                            });\n                            return true\n                        },\n                    }\n                }\n            }),\n            timeout: Duration.minutes(5),\n            memorySize: 128,\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,\n                OUTDATED_ASSETS_RETENTION_DAYS: `${props.outdatedAssetsRetentionDays ?? 30}`,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\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: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\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: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            httpVersion: HttpVersion.HTTP2_AND_3,\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.setupCloudFrontRouting(props),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n            logBucket: this.accessLogsBucket,\n            logFilePrefix: props.enableAccessLogsAnalysis ? CloudFrontAccessLogsAnalysis.getLogFilePrefix() : undefined,\n            logIncludesCookies: true,\n        });\n    }\n\n    /**\n     * Creates the CloudFront distribution behavior origin to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     */\n    private createNuxtAppHttpOrigin(): HttpOrigin {\n        return 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    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {};\n\n        // Specific ones first\n        if (props.enableApi) {\n            routingBehaviours = {...routingBehaviours, ...this.createApiRouteBehavior()};\n        }\n        if (props.enableSitemap) {\n            routingBehaviours = {...routingBehaviours, ...this.createSitemapRouteBehavior()};\n        }\n\n        routingBehaviours = {...routingBehaviours, ...this.createStaticAssetsRouteBehavior()};\n\n        return routingBehaviours;\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     */\n    private createNuxtAppCachePolicy(props: NuxtServerAppStackProps): CachePolicy {\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin and how the cache key is calculated.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.days(365),\n            queryStringBehavior: props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()),\n            headerBehavior: props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none(),\n            cookieBehavior: props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching Nuxt app API requests to the API gateway.\n     */\n    private createApiRouteBehavior(): Record<string, BehaviorOptions> {\n        const apiBehavior: BehaviorOptions = {\n            origin: this.httpOrigin,\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_ALL,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: this.appCachePolicy,\n            viewerProtocolPolicy: ViewerProtocolPolicy.HTTPS_ONLY\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['/api/*'] = apiBehavior;\n\n        return rules;\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            }),\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     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the sitemap assets\n     * to the S3 bucket that holds these sitemap assets.\n     *\n     * @private\n     */\n    private createSitemapRouteBehavior(): Record<string, BehaviorOptions> {\n        if (!this.sitemapBucket) {\n            throw new Error(\"Sitemap bucket must exist before creating sitemap route behavior.\");\n        }\n\n        const sitemapCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.sitemapBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\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        rules['*sitemap.xml'] = sitemapCacheConfig;\n        rules['*sitemap-gone.xml'] = sitemapCacheConfig;\n        rules['/sitemaps/*'] = sitemapCacheConfig;\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 with minimal storage load,\n     * we deploy the static assets of every deployment into the same folder but mark them with a deployment revision.\n     * By doing so, the files of previous deployments are retained to allow clients to continue to work with an older revision\n     * but gets cleaned up after a specified period of time via the cleanup Lambda function.\n     */\n    private configureDeployments(): BucketDeployment[] {\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                    exclude: asset.exclude,\n                })],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target.replace(/^\\/+/g, ''), // Remove leading slash\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? [`/${asset.pattern}`] : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n\n                metadata: {\n                    // Store build revision on every asset to allow cleanup of outdated assets\n                    revision: this.deploymentRevision,\n                },\n\n                // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory.\n                // Additionally, a high memory limit might speed up deployments.\n                memoryLimit: 1792\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: NuxtServerAppStackProps): 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: NuxtServerAppStackProps): 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 createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\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.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n\n    /**\n     * Creates a S3 bucket to store the access logs of the CloudFront distribution.\n     */\n    private createAccessLogsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-access-logs`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_PREFERRED,\n            // When the stack is destroyed, we expect everything to be deleted\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): CloudFrontAccessLogsAnalysis {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n        return new CloudFrontAccessLogsAnalysis(this, `${this.resourceIdPrefix}-access-logs-analysis`, {\n            bucket: this.accessLogsBucket,\n            resourcePrefix: `${this.resourceIdPrefix}-access-logs`,\n            accessLogCookies: props.accessLogCookies,\n        });\n    }\n}\n"]}
|
|
@@ -269,6 +269,7 @@ export class NuxtServerAppStack extends Stack {
|
|
|
269
269
|
*/
|
|
270
270
|
private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {
|
|
271
271
|
const functionName: string = `${this.resourceIdPrefix}-cleanup-function`;
|
|
272
|
+
const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');
|
|
272
273
|
|
|
273
274
|
const result: Function = new Function(this, functionName, {
|
|
274
275
|
functionName: functionName,
|
|
@@ -277,20 +278,22 @@ export class NuxtServerAppStack extends Stack {
|
|
|
277
278
|
architecture: Architecture.ARM_64,
|
|
278
279
|
layers: [new LayerVersion(this, `${functionName}-layer`, {
|
|
279
280
|
layerVersionName: `${functionName}-layer`,
|
|
280
|
-
code: Code.fromAsset(
|
|
281
|
+
code: Code.fromAsset(functionDirPath, {
|
|
281
282
|
assetHashType: AssetHashType.OUTPUT,
|
|
282
283
|
bundling: {
|
|
284
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
283
285
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
284
286
|
local: {
|
|
285
287
|
tryBundle(outputDir: string): boolean {
|
|
286
288
|
try {
|
|
287
|
-
execSync('
|
|
289
|
+
execSync('yarn install', {
|
|
290
|
+
cwd: functionDirPath
|
|
291
|
+
});
|
|
288
292
|
} catch {
|
|
289
293
|
return false;
|
|
290
294
|
}
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
recursive: true
|
|
295
|
+
fs.cpSync(`${functionDirPath}/node_modules`, `${outputDir}/nodejs/node_modules`, {
|
|
296
|
+
recursive: true,
|
|
294
297
|
});
|
|
295
298
|
return true
|
|
296
299
|
},
|
|
@@ -301,19 +304,21 @@ export class NuxtServerAppStack extends Stack {
|
|
|
301
304
|
description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`
|
|
302
305
|
})],
|
|
303
306
|
handler: 'index.handler',
|
|
304
|
-
code: Code.fromAsset(
|
|
307
|
+
code: Code.fromAsset(functionDirPath, {
|
|
305
308
|
assetHashType: AssetHashType.OUTPUT,
|
|
306
309
|
bundling: {
|
|
310
|
+
command: ['sh', '-c', 'echo "Docker build not supported. Please install yarn."'],
|
|
307
311
|
image: Runtime.NODEJS_20_X.bundlingImage,
|
|
308
312
|
local: {
|
|
309
313
|
tryBundle(outputDir: string): boolean {
|
|
310
314
|
try {
|
|
311
|
-
execSync('
|
|
315
|
+
execSync('yarn install && yarn build', {
|
|
316
|
+
cwd: functionDirPath
|
|
317
|
+
});
|
|
312
318
|
} catch {
|
|
313
319
|
return false;
|
|
314
320
|
}
|
|
315
|
-
|
|
316
|
-
fs.cpSync(path.join(__dirname, '../../functions/assets-cleanup/build/app'), outputDir, {
|
|
321
|
+
fs.cpSync(`${functionDirPath}/build/app`, outputDir, {
|
|
317
322
|
recursive: true
|
|
318
323
|
});
|
|
319
324
|
return true
|