cdk-nuxt 1.1.0 → 1.2.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -1
- package/lib/functions/assets-cleanup/build/app/index.js +94 -60
- package/lib/functions/assets-cleanup/build/app/index.js.map +1 -1
- package/lib/functions/assets-cleanup/index.js +95 -63
- package/lib/functions/assets-cleanup/index.ts +138 -70
- package/lib/stack/nuxt-app-static-assets.js +39 -13
- package/lib/stack/nuxt-app-static-assets.ts +40 -13
- package/lib/stack/server/nuxt-server-app-stack.d.ts +17 -5
- package/lib/stack/server/nuxt-server-app-stack.js +28 -8
- package/lib/stack/server/nuxt-server-app-stack.ts +40 -10
- package/package.json +2 -2
- package/lib/.DS_Store +0 -0
- package/lib/functions/.DS_Store +0 -0
- package/lib/functions/assets-cleanup/.DS_Store +0 -0
package/README.md
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
# CDK Nuxt 3
|
|
1
|
+
# AWS CDK Nuxt 3 Deployment Stack
|
|
2
2
|
|
|
3
3
|
<p>
|
|
4
4
|
<a href="https://github.com/ferdinandfrank/cdk-nuxt/actions/workflows/publish.yml"><img alt="Build" src="https://img.shields.io/github/actions/workflow/status/ferdinandfrank/cdk-nuxt/publish.yml?logo=github" /></a>
|
|
@@ -116,6 +116,11 @@ Whether to enable AWS X-Ray for the Nuxt Lambda function.
|
|
|
116
116
|
### enableSitemap?: boolean
|
|
117
117
|
Whether to enable a global Sitemap bucket which is permanently accessible through multiple deployments.
|
|
118
118
|
|
|
119
|
+
### outdatedAssetsRetentionDays?: boolean
|
|
120
|
+
The number of days to retain static assets of outdated deployments in the S3 bucket.
|
|
121
|
+
Useful to allow users to still access old assets after a new deployment when they are still browsing on an old version.
|
|
122
|
+
Defaults to 30 days.
|
|
123
|
+
|
|
119
124
|
### allowHeaders?: string[]
|
|
120
125
|
An array of headers to pass to the Nuxt app on SSR requests.
|
|
121
126
|
The more headers are passed, the weaker the cache performance will be, as the cache key
|
|
@@ -1,7 +1,51 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
const client_s3_1 = require("@aws-sdk/client-s3");
|
|
4
|
-
const
|
|
4
|
+
const consumers_1 = require("stream/consumers");
|
|
5
|
+
const MAX_DELETE_OBJECT_KEYS = 1000;
|
|
6
|
+
const ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;
|
|
7
|
+
/**
|
|
8
|
+
* Returns the current deployment revision (build timestamp).
|
|
9
|
+
*/
|
|
10
|
+
const getCurrentRevision = async (s3Client, bucketName) => {
|
|
11
|
+
const revisionFile = await s3Client.send(new client_s3_1.GetObjectCommand({
|
|
12
|
+
Bucket: bucketName,
|
|
13
|
+
Key: 'app-revision',
|
|
14
|
+
}));
|
|
15
|
+
return new Date(await (0, consumers_1.text)(revisionFile.Body));
|
|
16
|
+
};
|
|
17
|
+
/**
|
|
18
|
+
* Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.
|
|
19
|
+
*/
|
|
20
|
+
const filterOutdatedAssetKeys = (metadataResults, returnOlderThan) => {
|
|
21
|
+
return metadataResults
|
|
22
|
+
.filter(assetMetadata => assetMetadata.metadata.revision
|
|
23
|
+
? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()
|
|
24
|
+
: false)
|
|
25
|
+
.map(filteredAssets => filteredAssets.key);
|
|
26
|
+
};
|
|
27
|
+
/**
|
|
28
|
+
* Deletes the given assets.
|
|
29
|
+
*/
|
|
30
|
+
const deleteAssets = async (assetKeys, s3Client, bucketName) => {
|
|
31
|
+
let remainingAssetKeysToDelete = [...assetKeys];
|
|
32
|
+
const pendingDeletes = [];
|
|
33
|
+
while (remainingAssetKeysToDelete.length > 0) {
|
|
34
|
+
const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);
|
|
35
|
+
remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);
|
|
36
|
+
console.log('Deleting assets:', curDeleteBatch);
|
|
37
|
+
const pendingDelete = s3Client.send(new client_s3_1.DeleteObjectsCommand({
|
|
38
|
+
Bucket: bucketName,
|
|
39
|
+
Delete: {
|
|
40
|
+
Objects: curDeleteBatch.map(outdatedKey => {
|
|
41
|
+
return { Key: outdatedKey };
|
|
42
|
+
}),
|
|
43
|
+
},
|
|
44
|
+
}));
|
|
45
|
+
pendingDeletes.push(pendingDelete);
|
|
46
|
+
}
|
|
47
|
+
return await Promise.all(pendingDeletes);
|
|
48
|
+
};
|
|
5
49
|
exports.handler = async (event, context) => {
|
|
6
50
|
console.log('Starting cleanup of static assets older than 1 week...');
|
|
7
51
|
try {
|
|
@@ -10,68 +54,58 @@ exports.handler = async (event, context) => {
|
|
|
10
54
|
if (!bucketName) {
|
|
11
55
|
throw new Error("Static asset's bucket name not specified in environment!");
|
|
12
56
|
}
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
const deploymentFolders = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
16
|
-
Bucket: bucketName,
|
|
17
|
-
Delimiter: '/', // Only retrieve the top level folders
|
|
18
|
-
}));
|
|
19
|
-
// The result is a list of objects containing the keys with a trailing slash
|
|
20
|
-
const folderNames = deploymentFolders.CommonPrefixes?.map(folder => folder.Prefix?.slice(0, -1));
|
|
21
|
-
if (!folderNames) {
|
|
22
|
-
console.log('Canceled static assets cleanup as folder is empty.');
|
|
23
|
-
return;
|
|
57
|
+
if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {
|
|
58
|
+
throw new Error('Retain duration of static assets not specified!');
|
|
24
59
|
}
|
|
25
|
-
|
|
26
|
-
const
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
console.log('No outdated asset folders found.');
|
|
43
|
-
}
|
|
44
|
-
else {
|
|
45
|
-
console.log(`Deleting ${outdatedFolderNames.length} outdated folders...`);
|
|
46
|
-
// Unfortunately it's not possible to delete folders recursively 🙄
|
|
47
|
-
// so we need to query all the contents in order to delete them
|
|
48
|
-
const pendingPromises = outdatedFolderNames.map(folderName => {
|
|
49
|
-
return client
|
|
50
|
-
.send(new client_s3_1.ListObjectsV2Command({
|
|
60
|
+
const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);
|
|
61
|
+
const currentRevision = await getCurrentRevision(client, bucketName);
|
|
62
|
+
const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);
|
|
63
|
+
let assetKeysToDelete = [];
|
|
64
|
+
let lastToken = undefined;
|
|
65
|
+
do {
|
|
66
|
+
const curAssetsResult = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
67
|
+
Bucket: bucketName,
|
|
68
|
+
MaxKeys: 250,
|
|
69
|
+
ContinuationToken: lastToken,
|
|
70
|
+
}));
|
|
71
|
+
// Read object metadata in blocks of 10
|
|
72
|
+
let processableAssets = [...curAssetsResult.Contents];
|
|
73
|
+
while (processableAssets.length > 0) {
|
|
74
|
+
const assetsBatch = processableAssets.slice(0, 10);
|
|
75
|
+
processableAssets = processableAssets.slice(10);
|
|
76
|
+
const pendingMetadataRequests = assetsBatch.map(asset => client.send(new client_s3_1.HeadObjectCommand({
|
|
51
77
|
Bucket: bucketName,
|
|
52
|
-
|
|
53
|
-
}))
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
78
|
+
Key: asset.Key,
|
|
79
|
+
})));
|
|
80
|
+
const metadataResults = await Promise.all(pendingMetadataRequests);
|
|
81
|
+
// Assign metadata to assets
|
|
82
|
+
const metadataByAsset = metadataResults.map((metadataResult, index) => ({
|
|
83
|
+
key: assetsBatch[index].Key,
|
|
84
|
+
metadata: metadataResult.Metadata,
|
|
85
|
+
}));
|
|
86
|
+
const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);
|
|
87
|
+
assetKeysToDelete.push(...outdatedAssetKeys);
|
|
88
|
+
}
|
|
89
|
+
lastToken = curAssetsResult.NextContinuationToken;
|
|
90
|
+
} while (lastToken !== undefined);
|
|
91
|
+
if (assetKeysToDelete.length === 0) {
|
|
92
|
+
console.log('No outdated assets to delete found');
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
console.log('Deleting ' + assetKeysToDelete.length + ' assets...');
|
|
96
|
+
// Delete outdated assets (max. 1000 allowed per request)
|
|
97
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
98
|
+
const failed = results.reduce((previousResult, currentResult) => {
|
|
99
|
+
const currentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
100
|
+
if (currentError) {
|
|
101
|
+
console.error('Failed to delete outdated static assets', currentResult.Errors);
|
|
102
|
+
}
|
|
103
|
+
return previousResult || currentError;
|
|
104
|
+
}, false);
|
|
105
|
+
if (failed) {
|
|
106
|
+
throw new Error('Failed to delete outdated static assets');
|
|
73
107
|
}
|
|
74
|
-
console.log('Cleanup of old static assets finished
|
|
108
|
+
console.log('Cleanup of old static assets finished');
|
|
75
109
|
}
|
|
76
110
|
catch (error) {
|
|
77
111
|
console.error('### unexpected runtime error ###', error);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"/","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {DeleteObjectsCommand, ListObjectsV2Command, S3Client} from \"@aws-sdk/client-s3\";\n\nconst ONE_WEEK_IN_MILLISECONDS = 1000 * 60 * 60 * 24 * 7;\n\nexports.handler = async (event: any, context: any) => {\n console.log('Starting cleanup of static assets older than 1 week...');\n\n try {\n const client = new S3Client({region: process.env.AWS_REGION});\n const bucketName = process.env.STATIC_ASSETS_BUCKET;\n\n if (!bucketName) {\n throw new Error(\"Static asset's bucket name not specified in environment!\");\n }\n\n const deleteOlderThan = new Date(Date.now() - ONE_WEEK_IN_MILLISECONDS);\n\n // As we don't have hundreds of deployments per week, there's no need for pagination\n const deploymentFolders = await client.send(\n new ListObjectsV2Command({\n Bucket: bucketName,\n Delimiter: '/', // Only retrieve the top level folders\n })\n );\n\n // The result is a list of objects containing the keys with a trailing slash\n const folderNames = deploymentFolders.CommonPrefixes?.map(folder => folder.Prefix?.slice(0, -1))\n if (!folderNames) {\n console.log('Canceled static assets cleanup as folder is empty.');\n return;\n }\n\n // We sort the folders by their creation data and remove the latest one as this is the currently active one used in production\n const legacyFolderNames = folderNames.sort((a,b) => {\n // @ts-ignore\n const creationDateA = new Date(a);\n // @ts-ignore\n const creationDateB = new Date(b);\n\n return creationDateA.getTime() < creationDateB.getTime() ? -1 : 1;\n })\n const activeFolderName = legacyFolderNames.pop();\n console.log(`Detected assets folder \"${activeFolderName}\" as current production folder...`);\n\n // We want to get all outdated folders of our legacy (not productively used) folders for deletion\n const outdatedFolderNames = legacyFolderNames.filter(folderName => {\n // @ts-ignore\n const creationDate = new Date(folderName);\n return creationDate.getTime() < deleteOlderThan.getTime();\n }\n );\n\n if (!outdatedFolderNames || outdatedFolderNames.length === 0) {\n console.log('No outdated asset folders found.');\n } else {\n console.log(`Deleting ${outdatedFolderNames.length} outdated folders...`);\n\n // Unfortunately it's not possible to delete folders recursively 🙄\n // so we need to query all the contents in order to delete them\n const pendingPromises = outdatedFolderNames.map(folderName => {\n return client\n .send(\n new ListObjectsV2Command({\n Bucket: bucketName,\n Prefix: folderName,\n })\n )\n .then(outdatedAssets => {\n return client.send(\n new DeleteObjectsCommand({\n Bucket: bucketName,\n Delete: {\n Objects: outdatedAssets.Contents?.map(outdatedAsset => {\n return {Key: outdatedAsset.Key};\n }),\n },\n })\n );\n });\n });\n\n const results = await Promise.all(pendingPromises);\n results.forEach(result => {\n if (result.Errors && result.Errors.length) {\n const errorMsg = 'Failed to delete outdated static assets.';\n console.error(errorMsg, result.Errors);\n throw new Error(errorMsg);\n }\n });\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 console.log('Starting cleanup of static assets older than 1 week...');\n\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 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,9 +1,52 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
const client_s3_1 = require("@aws-sdk/client-s3");
|
|
4
|
-
const
|
|
4
|
+
const consumers_1 = require("stream/consumers");
|
|
5
|
+
const MAX_DELETE_OBJECT_KEYS = 1000;
|
|
6
|
+
const ONE_DAY_IN_MILLISECONDS = 1000 * 60 * 60 * 24;
|
|
7
|
+
/**
|
|
8
|
+
* Returns the current deployment revision (build timestamp).
|
|
9
|
+
*/
|
|
10
|
+
const getCurrentRevision = async (s3Client, bucketName) => {
|
|
11
|
+
const revisionFile = await s3Client.send(new client_s3_1.GetObjectCommand({
|
|
12
|
+
Bucket: bucketName,
|
|
13
|
+
Key: 'app-revision',
|
|
14
|
+
}));
|
|
15
|
+
return new Date(await (0, consumers_1.text)(revisionFile.Body));
|
|
16
|
+
};
|
|
17
|
+
/**
|
|
18
|
+
* Filters the given assets by inspecting their revision and returns those, that are older than the specified cutoff date.
|
|
19
|
+
*/
|
|
20
|
+
const filterOutdatedAssetKeys = (metadataResults, returnOlderThan) => {
|
|
21
|
+
return metadataResults
|
|
22
|
+
.filter(assetMetadata => assetMetadata.metadata.revision
|
|
23
|
+
? new Date(assetMetadata.metadata.revision).getTime() <= returnOlderThan.getTime()
|
|
24
|
+
: false)
|
|
25
|
+
.map(filteredAssets => filteredAssets.key);
|
|
26
|
+
};
|
|
27
|
+
/**
|
|
28
|
+
* Deletes the given assets.
|
|
29
|
+
*/
|
|
30
|
+
const deleteAssets = async (assetKeys, s3Client, bucketName) => {
|
|
31
|
+
let remainingAssetKeysToDelete = [...assetKeys];
|
|
32
|
+
const pendingDeletes = [];
|
|
33
|
+
while (remainingAssetKeysToDelete.length > 0) {
|
|
34
|
+
const curDeleteBatch = remainingAssetKeysToDelete.slice(0, MAX_DELETE_OBJECT_KEYS);
|
|
35
|
+
remainingAssetKeysToDelete = remainingAssetKeysToDelete.slice(MAX_DELETE_OBJECT_KEYS);
|
|
36
|
+
console.log('Deleting assets:', curDeleteBatch);
|
|
37
|
+
const pendingDelete = s3Client.send(new client_s3_1.DeleteObjectsCommand({
|
|
38
|
+
Bucket: bucketName,
|
|
39
|
+
Delete: {
|
|
40
|
+
Objects: curDeleteBatch.map(outdatedKey => {
|
|
41
|
+
return { Key: outdatedKey };
|
|
42
|
+
}),
|
|
43
|
+
},
|
|
44
|
+
}));
|
|
45
|
+
pendingDeletes.push(pendingDelete);
|
|
46
|
+
}
|
|
47
|
+
return await Promise.all(pendingDeletes);
|
|
48
|
+
};
|
|
5
49
|
exports.handler = async (event, context) => {
|
|
6
|
-
var _a;
|
|
7
50
|
console.log('Starting cleanup of static assets older than 1 week...');
|
|
8
51
|
try {
|
|
9
52
|
const client = new client_s3_1.S3Client({ region: process.env.AWS_REGION });
|
|
@@ -11,72 +54,61 @@ exports.handler = async (event, context) => {
|
|
|
11
54
|
if (!bucketName) {
|
|
12
55
|
throw new Error("Static asset's bucket name not specified in environment!");
|
|
13
56
|
}
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
const deploymentFolders = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
17
|
-
Bucket: bucketName,
|
|
18
|
-
Delimiter: '/', // Only retrieve the top level folders
|
|
19
|
-
}));
|
|
20
|
-
// The result is a list of objects containing the keys with a trailing slash
|
|
21
|
-
const folderNames = (_a = deploymentFolders.CommonPrefixes) === null || _a === void 0 ? void 0 : _a.map(folder => { var _a; return (_a = folder.Prefix) === null || _a === void 0 ? void 0 : _a.slice(0, -1); });
|
|
22
|
-
if (!folderNames) {
|
|
23
|
-
console.log('Canceled static assets cleanup as folder is empty.');
|
|
24
|
-
return;
|
|
57
|
+
if (!process.env.OUTDATED_ASSETS_RETENTION_DAYS) {
|
|
58
|
+
throw new Error('Retain duration of static assets not specified!');
|
|
25
59
|
}
|
|
26
|
-
|
|
27
|
-
const
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
console.log('No outdated asset folders found.');
|
|
44
|
-
}
|
|
45
|
-
else {
|
|
46
|
-
console.log(`Deleting ${outdatedFolderNames.length} outdated folders...`);
|
|
47
|
-
// Unfortunately it's not possible to delete folders recursively 🙄
|
|
48
|
-
// so we need to query all the contents in order to delete them
|
|
49
|
-
const pendingPromises = outdatedFolderNames.map(folderName => {
|
|
50
|
-
return client
|
|
51
|
-
.send(new client_s3_1.ListObjectsV2Command({
|
|
60
|
+
const retainAssetsInDays = Number.parseInt(process.env.OUTDATED_ASSETS_RETENTION_DAYS);
|
|
61
|
+
const currentRevision = await getCurrentRevision(client, bucketName);
|
|
62
|
+
const deleteOlderThan = new Date(currentRevision.getTime() - retainAssetsInDays * ONE_DAY_IN_MILLISECONDS);
|
|
63
|
+
let assetKeysToDelete = [];
|
|
64
|
+
let lastToken = undefined;
|
|
65
|
+
do {
|
|
66
|
+
const curAssetsResult = await client.send(new client_s3_1.ListObjectsV2Command({
|
|
67
|
+
Bucket: bucketName,
|
|
68
|
+
MaxKeys: 250,
|
|
69
|
+
ContinuationToken: lastToken,
|
|
70
|
+
}));
|
|
71
|
+
// Read object metadata in blocks of 10
|
|
72
|
+
let processableAssets = [...curAssetsResult.Contents];
|
|
73
|
+
while (processableAssets.length > 0) {
|
|
74
|
+
const assetsBatch = processableAssets.slice(0, 10);
|
|
75
|
+
processableAssets = processableAssets.slice(10);
|
|
76
|
+
const pendingMetadataRequests = assetsBatch.map(asset => client.send(new client_s3_1.HeadObjectCommand({
|
|
52
77
|
Bucket: bucketName,
|
|
53
|
-
|
|
54
|
-
}))
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
78
|
+
Key: asset.Key,
|
|
79
|
+
})));
|
|
80
|
+
const metadataResults = await Promise.all(pendingMetadataRequests);
|
|
81
|
+
// Assign metadata to assets
|
|
82
|
+
const metadataByAsset = metadataResults.map((metadataResult, index) => ({
|
|
83
|
+
key: assetsBatch[index].Key,
|
|
84
|
+
metadata: metadataResult.Metadata,
|
|
85
|
+
}));
|
|
86
|
+
const outdatedAssetKeys = filterOutdatedAssetKeys(metadataByAsset, deleteOlderThan);
|
|
87
|
+
assetKeysToDelete.push(...outdatedAssetKeys);
|
|
88
|
+
}
|
|
89
|
+
lastToken = curAssetsResult.NextContinuationToken;
|
|
90
|
+
} while (lastToken !== undefined);
|
|
91
|
+
if (assetKeysToDelete.length === 0) {
|
|
92
|
+
console.log('No outdated assets to delete found');
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
console.log('Deleting ' + assetKeysToDelete.length + ' assets...');
|
|
96
|
+
// Delete outdated assets (max. 1000 allowed per request)
|
|
97
|
+
const results = await deleteAssets(assetKeysToDelete, client, bucketName);
|
|
98
|
+
const failed = results.reduce((previousResult, currentResult) => {
|
|
99
|
+
const currentError = !!(currentResult.Errors && currentResult.Errors.length > 0);
|
|
100
|
+
if (currentError) {
|
|
101
|
+
console.error('Failed to delete outdated static assets', currentResult.Errors);
|
|
102
|
+
}
|
|
103
|
+
return previousResult || currentError;
|
|
104
|
+
}, false);
|
|
105
|
+
if (failed) {
|
|
106
|
+
throw new Error('Failed to delete outdated static assets');
|
|
75
107
|
}
|
|
76
|
-
console.log('Cleanup of old static assets finished
|
|
108
|
+
console.log('Cleanup of old static assets finished');
|
|
77
109
|
}
|
|
78
110
|
catch (error) {
|
|
79
111
|
console.error('### unexpected runtime error ###', error);
|
|
80
112
|
}
|
|
81
113
|
};
|
|
82
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {DeleteObjectsCommand, ListObjectsV2Command, S3Client} from \"@aws-sdk/client-s3\";\n\nconst ONE_WEEK_IN_MILLISECONDS = 1000 * 60 * 60 * 24 * 7;\n\nexports.handler = async (event: any, context: any) => {\n    console.log('Starting cleanup of static assets older than 1 week...');\n\n    try {\n        const client = new S3Client({region: process.env.AWS_REGION});\n        const bucketName = process.env.STATIC_ASSETS_BUCKET;\n\n        if (!bucketName) {\n            throw new Error(\"Static asset's bucket name not specified in environment!\");\n        }\n\n        const deleteOlderThan = new Date(Date.now() - ONE_WEEK_IN_MILLISECONDS);\n\n        // As we don't have hundreds of deployments per week, there's no need for pagination\n        const deploymentFolders = await client.send(\n            new ListObjectsV2Command({\n                Bucket: bucketName,\n                Delimiter: '/', // Only retrieve the top level folders\n            })\n        );\n\n        // The result is a list of objects containing the keys with a trailing slash\n        const folderNames = deploymentFolders.CommonPrefixes?.map(folder => folder.Prefix?.slice(0, -1))\n        if (!folderNames) {\n            console.log('Canceled static assets cleanup as folder is empty.');\n            return;\n        }\n\n        // We sort the folders by their creation data and remove the latest one as this is the currently active one used in production\n        const legacyFolderNames = folderNames.sort((a,b) => {\n            // @ts-ignore\n            const creationDateA = new Date(a);\n            // @ts-ignore\n            const creationDateB = new Date(b);\n\n            return creationDateA.getTime() < creationDateB.getTime() ? -1 : 1;\n        })\n        const activeFolderName = legacyFolderNames.pop();\n        console.log(`Detected assets folder \"${activeFolderName}\" as current production folder...`);\n\n        // We want to get all outdated folders of our legacy (not productively used) folders for deletion\n        const outdatedFolderNames = legacyFolderNames.filter(folderName => {\n                // @ts-ignore\n                const creationDate = new Date(folderName);\n                return creationDate.getTime() < deleteOlderThan.getTime();\n            }\n        );\n\n        if (!outdatedFolderNames || outdatedFolderNames.length === 0) {\n            console.log('No outdated asset folders found.');\n        } else {\n            console.log(`Deleting ${outdatedFolderNames.length} outdated folders...`);\n\n            // Unfortunately it's not possible to delete folders recursively 🙄\n            // so we need to query all the contents in order to delete them\n            const pendingPromises = outdatedFolderNames.map(folderName => {\n                return client\n                    .send(\n                        new ListObjectsV2Command({\n                            Bucket: bucketName,\n                            Prefix: folderName,\n                        })\n                    )\n                    .then(outdatedAssets => {\n                        return client.send(\n                            new DeleteObjectsCommand({\n                                Bucket: bucketName,\n                                Delete: {\n                                    Objects: outdatedAssets.Contents?.map(outdatedAsset => {\n                                        return {Key: outdatedAsset.Key};\n                                    }),\n                                },\n                            })\n                        );\n                    });\n            });\n\n            const results = await Promise.all(pendingPromises);\n            results.forEach(result => {\n                if (result.Errors && result.Errors.length) {\n                    const errorMsg = 'Failed to delete outdated static assets.';\n                    console.error(errorMsg, result.Errors);\n                    throw new Error(errorMsg);\n                }\n            });\n        }\n\n        console.log('Cleanup of old static assets finished.');\n    } catch (error) {\n        console.error('### unexpected runtime error ###', error);\n    }\n};"]}
|
|
114
|
+
//# 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    console.log('Starting cleanup of static assets older than 1 week...');\n\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        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};"]}
|