cdk-nuxt 0.10.0 → 0.10.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/index.d.ts +2 -0
- package/index.js +8 -0
- package/lib/functions/assets-cleanup/index.d.ts +1 -0
- package/lib/functions/assets-cleanup/index.js +82 -0
- package/lib/stack/nuxt-app-stack-props.d.ts +38 -0
- package/lib/stack/nuxt-app-stack-props.js +3 -0
- package/lib/stack/nuxt-app-stack-props.ts +0 -1
- package/lib/stack/nuxt-app-static-assets.d.ts +32 -0
- package/lib/stack/nuxt-app-static-assets.js +166 -0
- package/lib/stack/server/nuxt-server-app-stack.d.ts +201 -0
- package/lib/stack/server/nuxt-server-app-stack.js +434 -0
- package/lib/stack/static/nuxt-static-app-stack.d.ts +90 -0
- package/lib/stack/static/nuxt-static-app-stack.js +182 -0
- package/package.json +12 -11
package/index.d.ts
ADDED
package/index.js
ADDED
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.NuxtStaticAppStack = exports.NuxtServerAppStack = void 0;
|
|
4
|
+
var nuxt_server_app_stack_1 = require("./lib/stack/server/nuxt-server-app-stack");
|
|
5
|
+
Object.defineProperty(exports, "NuxtServerAppStack", { enumerable: true, get: function () { return nuxt_server_app_stack_1.NuxtServerAppStack; } });
|
|
6
|
+
var nuxt_static_app_stack_1 = require("./lib/stack/static/nuxt-static-app-stack");
|
|
7
|
+
Object.defineProperty(exports, "NuxtStaticAppStack", { enumerable: true, get: function () { return nuxt_static_app_stack_1.NuxtStaticAppStack; } });
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxrRkFBb0c7QUFBNUYsMkhBQUEsa0JBQWtCLE9BQUE7QUFDMUIsa0ZBQW9HO0FBQTVGLDJIQUFBLGtCQUFrQixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHtOdXh0U2VydmVyQXBwU3RhY2ssIE51eHRTZXJ2ZXJBcHBTdGFja1Byb3BzfSBmcm9tIFwiLi9saWIvc3RhY2svc2VydmVyL251eHQtc2VydmVyLWFwcC1zdGFja1wiXG5leHBvcnQge051eHRTdGF0aWNBcHBTdGFjaywgTnV4dFN0YXRpY0FwcFN0YWNrUHJvcHN9IGZyb20gXCIuL2xpYi9zdGFjay9zdGF0aWMvbnV4dC1zdGF0aWMtYXBwLXN0YWNrXCJcbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const client_s3_1 = require("@aws-sdk/client-s3");
|
|
4
|
+
const ONE_WEEK_IN_MILLISECONDS = 1000 * 60 * 60 * 24 * 7;
|
|
5
|
+
exports.handler = async (event, context) => {
|
|
6
|
+
var _a;
|
|
7
|
+
console.log('Starting cleanup of static assets older than 1 week...');
|
|
8
|
+
try {
|
|
9
|
+
const client = new client_s3_1.S3Client({ region: process.env.AWS_REGION });
|
|
10
|
+
const bucketName = process.env.STATIC_ASSETS_BUCKET;
|
|
11
|
+
if (!bucketName) {
|
|
12
|
+
throw new Error("Static asset's bucket name not specified in environment!");
|
|
13
|
+
}
|
|
14
|
+
const deleteOlderThan = new Date(Date.now() - ONE_WEEK_IN_MILLISECONDS);
|
|
15
|
+
// As we don't have hundreds of deployments per week, there's no need for pagination
|
|
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;
|
|
25
|
+
}
|
|
26
|
+
// We sort the folders by their creation data and remove the latest one as this is the currently active one used in production
|
|
27
|
+
const legacyFolderNames = folderNames.sort((a, b) => {
|
|
28
|
+
// @ts-ignore
|
|
29
|
+
const creationDateA = new Date(a);
|
|
30
|
+
// @ts-ignore
|
|
31
|
+
const creationDateB = new Date(b);
|
|
32
|
+
return creationDateA.getTime() < creationDateB.getTime() ? -1 : 1;
|
|
33
|
+
});
|
|
34
|
+
const activeFolderName = legacyFolderNames.pop();
|
|
35
|
+
console.log(`Detected assets folder "${activeFolderName}" as current production folder...`);
|
|
36
|
+
// We want to get all outdated folders of our legacy (not productively used) folders for deletion
|
|
37
|
+
const outdatedFolderNames = legacyFolderNames.filter(folderName => {
|
|
38
|
+
// @ts-ignore
|
|
39
|
+
const creationDate = new Date(folderName);
|
|
40
|
+
return creationDate.getTime() < deleteOlderThan.getTime();
|
|
41
|
+
});
|
|
42
|
+
if (!outdatedFolderNames || outdatedFolderNames.length === 0) {
|
|
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({
|
|
52
|
+
Bucket: bucketName,
|
|
53
|
+
Prefix: folderName,
|
|
54
|
+
}))
|
|
55
|
+
.then(outdatedAssets => {
|
|
56
|
+
var _a;
|
|
57
|
+
return client.send(new client_s3_1.DeleteObjectsCommand({
|
|
58
|
+
Bucket: bucketName,
|
|
59
|
+
Delete: {
|
|
60
|
+
Objects: (_a = outdatedAssets.Contents) === null || _a === void 0 ? void 0 : _a.map(outdatedAsset => {
|
|
61
|
+
return { Key: outdatedAsset.Key };
|
|
62
|
+
}),
|
|
63
|
+
},
|
|
64
|
+
}));
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
const results = await Promise.all(pendingPromises);
|
|
68
|
+
results.forEach(result => {
|
|
69
|
+
if (result.Errors && result.Errors.length) {
|
|
70
|
+
const errorMsg = 'Failed to delete outdated static assets.';
|
|
71
|
+
console.error(errorMsg, result.Errors);
|
|
72
|
+
throw new Error(errorMsg);
|
|
73
|
+
}
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
console.log('Cleanup of old static assets finished.');
|
|
77
|
+
}
|
|
78
|
+
catch (error) {
|
|
79
|
+
console.error('### unexpected runtime error ###', error);
|
|
80
|
+
}
|
|
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};"]}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { StackProps } from "aws-cdk-lib";
|
|
2
|
+
/**
|
|
3
|
+
* Defines the common props required to deploy Nuxt apps on AWS.
|
|
4
|
+
*/
|
|
5
|
+
export interface NuxtAppStackProps extends StackProps {
|
|
6
|
+
/**
|
|
7
|
+
* A string identifier for the project the Nuxt app is part of.
|
|
8
|
+
* A project might have multiple different services.
|
|
9
|
+
*/
|
|
10
|
+
readonly project: string;
|
|
11
|
+
/**
|
|
12
|
+
* A string identifier for the project's service the Nuxt app is created for.
|
|
13
|
+
* This can be seen as the name of the Nuxt app.
|
|
14
|
+
*/
|
|
15
|
+
readonly service: string;
|
|
16
|
+
/**
|
|
17
|
+
* A string to identify the environment of the Nuxt app. This enables us
|
|
18
|
+
* to deploy multiple different environments of the same Nuxt app, e.g., production and development.
|
|
19
|
+
*/
|
|
20
|
+
readonly environment: string;
|
|
21
|
+
/**
|
|
22
|
+
* The domain (without the protocol) at which the Nuxt app shall be publicly available.
|
|
23
|
+
* A DNS record will be automatically created in Route53 for the domain.
|
|
24
|
+
* This also supports subdomains.
|
|
25
|
+
* Examples: "example.com", "sub.example.com"
|
|
26
|
+
*/
|
|
27
|
+
readonly domain: string;
|
|
28
|
+
/**
|
|
29
|
+
* The id of the hosted zone to create a DNS record for the specified domain.
|
|
30
|
+
*/
|
|
31
|
+
readonly hostedZoneId: string;
|
|
32
|
+
/**
|
|
33
|
+
* The ARN of the certificate to use on CloudFront for the Nuxt app to make it accessible via HTTPS.
|
|
34
|
+
* The certificate must be issued for the specified domain in us-east-1 (global) regardless of the
|
|
35
|
+
* region specified via 'env.region' as CloudFront only works globally.
|
|
36
|
+
*/
|
|
37
|
+
readonly globalTlsCertificateArn: string;
|
|
38
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnV4dC1hcHAtc3RhY2stcHJvcHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJudXh0LWFwcC1zdGFjay1wcm9wcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtTdGFja1Byb3BzfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcblxuLyoqXG4gKiBEZWZpbmVzIHRoZSBjb21tb24gcHJvcHMgcmVxdWlyZWQgdG8gZGVwbG95IE51eHQgYXBwcyBvbiBBV1MuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTnV4dEFwcFN0YWNrUHJvcHMgZXh0ZW5kcyBTdGFja1Byb3BzIHtcblxuICAgIC8qKlxuICAgICAqIEEgc3RyaW5nIGlkZW50aWZpZXIgZm9yIHRoZSBwcm9qZWN0IHRoZSBOdXh0IGFwcCBpcyBwYXJ0IG9mLlxuICAgICAqIEEgcHJvamVjdCBtaWdodCBoYXZlIG11bHRpcGxlIGRpZmZlcmVudCBzZXJ2aWNlcy5cbiAgICAgKi9cbiAgICByZWFkb25seSBwcm9qZWN0OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBBIHN0cmluZyBpZGVudGlmaWVyIGZvciB0aGUgcHJvamVjdCdzIHNlcnZpY2UgdGhlIE51eHQgYXBwIGlzIGNyZWF0ZWQgZm9yLlxuICAgICAqIFRoaXMgY2FuIGJlIHNlZW4gYXMgdGhlIG5hbWUgb2YgdGhlIE51eHQgYXBwLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHNlcnZpY2U6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqIEEgc3RyaW5nIHRvIGlkZW50aWZ5IHRoZSBlbnZpcm9ubWVudCBvZiB0aGUgTnV4dCBhcHAuIFRoaXMgZW5hYmxlcyB1c1xuICAgICAqIHRvIGRlcGxveSBtdWx0aXBsZSBkaWZmZXJlbnQgZW52aXJvbm1lbnRzIG9mIHRoZSBzYW1lIE51eHQgYXBwLCBlLmcuLCBwcm9kdWN0aW9uIGFuZCBkZXZlbG9wbWVudC5cbiAgICAgKi9cbiAgICByZWFkb25seSBlbnZpcm9ubWVudDogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogVGhlIGRvbWFpbiAod2l0aG91dCB0aGUgcHJvdG9jb2wpIGF0IHdoaWNoIHRoZSBOdXh0IGFwcCBzaGFsbCBiZSBwdWJsaWNseSBhdmFpbGFibGUuXG4gICAgICogQSBETlMgcmVjb3JkIHdpbGwgYmUgYXV0b21hdGljYWxseSBjcmVhdGVkIGluIFJvdXRlNTMgZm9yIHRoZSBkb21haW4uXG4gICAgICogVGhpcyBhbHNvIHN1cHBvcnRzIHN1YmRvbWFpbnMuXG4gICAgICogRXhhbXBsZXM6IFwiZXhhbXBsZS5jb21cIiwgXCJzdWIuZXhhbXBsZS5jb21cIlxuICAgICAqL1xuICAgIHJlYWRvbmx5IGRvbWFpbjogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogVGhlIGlkIG9mIHRoZSBob3N0ZWQgem9uZSB0byBjcmVhdGUgYSBETlMgcmVjb3JkIGZvciB0aGUgc3BlY2lmaWVkIGRvbWFpbi5cbiAgICAgKi9cbiAgICByZWFkb25seSBob3N0ZWRab25lSWQ6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqIFRoZSBBUk4gb2YgdGhlIGNlcnRpZmljYXRlIHRvIHVzZSBvbiBDbG91ZEZyb250IGZvciB0aGUgTnV4dCBhcHAgdG8gbWFrZSBpdCBhY2Nlc3NpYmxlIHZpYSBIVFRQUy5cbiAgICAgKiBUaGUgY2VydGlmaWNhdGUgbXVzdCBiZSBpc3N1ZWQgZm9yIHRoZSBzcGVjaWZpZWQgZG9tYWluIGluIHVzLWVhc3QtMSAoZ2xvYmFsKSByZWdhcmRsZXNzIG9mIHRoZVxuICAgICAqIHJlZ2lvbiBzcGVjaWZpZWQgdmlhICdlbnYucmVnaW9uJyBhcyBDbG91ZEZyb250IG9ubHkgd29ya3MgZ2xvYmFsbHkuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZ2xvYmFsVGxzQ2VydGlmaWNhdGVBcm46IHN0cmluZztcblxufSJdfQ==
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { CacheControl } from "aws-cdk-lib/aws-s3-deployment";
|
|
2
|
+
export interface StaticAssetConfig {
|
|
3
|
+
/**
|
|
4
|
+
* The file pattern for the incoming requests that should be forwarded to the target path in the static assets S3 bucket
|
|
5
|
+
* with the appropriate cache and content settings defined in the same object.
|
|
6
|
+
*/
|
|
7
|
+
pattern: string;
|
|
8
|
+
/**
|
|
9
|
+
* The local directory to upload the files from.
|
|
10
|
+
*/
|
|
11
|
+
source: string;
|
|
12
|
+
/**
|
|
13
|
+
* The remote path at which to make the uploaded files from source accessible.
|
|
14
|
+
*/
|
|
15
|
+
target: string;
|
|
16
|
+
/**
|
|
17
|
+
* The content type to set for the files in the source folder when uploading them to the target.
|
|
18
|
+
*/
|
|
19
|
+
contentType: string;
|
|
20
|
+
/**
|
|
21
|
+
* The cache settings to use for the uploaded source files when accessing them on the target path with the specified pattern.
|
|
22
|
+
*/
|
|
23
|
+
cacheControl?: CacheControl[];
|
|
24
|
+
/**
|
|
25
|
+
* Whether to invalidate the files matching the config's pattern in the distribution's edge caches after the files are uploaded to the destination bucket.
|
|
26
|
+
*/
|
|
27
|
+
invalidateOnChange?: boolean;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Retrieves the static assets of the Nuxt app that shall be publicly available.
|
|
31
|
+
*/
|
|
32
|
+
export declare const getNuxtAppStaticAssetConfigs: () => StaticAssetConfig[];
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getNuxtAppStaticAssetConfigs = void 0;
|
|
4
|
+
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
/**
|
|
7
|
+
* Retrieves the custom static assets of the Nuxt app that shall be publicly available.
|
|
8
|
+
*/
|
|
9
|
+
const getNuxtAppCustomAssetConfigs = () => {
|
|
10
|
+
// The custom assets of server Nuxt apps are located in the src 'static' folder
|
|
11
|
+
// The custom assets of static Nuxt apps are located in the build files 'dist' folder
|
|
12
|
+
const customAssetsSourcePath = `./.output/public`;
|
|
13
|
+
const customAssetsTargetPath = '/';
|
|
14
|
+
return [
|
|
15
|
+
{
|
|
16
|
+
pattern: '*.txt',
|
|
17
|
+
source: customAssetsSourcePath,
|
|
18
|
+
target: customAssetsTargetPath,
|
|
19
|
+
contentType: 'text/plain; charset=UTF-8',
|
|
20
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.setPublic(), aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.days(1))],
|
|
21
|
+
},
|
|
22
|
+
{
|
|
23
|
+
pattern: '*.xml',
|
|
24
|
+
source: customAssetsSourcePath,
|
|
25
|
+
target: customAssetsTargetPath,
|
|
26
|
+
contentType: 'text/xml; charset=UTF-8',
|
|
27
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.setPublic(), aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.days(1))],
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
pattern: '.well-known/*',
|
|
31
|
+
source: customAssetsSourcePath,
|
|
32
|
+
target: customAssetsTargetPath,
|
|
33
|
+
contentType: 'application/json; charset=UTF-8',
|
|
34
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.setPublic(), aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.days(1))],
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
pattern: '*.ico',
|
|
38
|
+
source: customAssetsSourcePath,
|
|
39
|
+
target: customAssetsTargetPath,
|
|
40
|
+
contentType: 'image/x-icon',
|
|
41
|
+
},
|
|
42
|
+
{
|
|
43
|
+
pattern: '*.png',
|
|
44
|
+
source: customAssetsSourcePath,
|
|
45
|
+
target: customAssetsTargetPath,
|
|
46
|
+
contentType: 'image/png',
|
|
47
|
+
},
|
|
48
|
+
{
|
|
49
|
+
pattern: '*.jpg',
|
|
50
|
+
source: customAssetsSourcePath,
|
|
51
|
+
target: customAssetsTargetPath,
|
|
52
|
+
contentType: 'image/jpg',
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
pattern: '*.svg',
|
|
56
|
+
source: customAssetsSourcePath,
|
|
57
|
+
target: customAssetsTargetPath,
|
|
58
|
+
contentType: 'image/svg+xml',
|
|
59
|
+
},
|
|
60
|
+
{
|
|
61
|
+
pattern: '*.js',
|
|
62
|
+
source: customAssetsSourcePath,
|
|
63
|
+
target: customAssetsTargetPath,
|
|
64
|
+
contentType: 'application/javascript; charset=UTF-8',
|
|
65
|
+
// The js files in the custom static directory are usually not versioned
|
|
66
|
+
// whereby we want to prevent any caching issues when updating them -> cache for only 2 days
|
|
67
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.setPublic(), aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.days(2))],
|
|
68
|
+
}
|
|
69
|
+
];
|
|
70
|
+
};
|
|
71
|
+
/**
|
|
72
|
+
* Retrieves the static assets of the Nuxt app that are generated by the build of the app.
|
|
73
|
+
*/
|
|
74
|
+
const getNuxtAppBuildAssetConfigs = () => {
|
|
75
|
+
// The build assets required for CSR that are generated by 'nuxt build'
|
|
76
|
+
const buildAssetsSourcePath = './.output/public/_nuxt';
|
|
77
|
+
const buildAssetsTargetPath = ('/_nuxt/'); // Must match 'build.publicPath' in nuxt.config.js
|
|
78
|
+
return [
|
|
79
|
+
// Build Assets
|
|
80
|
+
{
|
|
81
|
+
pattern: '*.js',
|
|
82
|
+
target: buildAssetsTargetPath,
|
|
83
|
+
source: buildAssetsSourcePath,
|
|
84
|
+
contentType: 'application/javascript; charset=UTF-8',
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
pattern: '*.js.map',
|
|
88
|
+
target: buildAssetsTargetPath,
|
|
89
|
+
source: buildAssetsSourcePath,
|
|
90
|
+
contentType: 'application/json; charset=UTF-8',
|
|
91
|
+
},
|
|
92
|
+
{
|
|
93
|
+
pattern: '*.css',
|
|
94
|
+
target: buildAssetsTargetPath,
|
|
95
|
+
source: buildAssetsSourcePath,
|
|
96
|
+
contentType: 'text/css; charset=UTF-8',
|
|
97
|
+
},
|
|
98
|
+
{
|
|
99
|
+
pattern: '*.html',
|
|
100
|
+
target: buildAssetsTargetPath,
|
|
101
|
+
source: buildAssetsSourcePath,
|
|
102
|
+
contentType: 'text/html; charset=UTF-8',
|
|
103
|
+
// We do not want to cache html files on the browser as we want to be able to publish new changes quickly
|
|
104
|
+
// However, we do cache the html files on CloudFront for a performance boost and invalidate the cache on every deployment
|
|
105
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.setPublic(), aws_s3_deployment_1.CacheControl.sMaxAge(aws_cdk_lib_1.Duration.days(7))],
|
|
106
|
+
invalidateOnChange: true
|
|
107
|
+
},
|
|
108
|
+
// Manifest created by PWA module
|
|
109
|
+
{
|
|
110
|
+
pattern: 'manifest.*.json',
|
|
111
|
+
target: buildAssetsTargetPath,
|
|
112
|
+
source: buildAssetsSourcePath,
|
|
113
|
+
contentType: 'application/json; charset=UTF-8'
|
|
114
|
+
},
|
|
115
|
+
{
|
|
116
|
+
pattern: '*.png',
|
|
117
|
+
target: buildAssetsTargetPath,
|
|
118
|
+
source: buildAssetsSourcePath,
|
|
119
|
+
contentType: 'image/png',
|
|
120
|
+
},
|
|
121
|
+
{
|
|
122
|
+
pattern: '*.jpg',
|
|
123
|
+
target: buildAssetsTargetPath,
|
|
124
|
+
source: buildAssetsSourcePath,
|
|
125
|
+
contentType: 'image/jpg',
|
|
126
|
+
},
|
|
127
|
+
{
|
|
128
|
+
pattern: '*.svg',
|
|
129
|
+
target: buildAssetsTargetPath,
|
|
130
|
+
source: buildAssetsSourcePath,
|
|
131
|
+
contentType: 'image/svg+xml',
|
|
132
|
+
},
|
|
133
|
+
{
|
|
134
|
+
pattern: '*.eot',
|
|
135
|
+
target: buildAssetsTargetPath,
|
|
136
|
+
source: buildAssetsSourcePath,
|
|
137
|
+
contentType: 'application/vnd.ms-fontobject',
|
|
138
|
+
},
|
|
139
|
+
{
|
|
140
|
+
pattern: '*.ttf',
|
|
141
|
+
target: buildAssetsTargetPath,
|
|
142
|
+
source: buildAssetsSourcePath,
|
|
143
|
+
contentType: 'application/font-sfnt',
|
|
144
|
+
},
|
|
145
|
+
{
|
|
146
|
+
pattern: '*.woff',
|
|
147
|
+
target: buildAssetsTargetPath,
|
|
148
|
+
source: buildAssetsSourcePath,
|
|
149
|
+
contentType: 'font/woff',
|
|
150
|
+
},
|
|
151
|
+
{
|
|
152
|
+
pattern: '*.woff2',
|
|
153
|
+
target: buildAssetsTargetPath,
|
|
154
|
+
source: buildAssetsSourcePath,
|
|
155
|
+
contentType: 'font/woff2',
|
|
156
|
+
}
|
|
157
|
+
];
|
|
158
|
+
};
|
|
159
|
+
/**
|
|
160
|
+
* Retrieves the static assets of the Nuxt app that shall be publicly available.
|
|
161
|
+
*/
|
|
162
|
+
const getNuxtAppStaticAssetConfigs = () => {
|
|
163
|
+
return [...getNuxtAppBuildAssetConfigs(), ...getNuxtAppCustomAssetConfigs()];
|
|
164
|
+
};
|
|
165
|
+
exports.getNuxtAppStaticAssetConfigs = getNuxtAppStaticAssetConfigs;
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-app-static-assets.js","sourceRoot":"","sources":["nuxt-app-static-assets.ts"],"names":[],"mappings":"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","sourcesContent":["import {CacheControl} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {Duration} from \"aws-cdk-lib\";\n\nexport interface StaticAssetConfig {\n    /**\n     * The file pattern for the incoming requests that should be forwarded to the target path in the static assets S3 bucket\n     * with the appropriate cache and content settings defined in the same object.\n     */\n    pattern: string,\n\n    /**\n     * The local directory to upload the files from.\n     */\n    source: string,\n\n    /**\n     * The remote path at which to make the uploaded files from source accessible.\n     */\n    target: string,\n\n    /**\n     * The content type to set for the files in the source folder when uploading them to the target.\n     */\n    contentType: string,\n\n    /**\n     * The cache settings to use for the uploaded source files when accessing them on the target path with the specified pattern.\n     */\n    cacheControl?: CacheControl[],\n\n    /**\n     * Whether to invalidate the files matching the config's pattern in the distribution's edge caches after the files are uploaded to the destination bucket.\n     */\n    invalidateOnChange?: boolean;\n}\n\n/**\n * Retrieves the custom static assets of the Nuxt app that shall be publicly available.\n */\nconst getNuxtAppCustomAssetConfigs = (): StaticAssetConfig[] => {\n\n    // The custom assets of server Nuxt apps are located in the src 'static' folder\n    // The custom assets of static Nuxt apps are located in the build files 'dist' folder\n    const customAssetsSourcePath = `./.output/public`;\n    const customAssetsTargetPath = '/';\n\n    return [\n        {\n            pattern: '*.txt', // E.g., robots.txt, ads.txt, ...\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'text/plain; charset=UTF-8',\n            cacheControl: [CacheControl.setPublic(), CacheControl.maxAge(Duration.days(1))],\n        },\n        {\n            pattern: '*.xml', // E.g., sitemap.xml ...\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'text/xml; charset=UTF-8',\n            cacheControl: [CacheControl.setPublic(), CacheControl.maxAge(Duration.days(1))],\n        },\n        {\n            pattern: '.well-known/*', // Files for native app links\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'application/json; charset=UTF-8',\n            cacheControl: [CacheControl.setPublic(), CacheControl.maxAge(Duration.days(1))],\n        },\n        {\n            pattern: '*.ico', // Favicon\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'image/x-icon',\n        },\n        {\n            pattern: '*.png',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'image/png',\n        },\n        {\n            pattern: '*.jpg',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'image/jpg',\n        },\n        {\n            pattern: '*.svg',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'image/svg+xml',\n        },\n        {\n            pattern: '*.js',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'application/javascript; charset=UTF-8',\n            // The js files in the custom static directory are usually not versioned\n            // whereby we want to prevent any caching issues when updating them -> cache for only 2 days\n            cacheControl: [CacheControl.setPublic(), CacheControl.maxAge(Duration.days(2))],\n        }\n    ];\n};\n\n/**\n * Retrieves the static assets of the Nuxt app that are generated by the build of the app.\n */\nconst getNuxtAppBuildAssetConfigs = (): StaticAssetConfig[] => {\n\n    // The build assets required for CSR that are generated by 'nuxt build'\n    const buildAssetsSourcePath = './.output/public/_nuxt';\n    const buildAssetsTargetPath = ('/_nuxt/'); // Must match 'build.publicPath' in nuxt.config.js\n\n    return [\n\n        // Build Assets\n        {\n            pattern: '*.js',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'application/javascript; charset=UTF-8',\n        },\n        {\n            pattern: '*.js.map',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'application/json; charset=UTF-8',\n        },\n        {\n            pattern: '*.css',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'text/css; charset=UTF-8',\n        },\n        {\n            pattern: '*.html',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'text/html; charset=UTF-8',\n            // We do not want to cache html files on the browser as we want to be able to publish new changes quickly\n            // However, we do cache the html files on CloudFront for a performance boost and invalidate the cache on every deployment\n            cacheControl: [CacheControl.setPublic(), CacheControl.sMaxAge(Duration.days(7))],\n            invalidateOnChange: true\n        },\n\n        // Manifest created by PWA module\n        {\n            pattern: 'manifest.*.json',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'application/json; charset=UTF-8'\n        },\n        {\n            pattern: '*.png',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'image/png',\n        },\n        {\n            pattern: '*.jpg',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'image/jpg',\n        },\n        {\n            pattern: '*.svg',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'image/svg+xml',\n        },\n        {\n            pattern: '*.eot',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'application/vnd.ms-fontobject',\n        },\n        {\n            pattern: '*.ttf',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'application/font-sfnt',\n        },\n        {\n            pattern: '*.woff',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'font/woff',\n        },\n        {\n            pattern: '*.woff2',\n            target: buildAssetsTargetPath,\n            source: buildAssetsSourcePath,\n            contentType: 'font/woff2',\n        }\n    ]\n};\n\n/**\n * Retrieves the static assets of the Nuxt app that shall be publicly available.\n */\nexport const getNuxtAppStaticAssetConfigs = (): StaticAssetConfig[] => {\n    return [...getNuxtAppBuildAssetConfigs(), ...getNuxtAppCustomAssetConfigs()];\n};"]}
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
import { Stack } from 'aws-cdk-lib';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { IBucket } from "aws-cdk-lib/aws-s3";
|
|
4
|
+
import { NuxtAppStackProps } from "../nuxt-app-stack-props";
|
|
5
|
+
/**
|
|
6
|
+
* Defines the props required for the {@see NuxtServerAppStack}.
|
|
7
|
+
*/
|
|
8
|
+
export interface NuxtServerAppStackProps extends NuxtAppStackProps {
|
|
9
|
+
/**
|
|
10
|
+
* The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain
|
|
11
|
+
* and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.
|
|
12
|
+
* The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.
|
|
13
|
+
*/
|
|
14
|
+
readonly regionalTlsCertificateArn: string;
|
|
15
|
+
/**
|
|
16
|
+
* The memory size to apply to the Nuxt app's Lambda.
|
|
17
|
+
* Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).
|
|
18
|
+
*/
|
|
19
|
+
readonly memorySize?: number;
|
|
20
|
+
/**
|
|
21
|
+
* Whether to enable AWS X-Ray for the Nuxt Lambda function.
|
|
22
|
+
*/
|
|
23
|
+
readonly enableTracing?: boolean;
|
|
24
|
+
/**
|
|
25
|
+
* Whether to enable a global Sitemap bucket which is permanently accessible through multiple deployments.
|
|
26
|
+
*/
|
|
27
|
+
readonly enableSitemap?: boolean;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.
|
|
31
|
+
*/
|
|
32
|
+
export declare class NuxtServerAppStack extends Stack {
|
|
33
|
+
/**
|
|
34
|
+
* The identifier prefix of the resources created by the stack.
|
|
35
|
+
*
|
|
36
|
+
* @private
|
|
37
|
+
*/
|
|
38
|
+
private readonly resourceIdPrefix;
|
|
39
|
+
/**
|
|
40
|
+
* The identifier for the current deployment that is used as S3 folder name
|
|
41
|
+
* to store the static assets of the Nuxt app.
|
|
42
|
+
*
|
|
43
|
+
* @private
|
|
44
|
+
*/
|
|
45
|
+
private readonly deploymentRevision;
|
|
46
|
+
/**
|
|
47
|
+
* The identity to use for accessing the deployment assets on S3.
|
|
48
|
+
*
|
|
49
|
+
* @private
|
|
50
|
+
*/
|
|
51
|
+
private readonly cdnAccessIdentity;
|
|
52
|
+
/**
|
|
53
|
+
* The S3 bucket where the deployment assets gets stored.
|
|
54
|
+
*/
|
|
55
|
+
staticAssetsBucket: IBucket;
|
|
56
|
+
/**
|
|
57
|
+
* The S3 bucket where the sitemap assets gets stored.
|
|
58
|
+
*/
|
|
59
|
+
sitemapBucket: IBucket | undefined;
|
|
60
|
+
/**
|
|
61
|
+
* The Lambda function to render the Nuxt app on the server side.
|
|
62
|
+
*
|
|
63
|
+
* @private
|
|
64
|
+
*/
|
|
65
|
+
private readonly appLambdaFunction;
|
|
66
|
+
/**
|
|
67
|
+
* The Lambda function that cleanups the outdated static assets of the Nuxt app.
|
|
68
|
+
*
|
|
69
|
+
* @private
|
|
70
|
+
*/
|
|
71
|
+
private readonly cleanupLambdaFunction;
|
|
72
|
+
/**
|
|
73
|
+
* The API gateway to make the Lambda function to render the Nuxt app publicly available.
|
|
74
|
+
*
|
|
75
|
+
* @private
|
|
76
|
+
*/
|
|
77
|
+
private apiGateway;
|
|
78
|
+
/**
|
|
79
|
+
* The configs for the static assets of the Nuxt app that shall be publicly available.
|
|
80
|
+
*
|
|
81
|
+
* @private
|
|
82
|
+
*/
|
|
83
|
+
private staticAssetConfigs;
|
|
84
|
+
/**
|
|
85
|
+
* The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)
|
|
86
|
+
* or the S3 assets folder (with caching).
|
|
87
|
+
*
|
|
88
|
+
* @private
|
|
89
|
+
*/
|
|
90
|
+
private readonly cdn;
|
|
91
|
+
constructor(scope: Construct, id: string, props: NuxtServerAppStackProps);
|
|
92
|
+
/**
|
|
93
|
+
* Creates the identity to access the S3 deployment asset files via the CloudFront distribution.
|
|
94
|
+
*
|
|
95
|
+
* @private
|
|
96
|
+
*/
|
|
97
|
+
private createCdnAccessIdentity;
|
|
98
|
+
/**
|
|
99
|
+
* Creates the bucket to store the static deployment asset files of the Nuxt app.
|
|
100
|
+
*
|
|
101
|
+
* @private
|
|
102
|
+
*/
|
|
103
|
+
private createStaticAssetsBucket;
|
|
104
|
+
/**
|
|
105
|
+
* Creates the bucket to store the sitemap assets of the Nuxt app.
|
|
106
|
+
*
|
|
107
|
+
* @private
|
|
108
|
+
*/
|
|
109
|
+
private createSitemapBucket;
|
|
110
|
+
/**
|
|
111
|
+
* Creates the Lambda function to render the Nuxt app.
|
|
112
|
+
*
|
|
113
|
+
* @private
|
|
114
|
+
*/
|
|
115
|
+
private createAppLambdaFunction;
|
|
116
|
+
/**
|
|
117
|
+
* Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.
|
|
118
|
+
*
|
|
119
|
+
* @param props
|
|
120
|
+
* @private
|
|
121
|
+
*/
|
|
122
|
+
private createCleanupLambdaFunction;
|
|
123
|
+
/**
|
|
124
|
+
* Creates the API gateway to make the Nuxt app render Lambda function publicly available.
|
|
125
|
+
*
|
|
126
|
+
* @private
|
|
127
|
+
*/
|
|
128
|
+
private createApiGateway;
|
|
129
|
+
/**
|
|
130
|
+
* Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)
|
|
131
|
+
* or the S3 assets folder (with caching).
|
|
132
|
+
*
|
|
133
|
+
* @param props
|
|
134
|
+
* @private
|
|
135
|
+
*/
|
|
136
|
+
private createCloudFrontDistribution;
|
|
137
|
+
/**
|
|
138
|
+
* Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).
|
|
139
|
+
* Additionally, this automatically redirects HTTP requests to HTTPS.
|
|
140
|
+
*
|
|
141
|
+
* @private
|
|
142
|
+
*/
|
|
143
|
+
private createNuxtAppRouteBehavior;
|
|
144
|
+
private setupCloudFrontRouting;
|
|
145
|
+
/**
|
|
146
|
+
* Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.
|
|
147
|
+
* Even though we don't want to cache SSR requests, we still have to create this cache policy in order to
|
|
148
|
+
* forward required cookies, query params and headers. This doesn't make any sense, because if nothing
|
|
149
|
+
* is cached, one would expect, that anything would/could be forwarded, but anyway...
|
|
150
|
+
*/
|
|
151
|
+
private createSsrCachePolicy;
|
|
152
|
+
/**
|
|
153
|
+
* Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets
|
|
154
|
+
* to the S3 bucket that holds these static assets.
|
|
155
|
+
*
|
|
156
|
+
* @private
|
|
157
|
+
*/
|
|
158
|
+
private createStaticAssetsRouteBehavior;
|
|
159
|
+
/**
|
|
160
|
+
* Creates a behavior for the CloudFront distribution to route matching incoming requests for the sitemap assets
|
|
161
|
+
* to the S3 bucket that holds these sitemap assets.
|
|
162
|
+
*
|
|
163
|
+
* @private
|
|
164
|
+
*/
|
|
165
|
+
private createSitemapRouteBehavior;
|
|
166
|
+
/**
|
|
167
|
+
* Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.
|
|
168
|
+
* In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.
|
|
169
|
+
* The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up
|
|
170
|
+
* after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.
|
|
171
|
+
*/
|
|
172
|
+
private configureDeployments;
|
|
173
|
+
/**
|
|
174
|
+
* Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.
|
|
175
|
+
*
|
|
176
|
+
* @param props
|
|
177
|
+
* @private
|
|
178
|
+
*/
|
|
179
|
+
private findHostedZone;
|
|
180
|
+
/**
|
|
181
|
+
* Creates the DNS records to access the Nuxt app on the internet via the custom domain.
|
|
182
|
+
*
|
|
183
|
+
* @param props
|
|
184
|
+
* @private
|
|
185
|
+
*/
|
|
186
|
+
private createDnsRecords;
|
|
187
|
+
/**
|
|
188
|
+
* Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm
|
|
189
|
+
* and speed up initial SSR requests.
|
|
190
|
+
*
|
|
191
|
+
* @private
|
|
192
|
+
*/
|
|
193
|
+
private createAppPingRule;
|
|
194
|
+
/**
|
|
195
|
+
* Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger
|
|
196
|
+
* our cleanup Lambda function.
|
|
197
|
+
*
|
|
198
|
+
* @private
|
|
199
|
+
*/
|
|
200
|
+
private createCleanupTriggerRule;
|
|
201
|
+
}
|