cdk-nuxt 2.20.0 → 2.22.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/functions/assets-cleanup/build/app/index.js +40 -17
- package/lib/functions/assets-cleanup/build/app/index.js.map +1 -1
- package/lib/functions/assets-cleanup/index.js +41 -18
- package/lib/functions/assets-cleanup/index.ts +47 -22
- package/lib/stack/access-logs-analysis/AccessLogsTableConfig.js +5 -1
- package/lib/stack/access-logs-analysis/AccessLogsTableConfig.ts +4 -0
- package/lib/stack/server/NuxtServerAppStack.js +3 -3
- package/lib/stack/server/NuxtServerAppStack.ts +2 -2
- package/lib/stack/waf/CloudFrontWebAcl.js +20 -1
- package/lib/stack/waf/CloudFrontWebAcl.ts +19 -0
- package/lib/stack/waf/WafConfig.d.ts +2 -1
- package/lib/stack/waf/WafConfig.js +2 -2
- package/lib/stack/waf/WafConfig.ts +3 -2
- package/package.json +1 -1
|
@@ -60,55 +60,78 @@ exports.handler = async (event, context) => {
|
|
|
60
60
|
const currentRevision = await getCurrentRevision(client, bucketName);
|
|
61
61
|
const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);
|
|
62
62
|
console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);
|
|
63
|
+
console.log(`Current revision: ${currentRevision.toISOString()}, Retention days: ${retainAssetsInDays}`);
|
|
63
64
|
let assetKeysToDelete = [];
|
|
64
65
|
let lastToken = undefined;
|
|
66
|
+
let totalAssetsScanned = 0;
|
|
67
|
+
let totalBatches = 0;
|
|
68
|
+
const assetsDeleteResults = [];
|
|
65
69
|
do {
|
|
66
70
|
const curAssetsResult = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
67
71
|
Bucket: bucketName,
|
|
68
|
-
|
|
72
|
+
Prefix: '_nuxt/', // Only scan _nuxt/ directory where build assets are stored
|
|
73
|
+
MaxKeys: 1000,
|
|
69
74
|
ContinuationToken: lastToken,
|
|
70
75
|
}));
|
|
71
|
-
|
|
76
|
+
if (!curAssetsResult.Contents || curAssetsResult.Contents.length === 0) {
|
|
77
|
+
console.log('No assets found in current batch');
|
|
78
|
+
lastToken = curAssetsResult.NextContinuationToken;
|
|
79
|
+
continue;
|
|
80
|
+
}
|
|
81
|
+
totalAssetsScanned += curAssetsResult.Contents.length;
|
|
82
|
+
console.log(`Batch ${++totalBatches}: Scanning ${curAssetsResult.Contents.length} assets (Total: ${totalAssetsScanned})`);
|
|
83
|
+
// Read object metadata in blocks of 50 for better performance
|
|
72
84
|
let processableAssets = [...curAssetsResult.Contents];
|
|
73
85
|
while (processableAssets.length > 0) {
|
|
74
|
-
const assetsBatch = processableAssets.slice(0,
|
|
75
|
-
processableAssets = processableAssets.slice(
|
|
86
|
+
const assetsBatch = processableAssets.slice(0, 50);
|
|
87
|
+
processableAssets = processableAssets.slice(50);
|
|
76
88
|
const pendingMetadataRequests = assetsBatch.map(asset => client.send(new client_s3_1.HeadObjectCommand({
|
|
77
89
|
Bucket: bucketName,
|
|
78
90
|
Key: asset.Key,
|
|
79
|
-
}))
|
|
91
|
+
})).catch(error => {
|
|
92
|
+
console.warn(`Failed to get metadata for ${asset.Key}:`, error.message);
|
|
93
|
+
return { Metadata: {} }; // Return empty metadata on error
|
|
94
|
+
}));
|
|
80
95
|
const metadataResults = await Promise.all(pendingMetadataRequests);
|
|
81
96
|
// Assign metadata to assets
|
|
82
97
|
const metadataByAsset = metadataResults.map((metadataResult, index) => ({
|
|
83
98
|
key: assetsBatch[index].Key,
|
|
84
|
-
metadata: metadataResult.Metadata,
|
|
99
|
+
metadata: metadataResult.Metadata || {},
|
|
85
100
|
}));
|
|
86
101
|
const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);
|
|
87
102
|
assetKeysToDelete.push(...outdatedAssetKeys);
|
|
103
|
+
// Stream deletion: Delete in batches of 1000 to avoid memory issues and timeouts
|
|
104
|
+
if (assetKeysToDelete.length >= MAX_DELETE_OBJECT_KEYS) {
|
|
105
|
+
console.log(`Deleting batch of ${assetKeysToDelete.length} assets...`);
|
|
106
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
107
|
+
assetsDeleteResults.push(...results);
|
|
108
|
+
assetKeysToDelete = []; // Clear the array
|
|
109
|
+
}
|
|
88
110
|
}
|
|
89
111
|
lastToken = curAssetsResult.NextContinuationToken;
|
|
90
112
|
} while (lastToken !== undefined);
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
113
|
+
// Delete remaining assets
|
|
114
|
+
if (assetKeysToDelete.length > 0) {
|
|
115
|
+
console.log(`Deleting final batch of ${assetKeysToDelete.length} assets...`);
|
|
116
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
117
|
+
assetsDeleteResults.push(...results);
|
|
94
118
|
}
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
const currentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
100
|
-
if (currentError) {
|
|
119
|
+
// Check for deletion errors and report
|
|
120
|
+
const failed = assetsDeleteResults.reduce((previousResult, currentResult) => {
|
|
121
|
+
const hasCurrentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
122
|
+
if (hasCurrentError) {
|
|
101
123
|
console.error('Failed to delete outdated static assets', currentResult.Errors);
|
|
102
124
|
}
|
|
103
|
-
return previousResult ||
|
|
125
|
+
return previousResult || hasCurrentError;
|
|
104
126
|
}, false);
|
|
105
127
|
if (failed) {
|
|
106
128
|
throw new Error('Failed to delete outdated static assets');
|
|
107
129
|
}
|
|
108
|
-
console.log(
|
|
130
|
+
console.log(`Cleanup of old static assets finished. Total assets scanned: ${totalAssetsScanned}`);
|
|
109
131
|
}
|
|
110
132
|
catch (error) {
|
|
111
133
|
console.error('### unexpected runtime error ###', error);
|
|
134
|
+
throw error; // Re-throw to mark Lambda as failed
|
|
112
135
|
}
|
|
113
136
|
};
|
|
114
137
|
//# sourceMappingURL=index.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"/","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n DeleteObjectsCommand,\n DeleteObjectsCommandOutput,\n GetObjectCommand, HeadObjectCommand,\n ListObjectsV2Command,\n S3Client\n} from \"@aws-sdk/client-s3\";\nimport type {ListObjectsV2CommandOutput} from \"@aws-sdk/client-s3\";\nimport {text} from 'stream/consumers';\nimport {Readable} from \"stream\";\n\ninterface AssetMetadata {\n readonly key: string;\n readonly metadata: {[key: string]: string};\n}\n\nconst MAX_DELETE_OBJECT_KEYS = 1000;\nconst ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;\n\n/**\n * Returns the current deployment revision (build timestamp).\n */\nconst getCurrentRevision = async (s3Client: S3Client, bucketName: string): Promise<Date> => {\n const revisionFile = await s3Client.send(\n new GetObjectCommand({\n Bucket: bucketName,\n Key: 'app-revision',\n })\n );\n\n return new Date(await text(revisionFile.Body as Readable));\n};\n\n/**\n * Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.\n */\nconst filterOutdatedAssetKeys = (metadataResults: AssetMetadata[], returnOlderThan: Date): string[] => {\n return metadataResults\n .filter(assetMetadata =>\n assetMetadata.metadata.revision\n ? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()\n : false\n )\n .map(filteredAssets => filteredAssets.key);\n};\n\n/**\n * Deletes the given assets.\n */\nconst deleteAssets = async (\n assetKeys: string[],\n s3Client: S3Client,\n bucketName: string\n): Promise<DeleteObjectsCommandOutput[]> => {\n let remainingAssetKeysToDelete = [...assetKeys];\n const pendingDeletes = [];\n\n while (remainingAssetKeysToDelete.length > 0) {\n const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);\n remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);\n\n console.log('Deleting assets:', curDeleteBatch);\n\n const pendingDelete = s3Client.send(\n new DeleteObjectsCommand({\n Bucket: bucketName,\n Delete: {\n Objects: curDeleteBatch.map(outdatedKey => {\n return {Key: outdatedKey};\n }),\n },\n })\n );\n pendingDeletes.push(pendingDelete);\n }\n\n return await Promise.all(pendingDeletes);\n};\n\nexports.handler = async (event: any, context: any) => {\n try {\n const client = new S3Client({region: process.env.AWS_REGION});\n const bucketName = process.env.STATIC_ASSETS_BUCKET;\n if (!bucketName) {\n throw new Error(\"Static asset's bucket name not specified in environment!\");\n }\n\n if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {\n throw new Error('Retain duration of static assets not specified!');\n }\n const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);\n const currentRevision = await getCurrentRevision(client, bucketName);\n const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);\n\n console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);\n\n let assetKeysToDelete: string[] = [];\n let lastToken = undefined;\n\n do {\n const curAssetsResult: ListObjectsV2CommandOutput = await client.send(\n new ListObjectsV2Command({\n Bucket: bucketName,\n MaxKeys: 250,\n ContinuationToken: lastToken,\n })\n );\n\n // Read object metadata in blocks of 10\n let processableAssets = [...curAssetsResult.Contents!];\n\n while (processableAssets.length > 0) {\n const assetsBatch = processableAssets.slice(0, 10);\n processableAssets = processableAssets.slice(10);\n\n const pendingMetadataRequests = assetsBatch.map(asset =>\n client.send(\n new HeadObjectCommand({\n Bucket: bucketName,\n Key: asset.Key,\n })\n )\n );\n\n const metadataResults = await Promise.all(pendingMetadataRequests);\n\n // Assign metadata to assets\n const metadataByAsset: AssetMetadata[] = (metadataResults.map((metadataResult, index) => ({\n key: assetsBatch[index].Key,\n metadata: metadataResult.Metadata,\n })) as AssetMetadata[]);\n\n const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);\n assetKeysToDelete.push(...outdatedAssetKeys);\n }\n lastToken = curAssetsResult.NextContinuationToken;\n } while (lastToken !== undefined);\n\n if (assetKeysToDelete.length === 0) {\n console.log('No outdated assets to delete found');\n return;\n }\n\n console.log('Deleting ' + assetKeysToDelete.length + ' assets...');\n\n // Delete outdated assets (max. 1000 allowed per request)\n const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n const failed = results.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {\n const currentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);\n if (currentError) {\n console.error('Failed to delete outdated static assets', currentResult.Errors);\n }\n return previousResult || currentError;\n }, false);\n\n if (failed) {\n throw new Error('Failed to delete outdated static assets');\n }\n\n console.log('Cleanup of old static assets finished');\n } catch (error) {\n console.error('### unexpected runtime error ###', error);\n }\n};"]}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"/","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n DeleteObjectsCommand,\n DeleteObjectsCommandOutput,\n GetObjectCommand, HeadObjectCommand,\n ListObjectsV2Command,\n S3Client\n} from \"@aws-sdk/client-s3\";\nimport type {ListObjectsV2CommandOutput} from \"@aws-sdk/client-s3\";\nimport {text} from 'stream/consumers';\nimport {Readable} from \"stream\";\n\ninterface AssetMetadata {\n readonly key: string;\n readonly metadata: {[key: string]: string};\n}\n\nconst MAX_DELETE_OBJECT_KEYS = 1000;\nconst ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;\n\n/**\n * Returns the current deployment revision (build timestamp).\n */\nconst getCurrentRevision = async (s3Client: S3Client, bucketName: string): Promise<Date> => {\n const revisionFile = await s3Client.send(\n new GetObjectCommand({\n Bucket: bucketName,\n Key: 'app-revision',\n })\n );\n\n return new Date(await text(revisionFile.Body as Readable));\n};\n\n/**\n * Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.\n */\nconst filterOutdatedAssetKeys = (metadataResults: AssetMetadata[], returnOlderThan: Date): string[] => {\n return metadataResults\n .filter(assetMetadata =>\n assetMetadata.metadata.revision\n ? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()\n : false\n )\n .map(filteredAssets => filteredAssets.key);\n};\n\n/**\n * Deletes the given assets.\n */\nconst deleteAssets = async (\n assetKeys: string[],\n s3Client: S3Client,\n bucketName: string\n): Promise<DeleteObjectsCommandOutput[]> => {\n let remainingAssetKeysToDelete = [...assetKeys];\n const pendingDeletes = [];\n\n while (remainingAssetKeysToDelete.length > 0) {\n const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);\n remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);\n\n console.log('Deleting assets:', curDeleteBatch);\n\n const pendingDelete = s3Client.send(\n new DeleteObjectsCommand({\n Bucket: bucketName,\n Delete: {\n Objects: curDeleteBatch.map(outdatedKey => {\n return {Key: outdatedKey};\n }),\n },\n })\n );\n pendingDeletes.push(pendingDelete);\n }\n\n return await Promise.all(pendingDeletes);\n};\n\nexports.handler = async (event: any, context: any) => {\n try {\n const client = new S3Client({region: process.env.AWS_REGION});\n const bucketName = process.env.STATIC_ASSETS_BUCKET;\n if (!bucketName) {\n throw new Error(\"Static asset's bucket name not specified in environment!\");\n }\n\n if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {\n throw new Error('Retain duration of static assets not specified!');\n }\n const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);\n const currentRevision = await getCurrentRevision(client, bucketName);\n const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);\n\n console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);\n console.log(`Current revision: ${currentRevision.toISOString()}, Retention days: ${retainAssetsInDays}`);\n\n let assetKeysToDelete: string[] = [];\n let lastToken = undefined;\n let totalAssetsScanned = 0;\n let totalBatches = 0;\n const assetsDeleteResults: DeleteObjectsCommandOutput[] = [];\n\n do {\n const curAssetsResult: ListObjectsV2CommandOutput = await client.send(\n new ListObjectsV2Command({\n Bucket: bucketName,\n Prefix: '_nuxt/', // Only scan _nuxt/ directory where build assets are stored\n MaxKeys: 1000,\n ContinuationToken: lastToken,\n })\n );\n\n if (!curAssetsResult.Contents || curAssetsResult.Contents.length === 0) {\n console.log('No assets found in current batch');\n lastToken = curAssetsResult.NextContinuationToken;\n continue;\n }\n\n totalAssetsScanned += curAssetsResult.Contents.length;\n console.log(`Batch ${++totalBatches}: Scanning ${curAssetsResult.Contents.length} assets (Total: ${totalAssetsScanned})`);\n\n // Read object metadata in blocks of 50 for better performance\n let processableAssets = [...curAssetsResult.Contents];\n\n while (processableAssets.length > 0) {\n const assetsBatch = processableAssets.slice(0, 50);\n processableAssets = processableAssets.slice(50);\n\n const pendingMetadataRequests = assetsBatch.map(asset =>\n client.send(\n new HeadObjectCommand({\n Bucket: bucketName,\n Key: asset.Key,\n })\n ).catch(error => {\n console.warn(`Failed to get metadata for ${asset.Key}:`, error.message);\n return { Metadata: {} }; // Return empty metadata on error\n })\n );\n\n const metadataResults = await Promise.all(pendingMetadataRequests);\n\n // Assign metadata to assets\n const metadataByAsset: AssetMetadata[] = metadataResults.map((metadataResult, index) => ({\n key: assetsBatch[index].Key!,\n metadata: metadataResult.Metadata || {},\n }));\n\n const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);\n assetKeysToDelete.push(...outdatedAssetKeys);\n\n // Stream deletion: Delete in batches of 1000 to avoid memory issues and timeouts\n if (assetKeysToDelete.length >= MAX_DELETE_OBJECT_KEYS) {\n console.log(`Deleting batch of ${assetKeysToDelete.length} assets...`);\n const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n assetsDeleteResults.push(...results);\n assetKeysToDelete = []; // Clear the array\n }\n }\n lastToken = curAssetsResult.NextContinuationToken;\n } while (lastToken !== undefined);\n\n // Delete remaining assets\n if (assetKeysToDelete.length > 0) {\n console.log(`Deleting final batch of ${assetKeysToDelete.length} assets...`);\n const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n assetsDeleteResults.push(...results);\n }\n\n // Check for deletion errors and report\n const failed = assetsDeleteResults.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {\n const hasCurrentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);\n if (hasCurrentError) {\n console.error('Failed to delete outdated static assets', currentResult.Errors);\n }\n return previousResult || hasCurrentError;\n }, false);\n\n if (failed) {\n throw new Error('Failed to delete outdated static assets');\n }\n\n console.log(`Cleanup of old static assets finished. Total assets scanned: ${totalAssetsScanned}`);\n } catch (error) {\n console.error('### unexpected runtime error ###', error);\n throw error; // Re-throw to mark Lambda as failed\n }\n};"]}
|
|
@@ -60,55 +60,78 @@ exports.handler = async (event, context) => {
|
|
|
60
60
|
const currentRevision = await getCurrentRevision(client, bucketName);
|
|
61
61
|
const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);
|
|
62
62
|
console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);
|
|
63
|
+
console.log(`Current revision: ${currentRevision.toISOString()}, Retention days: ${retainAssetsInDays}`);
|
|
63
64
|
let assetKeysToDelete = [];
|
|
64
65
|
let lastToken = undefined;
|
|
66
|
+
let totalAssetsScanned = 0;
|
|
67
|
+
let totalBatches = 0;
|
|
68
|
+
const assetsDeleteResults = [];
|
|
65
69
|
do {
|
|
66
70
|
const curAssetsResult = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
67
71
|
Bucket: bucketName,
|
|
68
|
-
|
|
72
|
+
Prefix: '_nuxt/', // Only scan _nuxt/ directory where build assets are stored
|
|
73
|
+
MaxKeys: 1000,
|
|
69
74
|
ContinuationToken: lastToken,
|
|
70
75
|
}));
|
|
71
|
-
|
|
76
|
+
if (!curAssetsResult.Contents || curAssetsResult.Contents.length === 0) {
|
|
77
|
+
console.log('No assets found in current batch');
|
|
78
|
+
lastToken = curAssetsResult.NextContinuationToken;
|
|
79
|
+
continue;
|
|
80
|
+
}
|
|
81
|
+
totalAssetsScanned += curAssetsResult.Contents.length;
|
|
82
|
+
console.log(`Batch ${++totalBatches}: Scanning ${curAssetsResult.Contents.length} assets (Total: ${totalAssetsScanned})`);
|
|
83
|
+
// Read object metadata in blocks of 50 for better performance
|
|
72
84
|
let processableAssets = [...curAssetsResult.Contents];
|
|
73
85
|
while (processableAssets.length > 0) {
|
|
74
|
-
const assetsBatch = processableAssets.slice(0,
|
|
75
|
-
processableAssets = processableAssets.slice(
|
|
86
|
+
const assetsBatch = processableAssets.slice(0, 50);
|
|
87
|
+
processableAssets = processableAssets.slice(50);
|
|
76
88
|
const pendingMetadataRequests = assetsBatch.map(asset => client.send(new client_s3_1.HeadObjectCommand({
|
|
77
89
|
Bucket: bucketName,
|
|
78
90
|
Key: asset.Key,
|
|
79
|
-
}))
|
|
91
|
+
})).catch(error => {
|
|
92
|
+
console.warn(`Failed to get metadata for ${asset.Key}:`, error.message);
|
|
93
|
+
return { Metadata: {} }; // Return empty metadata on error
|
|
94
|
+
}));
|
|
80
95
|
const metadataResults = await Promise.all(pendingMetadataRequests);
|
|
81
96
|
// Assign metadata to assets
|
|
82
97
|
const metadataByAsset = metadataResults.map((metadataResult, index) => ({
|
|
83
98
|
key: assetsBatch[index].Key,
|
|
84
|
-
metadata: metadataResult.Metadata,
|
|
99
|
+
metadata: metadataResult.Metadata || {},
|
|
85
100
|
}));
|
|
86
101
|
const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);
|
|
87
102
|
assetKeysToDelete.push(...outdatedAssetKeys);
|
|
103
|
+
// Stream deletion: Delete in batches of 1000 to avoid memory issues and timeouts
|
|
104
|
+
if (assetKeysToDelete.length >= MAX_DELETE_OBJECT_KEYS) {
|
|
105
|
+
console.log(`Deleting batch of ${assetKeysToDelete.length} assets...`);
|
|
106
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
107
|
+
assetsDeleteResults.push(...results);
|
|
108
|
+
assetKeysToDelete = []; // Clear the array
|
|
109
|
+
}
|
|
88
110
|
}
|
|
89
111
|
lastToken = curAssetsResult.NextContinuationToken;
|
|
90
112
|
} while (lastToken !== undefined);
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
113
|
+
// Delete remaining assets
|
|
114
|
+
if (assetKeysToDelete.length > 0) {
|
|
115
|
+
console.log(`Deleting final batch of ${assetKeysToDelete.length} assets...`);
|
|
116
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
117
|
+
assetsDeleteResults.push(...results);
|
|
94
118
|
}
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
const currentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
100
|
-
if (currentError) {
|
|
119
|
+
// Check for deletion errors and report
|
|
120
|
+
const failed = assetsDeleteResults.reduce((previousResult, currentResult) => {
|
|
121
|
+
const hasCurrentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
122
|
+
if (hasCurrentError) {
|
|
101
123
|
console.error('Failed to delete outdated static assets', currentResult.Errors);
|
|
102
124
|
}
|
|
103
|
-
return previousResult ||
|
|
125
|
+
return previousResult || hasCurrentError;
|
|
104
126
|
}, false);
|
|
105
127
|
if (failed) {
|
|
106
128
|
throw new Error('Failed to delete outdated static assets');
|
|
107
129
|
}
|
|
108
|
-
console.log(
|
|
130
|
+
console.log(`Cleanup of old static assets finished. Total assets scanned: ${totalAssetsScanned}`);
|
|
109
131
|
}
|
|
110
132
|
catch (error) {
|
|
111
133
|
console.error('### unexpected runtime error ###', error);
|
|
134
|
+
throw error; // Re-throw to mark Lambda as failed
|
|
112
135
|
}
|
|
113
136
|
};
|
|
114
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;AAAA,kDAM4B;AAE5B,gDAAsC;AAQtC,MAAM,sBAAsB,GAAG,IAAI,CAAC;AACpC,MAAM,uBAAuB,GAAG,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC;AAEpD;;GAEG;AACH,MAAM,kBAAkB,GAAG,KAAK,EAAE,QAAkB,EAAE,UAAkB,EAAiB,EAAE;IACvF,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,IAAI,CACpC,IAAI,4BAAgB,CAAC;QACjB,MAAM,EAAE,UAAU;QAClB,GAAG,EAAE,cAAc;KACtB,CAAC,CACL,CAAC;IAEF,OAAO,IAAI,IAAI,CAAC,MAAM,IAAA,gBAAI,EAAC,YAAY,CAAC,IAAgB,CAAC,CAAC,CAAC;AAC/D,CAAC,CAAC;AAEF;;GAEG;AACH,MAAM,uBAAuB,GAAG,CAAC,eAAgC,EAAE,eAAqB,EAAY,EAAE;IAClG,OAAO,eAAe;SACjB,MAAM,CAAC,aAAa,CAAC,EAAE,CACpB,aAAa,CAAC,QAAQ,CAAC,QAAQ;QAC3B,CAAC,CAAC,IAAI,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,OAAO,EAAE,IAAI,eAAe,CAAC,OAAO,EAAE;QAClF,CAAC,CAAC,KAAK,CACd;SACA,GAAG,CAAC,cAAc,CAAC,EAAE,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;AACnD,CAAC,CAAC;AAEF;;GAEG;AACH,MAAM,YAAY,GAAG,KAAK,EACtB,SAAmB,EACnB,QAAkB,EAClB,UAAkB,EACmB,EAAE;IACvC,IAAI,0BAA0B,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC;IAChD,MAAM,cAAc,GAAG,EAAE,CAAC;IAE1B,OAAO,0BAA0B,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC3C,MAAM,cAAc,GAAG,0BAA0B,CAAC,KAAK,CAAC,CAAC,EAAE,sBAAsB,CAAC,CAAC;QACnF,0BAA0B,GAAG,0BAA0B,CAAC,KAAK,CAAC,sBAAsB,CAAC,CAAC;QAEtF,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,cAAc,CAAC,CAAC;QAEhD,MAAM,aAAa,GAAG,QAAQ,CAAC,IAAI,CAC/B,IAAI,gCAAoB,CAAC;YACrB,MAAM,EAAE,UAAU;YAClB,MAAM,EAAE;gBACJ,OAAO,EAAE,cAAc,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE;oBACtC,OAAO,EAAC,GAAG,EAAE,WAAW,EAAC,CAAC;gBAC9B,CAAC,CAAC;aACL;SACJ,CAAC,CACL,CAAC;QACF,cAAc,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IACvC,CAAC;IAED,OAAO,MAAM,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;AAC7C,CAAC,CAAC;AAEF,OAAO,CAAC,OAAO,GAAG,KAAK,EAAE,KAAU,EAAE,OAAY,EAAE,EAAE;IACjD,IAAI,CAAC;QACD,MAAM,MAAM,GAAG,IAAI,oBAAQ,CAAC,EAAC,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,UAAU,EAAC,CAAC,CAAC;QAC9D,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC;QACpD,IAAI,CAAC,UAAU,EAAE,CAAC;YACd,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;QAChF,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,8BAA8B,EAAE,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;QACvE,CAAC;QACD,MAAM,kBAAkB,GAAG,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAC;QACvF,MAAM,eAAe,GAAG,MAAM,kBAAkB,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;QACrE,MAAM,eAAe,GAAG,IAAI,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,GAAG,kBAAkB,GAAG,uBAAuB,CAAC,CAAC;QAE3G,OAAO,CAAC,GAAG,CAAC,gDAAgD,eAAe,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC;QAEhG,IAAI,iBAAiB,GAAa,EAAE,CAAC;QACrC,IAAI,SAAS,GAAG,SAAS,CAAC;QAE1B,GAAG,CAAC;YACA,MAAM,eAAe,GAA+B,MAAM,MAAM,CAAC,IAAI,CACjE,IAAI,gCAAoB,CAAC;gBACrB,MAAM,EAAE,UAAU;gBAClB,OAAO,EAAE,GAAG;gBACZ,iBAAiB,EAAE,SAAS;aAC/B,CAAC,CACL,CAAC;YAEF,uCAAuC;YACvC,IAAI,iBAAiB,GAAG,CAAC,GAAG,eAAe,CAAC,QAAS,CAAC,CAAC;YAEvD,OAAO,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAClC,MAAM,WAAW,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACnD,iBAAiB,GAAG,iBAAiB,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBAEhD,MAAM,uBAAuB,GAAG,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CACpD,MAAM,CAAC,IAAI,CACP,IAAI,6BAAiB,CAAC;oBAClB,MAAM,EAAE,UAAU;oBAClB,GAAG,EAAE,KAAK,CAAC,GAAG;iBACjB,CAAC,CACL,CACJ,CAAC;gBAEF,MAAM,eAAe,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;gBAEnE,4BAA4B;gBAC5B,MAAM,eAAe,GAAqB,eAAe,CAAC,GAAG,CAAC,CAAC,cAAc,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;oBACtF,GAAG,EAAE,WAAW,CAAC,KAAK,CAAC,CAAC,GAAG;oBAC3B,QAAQ,EAAE,cAAc,CAAC,QAAQ;iBACpC,CAAC,CAAqB,CAAC;gBAExB,MAAM,iBAAiB,GAAG,uBAAuB,CAAC,eAAe,EAAE,eAAe,CAAC,CAAC;gBACpF,iBAAiB,CAAC,IAAI,CAAC,GAAG,iBAAiB,CAAC,CAAC;YACjD,CAAC;YACD,SAAS,GAAG,eAAe,CAAC,qBAAqB,CAAC;QACtD,CAAC,QAAQ,SAAS,KAAK,SAAS,EAAE;QAElC,IAAI,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjC,OAAO,CAAC,GAAG,CAAC,oCAAoC,CAAC,CAAC;YAClD,OAAO;QACX,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,WAAW,GAAG,iBAAiB,CAAC,MAAM,GAAG,YAAY,CAAC,CAAC;QAEnE,yDAAyD;QACzD,MAAM,OAAO,GAAG,MAAM,YAAY,CAAC,iBAAiB,EAAE,MAAM,EAAE,UAAU,CAAC,CAAC;QAC1E,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,cAAuB,EAAE,aAAyC,EAAW,EAAE;YAC1G,MAAM,YAAY,GAAY,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,IAAI,aAAa,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAC1F,IAAI,YAAY,EAAE,CAAC;gBACf,OAAO,CAAC,KAAK,CAAC,yCAAyC,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC;YACnF,CAAC;YACD,OAAO,cAAc,IAAI,YAAY,CAAC;QAC1C,CAAC,EAAE,KAAK,CAAC,CAAC;QAEV,IAAI,MAAM,EAAE,CAAC;YACT,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;QAC/D,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,uCAAuC,CAAC,CAAC;IACzD,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,kCAAkC,EAAE,KAAK,CAAC,CAAC;IAC7D,CAAC;AACL,CAAC,CAAC","sourcesContent":["import {\n    DeleteObjectsCommand,\n    DeleteObjectsCommandOutput,\n    GetObjectCommand, HeadObjectCommand,\n    ListObjectsV2Command,\n    S3Client\n} from \"@aws-sdk/client-s3\";\nimport type {ListObjectsV2CommandOutput} from \"@aws-sdk/client-s3\";\nimport {text} from 'stream/consumers';\nimport {Readable} from \"stream\";\n\ninterface AssetMetadata {\n    readonly key: string;\n    readonly metadata: {[key: string]: string};\n}\n\nconst MAX_DELETE_OBJECT_KEYS = 1000;\nconst ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;\n\n/**\n * Returns the current deployment revision (build timestamp).\n */\nconst getCurrentRevision = async (s3Client: S3Client, bucketName: string): Promise<Date> => {\n    const revisionFile = await s3Client.send(\n        new GetObjectCommand({\n            Bucket: bucketName,\n            Key: 'app-revision',\n        })\n    );\n\n    return new Date(await text(revisionFile.Body as Readable));\n};\n\n/**\n * Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.\n */\nconst filterOutdatedAssetKeys = (metadataResults: AssetMetadata[], returnOlderThan: Date): string[] => {\n    return metadataResults\n        .filter(assetMetadata =>\n            assetMetadata.metadata.revision\n                ? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()\n                : false\n        )\n        .map(filteredAssets => filteredAssets.key);\n};\n\n/**\n * Deletes the given assets.\n */\nconst deleteAssets = async (\n    assetKeys: string[],\n    s3Client: S3Client,\n    bucketName: string\n): Promise<DeleteObjectsCommandOutput[]> => {\n    let remainingAssetKeysToDelete = [...assetKeys];\n    const pendingDeletes = [];\n\n    while (remainingAssetKeysToDelete.length > 0) {\n        const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);\n        remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);\n\n        console.log('Deleting assets:', curDeleteBatch);\n\n        const pendingDelete = s3Client.send(\n            new DeleteObjectsCommand({\n                Bucket: bucketName,\n                Delete: {\n                    Objects: curDeleteBatch.map(outdatedKey => {\n                        return {Key: outdatedKey};\n                    }),\n                },\n            })\n        );\n        pendingDeletes.push(pendingDelete);\n    }\n\n    return await Promise.all(pendingDeletes);\n};\n\nexports.handler = async (event: any, context: any) => {\n    try {\n        const client = new S3Client({region: process.env.AWS_REGION});\n        const bucketName = process.env.STATIC_ASSETS_BUCKET;\n        if (!bucketName) {\n            throw new Error(\"Static asset's bucket name not specified in environment!\");\n        }\n\n        if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {\n            throw new Error('Retain duration of static assets not specified!');\n        }\n        const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);\n        const currentRevision = await getCurrentRevision(client, bucketName);\n        const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);\n\n        console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);\n\n        let assetKeysToDelete: string[] = [];\n        let lastToken = undefined;\n\n        do {\n            const curAssetsResult: ListObjectsV2CommandOutput = await client.send(\n                new ListObjectsV2Command({\n                    Bucket: bucketName,\n                    MaxKeys: 250,\n                    ContinuationToken: lastToken,\n                })\n            );\n\n            // Read object metadata in blocks of 10\n            let processableAssets = [...curAssetsResult.Contents!];\n\n            while (processableAssets.length > 0) {\n                const assetsBatch = processableAssets.slice(0, 10);\n                processableAssets = processableAssets.slice(10);\n\n                const pendingMetadataRequests = assetsBatch.map(asset =>\n                    client.send(\n                        new HeadObjectCommand({\n                            Bucket: bucketName,\n                            Key: asset.Key,\n                        })\n                    )\n                );\n\n                const metadataResults = await Promise.all(pendingMetadataRequests);\n\n                // Assign metadata to assets\n                const metadataByAsset: AssetMetadata[] = (metadataResults.map((metadataResult, index) => ({\n                    key: assetsBatch[index].Key,\n                    metadata: metadataResult.Metadata,\n                })) as AssetMetadata[]);\n\n                const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);\n                assetKeysToDelete.push(...outdatedAssetKeys);\n            }\n            lastToken = curAssetsResult.NextContinuationToken;\n        } while (lastToken !== undefined);\n\n        if (assetKeysToDelete.length === 0) {\n            console.log('No outdated assets to delete found');\n            return;\n        }\n\n        console.log('Deleting ' + assetKeysToDelete.length + ' assets...');\n\n        // Delete outdated assets (max. 1000 allowed per request)\n        const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n        const failed = results.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {\n            const currentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);\n            if (currentError) {\n                console.error('Failed to delete outdated static assets', currentResult.Errors);\n            }\n            return previousResult || currentError;\n        }, false);\n\n        if (failed) {\n            throw new Error('Failed to delete outdated static assets');\n        }\n\n        console.log('Cleanup of old static assets finished');\n    } catch (error) {\n        console.error('### unexpected runtime error ###', error);\n    }\n};"]}
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    DeleteObjectsCommand,\n    DeleteObjectsCommandOutput,\n    GetObjectCommand, HeadObjectCommand,\n    ListObjectsV2Command,\n    S3Client\n} from \"@aws-sdk/client-s3\";\nimport type {ListObjectsV2CommandOutput} from \"@aws-sdk/client-s3\";\nimport {text} from 'stream/consumers';\nimport {Readable} from \"stream\";\n\ninterface AssetMetadata {\n    readonly key: string;\n    readonly metadata: {[key: string]: string};\n}\n\nconst MAX_DELETE_OBJECT_KEYS = 1000;\nconst ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;\n\n/**\n * Returns the current deployment revision (build timestamp).\n */\nconst getCurrentRevision = async (s3Client: S3Client, bucketName: string): Promise<Date> => {\n    const revisionFile = await s3Client.send(\n        new GetObjectCommand({\n            Bucket: bucketName,\n            Key: 'app-revision',\n        })\n    );\n\n    return new Date(await text(revisionFile.Body as Readable));\n};\n\n/**\n * Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.\n */\nconst filterOutdatedAssetKeys = (metadataResults: AssetMetadata[], returnOlderThan: Date): string[] => {\n    return metadataResults\n        .filter(assetMetadata =>\n            assetMetadata.metadata.revision\n                ? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()\n                : false\n        )\n        .map(filteredAssets => filteredAssets.key);\n};\n\n/**\n * Deletes the given assets.\n */\nconst deleteAssets = async (\n    assetKeys: string[],\n    s3Client: S3Client,\n    bucketName: string\n): Promise<DeleteObjectsCommandOutput[]> => {\n    let remainingAssetKeysToDelete = [...assetKeys];\n    const pendingDeletes = [];\n\n    while (remainingAssetKeysToDelete.length > 0) {\n        const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);\n        remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);\n\n        console.log('Deleting assets:', curDeleteBatch);\n\n        const pendingDelete = s3Client.send(\n            new DeleteObjectsCommand({\n                Bucket: bucketName,\n                Delete: {\n                    Objects: curDeleteBatch.map(outdatedKey => {\n                        return {Key: outdatedKey};\n                    }),\n                },\n            })\n        );\n        pendingDeletes.push(pendingDelete);\n    }\n\n    return await Promise.all(pendingDeletes);\n};\n\nexports.handler = async (event: any, context: any) => {\n    try {\n        const client = new S3Client({region: process.env.AWS_REGION});\n        const bucketName = process.env.STATIC_ASSETS_BUCKET;\n        if (!bucketName) {\n            throw new Error(\"Static asset's bucket name not specified in environment!\");\n        }\n\n        if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {\n            throw new Error('Retain duration of static assets not specified!');\n        }\n        const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);\n        const currentRevision = await getCurrentRevision(client, bucketName);\n        const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);\n\n        console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);\n        console.log(`Current revision: ${currentRevision.toISOString()}, Retention days: ${retainAssetsInDays}`);\n\n        let assetKeysToDelete: string[] = [];\n        let lastToken = undefined;\n        let totalAssetsScanned = 0;\n        let totalBatches = 0;\n        const assetsDeleteResults: DeleteObjectsCommandOutput[] = [];\n\n        do {\n            const curAssetsResult: ListObjectsV2CommandOutput = await client.send(\n                new ListObjectsV2Command({\n                    Bucket: bucketName,\n                    Prefix: '_nuxt/', // Only scan _nuxt/ directory where build assets are stored\n                    MaxKeys: 1000,\n                    ContinuationToken: lastToken,\n                })\n            );\n\n            if (!curAssetsResult.Contents || curAssetsResult.Contents.length === 0) {\n                console.log('No assets found in current batch');\n                lastToken = curAssetsResult.NextContinuationToken;\n                continue;\n            }\n\n            totalAssetsScanned += curAssetsResult.Contents.length;\n            console.log(`Batch ${++totalBatches}: Scanning ${curAssetsResult.Contents.length} assets (Total: ${totalAssetsScanned})`);\n\n            // Read object metadata in blocks of 50 for better performance\n            let processableAssets = [...curAssetsResult.Contents];\n\n            while (processableAssets.length > 0) {\n                const assetsBatch = processableAssets.slice(0, 50);\n                processableAssets = processableAssets.slice(50);\n\n                const pendingMetadataRequests = assetsBatch.map(asset =>\n                    client.send(\n                        new HeadObjectCommand({\n                            Bucket: bucketName,\n                            Key: asset.Key,\n                        })\n                    ).catch(error => {\n                        console.warn(`Failed to get metadata for ${asset.Key}:`, error.message);\n                        return { Metadata: {} }; // Return empty metadata on error\n                    })\n                );\n\n                const metadataResults = await Promise.all(pendingMetadataRequests);\n\n                // Assign metadata to assets\n                const metadataByAsset: AssetMetadata[] = metadataResults.map((metadataResult, index) => ({\n                    key: assetsBatch[index].Key!,\n                    metadata: metadataResult.Metadata || {},\n                }));\n\n                const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);\n                assetKeysToDelete.push(...outdatedAssetKeys);\n\n                // Stream deletion: Delete in batches of 1000 to avoid memory issues and timeouts\n                if (assetKeysToDelete.length >= MAX_DELETE_OBJECT_KEYS) {\n                    console.log(`Deleting batch of ${assetKeysToDelete.length} assets...`);\n                    const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n                    assetsDeleteResults.push(...results);\n                    assetKeysToDelete = []; // Clear the array\n                }\n            }\n            lastToken = curAssetsResult.NextContinuationToken;\n        } while (lastToken !== undefined);\n\n        // Delete remaining assets\n        if (assetKeysToDelete.length > 0) {\n            console.log(`Deleting final batch of ${assetKeysToDelete.length} assets...`);\n            const results = await deleteAssets(assetKeysToDelete, client, bucketName);\n            assetsDeleteResults.push(...results);\n        }\n\n        // Check for deletion errors and report\n        const failed = assetsDeleteResults.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {\n            const hasCurrentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);\n            if (hasCurrentError) {\n                console.error('Failed to delete outdated static assets', currentResult.Errors);\n            }\n            return previousResult || hasCurrentError;\n        }, false);\n\n        if (failed) {\n            throw new Error('Failed to delete outdated static assets');\n        }\n\n        console.log(`Cleanup of old static assets finished. Total assets scanned: ${totalAssetsScanned}`);\n    } catch (error) {\n        console.error('### unexpected runtime error ###', error);\n        throw error; // Re-throw to mark Lambda as failed\n    }\n};"]}
|
|
@@ -93,25 +93,39 @@ exports.handler = async (event: any, context: any) => {
|
|
|
93
93
|
const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);
|
|
94
94
|
|
|
95
95
|
console.log(`Starting cleanup of static assets older than ${deleteOlderThan.toISOString()}...`);
|
|
96
|
+
console.log(`Current revision: ${currentRevision.toISOString()}, Retention days: ${retainAssetsInDays}`);
|
|
96
97
|
|
|
97
98
|
let assetKeysToDelete: string[] = [];
|
|
98
99
|
let lastToken = undefined;
|
|
100
|
+
let totalAssetsScanned = 0;
|
|
101
|
+
let totalBatches = 0;
|
|
102
|
+
const assetsDeleteResults: DeleteObjectsCommandOutput[] = [];
|
|
99
103
|
|
|
100
104
|
do {
|
|
101
105
|
const curAssetsResult: ListObjectsV2CommandOutput = await client.send(
|
|
102
106
|
new ListObjectsV2Command({
|
|
103
107
|
Bucket: bucketName,
|
|
104
|
-
|
|
108
|
+
Prefix: '_nuxt/', // Only scan _nuxt/ directory where build assets are stored
|
|
109
|
+
MaxKeys: 1000,
|
|
105
110
|
ContinuationToken: lastToken,
|
|
106
111
|
})
|
|
107
112
|
);
|
|
108
113
|
|
|
109
|
-
|
|
110
|
-
|
|
114
|
+
if (!curAssetsResult.Contents || curAssetsResult.Contents.length === 0) {
|
|
115
|
+
console.log('No assets found in current batch');
|
|
116
|
+
lastToken = curAssetsResult.NextContinuationToken;
|
|
117
|
+
continue;
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
totalAssetsScanned += curAssetsResult.Contents.length;
|
|
121
|
+
console.log(`Batch ${++totalBatches}: Scanning ${curAssetsResult.Contents.length} assets (Total: ${totalAssetsScanned})`);
|
|
122
|
+
|
|
123
|
+
// Read object metadata in blocks of 50 for better performance
|
|
124
|
+
let processableAssets = [...curAssetsResult.Contents];
|
|
111
125
|
|
|
112
126
|
while (processableAssets.length > 0) {
|
|
113
|
-
const assetsBatch = processableAssets.slice(0,
|
|
114
|
-
processableAssets = processableAssets.slice(
|
|
127
|
+
const assetsBatch = processableAssets.slice(0, 50);
|
|
128
|
+
processableAssets = processableAssets.slice(50);
|
|
115
129
|
|
|
116
130
|
const pendingMetadataRequests = assetsBatch.map(asset =>
|
|
117
131
|
client.send(
|
|
@@ -119,46 +133,57 @@ exports.handler = async (event: any, context: any) => {
|
|
|
119
133
|
Bucket: bucketName,
|
|
120
134
|
Key: asset.Key,
|
|
121
135
|
})
|
|
122
|
-
)
|
|
136
|
+
).catch(error => {
|
|
137
|
+
console.warn(`Failed to get metadata for ${asset.Key}:`, error.message);
|
|
138
|
+
return { Metadata: {} }; // Return empty metadata on error
|
|
139
|
+
})
|
|
123
140
|
);
|
|
124
141
|
|
|
125
142
|
const metadataResults = await Promise.all(pendingMetadataRequests);
|
|
126
143
|
|
|
127
144
|
// Assign metadata to assets
|
|
128
|
-
const metadataByAsset: AssetMetadata[] =
|
|
129
|
-
key: assetsBatch[index].Key
|
|
130
|
-
metadata: metadataResult.Metadata,
|
|
131
|
-
}))
|
|
145
|
+
const metadataByAsset: AssetMetadata[] = metadataResults.map((metadataResult, index) => ({
|
|
146
|
+
key: assetsBatch[index].Key!,
|
|
147
|
+
metadata: metadataResult.Metadata || {},
|
|
148
|
+
}));
|
|
132
149
|
|
|
133
150
|
const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);
|
|
134
151
|
assetKeysToDelete.push(...outdatedAssetKeys);
|
|
152
|
+
|
|
153
|
+
// Stream deletion: Delete in batches of 1000 to avoid memory issues and timeouts
|
|
154
|
+
if (assetKeysToDelete.length >= MAX_DELETE_OBJECT_KEYS) {
|
|
155
|
+
console.log(`Deleting batch of ${assetKeysToDelete.length} assets...`);
|
|
156
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
157
|
+
assetsDeleteResults.push(...results);
|
|
158
|
+
assetKeysToDelete = []; // Clear the array
|
|
159
|
+
}
|
|
135
160
|
}
|
|
136
161
|
lastToken = curAssetsResult.NextContinuationToken;
|
|
137
162
|
} while (lastToken !== undefined);
|
|
138
163
|
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
164
|
+
// Delete remaining assets
|
|
165
|
+
if (assetKeysToDelete.length > 0) {
|
|
166
|
+
console.log(`Deleting final batch of ${assetKeysToDelete.length} assets...`);
|
|
167
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
168
|
+
assetsDeleteResults.push(...results);
|
|
142
169
|
}
|
|
143
170
|
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
const failed = results.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {
|
|
149
|
-
const currentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
150
|
-
if (currentError) {
|
|
171
|
+
// Check for deletion errors and report
|
|
172
|
+
const failed = assetsDeleteResults.reduce((previousResult: boolean, currentResult: DeleteObjectsCommandOutput): boolean => {
|
|
173
|
+
const hasCurrentError: boolean = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
174
|
+
if (hasCurrentError) {
|
|
151
175
|
console.error('Failed to delete outdated static assets', currentResult.Errors);
|
|
152
176
|
}
|
|
153
|
-
return previousResult ||
|
|
177
|
+
return previousResult || hasCurrentError;
|
|
154
178
|
}, false);
|
|
155
179
|
|
|
156
180
|
if (failed) {
|
|
157
181
|
throw new Error('Failed to delete outdated static assets');
|
|
158
182
|
}
|
|
159
183
|
|
|
160
|
-
console.log(
|
|
184
|
+
console.log(`Cleanup of old static assets finished. Total assets scanned: ${totalAssetsScanned}`);
|
|
161
185
|
} catch (error) {
|
|
162
186
|
console.error('### unexpected runtime error ###', error);
|
|
187
|
+
throw error; // Re-throw to mark Lambda as failed
|
|
163
188
|
}
|
|
164
189
|
};
|
|
@@ -160,8 +160,12 @@ class AccessLogsTableConfig {
|
|
|
160
160
|
name: 'sc_range_end',
|
|
161
161
|
type: aws_glue_alpha_1.Schema.BIG_INT,
|
|
162
162
|
},
|
|
163
|
+
{
|
|
164
|
+
name: 'sc_status',
|
|
165
|
+
type: aws_glue_alpha_1.Schema.STRING,
|
|
166
|
+
}
|
|
163
167
|
];
|
|
164
168
|
}
|
|
165
169
|
}
|
|
166
170
|
exports.AccessLogsTableConfig = AccessLogsTableConfig;
|
|
167
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AccessLogsTableConfig.js","sourceRoot":"","sources":["AccessLogsTableConfig.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Provides the column descriptions for both of the access log tables.\n */\nimport {type Column, Schema} from '@aws-cdk/aws-glue-alpha';\n\nexport class AccessLogsTableConfig {\n  public static getPartitionKeys(): Column[] {\n    return [\n      {\n        name: 'year',\n        type: Schema.STRING,\n      },\n      {\n        name: 'month',\n        type: Schema.STRING,\n      },\n      {\n        name: 'day',\n        type: Schema.STRING,\n      },\n      {\n        name: 'hour',\n        type: Schema.STRING,\n      },\n    ];\n  }\n\n  public static getTableColumns(): Column[] {\n    return [\n      {\n        name: 'date',\n        type: Schema.DATE,\n      },\n      {\n        name: 'time',\n        type: Schema.STRING,\n      },\n      {\n        name: 'location',\n        type: Schema.STRING,\n      },\n      {\n        name: 'bytes',\n        type: Schema.BIG_INT,\n      },\n      {\n        name: 'request_ip',\n        type: Schema.STRING,\n      },\n      {\n        name: 'method',\n        type: Schema.STRING,\n      },\n      {\n        name: 'host',\n        type: Schema.STRING,\n      },\n      {\n        name: 'uri',\n        type: Schema.STRING,\n      },\n      {\n        name: 'status',\n        type: Schema.INTEGER,\n      },\n      {\n        name: 'referrer',\n        type: Schema.STRING,\n      },\n      {\n        name: 'user_agent',\n        type: Schema.STRING,\n      },\n      {\n        name: 'query_string',\n        type: Schema.STRING,\n      },\n      {\n        name: 'cookie',\n        type: Schema.STRING,\n      },\n      {\n        name: 'result_type',\n        type: Schema.STRING,\n      },\n      {\n        name: 'request_id',\n        type: Schema.STRING,\n      },\n      {\n        name: 'host_header',\n        type: Schema.STRING,\n      },\n      {\n        name: 'request_protocol',\n        type: Schema.STRING,\n      },\n      {\n        name: 'request_bytes',\n        type: Schema.BIG_INT,\n      },\n      {\n        name: 'time_taken',\n        type: Schema.FLOAT,\n      },\n      {\n        name: 'xforwarded_for',\n        type: Schema.STRING,\n      },\n      {\n        name: 'ssl_protocol',\n        type: Schema.STRING,\n      },\n      {\n        name: 'ssl_cipher',\n        type: Schema.STRING,\n      },\n      {\n        name: 'response_result_type',\n        type: Schema.STRING,\n      },\n      {\n        name: 'http_version',\n        type: Schema.STRING,\n      },\n      {\n        name: 'fle_status',\n        type: Schema.STRING,\n      },\n      {\n        name: 'fle_encrypted_fields',\n        type: Schema.INTEGER,\n      },\n      {\n        name: 'c_port',\n        type: Schema.INTEGER,\n      },\n      {\n        name: 'time_to_first_byte',\n        type: Schema.FLOAT,\n      },\n      {\n        name: 'x_edge_detailed_result_type',\n        type: Schema.STRING,\n      },\n      {\n        name: 'sc_content_type',\n        type: Schema.STRING,\n      },\n      {\n        name: 'sc_content_len',\n        type: Schema.BIG_INT,\n      },\n      {\n        name: 'sc_range_start',\n        type: Schema.BIG_INT,\n      },\n      {\n        name: 'sc_range_end',\n        type: Schema.BIG_INT,\n      },\n      {\n        name: 'sc_status',\n        type: Schema.STRING,\n      }\n    ];\n  }\n}\n"]}
|
|
@@ -171,8 +171,8 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
171
171
|
code: aws_lambda_1.Code.fromAsset(`${functionDirPath}/build/app`, {
|
|
172
172
|
exclude: ['*.d.ts']
|
|
173
173
|
}),
|
|
174
|
-
timeout: aws_cdk_lib_1.Duration.minutes(
|
|
175
|
-
memorySize:
|
|
174
|
+
timeout: aws_cdk_lib_1.Duration.minutes(15),
|
|
175
|
+
memorySize: 512,
|
|
176
176
|
environment: {
|
|
177
177
|
STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,
|
|
178
178
|
OUTDATED_ASSETS_RETENTION_DAYS: `${(_a = props.outdatedAssetsRetentionDays) !== null && _a !== void 0 ? _a : 30}`,
|
|
@@ -568,4 +568,4 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
568
568
|
}
|
|
569
569
|
}
|
|
570
570
|
exports.NuxtServerAppStack = NuxtServerAppStack;
|
|
571
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    type BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution, HttpVersion,\n    type IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy, OriginRequestPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy,OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, 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, type IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3BucketOrigin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, type StaticAssetConfig} from \"../NuxtAppStaticAssets\";\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, mkdirSync, existsSync} from \"fs\";\nimport {type NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, HttpMethod, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\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 that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app and uses to generate a cache key.\n     */\n    private appCachePolicy: CachePolicy;\n\n    /**\n     * The origin request policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app without affecting the cache key.\n     */\n    private appRequestPolicy: OriginRequestPolicy | undefined;\n\n    /**\n     * The behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt Lambda function (via API gateway).\n     */\n    private nuxtServerRouteBehavior: BehaviorOptions;\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, {\n            ...props,\n\n            // Force cross-region references if a WAF ACL is used outside us-east-1\n            crossRegionReferences: props.webAclArn !== undefined && props.env?.region !== 'us-east-1' ? true : props.crossRegionReferences,\n        });\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.appRequestPolicy = this.createNuxtAppRequestPolicy(props)\n        this.nuxtServerRouteBehavior = this.createNuxtServerRouteBehavior()\n\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 appRevision = new Date().toISOString();\n\n        const dir = path.join(props.rootDir ?? '.', '.output', 'public');\n        mkdirSync(dir, { recursive: true });\n        writeFileSync(path.join(dir, 'app-revision'), 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}-nuxt`;\n\n        const appLogGroup = new LogGroup(this, `${funcName}-logs`, {\n            logGroupName: `/aws/lambda/${funcName}`,\n            retention: RetentionDays.ONE_MONTH,\n        });\n        appLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\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            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            logGroup: appLogGroup,\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     * Note that we use the bundled AWS SDK for Node to avoid the need for a custom layer\n     * which restricts the consumer to a specific yarn or npm version.\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup`;\n        const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');\n\n        const cleanupLogGroup = new LogGroup(this, `${functionName}-logs`, {\n            logGroupName: `/aws/lambda/${functionName}`,\n            retention: RetentionDays.TWO_WEEKS,\n        });\n        cleanupLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\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            handler: 'index.handler',\n            code: Code.fromAsset(`${functionDirPath}/build/app`, {\n                exclude: ['*.d.ts']\n            }),\n            timeout: Duration.minutes(5),\n            memorySize: 128,\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            logGroup: cleanupLogGroup,\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: [\n                HttpMethod.GET,\n                HttpMethod.HEAD,\n                HttpMethod.OPTIONS,\n                HttpMethod.POST,\n                HttpMethod.PUT,\n                HttpMethod.PATCH,\n                HttpMethod.DELETE,\n            ],\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.nuxtServerRouteBehavior,\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 ? 'unprocessed' : undefined,\n            logIncludesCookies: props.enableAccessLogsAnalysis,\n            webAclId: props.webAclArn,\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 createNuxtServerRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: this.appRequestPolicy,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {\n\n            // Nuxt I18n files are served via a server route\n            '/_i18n/*': this.nuxtServerRouteBehavior,\n        };\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        // Add custom server routes before static assets to ensure they take precedence\n        if (props.serverRoutes && props.serverRoutes.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createServerRouteBehavior(props.serverRoutes)};\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.cacheKeyQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.cacheKeyQueryParams) : (props.denyCacheKeyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyCacheKeyQueryParams) : (props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()))),\n            headerBehavior: props.cacheKeyHeaders?.length ? CacheHeaderBehavior.allowList(...props.cacheKeyHeaders) : (props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none()),\n            cookieBehavior: props.cacheKeyCookies?.length ? CacheCookieBehavior.allowList(...props.cacheKeyCookies) : (props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none()),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates an origin request policy for the Nuxt app route behavior of the CloudFront distribution.\n     * No policy is created if no explicit config is provided.\n     */\n    private createNuxtAppRequestPolicy(props: NuxtServerAppStackProps): OriginRequestPolicy|undefined {\n\n        // If no explicit config is provided, we want to use the default from Cloudfront\n        const hasAnyConfig = props.forwardQueryParams?.length || props.forwardHeaders?.length || props.forwardCookies?.length;\n        if (!hasAnyConfig) {\n            return undefined;\n        }\n\n        return new OriginRequestPolicy(this, `${this.resourceIdPrefix}-request-policy`, {\n            originRequestPolicyName: `${this.resourceIdPrefix}-cdn-request-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin without affecting the cache key.`,\n            queryStringBehavior: props.forwardQueryParams?.length ? OriginRequestQueryStringBehavior.allowList(...props.forwardQueryParams) : OriginRequestQueryStringBehavior.all(),\n            headerBehavior: props.forwardHeaders?.length ? OriginRequestHeaderBehavior.allowList(...props.forwardHeaders) : OriginRequestHeaderBehavior.none(),\n            cookieBehavior: props.forwardCookies?.length ? OriginRequestCookieBehavior.allowList(...props.forwardCookies) : OriginRequestCookieBehavior.none(),\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            originRequestPolicy: this.appRequestPolicy,\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 behaviors for the CloudFront distribution to route specified path patterns to the SSR origin.\n     * This allows server endpoints that use file-like URLs (e.g., /sitemap.xml from @nuxtjs/sitemap) \n     * to be handled by the Lambda function for dynamic content generation.\n     */\n    private createServerRouteBehavior(serverRoutes: string[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n        \n        serverRoutes.forEach(route => {\n            rules[route] = this.nuxtServerRouteBehavior;\n        });\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: S3BucketOrigin.withOriginAccessIdentity(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: S3BucketOrigin.withOriginAccessIdentity(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        const logGroup = new LogGroup(this, `${this.resourceIdPrefix}-assets-deployment-logs`, {\n            logGroupName: `/aws/lambda/${this.resourceIdPrefix}-assets-deployment`,\n            retention: RetentionDays.ONE_DAY,\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => 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                logGroup: logGroup,\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: '2', 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\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): void {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n\n        const { CloudFrontAccessLogsAnalysis } = require('../access-logs-analysis/CloudFrontAccessLogsAnalysis');\n\n        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"]}
|
|
571
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    type BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution, HttpVersion,\n    type IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy, OriginRequestPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy,OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, 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, type IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3BucketOrigin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, type StaticAssetConfig} from \"../NuxtAppStaticAssets\";\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, mkdirSync, existsSync} from \"fs\";\nimport {type NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, HttpMethod, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\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 that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app and uses to generate a cache key.\n     */\n    private appCachePolicy: CachePolicy;\n\n    /**\n     * The origin request policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app without affecting the cache key.\n     */\n    private appRequestPolicy: OriginRequestPolicy | undefined;\n\n    /**\n     * The behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt Lambda function (via API gateway).\n     */\n    private nuxtServerRouteBehavior: BehaviorOptions;\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, {\n            ...props,\n\n            // Force cross-region references if a WAF ACL is used outside us-east-1\n            crossRegionReferences: props.webAclArn !== undefined && props.env?.region !== 'us-east-1' ? true : props.crossRegionReferences,\n        });\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.appRequestPolicy = this.createNuxtAppRequestPolicy(props)\n        this.nuxtServerRouteBehavior = this.createNuxtServerRouteBehavior()\n\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 appRevision = new Date().toISOString();\n\n        const dir = path.join(props.rootDir ?? '.', '.output', 'public');\n        mkdirSync(dir, { recursive: true });\n        writeFileSync(path.join(dir, 'app-revision'), 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}-nuxt`;\n\n        const appLogGroup = new LogGroup(this, `${funcName}-logs`, {\n            logGroupName: `/aws/lambda/${funcName}`,\n            retention: RetentionDays.ONE_MONTH,\n        });\n        appLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\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            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            logGroup: appLogGroup,\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     * Note that we use the bundled AWS SDK for Node to avoid the need for a custom layer\n     * which restricts the consumer to a specific yarn or npm version.\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup`;\n        const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');\n\n        const cleanupLogGroup = new LogGroup(this, `${functionName}-logs`, {\n            logGroupName: `/aws/lambda/${functionName}`,\n            retention: RetentionDays.TWO_WEEKS,\n        });\n        cleanupLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\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            handler: 'index.handler',\n            code: Code.fromAsset(`${functionDirPath}/build/app`, {\n                exclude: ['*.d.ts']\n            }),\n            timeout: Duration.minutes(15),\n            memorySize: 512,\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            logGroup: cleanupLogGroup,\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: [\n                HttpMethod.GET,\n                HttpMethod.HEAD,\n                HttpMethod.OPTIONS,\n                HttpMethod.POST,\n                HttpMethod.PUT,\n                HttpMethod.PATCH,\n                HttpMethod.DELETE,\n            ],\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.nuxtServerRouteBehavior,\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 ? 'unprocessed' : undefined,\n            logIncludesCookies: props.enableAccessLogsAnalysis,\n            webAclId: props.webAclArn,\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 createNuxtServerRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: this.appRequestPolicy,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {\n\n            // Nuxt I18n files are served via a server route\n            '/_i18n/*': this.nuxtServerRouteBehavior,\n        };\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        // Add custom server routes before static assets to ensure they take precedence\n        if (props.serverRoutes && props.serverRoutes.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createServerRouteBehavior(props.serverRoutes)};\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.cacheKeyQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.cacheKeyQueryParams) : (props.denyCacheKeyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyCacheKeyQueryParams) : (props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()))),\n            headerBehavior: props.cacheKeyHeaders?.length ? CacheHeaderBehavior.allowList(...props.cacheKeyHeaders) : (props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none()),\n            cookieBehavior: props.cacheKeyCookies?.length ? CacheCookieBehavior.allowList(...props.cacheKeyCookies) : (props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none()),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates an origin request policy for the Nuxt app route behavior of the CloudFront distribution.\n     * No policy is created if no explicit config is provided.\n     */\n    private createNuxtAppRequestPolicy(props: NuxtServerAppStackProps): OriginRequestPolicy|undefined {\n\n        // If no explicit config is provided, we want to use the default from Cloudfront\n        const hasAnyConfig = props.forwardQueryParams?.length || props.forwardHeaders?.length || props.forwardCookies?.length;\n        if (!hasAnyConfig) {\n            return undefined;\n        }\n\n        return new OriginRequestPolicy(this, `${this.resourceIdPrefix}-request-policy`, {\n            originRequestPolicyName: `${this.resourceIdPrefix}-cdn-request-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin without affecting the cache key.`,\n            queryStringBehavior: props.forwardQueryParams?.length ? OriginRequestQueryStringBehavior.allowList(...props.forwardQueryParams) : OriginRequestQueryStringBehavior.all(),\n            headerBehavior: props.forwardHeaders?.length ? OriginRequestHeaderBehavior.allowList(...props.forwardHeaders) : OriginRequestHeaderBehavior.none(),\n            cookieBehavior: props.forwardCookies?.length ? OriginRequestCookieBehavior.allowList(...props.forwardCookies) : OriginRequestCookieBehavior.none(),\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            originRequestPolicy: this.appRequestPolicy,\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 behaviors for the CloudFront distribution to route specified path patterns to the SSR origin.\n     * This allows server endpoints that use file-like URLs (e.g., /sitemap.xml from @nuxtjs/sitemap) \n     * to be handled by the Lambda function for dynamic content generation.\n     */\n    private createServerRouteBehavior(serverRoutes: string[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n        \n        serverRoutes.forEach(route => {\n            rules[route] = this.nuxtServerRouteBehavior;\n        });\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: S3BucketOrigin.withOriginAccessIdentity(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: S3BucketOrigin.withOriginAccessIdentity(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        const logGroup = new LogGroup(this, `${this.resourceIdPrefix}-assets-deployment-logs`, {\n            logGroupName: `/aws/lambda/${this.resourceIdPrefix}-assets-deployment`,\n            retention: RetentionDays.ONE_DAY,\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => 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                logGroup: logGroup,\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: '2', 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\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): void {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n\n        const { CloudFrontAccessLogsAnalysis } = require('../access-logs-analysis/CloudFrontAccessLogsAnalysis');\n\n        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"]}
|
|
@@ -309,8 +309,8 @@ export class NuxtServerAppStack extends Stack {
|
|
|
309
309
|
code: Code.fromAsset(`${functionDirPath}/build/app`, {
|
|
310
310
|
exclude: ['*.d.ts']
|
|
311
311
|
}),
|
|
312
|
-
timeout: Duration.minutes(
|
|
313
|
-
memorySize:
|
|
312
|
+
timeout: Duration.minutes(15),
|
|
313
|
+
memorySize: 512,
|
|
314
314
|
environment: {
|
|
315
315
|
STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,
|
|
316
316
|
OUTDATED_ASSETS_RETENTION_DAYS: `${props.outdatedAssetsRetentionDays ?? 30}`,
|
|
@@ -71,6 +71,7 @@ class CloudFrontWebAcl extends constructs_1.Construct {
|
|
|
71
71
|
});
|
|
72
72
|
}
|
|
73
73
|
// Rate limiting (DDoS protection)
|
|
74
|
+
// Exclude static assets (build files, chunks) from rate limiting
|
|
74
75
|
if (config.rateLimit) {
|
|
75
76
|
rules.push({
|
|
76
77
|
name: 'RateLimiting',
|
|
@@ -79,6 +80,24 @@ class CloudFrontWebAcl extends constructs_1.Construct {
|
|
|
79
80
|
rateBasedStatement: {
|
|
80
81
|
limit: config.rateLimit,
|
|
81
82
|
aggregateKeyType: 'IP',
|
|
83
|
+
// Exclude requests to /_nuxt/* paths (build files)
|
|
84
|
+
scopeDownStatement: {
|
|
85
|
+
notStatement: {
|
|
86
|
+
statement: {
|
|
87
|
+
byteMatchStatement: {
|
|
88
|
+
searchString: '/_nuxt/',
|
|
89
|
+
fieldToMatch: {
|
|
90
|
+
uriPath: {},
|
|
91
|
+
},
|
|
92
|
+
textTransformations: [{
|
|
93
|
+
priority: 0,
|
|
94
|
+
type: 'NONE',
|
|
95
|
+
}],
|
|
96
|
+
positionalConstraint: 'STARTS_WITH',
|
|
97
|
+
},
|
|
98
|
+
},
|
|
99
|
+
},
|
|
100
|
+
},
|
|
82
101
|
},
|
|
83
102
|
},
|
|
84
103
|
action: { block: {} },
|
|
@@ -219,4 +238,4 @@ class CloudFrontWebAcl extends constructs_1.Construct {
|
|
|
219
238
|
}
|
|
220
239
|
}
|
|
221
240
|
exports.CloudFrontWebAcl = CloudFrontWebAcl;
|
|
222
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CloudFrontWebAcl.js","sourceRoot":"","sources":["CloudFrontWebAcl.ts"],"names":[],"mappings":"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","sourcesContent":["import {Construct} from 'constructs';\nimport {CfnWebACL, CfnIPSet} from 'aws-cdk-lib/aws-wafv2';\nimport {type WafConfig, DEFAULT_NUXT_WAF_CONFIG} from './WafConfig';\n\n/**\n * Properties for CloudFrontWebAcl construct.\n */\nexport interface CloudFrontWebAclProps {\n    /**\n     * The name prefix for the Web ACL and related resources.\n     */\n    readonly name: string;\n\n    /**\n     * WAF configuration options.\n     */\n    readonly config: WafConfig;\n}\n\n/**\n * A construct that creates an AWS WAF Web ACL for CloudFront distributions.\n * Provides protection against common web exploits, bots, and DDoS attacks.\n */\nexport class CloudFrontWebAcl extends Construct {\n    /**\n     * The Web ACL resource.\n     */\n    public readonly webAcl: CfnWebACL;\n\n    constructor(scope: Construct, id: string, props: CloudFrontWebAclProps) {\n        super(scope, id);\n\n        const config = {...DEFAULT_NUXT_WAF_CONFIG, ...props.config};\n        const rules: CfnWebACL.RuleProperty[] = [];\n        let priority = 0;\n\n        // IP allowlist (highest priority - bypasses all other rules)\n        if (config.allowedIpAddresses && config.allowedIpAddresses.length > 0) {\n            const allowedIpSet = this.createIpSet(`${props.name}-allowed-ips`, config.allowedIpAddresses);\n            rules.push({\n                name: 'AllowedIpAddresses',\n                priority: priority++,\n                statement: {\n                    ipSetReferenceStatement: {\n                        arn: allowedIpSet.attrArn,\n                    },\n                },\n                action: {allow: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AllowedIps`,\n                },\n            });\n        }\n\n        // IP blocklist\n        if (config.blockedIpAddresses && config.blockedIpAddresses.length > 0) {\n            const blockedIpSet = this.createIpSet(`${props.name}-blocked-ips`, config.blockedIpAddresses);\n            rules.push({\n                name: 'BlockedIpAddresses',\n                priority: priority++,\n                statement: {\n                    ipSetReferenceStatement: {\n                        arn: blockedIpSet.attrArn,\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}BlockedIps`,\n                },\n            });\n        }\n\n        // Geo-blocking\n        if (config.blockedCountries && config.blockedCountries.length > 0) {\n            rules.push({\n                name: 'GeoBlocking',\n                priority: priority++,\n                statement: {\n                    geoMatchStatement: {\n                        countryCodes: config.blockedCountries,\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}GeoBlocking`,\n                },\n            });\n        }\n\n        // Rate limiting (DDoS protection)\n        if (config.rateLimit) {\n            rules.push({\n                name: 'RateLimiting',\n                priority: priority++,\n                statement: {\n                    rateBasedStatement: {\n                        limit: config.rateLimit,\n                        aggregateKeyType: 'IP',\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}RateLimit`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Common Rule Set\n        if (config.enableCommonRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesCommonRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesCommonRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}CommonRuleSet`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Known Bad Inputs\n        if (config.enableKnownBadInputsRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesKnownBadInputsRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesKnownBadInputsRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}KnownBadInputs`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Anonymous IP List\n        if (config.enableAnonymousIpRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesAnonymousIpList',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesAnonymousIpList',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AnonymousIpList`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Amazon IP Reputation List\n        if (config.enableAmazonIpReputationRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesAmazonIpReputationList',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesAmazonIpReputationList',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AmazonIpReputation`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Bot Control\n        if (config.enableBotControlRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesBotControlRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesBotControlRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}BotControl`,\n                },\n            });\n        }\n\n        // Custom rules (added at the end with automatically assigned priorities)\n        if (config.customRules && config.customRules.length > 0) {\n            config.customRules.forEach((customRule) => {\n                rules.push({\n                    ...customRule,\n                    priority: priority++,\n                } as CfnWebACL.RuleProperty);\n            });\n        }\n\n        // Create the Web ACL\n        this.webAcl = new CfnWebACL(this, 'WebAcl', {\n            name: props.name,\n            scope: 'CLOUDFRONT',\n            defaultAction: {allow: {}},\n            rules,\n            visibilityConfig: {\n                sampledRequestsEnabled: true,\n                cloudWatchMetricsEnabled: true,\n                metricName: `${config.metricsPrefix}WebAcl`,\n            },\n        });\n    }\n\n    /**\n     * Creates an IP set for WAF rules.\n     */\n    private createIpSet(name: string, addresses: string[]): CfnIPSet {\n        return new CfnIPSet(this, name, {\n            name,\n            scope: 'CLOUDFRONT',\n            ipAddressVersion: 'IPV4',\n            addresses,\n        });\n    }\n}\n\n"]}
|
|
241
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CloudFrontWebAcl.js","sourceRoot":"","sources":["CloudFrontWebAcl.ts"],"names":[],"mappings":"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","sourcesContent":["import {Construct} from 'constructs';\nimport {CfnWebACL, CfnIPSet} from 'aws-cdk-lib/aws-wafv2';\nimport {type WafConfig, DEFAULT_NUXT_WAF_CONFIG} from './WafConfig';\n\n/**\n * Properties for CloudFrontWebAcl construct.\n */\nexport interface CloudFrontWebAclProps {\n    /**\n     * The name prefix for the Web ACL and related resources.\n     */\n    readonly name: string;\n\n    /**\n     * WAF configuration options.\n     */\n    readonly config: WafConfig;\n}\n\n/**\n * A construct that creates an AWS WAF Web ACL for CloudFront distributions.\n * Provides protection against common web exploits, bots, and DDoS attacks.\n */\nexport class CloudFrontWebAcl extends Construct {\n    /**\n     * The Web ACL resource.\n     */\n    public readonly webAcl: CfnWebACL;\n\n    constructor(scope: Construct, id: string, props: CloudFrontWebAclProps) {\n        super(scope, id);\n\n        const config = {...DEFAULT_NUXT_WAF_CONFIG, ...props.config};\n        const rules: CfnWebACL.RuleProperty[] = [];\n        let priority = 0;\n\n        // IP allowlist (highest priority - bypasses all other rules)\n        if (config.allowedIpAddresses && config.allowedIpAddresses.length > 0) {\n            const allowedIpSet = this.createIpSet(`${props.name}-allowed-ips`, config.allowedIpAddresses);\n            rules.push({\n                name: 'AllowedIpAddresses',\n                priority: priority++,\n                statement: {\n                    ipSetReferenceStatement: {\n                        arn: allowedIpSet.attrArn,\n                    },\n                },\n                action: {allow: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AllowedIps`,\n                },\n            });\n        }\n\n        // IP blocklist\n        if (config.blockedIpAddresses && config.blockedIpAddresses.length > 0) {\n            const blockedIpSet = this.createIpSet(`${props.name}-blocked-ips`, config.blockedIpAddresses);\n            rules.push({\n                name: 'BlockedIpAddresses',\n                priority: priority++,\n                statement: {\n                    ipSetReferenceStatement: {\n                        arn: blockedIpSet.attrArn,\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}BlockedIps`,\n                },\n            });\n        }\n\n        // Geo-blocking\n        if (config.blockedCountries && config.blockedCountries.length > 0) {\n            rules.push({\n                name: 'GeoBlocking',\n                priority: priority++,\n                statement: {\n                    geoMatchStatement: {\n                        countryCodes: config.blockedCountries,\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}GeoBlocking`,\n                },\n            });\n        }\n\n        // Rate limiting (DDoS protection)\n        // Exclude static assets (build files, chunks) from rate limiting\n        if (config.rateLimit) {\n            rules.push({\n                name: 'RateLimiting',\n                priority: priority++,\n                statement: {\n                    rateBasedStatement: {\n                        limit: config.rateLimit,\n                        aggregateKeyType: 'IP',\n                        // Exclude requests to /_nuxt/* paths (build files)\n                        scopeDownStatement: {\n                            notStatement: {\n                                statement: {\n                                    byteMatchStatement: {\n                                        searchString: '/_nuxt/',\n                                        fieldToMatch: {\n                                            uriPath: {},\n                                        },\n                                        textTransformations: [{\n                                            priority: 0,\n                                            type: 'NONE',\n                                        }],\n                                        positionalConstraint: 'STARTS_WITH',\n                                    },\n                                },\n                            },\n                        },\n                    },\n                },\n                action: {block: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}RateLimit`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Common Rule Set\n        if (config.enableCommonRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesCommonRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesCommonRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}CommonRuleSet`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Known Bad Inputs\n        if (config.enableKnownBadInputsRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesKnownBadInputsRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesKnownBadInputsRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}KnownBadInputs`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Anonymous IP List\n        if (config.enableAnonymousIpRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesAnonymousIpList',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesAnonymousIpList',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AnonymousIpList`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Amazon IP Reputation List\n        if (config.enableAmazonIpReputationRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesAmazonIpReputationList',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesAmazonIpReputationList',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}AmazonIpReputation`,\n                },\n            });\n        }\n\n        // AWS Managed Rules - Bot Control\n        if (config.enableBotControlRuleSet) {\n            rules.push({\n                name: 'AWSManagedRulesBotControlRuleSet',\n                priority: priority++,\n                statement: {\n                    managedRuleGroupStatement: {\n                        vendorName: 'AWS',\n                        name: 'AWSManagedRulesBotControlRuleSet',\n                    },\n                },\n                overrideAction: {none: {}},\n                visibilityConfig: {\n                    sampledRequestsEnabled: true,\n                    cloudWatchMetricsEnabled: true,\n                    metricName: `${config.metricsPrefix}BotControl`,\n                },\n            });\n        }\n\n        // Custom rules (added at the end with automatically assigned priorities)\n        if (config.customRules && config.customRules.length > 0) {\n            config.customRules.forEach((customRule) => {\n                rules.push({\n                    ...customRule,\n                    priority: priority++,\n                } as CfnWebACL.RuleProperty);\n            });\n        }\n\n        // Create the Web ACL\n        this.webAcl = new CfnWebACL(this, 'WebAcl', {\n            name: props.name,\n            scope: 'CLOUDFRONT',\n            defaultAction: {allow: {}},\n            rules,\n            visibilityConfig: {\n                sampledRequestsEnabled: true,\n                cloudWatchMetricsEnabled: true,\n                metricName: `${config.metricsPrefix}WebAcl`,\n            },\n        });\n    }\n\n    /**\n     * Creates an IP set for WAF rules.\n     */\n    private createIpSet(name: string, addresses: string[]): CfnIPSet {\n        return new CfnIPSet(this, name, {\n            name,\n            scope: 'CLOUDFRONT',\n            ipAddressVersion: 'IPV4',\n            addresses,\n        });\n    }\n}\n\n"]}
|
|
@@ -94,6 +94,7 @@ export class CloudFrontWebAcl extends Construct {
|
|
|
94
94
|
}
|
|
95
95
|
|
|
96
96
|
// Rate limiting (DDoS protection)
|
|
97
|
+
// Exclude static assets (build files, chunks) from rate limiting
|
|
97
98
|
if (config.rateLimit) {
|
|
98
99
|
rules.push({
|
|
99
100
|
name: 'RateLimiting',
|
|
@@ -102,6 +103,24 @@ export class CloudFrontWebAcl extends Construct {
|
|
|
102
103
|
rateBasedStatement: {
|
|
103
104
|
limit: config.rateLimit,
|
|
104
105
|
aggregateKeyType: 'IP',
|
|
106
|
+
// Exclude requests to /_nuxt/* paths (build files)
|
|
107
|
+
scopeDownStatement: {
|
|
108
|
+
notStatement: {
|
|
109
|
+
statement: {
|
|
110
|
+
byteMatchStatement: {
|
|
111
|
+
searchString: '/_nuxt/',
|
|
112
|
+
fieldToMatch: {
|
|
113
|
+
uriPath: {},
|
|
114
|
+
},
|
|
115
|
+
textTransformations: [{
|
|
116
|
+
priority: 0,
|
|
117
|
+
type: 'NONE',
|
|
118
|
+
}],
|
|
119
|
+
positionalConstraint: 'STARTS_WITH',
|
|
120
|
+
},
|
|
121
|
+
},
|
|
122
|
+
},
|
|
123
|
+
},
|
|
105
124
|
},
|
|
106
125
|
},
|
|
107
126
|
action: {block: {}},
|
|
@@ -39,8 +39,9 @@ export interface WafConfig {
|
|
|
39
39
|
/**
|
|
40
40
|
* The maximum number of requests allowed from a single IP within a 5-minute period
|
|
41
41
|
* to protect against DDoS attacks.
|
|
42
|
+
* Does not apply to build files server under the `/_nuxt/` path.
|
|
42
43
|
* Can be disabled by setting it to `undefined`.
|
|
43
|
-
* @default
|
|
44
|
+
* @default 300
|
|
44
45
|
*/
|
|
45
46
|
readonly rateLimit?: number | undefined;
|
|
46
47
|
/**
|
|
@@ -10,7 +10,7 @@ exports.DEFAULT_NUXT_WAF_CONFIG = {
|
|
|
10
10
|
enableAnonymousIpRuleSet: false,
|
|
11
11
|
enableAmazonIpReputationRuleSet: true,
|
|
12
12
|
enableBotControlRuleSet: false,
|
|
13
|
-
rateLimit:
|
|
13
|
+
rateLimit: 300,
|
|
14
14
|
metricsPrefix: 'WafMetrics',
|
|
15
15
|
};
|
|
16
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiV2FmQ29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiV2FmQ29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQTJHQTs7R0FFRztBQUNVLFFBQUEsdUJBQXVCLEdBQXVCO0lBQ3ZELG1CQUFtQixFQUFFLElBQUk7SUFDekIsMkJBQTJCLEVBQUUsSUFBSTtJQUNqQyx3QkFBd0IsRUFBRSxLQUFLO0lBQy9CLCtCQUErQixFQUFFLElBQUk7SUFDckMsdUJBQXVCLEVBQUUsS0FBSztJQUM5QixTQUFTLEVBQUUsR0FBRztJQUNkLGFBQWEsRUFBRSxZQUFZO0NBQzlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvbmZpZ3VyYXRpb24gb3B0aW9ucyBmb3IgQVdTIFdBRiBXZWIgQUNMLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFdhZkNvbmZpZyB7XG5cbiAgICAvKipcbiAgICAgKiBXaGV0aGVyIHRvIGVuYWJsZSBBV1MgbWFuYWdlZCBydWxlIGZvciBjb21tb24gZXhwbG9pdHMgKFNRTCBpbmplY3Rpb24sIFhTUywgZXRjLikuXG4gICAgICogVGhpcyBydWxlIGdyb3VwIGNvbnRhaW5zIHJ1bGVzIHRoYXQgYmxvY2sgcmVxdWVzdCBwYXR0ZXJucyBhc3NvY2lhdGVkIHdpdGggZXhwbG9pdGF0aW9uXG4gICAgICogb2YgdnVsbmVyYWJpbGl0aWVzIHNwZWNpZmljIHRvIHdlYiBhcHBsaWNhdGlvbnMuXG4gICAgICogQGRlZmF1bHQgdHJ1ZVxuICAgICAqL1xuICAgIHJlYWRvbmx5IGVuYWJsZUNvbW1vblJ1bGVTZXQ/OiBib29sZWFuO1xuXG4gICAgLyoqXG4gICAgICogV2hldGhlciB0byBlbmFibGUgQVdTIG1hbmFnZWQgcnVsZSBmb3Iga25vd24gYmFkIGlucHV0cy5cbiAgICAgKiBUaGlzIHJ1bGUgZ3JvdXAgY29udGFpbnMgcnVsZXMgdG8gYmxvY2sgcmVxdWVzdCBwYXR0ZXJucyBrbm93biB0byBiZSBpbnZhbGlkXG4gICAgICogYW5kIGFzc29jaWF0ZWQgd2l0aCBleHBsb2l0YXRpb24gb3IgZGlzY292ZXJ5IG9mIHZ1bG5lcmFiaWxpdGllcy5cbiAgICAgKiBAZGVmYXVsdCB0cnVlXG4gICAgICovXG4gICAgcmVhZG9ubHkgZW5hYmxlS25vd25CYWRJbnB1dHNSdWxlU2V0PzogYm9vbGVhbjtcblxuICAgIC8qKlxuICAgICAqIFdoZXRoZXIgdG8gZW5hYmxlIEFXUyBtYW5hZ2VkIHJ1bGUgZm9yIGFub255bW91cyBJUCBhZGRyZXNzZXMuXG4gICAgICogVGhpcyBydWxlIGdyb3VwIGNvbnRhaW5zIHJ1bGVzIHRvIGJsb2NrIHJlcXVlc3RzIGZyb20gc2VydmljZXMgdGhhdCBhbGxvd1xuICAgICAqIG9iZnVzY2F0aW9uIG9mIHZpZXdlciBpZGVudGl0eSAoVlBOcywgcHJveGllcywgVG9yIG5vZGVzLCBob3N0aW5nIHByb3ZpZGVycykuXG4gICAgICogQGRlZmF1bHQgZmFsc2VcbiAgICAgKi9cbiAgICByZWFkb25seSBlbmFibGVBbm9ueW1vdXNJcFJ1bGVTZXQ/OiBib29sZWFuO1xuXG4gICAgLyoqXG4gICAgICogV2hldGhlciB0byBlbmFibGUgQVdTIG1hbmFnZWQgcnVsZSBmb3IgQW1hem9uIElQIHJlcHV0YXRpb24gbGlzdC5cbiAgICAgKiBUaGlzIHJ1bGUgZ3JvdXAgY29udGFpbnMgcnVsZXMgYmFzZWQgb24gQW1hem9uIHRocmVhdCBpbnRlbGxpZ2VuY2UuXG4gICAgICogQGRlZmF1bHQgdHJ1ZVxuICAgICAqL1xuICAgIHJlYWRvbmx5IGVuYWJsZUFtYXpvbklwUmVwdXRhdGlvblJ1bGVTZXQ/OiBib29sZWFuO1xuXG4gICAgLyoqXG4gICAgICogV2hldGhlciB0byBlbmFibGUgQVdTIG1hbmFnZWQgcnVsZSBmb3IgYm90IGNvbnRyb2wuXG4gICAgICogVGhpcyBydWxlIGdyb3VwIHByb3ZpZGVzIHByb3RlY3Rpb24gYWdhaW5zdCBhdXRvbWF0ZWQgYm90cy5cbiAgICAgKiBOb3RlOiBUaGlzIGlzIGEgcGFpZCBmZWF0dXJlIHdpdGggYWRkaXRpb25hbCBjb3N0cy5cbiAgICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgICAqL1xuICAgIHJlYWRvbmx5IGVuYWJsZUJvdENvbnRyb2xSdWxlU2V0PzogYm9vbGVhbjtcblxuICAgIC8qKlxuICAgICAqIFRoZSBtYXhpbXVtIG51bWJlciBvZiByZXF1ZXN0cyBhbGxvd2VkIGZyb20gYSBzaW5nbGUgSVAgd2l0aGluIGEgNS1taW51dGUgcGVyaW9kXG4gICAgICogdG8gcHJvdGVjdCBhZ2FpbnN0IEREb1MgYXR0YWNrcy5cbiAgICAgKiBEb2VzIG5vdCBhcHBseSB0byBidWlsZCBmaWxlcyBzZXJ2ZXIgdW5kZXIgdGhlIGAvX251eHQvYCBwYXRoLlxuICAgICAqIENhbiBiZSBkaXNhYmxlZCBieSBzZXR0aW5nIGl0IHRvIGB1bmRlZmluZWRgLlxuICAgICAqIEBkZWZhdWx0IDMwMFxuICAgICAqL1xuICAgIHJlYWRvbmx5IHJhdGVMaW1pdD86IG51bWJlciB8IHVuZGVmaW5lZDtcblxuICAgIC8qKlxuICAgICAqIEFycmF5IG9mIGNvdW50cnkgY29kZXMgKElTTyAzMTY2LTEgYWxwaGEtMikgdG8gYmxvY2suXG4gICAgICogRXhhbXBsZTogWydDTicsICdSVScsICdLUCddXG4gICAgICovXG4gICAgcmVhZG9ubHkgYmxvY2tlZENvdW50cmllcz86IHN0cmluZ1tdO1xuXG4gICAgLyoqXG4gICAgICogQ3VzdG9tIElQIGFkZHJlc3NlcyBvciBDSURSIHJhbmdlcyB0byBibG9jay5cbiAgICAgKiBFeGFtcGxlOiBbJzE5Mi4wLjIuMC8yNCcsICcxOTguNTEuMTAwLjQyLzMyJ11cbiAgICAgKi9cbiAgICByZWFkb25seSBibG9ja2VkSXBBZGRyZXNzZXM/OiBzdHJpbmdbXTtcblxuICAgIC8qKlxuICAgICAqIEN1c3RvbSBJUCBhZGRyZXNzZXMgb3IgQ0lEUiByYW5nZXMgdG8gYWxsb3cgKGJ5cGFzcyBhbGwgcnVsZXMpLlxuICAgICAqIEV4YW1wbGU6IFsnMjAzLjAuMTEzLjAvMjQnXVxuICAgICAqL1xuICAgIHJlYWRvbmx5IGFsbG93ZWRJcEFkZHJlc3Nlcz86IHN0cmluZ1tdO1xuXG4gICAgLyoqXG4gICAgICogQ2xvdWRXYXRjaCBtZXRyaWNzIG5hbWUgcHJlZml4IGZvciB0aGUgV0FGLlxuICAgICAqIEBkZWZhdWx0ICdXYWZNZXRyaWNzJ1xuICAgICAqL1xuICAgIHJlYWRvbmx5IG1ldHJpY3NQcmVmaXg/OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBDdXN0b20gV0FGIHJ1bGVzIHRvIGFkZCBhdCB0aGUgZW5kIG9mIHRoZSBydWxlIHNldC5cbiAgICAgKiBUaGVzZSBydWxlcyB3aWxsIGJlIGFwcGVuZGVkIGFmdGVyIGFsbCBidWlsdC1pbiBtYW5hZ2VkIHJ1bGVzLlxuICAgICAqXG4gICAgICogTm90ZTogRG8gbm90IHNldCB0aGUgYHByaW9yaXR5YCBmaWVsZCAtIGl0IHdpbGwgYmUgYXV0b21hdGljYWxseSBhc3NpZ25lZC5cbiAgICAgKlxuICAgICAqIEV4YW1wbGU6XG4gICAgICogYGBgdHlwZXNjcmlwdFxuICAgICAqIGN1c3RvbVJ1bGVzOiBbe1xuICAgICAqICAgbmFtZTogJ0Jsb2NrU3BlY2lmaWNVc2VyQWdlbnQnLFxuICAgICAqICAgc3RhdGVtZW50OiB7XG4gICAgICogICAgIGJ5dGVNYXRjaFN0YXRlbWVudDoge1xuICAgICAqICAgICAgIHNlYXJjaFN0cmluZzogJ0JhZEJvdCcsXG4gICAgICogICAgICAgZmllbGRUb01hdGNoOiB7IHNpbmdsZUhlYWRlcjogeyBuYW1lOiAndXNlci1hZ2VudCcgfSB9LFxuICAgICAqICAgICAgIHRleHRUcmFuc2Zvcm1hdGlvbnM6IFt7IHByaW9yaXR5OiAwLCB0eXBlOiAnTE9XRVJDQVNFJyB9XSxcbiAgICAgKiAgICAgICBwb3NpdGlvbmFsQ29uc3RyYWludDogJ0NPTlRBSU5TJ1xuICAgICAqICAgICB9XG4gICAgICogICB9LFxuICAgICAqICAgYWN0aW9uOiB7IGJsb2NrOiB7fSB9LFxuICAgICAqICAgdmlzaWJpbGl0eUNvbmZpZzoge1xuICAgICAqICAgICBzYW1wbGVkUmVxdWVzdHNFbmFibGVkOiB0cnVlLFxuICAgICAqICAgICBjbG91ZFdhdGNoTWV0cmljc0VuYWJsZWQ6IHRydWUsXG4gICAgICogICAgIG1ldHJpY05hbWU6ICdCbG9ja1NwZWNpZmljVXNlckFnZW50J1xuICAgICAqICAgfVxuICAgICAqIH1dXG4gICAgICogYGBgXG4gICAgICovXG4gICAgcmVhZG9ubHkgY3VzdG9tUnVsZXM/OiBPbWl0PGltcG9ydCgnYXdzLWNkay1saWIvYXdzLXdhZnYyJykuQ2ZuV2ViQUNMLlJ1bGVQcm9wZXJ0eSwgJ3ByaW9yaXR5Jz5bXTtcbn1cblxuLyoqXG4gKiBEZWZhdWx0IGNvbmZpZ3VyYXRpb24gZm9yIFdBRiBvcHRpbWl6ZWQgZm9yIE51eHQgYXBwbGljYXRpb25zLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9OVVhUX1dBRl9DT05GSUc6IFBhcnRpYWw8V2FmQ29uZmlnPiA9IHtcbiAgICBlbmFibGVDb21tb25SdWxlU2V0OiB0cnVlLFxuICAgIGVuYWJsZUtub3duQmFkSW5wdXRzUnVsZVNldDogdHJ1ZSxcbiAgICBlbmFibGVBbm9ueW1vdXNJcFJ1bGVTZXQ6IGZhbHNlLFxuICAgIGVuYWJsZUFtYXpvbklwUmVwdXRhdGlvblJ1bGVTZXQ6IHRydWUsXG4gICAgZW5hYmxlQm90Q29udHJvbFJ1bGVTZXQ6IGZhbHNlLFxuICAgIHJhdGVMaW1pdDogMzAwLFxuICAgIG1ldHJpY3NQcmVmaXg6ICdXYWZNZXRyaWNzJyxcbn07XG5cbiJdfQ==
|
|
@@ -45,8 +45,9 @@ export interface WafConfig {
|
|
|
45
45
|
/**
|
|
46
46
|
* The maximum number of requests allowed from a single IP within a 5-minute period
|
|
47
47
|
* to protect against DDoS attacks.
|
|
48
|
+
* Does not apply to build files server under the `/_nuxt/` path.
|
|
48
49
|
* Can be disabled by setting it to `undefined`.
|
|
49
|
-
* @default
|
|
50
|
+
* @default 300
|
|
50
51
|
*/
|
|
51
52
|
readonly rateLimit?: number | undefined;
|
|
52
53
|
|
|
@@ -113,7 +114,7 @@ export const DEFAULT_NUXT_WAF_CONFIG: Partial<WafConfig> = {
|
|
|
113
114
|
enableAnonymousIpRuleSet: false,
|
|
114
115
|
enableAmazonIpReputationRuleSet: true,
|
|
115
116
|
enableBotControlRuleSet: false,
|
|
116
|
-
rateLimit:
|
|
117
|
+
rateLimit: 300,
|
|
117
118
|
metricsPrefix: 'WafMetrics',
|
|
118
119
|
};
|
|
119
120
|
|