cdk-nuxt 0.2.10 → 0.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,14 +1,5 @@
1
- # Welcome to your CDK TypeScript project!
1
+ # Easy Nuxt deployments on AWS via CDK
2
2
 
3
- This is a blank project for TypeScript development with CDK.
3
+ This package allows deployments of Nuxt applications via CDK on AWS by using Route53, API Gateway, Cloudfront, Lambda and S3.
4
4
 
5
- The `cdk.json` file tells the CDK Toolkit how to execute your app.
6
-
7
- ## Useful commands
8
-
9
- * `npm run build` compile typescript to js
10
- * `npm run watch` watch for changes and compile
11
- * `npm run test` perform the jest unit tests
12
- * `cdk deploy` deploy this stack to your default AWS account/region
13
- * `cdk diff` compare deployed stack with current state
14
- * `cdk synth` emits the synthesized CloudFormation template
5
+ This package is currently in development and will be published with a full Readme shortly.
package/index.d.ts CHANGED
@@ -1,2 +1,3 @@
1
+ export { AppStackProps } from "./lib/stack/app-stack-props";
1
2
  export { NuxtAppStack, NuxtAppStackProps } from "./lib/stack/nuxt-app-stack";
2
3
  export { NuxtAppAssetsCleanupStack, NuxtAppAssetsCleanupProps } from "./lib/stack/nuxt-app-assets-cleanup-stack";
package/index.js CHANGED
@@ -1,7 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.NuxtAppAssetsCleanupStack = exports.NuxtAppStack = void 0;
3
4
  var nuxt_app_stack_1 = require("./lib/stack/nuxt-app-stack");
4
5
  Object.defineProperty(exports, "NuxtAppStack", { enumerable: true, get: function () { return nuxt_app_stack_1.NuxtAppStack; } });
5
6
  var nuxt_app_assets_cleanup_stack_1 = require("./lib/stack/nuxt-app-assets-cleanup-stack");
6
7
  Object.defineProperty(exports, "NuxtAppAssetsCleanupStack", { enumerable: true, get: function () { return nuxt_app_assets_cleanup_stack_1.NuxtAppAssetsCleanupStack; } });
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLDZEQUEwRTtBQUFsRSw4R0FBQSxZQUFZLE9BQUE7QUFDcEIsMkZBQThHO0FBQXRHLDBJQUFBLHlCQUF5QixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHtOdXh0QXBwU3RhY2ssIE51eHRBcHBTdGFja1Byb3BzfSBmcm9tIFwiLi9saWIvc3RhY2svbnV4dC1hcHAtc3RhY2tcIlxuZXhwb3J0IHtOdXh0QXBwQXNzZXRzQ2xlYW51cFN0YWNrLCBOdXh0QXBwQXNzZXRzQ2xlYW51cFByb3BzfSBmcm9tIFwiLi9saWIvc3RhY2svbnV4dC1hcHAtYXNzZXRzLWNsZWFudXAtc3RhY2tcIiJdfQ==
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFDQSw2REFBMEU7QUFBbEUsOEdBQUEsWUFBWSxPQUFBO0FBQ3BCLDJGQUE4RztBQUF0RywwSUFBQSx5QkFBeUIsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7QXBwU3RhY2tQcm9wc30gZnJvbSBcIi4vbGliL3N0YWNrL2FwcC1zdGFjay1wcm9wc1wiXG5leHBvcnQge051eHRBcHBTdGFjaywgTnV4dEFwcFN0YWNrUHJvcHN9IGZyb20gXCIuL2xpYi9zdGFjay9udXh0LWFwcC1zdGFja1wiXG5leHBvcnQge051eHRBcHBBc3NldHNDbGVhbnVwU3RhY2ssIE51eHRBcHBBc3NldHNDbGVhbnVwUHJvcHN9IGZyb20gXCIuL2xpYi9zdGFjay9udXh0LWFwcC1hc3NldHMtY2xlYW51cC1zdGFja1wiIl19
package/lib/cli/deploy.js CHANGED
@@ -3,54 +3,71 @@
3
3
  const shell = require("shelljs");
4
4
  const path = require("path");
5
5
 
6
+ const logPrefix = 'Nuxt Deployment';
7
+
8
+ shell.echo(`${logPrefix}: Starting deployment...`);
9
+
6
10
  const deploymentFolder = '.nuxt/cdk-deployment';
7
11
  const deploymentSourceFolder = `${deploymentFolder}/src`;
8
12
  const deploymentLayerFolder = `${deploymentFolder}/layer/nodejs`;
13
+ const rootFolder = process.cwd();
9
14
 
10
- // Refresh the deployment folder to a clean state
11
- shell.echo('Refreshing deployment folder...');
15
+ // Refresh the cdk output folder to have a clean state and prevent persisting outdated outputs
16
+ shell.echo(`${logPrefix}: Deleting outdated CDK files...`);
17
+ shell.rm('-rf', 'cdk.out');
18
+
19
+ // Refresh the deployment folder to have a clean state
20
+ shell.echo(`${logPrefix}: Refreshing deployment folder [${deploymentFolder}]...`);
12
21
  shell.rm('-rf', deploymentFolder);
13
22
  shell.mkdir('-p', `${deploymentSourceFolder}/.nuxt/dist`);
14
23
  shell.mkdir('-p', deploymentLayerFolder);
15
24
 
16
25
  // Preparing the assets cleanup lambda function
17
- shell.echo('Preparing the assets cleanup lambda function...');
26
+ shell.echo(`${logPrefix}: Installing the assets cleanup lambda function...`);
18
27
  shell.cd(path.join(__dirname, '../functions/assets_cleanup'));
19
28
  if (shell.exec('yarn install --frozen-lockfile && yarn install-layer').code !== 0) {
20
- shell.echo('Error: Installation of assets cleanup lambda function failed.');
29
+ shell.echo(`${logPrefix} Error: Installation of assets cleanup lambda function failed.`);
21
30
  shell.exit(1);
22
31
  }
32
+ shell.echo(`${logPrefix}: Building the assets cleanup lambda function...`);
23
33
  if (shell.exec('yarn build').code !== 0) {
24
- shell.echo('Error: Build of assets cleanup lambda function failed.');
34
+ shell.echo(`${logPrefix} Error: Build of assets cleanup lambda function failed.`);
25
35
  shell.exit(1);
26
36
  }
27
- shell.cd('../../../../..');
37
+ shell.cd(rootFolder);
28
38
 
29
39
  // Copy our files required for SSR
30
- shell.echo('Copying ssr files for Nuxt app...');
40
+ shell.echo(`${logPrefix}: Copying ssr files to deployment folder...`);
31
41
  shell.cp('-r', '.nuxt/dist/server', `${deploymentSourceFolder}/.nuxt/dist`);
32
42
  shell.cp(path.join(__dirname, '../functions/app/index.js'), deploymentSourceFolder);
33
43
 
44
+ // Copy the config files
45
+ shell.echo(`${logPrefix}: Copying config files to deployment folder...`);
46
+ shell.cp('.env', deploymentSourceFolder);
47
+ shell.cp('nuxt.config.js', deploymentSourceFolder);
48
+
34
49
  // Prepare the Lambda layer for the node_modules required for SSR of the Nuxt app
35
- shell.echo('Preparing the lambda layer for Nuxt app...');
50
+ shell.echo(`${logPrefix}: Preparing the node_modules lambda layer in deployment folder...`);
36
51
  shell.cp('package.json', deploymentLayerFolder);
37
52
  shell.cp('yarn.lock', deploymentLayerFolder);
53
+ shell.cp('.yarnclean', deploymentLayerFolder);
38
54
  shell.cd(deploymentLayerFolder);
55
+
56
+ // We do not want to install any dependencies listed under 'devDendencies'
57
+ // Usually the --production flag should do the trick but somehow still installs more dependencies as required
58
+ shell.exec('sed -i \'\' \'/\\"devDependencies\\"/,/}/ d\' package.json')
39
59
  if (shell.exec('yarn install --production --frozen-lockfile').code !== 0) {
40
- shell.echo('Error: Installation of lambda layer failed.');
60
+ shell.echo(`${logPrefix} Error: Installation of lambda layer failed.`);
41
61
  shell.exit(1);
42
62
  }
43
- shell.cd('../../../..');
44
-
45
- // Copy the config files
46
- shell.echo('Copying config files...');
47
- shell.cp('.env', deploymentSourceFolder);
48
- shell.cp('nuxt.config.js', deploymentSourceFolder);
63
+ shell.cd(rootFolder);
49
64
 
50
65
  // Run the deployment
66
+ shell.echo(`${logPrefix}: Running deployment to AWS via CDK...`);
51
67
  if (shell.exec('yarn cdk deploy --require-approval never --all --app="yarn ts-node stack/index.ts"').code !== 0) {
52
- shell.echo('Error: CDK deployment failed.');
68
+ shell.echo(`${logPrefix} Error: CDK deployment failed.`);
53
69
  shell.exit(1);
54
70
  }
55
71
 
56
- shell.echo('CDK deployment succeeded.');
72
+ shell.echo(`${logPrefix}: CDK deployment successful.`);
73
+
@@ -15,7 +15,7 @@ exports.handler = async (event, context) => {
15
15
  // As we don't have hundreds of deployments per week, there's no need for pagination
16
16
  const deploymentFolders = await client.send(new client_s3_1.ListObjectsV2Command({
17
17
  Bucket: bucketName,
18
- Delimiter: '/',
18
+ Delimiter: '/', // Only retrieve the top level folders
19
19
  }));
20
20
  // The result is a list of objects containing the keys with a trailing slash
21
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); });
@@ -79,4 +79,4 @@ exports.handler = async (event, context) => {
79
79
  console.error('### unexpected runtime error ###', error);
80
80
  }
81
81
  };
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};"]}
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};"]}
@@ -10,8 +10,8 @@
10
10
  },
11
11
  "devDependencies": {
12
12
  "@types/node": "^16.11.0",
13
- "ts-node": "^10.3.0",
14
- "typescript": "^4.4.0"
13
+ "ts-node": "^10.5.0",
14
+ "typescript": "^4.5.5"
15
15
  },
16
16
  "dependencies": {
17
17
  "@aws-sdk/client-s3": "^3.37.0"
@@ -909,7 +909,7 @@ make-error@^1.1.1:
909
909
  resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2"
910
910
  integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==
911
911
 
912
- ts-node@^10.3.0:
912
+ ts-node@^10.5.0:
913
913
  version "10.5.0"
914
914
  resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-10.5.0.tgz#618bef5854c1fbbedf5e31465cbb224a1d524ef9"
915
915
  integrity sha512-6kEJKwVxAJ35W4akuiysfKwKmjkbYxwQMTBaAxo9KKAx/Yd26mPUyhGz3ji+EsJoAgrLqVsYHNuuYwQe22lbtw==
@@ -938,7 +938,7 @@ tslib@^2.3.0:
938
938
  resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.3.1.tgz#e8a335add5ceae51aa261d32a490158ef042ef01"
939
939
  integrity sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==
940
940
 
941
- typescript@^4.4.0:
941
+ typescript@^4.5.5:
942
942
  version "4.5.5"
943
943
  resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.5.5.tgz#d8c953832d28924a9e3d37c73d729c846c5896f3"
944
944
  integrity sha512-TCTIul70LyWe6IJWT8QSYeA54WQe8EjQFU4wY52Fasj5UKx88LNYKCgBEHcOMOrFF1rKGbD8v/xcNWVUq9SymA==
@@ -1,6 +1,22 @@
1
1
  import { StackProps } from "aws-cdk-lib";
2
+ /**
3
+ * Defines the common props required for the individual stacks
4
+ * of our Nuxt CDK app.
5
+ */
2
6
  export interface AppStackProps extends StackProps {
7
+ /**
8
+ * A string identifier for the project the Nuxt app is part of.
9
+ * A project might have multiple different services.
10
+ */
3
11
  readonly project: string;
12
+ /**
13
+ * A string identifier for the project's service the Nuxt app is created for.
14
+ * This can be seen as the name of the Nuxt app.
15
+ */
4
16
  readonly service: string;
17
+ /**
18
+ * A string to identify the environment of the Nuxt app. This enables us
19
+ * to deploy multiple different environments of the same Nuxt app, e.g., production and development.
20
+ */
5
21
  readonly environment: string;
6
22
  }
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLXN0YWNrLXByb3BzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiYXBwLXN0YWNrLXByb3BzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1N0YWNrUHJvcHN9IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuXG5leHBvcnQgaW50ZXJmYWNlIEFwcFN0YWNrUHJvcHMgZXh0ZW5kcyBTdGFja1Byb3BzIHtcbiAgICByZWFkb25seSBwcm9qZWN0OiBzdHJpbmc7XG4gICAgcmVhZG9ubHkgc2VydmljZTogc3RyaW5nO1xuICAgIHJlYWRvbmx5IGVudmlyb25tZW50OiBzdHJpbmc7XG59Il19
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLXN0YWNrLXByb3BzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiYXBwLXN0YWNrLXByb3BzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1N0YWNrUHJvcHN9IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuXG4vKipcbiAqIERlZmluZXMgdGhlIGNvbW1vbiBwcm9wcyByZXF1aXJlZCBmb3IgdGhlIGluZGl2aWR1YWwgc3RhY2tzXG4gKiBvZiBvdXIgTnV4dCBDREsgYXBwLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEFwcFN0YWNrUHJvcHMgZXh0ZW5kcyBTdGFja1Byb3BzIHtcblxuICAgIC8qKlxuICAgICAqIEEgc3RyaW5nIGlkZW50aWZpZXIgZm9yIHRoZSBwcm9qZWN0IHRoZSBOdXh0IGFwcCBpcyBwYXJ0IG9mLlxuICAgICAqIEEgcHJvamVjdCBtaWdodCBoYXZlIG11bHRpcGxlIGRpZmZlcmVudCBzZXJ2aWNlcy5cbiAgICAgKi9cbiAgICByZWFkb25seSBwcm9qZWN0OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBBIHN0cmluZyBpZGVudGlmaWVyIGZvciB0aGUgcHJvamVjdCdzIHNlcnZpY2UgdGhlIE51eHQgYXBwIGlzIGNyZWF0ZWQgZm9yLlxuICAgICAqIFRoaXMgY2FuIGJlIHNlZW4gYXMgdGhlIG5hbWUgb2YgdGhlIE51eHQgYXBwLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHNlcnZpY2U6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqIEEgc3RyaW5nIHRvIGlkZW50aWZ5IHRoZSBlbnZpcm9ubWVudCBvZiB0aGUgTnV4dCBhcHAuIFRoaXMgZW5hYmxlcyB1c1xuICAgICAqIHRvIGRlcGxveSBtdWx0aXBsZSBkaWZmZXJlbnQgZW52aXJvbm1lbnRzIG9mIHRoZSBzYW1lIE51eHQgYXBwLCBlLmcuLCBwcm9kdWN0aW9uIGFuZCBkZXZlbG9wbWVudC5cbiAgICAgKi9cbiAgICByZWFkb25seSBlbnZpcm9ubWVudDogc3RyaW5nO1xufSJdfQ==
@@ -1,7 +1,26 @@
1
1
  import {StackProps} from "aws-cdk-lib";
2
2
 
3
+ /**
4
+ * Defines the common props required for the individual stacks
5
+ * of our Nuxt CDK app.
6
+ */
3
7
  export interface AppStackProps extends StackProps {
8
+
9
+ /**
10
+ * A string identifier for the project the Nuxt app is part of.
11
+ * A project might have multiple different services.
12
+ */
4
13
  readonly project: string;
14
+
15
+ /**
16
+ * A string identifier for the project's service the Nuxt app is created for.
17
+ * This can be seen as the name of the Nuxt app.
18
+ */
5
19
  readonly service: string;
20
+
21
+ /**
22
+ * A string to identify the environment of the Nuxt app. This enables us
23
+ * to deploy multiple different environments of the same Nuxt app, e.g., production and development.
24
+ */
6
25
  readonly environment: string;
7
26
  }
@@ -2,21 +2,52 @@ import { IBucket } from "aws-cdk-lib/aws-s3";
2
2
  import { Stack } from "aws-cdk-lib";
3
3
  import { AppStackProps } from "./app-stack-props";
4
4
  import { Construct } from "constructs";
5
+ /**
6
+ * Defines the props required for the {@see NuxtAppAssetsCleanupStack}.
7
+ */
5
8
  export interface NuxtAppAssetsCleanupProps extends AppStackProps {
9
+ /**
10
+ * The S3 bucket where the static assets of the Nuxt app are located.
11
+ */
6
12
  readonly staticAssetsBucket: IBucket;
7
- readonly sentryDsn: string;
8
13
  }
9
14
  /**
10
- * Contains a scheduled lambda function, that deletes outdated static assets of the PWA from S3.
15
+ * Creates a scheduled lambda function that deletes outdated static assets of the Nuxt app from S3
16
+ * to keep it nice without any outdated files that aren't used anymore.
11
17
  */
12
18
  export declare class NuxtAppAssetsCleanupStack extends Stack {
19
+ /**
20
+ * The identifier prefix of the resources created by the stack.
21
+ *
22
+ * @private
23
+ */
13
24
  private readonly resourceIdPrefix;
14
- private staticAssetsBucket;
15
- private readonly layer;
25
+ /**
26
+ * A reference to the created lambda function that cleanups the outdated static assets of the Nuxt app.
27
+ *
28
+ * @private
29
+ */
16
30
  private readonly lambdaFunction;
17
- private scheduledRule;
18
31
  constructor(scope: Construct, id: string, props: NuxtAppAssetsCleanupProps);
19
- private createLayer;
32
+ /**
33
+ * Creates the lambda function that cleanups the outdated static assets of the Nuxt app.
34
+ *
35
+ * @param props
36
+ * @private
37
+ */
20
38
  private createLambdaFunction;
39
+ /**
40
+ * Creates a lambda layer for the cleanup function that holds the required node_modules.
41
+ *
42
+ * @private
43
+ */
44
+ private createNodeModulesLayer;
45
+ /**
46
+ * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger
47
+ * our cleanup lamda function.
48
+ *
49
+ * @param props
50
+ * @private
51
+ */
21
52
  private createRule;
22
53
  }
@@ -8,58 +8,71 @@ const aws_logs_1 = require("aws-cdk-lib/aws-logs");
8
8
  const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets");
9
9
  const path = require("path");
10
10
  /**
11
- * Contains a scheduled lambda function, that deletes outdated static assets of the PWA from S3.
11
+ * Creates a scheduled lambda function that deletes outdated static assets of the Nuxt app from S3
12
+ * to keep it nice without any outdated files that aren't used anymore.
12
13
  */
13
14
  class NuxtAppAssetsCleanupStack extends aws_cdk_lib_1.Stack {
14
15
  constructor(scope, id, props) {
15
16
  super(scope, id, props);
16
- // Note, that this stack can't be tagged independently of the NuxtAppStack, as both are sharing the
17
- // same scope (CDK App). This would overwrite previous set tags for the CDK App (last one wins).
18
17
  this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;
19
- this.staticAssetsBucket = props.staticAssetsBucket;
20
- this.layer = this.createLayer();
21
18
  this.lambdaFunction = this.createLambdaFunction(props);
22
- this.scheduledRule = this.createRule(props);
23
- }
24
- createLayer() {
25
- const layerName = `${this.resourceIdPrefix}-layer`;
26
- return new aws_lambda_1.LayerVersion(this, layerName, {
27
- layerVersionName: layerName,
28
- code: aws_lambda_1.Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),
29
- compatibleRuntimes: [aws_lambda_1.Runtime.NODEJS_14_X],
30
- });
19
+ this.createRule(props);
31
20
  }
21
+ /**
22
+ * Creates the lambda function that cleanups the outdated static assets of the Nuxt app.
23
+ *
24
+ * @param props
25
+ * @private
26
+ */
32
27
  createLambdaFunction(props) {
33
28
  const functionName = `${this.resourceIdPrefix}-function`;
34
29
  const result = new aws_lambda_1.Function(this, functionName, {
35
30
  functionName: functionName,
31
+ description: `Auto-deletes the outdated static assets in the ${props.staticAssetsBucket.bucketName} S3 bucket.`,
36
32
  runtime: aws_lambda_1.Runtime.NODEJS_14_X,
37
33
  architecture: aws_lambda_1.Architecture.ARM_64,
38
- layers: [this.layer],
34
+ layers: [this.createNodeModulesLayer()],
39
35
  handler: 'index.handler',
40
36
  code: aws_lambda_1.Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/app')),
41
37
  timeout: aws_cdk_lib_1.Duration.minutes(1),
42
38
  memorySize: 128,
43
39
  logRetention: aws_logs_1.RetentionDays.TWO_WEEKS,
44
40
  environment: {
45
- STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,
46
- SENTRY_DSN: props.sentryDsn,
41
+ STATIC_ASSETS_BUCKET: props.staticAssetsBucket.bucketName,
47
42
  ENVIRONMENT: props.environment,
48
43
  AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',
49
44
  NODE_OPTIONS: '--enable-source-maps',
50
45
  },
51
46
  });
52
47
  // grant function access to S3 bucket
53
- this.staticAssetsBucket.grantRead(result);
54
- this.staticAssetsBucket.grantDelete(result);
48
+ props.staticAssetsBucket.grantRead(result);
49
+ props.staticAssetsBucket.grantDelete(result);
55
50
  return result;
56
51
  }
52
+ /**
53
+ * Creates a lambda layer for the cleanup function that holds the required node_modules.
54
+ *
55
+ * @private
56
+ */
57
+ createNodeModulesLayer() {
58
+ return new aws_lambda_1.LayerVersion(this, `${this.resourceIdPrefix}-layer`, {
59
+ layerVersionName: `${this.resourceIdPrefix}-layer`,
60
+ code: aws_lambda_1.Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),
61
+ compatibleRuntimes: [aws_lambda_1.Runtime.NODEJS_14_X],
62
+ description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`
63
+ });
64
+ }
65
+ /**
66
+ * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger
67
+ * our cleanup lamda function.
68
+ *
69
+ * @param props
70
+ * @private
71
+ */
57
72
  createRule(props) {
58
- const ruleName = `${this.resourceIdPrefix}-scheduled-rule`;
59
- // Schedule every tuesday at 03:30 AM GMT
60
- return new aws_events_1.Rule(this, ruleName, {
61
- ruleName,
62
- description: `Triggers a cleanup of outdated static assets of the ${props.project} app.`,
73
+ new aws_events_1.Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {
74
+ ruleName: `${this.resourceIdPrefix}-scheduler`,
75
+ description: `Triggers a cleanup of the outdated static assets at the ${props.staticAssetsBucket.bucketName} S3 bucket.`,
63
76
  enabled: true,
64
77
  schedule: aws_events_1.Schedule.cron({ weekDay: '3', hour: '3', minute: '30' }),
65
78
  targets: [new aws_events_targets_1.LambdaFunction(this.lambdaFunction)],
@@ -67,4 +80,4 @@ class NuxtAppAssetsCleanupStack extends aws_cdk_lib_1.Stack {
67
80
  }
68
81
  }
69
82
  exports.NuxtAppAssetsCleanupStack = NuxtAppAssetsCleanupStack;
70
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-app-assets-cleanup-stack.js","sourceRoot":"","sources":["nuxt-app-assets-cleanup-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {Duration, Stack} from \"aws-cdk-lib\";\nimport {AppStackProps} from \"./app-stack-props\";\nimport {Architecture, Code, LayerVersion, Runtime, Function} from \"aws-cdk-lib/aws-lambda\";\nimport {Rule, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {Construct} from \"constructs\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\n\nexport interface NuxtAppAssetsCleanupProps extends AppStackProps {\n    readonly staticAssetsBucket: IBucket;\n    readonly sentryDsn: string;\n}\n\n/**\n * Contains a scheduled lambda function, that deletes outdated static assets of the PWA from S3.\n */\nexport class NuxtAppAssetsCleanupStack extends Stack {\n\n    private readonly resourceIdPrefix: string;\n    private staticAssetsBucket: IBucket;\n    private readonly layer: LayerVersion;\n    private readonly lambdaFunction: Function;\n    private scheduledRule: Rule;\n\n    constructor(scope: Construct, id: string, props: NuxtAppAssetsCleanupProps) {\n        super(scope, id, props);\n\n        // Note, that this stack can't be tagged independently of the NuxtAppStack, as both are sharing the\n        // same scope (CDK App). This would overwrite previous set tags for the CDK App (last one wins).\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.staticAssetsBucket = props.staticAssetsBucket;\n        this.layer = this.createLayer();\n        this.lambdaFunction = this.createLambdaFunction(props);\n        this.scheduledRule = this.createRule(props);\n    }\n\n    private createLayer(): LayerVersion {\n        const layerName = `${this.resourceIdPrefix}-layer`;\n\n        return new LayerVersion(this, layerName, {\n            layerVersionName: layerName,\n            code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),\n            compatibleRuntimes: [Runtime.NODEJS_14_X],\n        });\n    }\n\n    private createLambdaFunction(props: NuxtAppAssetsCleanupProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-function`;\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            runtime: Runtime.NODEJS_14_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.layer],\n            handler: 'index.handler',\n            code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/app')),\n            timeout: Duration.minutes(1),\n            memorySize: 128,\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,\n                SENTRY_DSN: props.sentryDsn,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    private createRule(props: NuxtAppAssetsCleanupProps): Rule {\n        const ruleName = `${this.resourceIdPrefix}-scheduled-rule`;\n\n        // Schedule every tuesday at 03:30 AM GMT\n        return new Rule(this, ruleName, {\n            ruleName,\n            description: `Triggers a cleanup of outdated static assets of the ${props.project} app.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.lambdaFunction)],\n        });\n    }\n}"]}
83
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-app-assets-cleanup-stack.js","sourceRoot":"","sources":["nuxt-app-assets-cleanup-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {Duration, Stack} from \"aws-cdk-lib\";\nimport {AppStackProps} from \"./app-stack-props\";\nimport {Architecture, Code, LayerVersion, Runtime, Function} from \"aws-cdk-lib/aws-lambda\";\nimport {Rule, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {Construct} from \"constructs\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\n\n/**\n * Defines the props required for the {@see NuxtAppAssetsCleanupStack}.\n */\nexport interface NuxtAppAssetsCleanupProps extends AppStackProps {\n    /**\n     * The S3 bucket where the static assets of the Nuxt app are located.\n     */\n    readonly staticAssetsBucket: IBucket;\n}\n\n/**\n * Creates a scheduled lambda function that deletes outdated static assets of the Nuxt app from S3\n * to keep it nice without any outdated files that aren't used anymore.\n */\nexport class NuxtAppAssetsCleanupStack 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     * A reference to the created lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly lambdaFunction: Function;\n\n    constructor(scope: Construct, id: string, props: NuxtAppAssetsCleanupProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.lambdaFunction = this.createLambdaFunction(props);\n        this.createRule(props);\n    }\n\n    /**\n     * Creates the lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private createLambdaFunction(props: NuxtAppAssetsCleanupProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-function`;\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${props.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_14_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.createNodeModulesLayer()],\n            handler: 'index.handler',\n            code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/app')),\n            timeout: Duration.minutes(1),\n            memorySize: 128,\n            logRetention: RetentionDays.TWO_WEEKS,\n            environment: {\n                STATIC_ASSETS_BUCKET: props.staticAssetsBucket.bucketName,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        props.staticAssetsBucket.grantRead(result);\n        props.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    /**\n     * Creates a lambda layer for the cleanup function that holds the required node_modules.\n     *\n     * @private\n     */\n    private createNodeModulesLayer(): LayerVersion {\n        return new LayerVersion(this, `${this.resourceIdPrefix}-layer`, {\n            layerVersionName: `${this.resourceIdPrefix}-layer`,\n            code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),\n            compatibleRuntimes: [Runtime.NODEJS_14_X],\n            description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`\n        });\n    }\n\n    /**\n     * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger\n     * our cleanup lamda function.\n     *\n     * @param props\n     * @private\n     */\n    private createRule(props: NuxtAppAssetsCleanupProps): 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 ${props.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.lambdaFunction)],\n        });\n    }\n}"]}
@@ -8,61 +8,66 @@ import {RetentionDays} from "aws-cdk-lib/aws-logs";
8
8
  import {LambdaFunction} from "aws-cdk-lib/aws-events-targets";
9
9
  import * as path from "path";
10
10
 
11
+ /**
12
+ * Defines the props required for the {@see NuxtAppAssetsCleanupStack}.
13
+ */
11
14
  export interface NuxtAppAssetsCleanupProps extends AppStackProps {
15
+ /**
16
+ * The S3 bucket where the static assets of the Nuxt app are located.
17
+ */
12
18
  readonly staticAssetsBucket: IBucket;
13
- readonly sentryDsn: string;
14
19
  }
15
20
 
16
21
  /**
17
- * Contains a scheduled lambda function, that deletes outdated static assets of the PWA from S3.
22
+ * Creates a scheduled lambda function that deletes outdated static assets of the Nuxt app from S3
23
+ * to keep it nice without any outdated files that aren't used anymore.
18
24
  */
19
25
  export class NuxtAppAssetsCleanupStack extends Stack {
20
26
 
27
+ /**
28
+ * The identifier prefix of the resources created by the stack.
29
+ *
30
+ * @private
31
+ */
21
32
  private readonly resourceIdPrefix: string;
22
- private staticAssetsBucket: IBucket;
23
- private readonly layer: LayerVersion;
33
+
34
+ /**
35
+ * A reference to the created lambda function that cleanups the outdated static assets of the Nuxt app.
36
+ *
37
+ * @private
38
+ */
24
39
  private readonly lambdaFunction: Function;
25
- private scheduledRule: Rule;
26
40
 
27
41
  constructor(scope: Construct, id: string, props: NuxtAppAssetsCleanupProps) {
28
42
  super(scope, id, props);
29
43
 
30
- // Note, that this stack can't be tagged independently of the NuxtAppStack, as both are sharing the
31
- // same scope (CDK App). This would overwrite previous set tags for the CDK App (last one wins).
32
-
33
44
  this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;
34
- this.staticAssetsBucket = props.staticAssetsBucket;
35
- this.layer = this.createLayer();
36
45
  this.lambdaFunction = this.createLambdaFunction(props);
37
- this.scheduledRule = this.createRule(props);
38
- }
39
-
40
- private createLayer(): LayerVersion {
41
- const layerName = `${this.resourceIdPrefix}-layer`;
42
-
43
- return new LayerVersion(this, layerName, {
44
- layerVersionName: layerName,
45
- code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),
46
- compatibleRuntimes: [Runtime.NODEJS_14_X],
47
- });
46
+ this.createRule(props);
48
47
  }
49
48
 
49
+ /**
50
+ * Creates the lambda function that cleanups the outdated static assets of the Nuxt app.
51
+ *
52
+ * @param props
53
+ * @private
54
+ */
50
55
  private createLambdaFunction(props: NuxtAppAssetsCleanupProps): Function {
51
56
  const functionName: string = `${this.resourceIdPrefix}-function`;
52
57
 
53
58
  const result: Function = new Function(this, functionName, {
54
59
  functionName: functionName,
60
+ description: `Auto-deletes the outdated static assets in the ${props.staticAssetsBucket.bucketName} S3 bucket.`,
55
61
  runtime: Runtime.NODEJS_14_X,
56
62
  architecture: Architecture.ARM_64,
57
- layers: [this.layer],
63
+ layers: [this.createNodeModulesLayer()],
58
64
  handler: 'index.handler',
59
65
  code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/app')),
60
66
  timeout: Duration.minutes(1),
61
67
  memorySize: 128,
62
68
  logRetention: RetentionDays.TWO_WEEKS,
63
69
  environment: {
64
- STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,
65
- SENTRY_DSN: props.sentryDsn,
70
+ STATIC_ASSETS_BUCKET: props.staticAssetsBucket.bucketName,
66
71
  ENVIRONMENT: props.environment,
67
72
  AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',
68
73
  NODE_OPTIONS: '--enable-source-maps',
@@ -70,19 +75,37 @@ export class NuxtAppAssetsCleanupStack extends Stack {
70
75
  });
71
76
 
72
77
  // grant function access to S3 bucket
73
- this.staticAssetsBucket.grantRead(result);
74
- this.staticAssetsBucket.grantDelete(result);
78
+ props.staticAssetsBucket.grantRead(result);
79
+ props.staticAssetsBucket.grantDelete(result);
75
80
 
76
81
  return result;
77
82
  }
78
83
 
79
- private createRule(props: NuxtAppAssetsCleanupProps): Rule {
80
- const ruleName = `${this.resourceIdPrefix}-scheduled-rule`;
84
+ /**
85
+ * Creates a lambda layer for the cleanup function that holds the required node_modules.
86
+ *
87
+ * @private
88
+ */
89
+ private createNodeModulesLayer(): LayerVersion {
90
+ return new LayerVersion(this, `${this.resourceIdPrefix}-layer`, {
91
+ layerVersionName: `${this.resourceIdPrefix}-layer`,
92
+ code: Code.fromAsset(path.join(__dirname, '../functions/assets_cleanup/build/layer')),
93
+ compatibleRuntimes: [Runtime.NODEJS_14_X],
94
+ description: `Provides the node_modules required for the ${this.resourceIdPrefix} lambda function.`
95
+ });
96
+ }
81
97
 
82
- // Schedule every tuesday at 03:30 AM GMT
83
- return new Rule(this, ruleName, {
84
- ruleName,
85
- description: `Triggers a cleanup of outdated static assets of the ${props.project} app.`,
98
+ /**
99
+ * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger
100
+ * our cleanup lamda function.
101
+ *
102
+ * @param props
103
+ * @private
104
+ */
105
+ private createRule(props: NuxtAppAssetsCleanupProps): void {
106
+ new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {
107
+ ruleName: `${this.resourceIdPrefix}-scheduler`,
108
+ description: `Triggers a cleanup of the outdated static assets at the ${props.staticAssetsBucket.bucketName} S3 bucket.`,
86
109
  enabled: true,
87
110
  schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),
88
111
  targets: [new LambdaFunction(this.lambdaFunction)],