cdk-nuxt 0.6.2 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/functions/cloudfront/redirect-to-index.js +37 -0
- package/lib/stack/nuxt-app-static-assets.js +2 -2
- package/lib/stack/nuxt-app-static-assets.ts +1 -1
- package/lib/stack/server/nuxt-server-app-stack.js +2 -2
- package/lib/stack/server/nuxt-server-app-stack.ts +1 -1
- package/lib/stack/static/nuxt-static-app-stack.js +25 -7
- package/lib/stack/static/nuxt-static-app-stack.ts +29 -7
- package/package.json +1 -1
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CloudFront function to redirect incoming requests to their corresponding S3 file.
|
|
3
|
+
*/
|
|
4
|
+
function handler(event) {
|
|
5
|
+
// Const and var are not supported
|
|
6
|
+
var request = event.request;
|
|
7
|
+
var uri = request.uri;
|
|
8
|
+
|
|
9
|
+
// The main entry point
|
|
10
|
+
if (uri === '/') {
|
|
11
|
+
request.uri = '/index.html';
|
|
12
|
+
return request;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
// The path suffixes that shall redirect to the main entry point
|
|
16
|
+
var indexPathAliases = ['/', '/index.html']
|
|
17
|
+
for (var i = 0; i < indexPathAliases.length; i++) {
|
|
18
|
+
var alias = indexPathAliases[i];
|
|
19
|
+
if (uri.endsWith(alias)) {
|
|
20
|
+
var host = request.headers.host.value;
|
|
21
|
+
|
|
22
|
+
return {
|
|
23
|
+
statusCode: 301,
|
|
24
|
+
statusDescription: 'Moved Permanently',
|
|
25
|
+
headers:
|
|
26
|
+
{ "location": { "value": `https://${host}${uri.slice(0, -alias.length)}` } }
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// When the URI is missing a file extension, we assume the main entry file (index.html) for the current path is requested
|
|
32
|
+
if (!uri.includes('.')) {
|
|
33
|
+
request.uri += '/index.html';
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
return request;
|
|
37
|
+
}
|
|
@@ -16,7 +16,7 @@ const getNuxtAppCustomAssetConfigs = (nuxtConfig) => {
|
|
|
16
16
|
const commonCustomAssets = [
|
|
17
17
|
// Custom Static Assets
|
|
18
18
|
{
|
|
19
|
-
pattern: '
|
|
19
|
+
pattern: '*.txt',
|
|
20
20
|
source: customAssetsSourcePath,
|
|
21
21
|
target: customAssetsTargetPath,
|
|
22
22
|
contentType: 'text/plain; charset=UTF-8',
|
|
@@ -182,4 +182,4 @@ const getNuxtAppStaticAssetConfigs = (nuxtConfig) => {
|
|
|
182
182
|
return [...getNuxtAppBuildAssetConfigs(nuxtConfig), ...getNuxtAppCustomAssetConfigs(nuxtConfig)];
|
|
183
183
|
};
|
|
184
184
|
exports.getNuxtAppStaticAssetConfigs = getNuxtAppStaticAssetConfigs;
|
|
185
|
-
//# 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\";\nimport {NuxtConfig} from \"./nuxt-config\";\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 * These should match the files in the 'static' directory.\n */\nconst getNuxtAppCustomAssetConfigs = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    const isStaticApp = nuxtConfig.target === 'static';\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 = isStaticApp ? './dist' : `./${nuxtConfig.srcDir ? (nuxtConfig.srcDir + '/') : ''}static`;\n    const customAssetsTargetPath = '/';\n\n    const commonCustomAssets: StaticAssetConfig[] = [\n\n        // Custom Static Assets\n        {\n            pattern: 'robots.txt', // See https://developers.google.com/search/docs/advanced/robots/intro\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: '*sitemap.xml', // Support various sitemaps, e.g., sitemap.xml or google-news-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: 'sitemaps/*.xml', // Nested sitemap files for sitemap index\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\n    if (isStaticApp) {\n        return commonCustomAssets;\n    }\n\n    // Server Nuxt apps might maintain additional custom assets that are already included in the build assets for static Nuxt apps\n    return [...commonCustomAssets,\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 * These should match the files in '.nuxt/dist/client' for server apps or in 'dist' for static apps.\n */\nconst getNuxtAppBuildAssetConfigs = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    const isStaticApp = nuxtConfig.target === 'static';\n\n    // The build assets required for CSR that are generated by 'nuxt build'\n    const buildAssetsSourcePath = isStaticApp ? './dist' : './.nuxt/dist/client';\n    const buildAssetsTargetPath = isStaticApp ? '/' : (nuxtConfig.build?.publicPath ?? '/_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 = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    return [...getNuxtAppBuildAssetConfigs(nuxtConfig), ...getNuxtAppCustomAssetConfigs(nuxtConfig)];\n};"]}
|
|
185
|
+
//# 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\";\nimport {NuxtConfig} from \"./nuxt-config\";\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 * These should match the files in the 'static' directory.\n */\nconst getNuxtAppCustomAssetConfigs = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    const isStaticApp = nuxtConfig.target === 'static';\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 = isStaticApp ? './dist' : `./${nuxtConfig.srcDir ? (nuxtConfig.srcDir + '/') : ''}static`;\n    const customAssetsTargetPath = '/';\n\n    const commonCustomAssets: StaticAssetConfig[] = [\n\n        // Custom Static Assets\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: '*sitemap.xml', // Support various sitemaps, e.g., sitemap.xml or google-news-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: 'sitemaps/*.xml', // Nested sitemap files for sitemap index\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\n    if (isStaticApp) {\n        return commonCustomAssets;\n    }\n\n    // Server Nuxt apps might maintain additional custom assets that are already included in the build assets for static Nuxt apps\n    return [...commonCustomAssets,\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 * These should match the files in '.nuxt/dist/client' for server apps or in 'dist' for static apps.\n */\nconst getNuxtAppBuildAssetConfigs = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    const isStaticApp = nuxtConfig.target === 'static';\n\n    // The build assets required for CSR that are generated by 'nuxt build'\n    const buildAssetsSourcePath = isStaticApp ? './dist' : './.nuxt/dist/client';\n    const buildAssetsTargetPath = isStaticApp ? '/' : (nuxtConfig.build?.publicPath ?? '/_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 = (nuxtConfig: NuxtConfig): StaticAssetConfig[] => {\n    return [...getNuxtAppBuildAssetConfigs(nuxtConfig), ...getNuxtAppCustomAssetConfigs(nuxtConfig)];\n};"]}
|
|
@@ -51,7 +51,7 @@ const getNuxtAppCustomAssetConfigs = (nuxtConfig: NuxtConfig): StaticAssetConfig
|
|
|
51
51
|
|
|
52
52
|
// Custom Static Assets
|
|
53
53
|
{
|
|
54
|
-
pattern: '
|
|
54
|
+
pattern: '*.txt', // E.g., robots.txt, ads.txt, ...
|
|
55
55
|
source: customAssetsSourcePath,
|
|
56
56
|
target: customAssetsTargetPath,
|
|
57
57
|
contentType: 'text/plain; charset=UTF-8',
|
|
@@ -296,7 +296,7 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
296
296
|
cacheControl: (_a = asset.cacheControl) !== null && _a !== void 0 ? _a : defaultCacheConfig,
|
|
297
297
|
contentType: asset.contentType,
|
|
298
298
|
distribution: asset.invalidateOnChange ? this.cdn : undefined,
|
|
299
|
-
distributionPaths: asset.invalidateOnChange ?
|
|
299
|
+
distributionPaths: asset.invalidateOnChange ? [`/${asset.pattern}`] : undefined,
|
|
300
300
|
logRetention: aws_logs_1.RetentionDays.ONE_DAY,
|
|
301
301
|
memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory
|
|
302
302
|
});
|
|
@@ -386,4 +386,4 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
|
|
|
386
386
|
}
|
|
387
387
|
}
|
|
388
388
|
exports.NuxtServerAppStack = NuxtServerAppStack;
|
|
389
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-server-app-stack.js","sourceRoot":"","sources":["nuxt-server-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution,\n    ICachePolicy,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, LayerVersion, Runtime, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {HttpMethod} from \"aws-cdk-lib/aws-stepfunctions-tasks\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {HttpLambdaIntegration} from '@aws-cdk/aws-apigatewayv2-integrations-alpha';\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"@aws-cdk/aws-apigatewayv2-alpha\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../nuxt-app-static-assets\";\nimport * as fs from \"fs\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport {NuxtAppStackProps} from \"../nuxt-app-stack-props\";\nimport * as path from \"path\";\n\n/**\n * Defines the props required for the {@see NuxtServerAppStack}.\n */\nexport interface NuxtServerAppStackProps extends NuxtAppStackProps {\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The memory size to apply to the Nuxt app's Lambda.\n     * Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).\n     */\n    readonly memorySize?: number;\n\n    /**\n     * Whether to enable AWS X-Ray for the Nuxt Lambda function.\n     */\n    readonly enableTracing?: boolean;\n}\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used as S3 folder name\n     * to store the static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: Function;\n\n    /**\n     * The API gateway to make the Lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtServerAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = new Date().toISOString();\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            accessControl: BucketAccessControl.AUTHENTICATED_READ,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private createSsrLambdaLayer(): LayerVersion {\n        const layerName = `${this.resourceIdPrefix}-ssr-layer`;\n        return new LayerVersion(this, layerName, {\n            layerVersionName: layerName,\n            code: Code.fromAsset('.nuxt/cdk-deployment/layer'),\n            compatibleRuntimes: [Runtime.NODEJS_12_X],\n            description: `Provides the node_modules required for SSR of ${this.resourceIdPrefix}.`,\n        });\n    }\n\n    /**\n     * Creates the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-app-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_12_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.createSsrLambdaLayer()],\n            handler: 'index.handler',\n            code: Code.fromAsset('.nuxt/cdk-deployment/src', {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup-function`;\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_14_X,\n            architecture: Architecture.ARM_64,\n            layers: [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            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                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render Lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [HttpMethod.GET, HttpMethod.HEAD],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.createStaticAssetsRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(2),\n                readTimeout: Duration.seconds(10),\n                protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n            }),\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.createSsrCachePolicy(),\n        };\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     * Even though we don't want to cache SSR requests, we still have to create this cache policy in order to\n     * forward required cookies, query params and headers. This doesn't make any sense, because if nothing\n     * is cached, one would expect, that anything would/could be forwarded, but anyway...\n     */\n    private createSsrCachePolicy(): ICachePolicy {\n\n        // The headers to make accessible in the Nuxt app code.\n        // There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.\n        const headers = [\n            'User-Agent', // Required to distinguish between mobile and desktop template\n            'Authorization', // For authorization\n            'Host' // To access the domain name on SSR requests\n        ];\n\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Passes all required request data to the ${this.resourceIdPrefix} origin.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.seconds(1), // The max TTL must not be 0 for a cache policy\n            queryStringBehavior: CacheQueryStringBehavior.all(),\n            headerBehavior: CacheHeaderBehavior.allowList(...headers),\n            cookieBehavior: CacheCookieBehavior.all(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n                originPath: this.deploymentRevision,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: this.deploymentRevision + asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? (asset.pattern.endsWith('.html') ? ['/', '/*/', `/${asset.pattern}`] : [`/${asset.pattern}`]) : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtServerAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtServerAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n}\n"]}
|
|
389
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-server-app-stack.js","sourceRoot":"","sources":["nuxt-server-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution,\n    ICachePolicy,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, LayerVersion, Runtime, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {HttpMethod} from \"aws-cdk-lib/aws-stepfunctions-tasks\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {HttpLambdaIntegration} from '@aws-cdk/aws-apigatewayv2-integrations-alpha';\nimport {DomainName, EndpointType, HttpApi, SecurityPolicy} from \"@aws-cdk/aws-apigatewayv2-alpha\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../nuxt-app-static-assets\";\nimport * as fs from \"fs\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport {NuxtAppStackProps} from \"../nuxt-app-stack-props\";\nimport * as path from \"path\";\n\n/**\n * Defines the props required for the {@see NuxtServerAppStack}.\n */\nexport interface NuxtServerAppStackProps extends NuxtAppStackProps {\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The memory size to apply to the Nuxt app's Lambda.\n     * Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).\n     */\n    readonly memorySize?: number;\n\n    /**\n     * Whether to enable AWS X-Ray for the Nuxt Lambda function.\n     */\n    readonly enableTracing?: boolean;\n}\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used as S3 folder name\n     * to store the static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: Function;\n\n    /**\n     * The API gateway to make the Lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtServerAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = new Date().toISOString();\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            accessControl: BucketAccessControl.AUTHENTICATED_READ,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates a Lambda layer with the node_modules required to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private createSsrLambdaLayer(): LayerVersion {\n        const layerName = `${this.resourceIdPrefix}-ssr-layer`;\n        return new LayerVersion(this, layerName, {\n            layerVersionName: layerName,\n            code: Code.fromAsset('.nuxt/cdk-deployment/layer'),\n            compatibleRuntimes: [Runtime.NODEJS_12_X],\n            description: `Provides the node_modules required for SSR of ${this.resourceIdPrefix}.`,\n        });\n    }\n\n    /**\n     * Creates the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-app-function`;\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_12_X,\n            architecture: Architecture.ARM_64,\n            layers: [this.createSsrLambdaLayer()],\n            handler: 'index.handler',\n            code: Code.fromAsset('.nuxt/cdk-deployment/src', {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            logRetention: RetentionDays.ONE_MONTH,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @param props\n     * @private\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup-function`;\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_14_X,\n            architecture: Architecture.ARM_64,\n            layers: [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            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                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render Lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [HttpMethod.GET, HttpMethod.HEAD],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtAppRouteBehavior(),\n            additionalBehaviors: this.createStaticAssetsRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(2),\n                readTimeout: Duration.seconds(10),\n                protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n            }),\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: undefined,\n            cachePolicy: this.createSsrCachePolicy(),\n        };\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     * Even though we don't want to cache SSR requests, we still have to create this cache policy in order to\n     * forward required cookies, query params and headers. This doesn't make any sense, because if nothing\n     * is cached, one would expect, that anything would/could be forwarded, but anyway...\n     */\n    private createSsrCachePolicy(): ICachePolicy {\n\n        // The headers to make accessible in the Nuxt app code.\n        // There is no 'CacheHeaderBehavior.all()' option, so we have to explicitly define them.\n        const headers = [\n            'User-Agent', // Required to distinguish between mobile and desktop template\n            'Authorization', // For authorization\n            'Host' // To access the domain name on SSR requests\n        ];\n\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Passes all required request data to the ${this.resourceIdPrefix} origin.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.seconds(1), // The max TTL must not be 0 for a cache policy\n            queryStringBehavior: CacheQueryStringBehavior.all(),\n            headerBehavior: CacheHeaderBehavior.allowList(...headers),\n            cookieBehavior: CacheCookieBehavior.all(),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n                originPath: this.deploymentRevision,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the Lambda function in the {@see NuxtAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: this.deploymentRevision + asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? [`/${asset.pattern}`] : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtServerAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtServerAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '3', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n}\n"]}
|
|
@@ -428,7 +428,7 @@ export class NuxtServerAppStack extends Stack {
|
|
|
428
428
|
cacheControl: asset.cacheControl ?? defaultCacheConfig,
|
|
429
429
|
contentType: asset.contentType,
|
|
430
430
|
distribution: asset.invalidateOnChange ? this.cdn : undefined,
|
|
431
|
-
distributionPaths: asset.invalidateOnChange ?
|
|
431
|
+
distributionPaths: asset.invalidateOnChange ? [`/${asset.pattern}`] : undefined,
|
|
432
432
|
logRetention: RetentionDays.ONE_DAY,
|
|
433
433
|
memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory
|
|
434
434
|
})
|
|
@@ -12,6 +12,7 @@ const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
|
|
|
12
12
|
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
13
13
|
const nuxt_app_static_assets_1 = require("../nuxt-app-static-assets");
|
|
14
14
|
const fs = require("fs");
|
|
15
|
+
const path = require("path");
|
|
15
16
|
/**
|
|
16
17
|
* CDK stack to deploy a static generated Nuxt app (target=static) on AWS with S3 and CloudFront.
|
|
17
18
|
*/
|
|
@@ -47,10 +48,8 @@ class NuxtStaticAppStack extends aws_cdk_lib_1.Stack {
|
|
|
47
48
|
// The bucket and all of its objects can be deleted, because all the content is managed in this project
|
|
48
49
|
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
49
50
|
autoDeleteObjects: true,
|
|
50
|
-
//
|
|
51
|
-
|
|
52
|
-
websiteIndexDocument: 'index.html',
|
|
53
|
-
websiteErrorDocument: '200.html'
|
|
51
|
+
// We only want the files to be reachable by our custom domain to prevent duplicate content issues
|
|
52
|
+
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL
|
|
54
53
|
});
|
|
55
54
|
bucket.grantRead(this.cdnAccessIdentity);
|
|
56
55
|
return bucket;
|
|
@@ -69,7 +68,15 @@ class NuxtStaticAppStack extends aws_cdk_lib_1.Stack {
|
|
|
69
68
|
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2018,
|
|
70
69
|
certificate: aws_certificatemanager_1.Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),
|
|
71
70
|
defaultBehavior: this.createNuxtStaticAppRouteBehavior(),
|
|
72
|
-
priceClass: aws_cloudfront_1.PriceClass.PRICE_CLASS_100,
|
|
71
|
+
priceClass: aws_cloudfront_1.PriceClass.PRICE_CLASS_100,
|
|
72
|
+
errorResponses: [404, 403].map(errorCode => {
|
|
73
|
+
return {
|
|
74
|
+
ttl: aws_cdk_lib_1.Duration.seconds(10),
|
|
75
|
+
httpStatus: errorCode,
|
|
76
|
+
responseHttpStatus: errorCode,
|
|
77
|
+
responsePagePath: '/200.html'
|
|
78
|
+
};
|
|
79
|
+
})
|
|
73
80
|
});
|
|
74
81
|
}
|
|
75
82
|
/**
|
|
@@ -79,6 +86,13 @@ class NuxtStaticAppStack extends aws_cdk_lib_1.Stack {
|
|
|
79
86
|
* @private
|
|
80
87
|
*/
|
|
81
88
|
createNuxtStaticAppRouteBehavior() {
|
|
89
|
+
const redirectFunction = new aws_cloudfront_1.Function(this, `${this.resourceIdPrefix}-redirect-to-index`, {
|
|
90
|
+
functionName: `${this.resourceIdPrefix}-redirect-to-index`,
|
|
91
|
+
comment: `Redirects incoming requests to the ${this.resourceIdPrefix} service to their corresponding S3 bucket file.`,
|
|
92
|
+
code: aws_cloudfront_1.FunctionCode.fromFile({
|
|
93
|
+
filePath: path.join(__dirname, '../../functions/cloudfront/redirect-to-index.js')
|
|
94
|
+
}),
|
|
95
|
+
});
|
|
82
96
|
return {
|
|
83
97
|
origin: new aws_cloudfront_origins_1.S3Origin(this.staticAssetsBucket, {
|
|
84
98
|
connectionAttempts: 2,
|
|
@@ -89,7 +103,11 @@ class NuxtStaticAppStack extends aws_cdk_lib_1.Stack {
|
|
|
89
103
|
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
|
|
90
104
|
cachedMethods: aws_cloudfront_1.CachedMethods.CACHE_GET_HEAD_OPTIONS,
|
|
91
105
|
cachePolicy: aws_cloudfront_1.CachePolicy.CACHING_OPTIMIZED,
|
|
92
|
-
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS
|
|
106
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
107
|
+
functionAssociations: [{
|
|
108
|
+
function: redirectFunction,
|
|
109
|
+
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST,
|
|
110
|
+
}]
|
|
93
111
|
};
|
|
94
112
|
}
|
|
95
113
|
/**
|
|
@@ -161,4 +179,4 @@ class NuxtStaticAppStack extends aws_cdk_lib_1.Stack {
|
|
|
161
179
|
}
|
|
162
180
|
}
|
|
163
181
|
exports.NuxtStaticAppStack = NuxtStaticAppStack;
|
|
164
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-static-app-stack.js","sourceRoot":"","sources":["nuxt-static-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CachedMethods,\n    CachePolicy,\n    Distribution,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Bucket, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../nuxt-app-static-assets\";\nimport * as fs from \"fs\";\nimport {NuxtAppStackProps} from \"../nuxt-app-stack-props\";\n\n/**\n * Defines the props required for the {@see NuxtStaticAppStack}.\n */\nexport interface NuxtStaticAppStackProps extends NuxtAppStackProps {}\n\n/**\n * CDK stack to deploy a static generated Nuxt app (target=static) on AWS with S3 and CloudFront.\n */\nexport class NuxtStaticAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the S3 bucket (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtStaticAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n            // Configure S3 as a website host\n            publicReadAccess: true,\n            websiteIndexDocument: 'index.html',\n            websiteErrorDocument: '200.html'\n        });\n\n        bucket.grantRead(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the S3 bucket (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtStaticAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtStaticAppRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the S3 bucket.\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtStaticAppRouteBehavior(): BehaviorOptions {\n        return {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS\n        };\n    }\n\n    /**\n     * Uploads the assets of the Nuxt app as defined in {@see getNuxtStaticAppStaticAssetConfigs} to the S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the Lambda function in the {@see NuxtStaticAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? (asset.pattern.endsWith('.html') ? ['/*'] : [`/${asset.pattern}`]) : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtStaticAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtStaticAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n}\n"]}
|
|
182
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nuxt-static-app-stack.js","sourceRoot":"","sources":["nuxt-static-app-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    BehaviorOptions,\n    CachedMethods,\n    CachePolicy,\n    Distribution,\n    IOriginAccessIdentity,\n    OriginAccessIdentity,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy,\n    Function, FunctionCode, FunctionEventType\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, CacheControl, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {RetentionDays} from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from \"../nuxt-app-static-assets\";\nimport * as fs from \"fs\";\nimport {NuxtAppStackProps} from \"../nuxt-app-stack-props\";\nimport * as path from \"path\";\n\n/**\n * Defines the props required for the {@see NuxtStaticAppStack}.\n */\nexport interface NuxtStaticAppStackProps extends NuxtAppStackProps {}\n\n/**\n * CDK stack to deploy a static generated Nuxt app (target=static) on AWS with S3 and CloudFront.\n */\nexport class NuxtStaticAppStack extends Stack {\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: IBucket;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution to route incoming requests to the S3 bucket (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtStaticAppStackProps) {\n        super(scope, id, props);\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.nuxtConfig);\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): IBucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n\n            // We only want the files to be reachable by our custom domain to prevent duplicate content issues\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL\n        });\n\n        bucket.grantRead(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the S3 bucket (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtStaticAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            defaultBehavior: this.createNuxtStaticAppRouteBehavior(),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n            errorResponses: [404, 403].map(errorCode => {\n                return {\n                    ttl: Duration.seconds(10),\n                    httpStatus: errorCode,\n                    responseHttpStatus: errorCode,\n                    responsePagePath: '/200.html'\n                }\n            })\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming requests to the S3 bucket.\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     *\n     * @private\n     */\n    private createNuxtStaticAppRouteBehavior(): BehaviorOptions {\n\n        const redirectFunction = new Function(this, `${this.resourceIdPrefix}-redirect-to-index`, {\n            functionName: `${this.resourceIdPrefix}-redirect-to-index`,\n            comment: `Redirects incoming requests to the ${this.resourceIdPrefix} service to their corresponding S3 bucket file.`,\n            code: FunctionCode.fromFile({\n                filePath: path.join(__dirname, '../../functions/cloudfront/redirect-to-index.js')\n            }),\n        });\n\n        return {\n            origin: new S3Origin(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            functionAssociations: [{\n                function: redirectFunction,\n                eventType: FunctionEventType.VIEWER_REQUEST,\n            }]\n        };\n    }\n\n    /**\n     * Uploads the assets of the Nuxt app as defined in {@see getNuxtStaticAppStaticAssetConfigs} to the S3 bucket.\n     * In order to enable a zero-downtime deployment, we use a new subdirectory (revision) for every deployment.\n     * The previous versions are retained to allow clients to continue to work with an older revision but gets cleaned up\n     * after a specified period of time via the Lambda function in the {@see NuxtStaticAppAssetsCleanupStack}.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const defaultCacheConfig = [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(365)),\n            CacheControl.fromString('immutable'),\n        ];\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        return this.staticAssetConfigs.filter(asset => fs.existsSync(asset.source)).map((asset, assetIndex) => {\n            return new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source)],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target,\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl ?? defaultCacheConfig,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: asset.invalidateOnChange ? (asset.pattern.endsWith('.html') ? ['/*'] : [`/${asset.pattern}`]) : undefined,\n                logRetention: RetentionDays.ONE_DAY,\n                memoryLimit: 256 // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory\n            })\n        });\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtStaticAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtStaticAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n}\n"]}
|
|
@@ -11,9 +11,10 @@ import {
|
|
|
11
11
|
OriginAccessIdentity,
|
|
12
12
|
PriceClass,
|
|
13
13
|
SecurityPolicyProtocol,
|
|
14
|
-
ViewerProtocolPolicy
|
|
14
|
+
ViewerProtocolPolicy,
|
|
15
|
+
Function, FunctionCode, FunctionEventType
|
|
15
16
|
} from "aws-cdk-lib/aws-cloudfront";
|
|
16
|
-
import {Bucket, IBucket} from "aws-cdk-lib/aws-s3";
|
|
17
|
+
import {BlockPublicAccess, Bucket, BucketAccessControl, IBucket} from "aws-cdk-lib/aws-s3";
|
|
17
18
|
import {AaaaRecord, ARecord, HostedZone, IHostedZone, RecordTarget} from "aws-cdk-lib/aws-route53";
|
|
18
19
|
import {BucketDeployment, CacheControl, Source, StorageClass} from "aws-cdk-lib/aws-s3-deployment";
|
|
19
20
|
import {S3Origin} from "aws-cdk-lib/aws-cloudfront-origins";
|
|
@@ -22,6 +23,7 @@ import {RetentionDays} from "aws-cdk-lib/aws-logs";
|
|
|
22
23
|
import {getNuxtAppStaticAssetConfigs, StaticAssetConfig} from "../nuxt-app-static-assets";
|
|
23
24
|
import * as fs from "fs";
|
|
24
25
|
import {NuxtAppStackProps} from "../nuxt-app-stack-props";
|
|
26
|
+
import * as path from "path";
|
|
25
27
|
|
|
26
28
|
/**
|
|
27
29
|
* Defines the props required for the {@see NuxtStaticAppStack}.
|
|
@@ -100,10 +102,9 @@ export class NuxtStaticAppStack extends Stack {
|
|
|
100
102
|
// The bucket and all of its objects can be deleted, because all the content is managed in this project
|
|
101
103
|
removalPolicy: RemovalPolicy.DESTROY,
|
|
102
104
|
autoDeleteObjects: true,
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
websiteErrorDocument: '200.html'
|
|
105
|
+
|
|
106
|
+
// We only want the files to be reachable by our custom domain to prevent duplicate content issues
|
|
107
|
+
blockPublicAccess: BlockPublicAccess.BLOCK_ALL
|
|
107
108
|
});
|
|
108
109
|
|
|
109
110
|
bucket.grantRead(this.cdnAccessIdentity);
|
|
@@ -127,6 +128,14 @@ export class NuxtStaticAppStack extends Stack {
|
|
|
127
128
|
certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),
|
|
128
129
|
defaultBehavior: this.createNuxtStaticAppRouteBehavior(),
|
|
129
130
|
priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe
|
|
131
|
+
errorResponses: [404, 403].map(errorCode => {
|
|
132
|
+
return {
|
|
133
|
+
ttl: Duration.seconds(10),
|
|
134
|
+
httpStatus: errorCode,
|
|
135
|
+
responseHttpStatus: errorCode,
|
|
136
|
+
responsePagePath: '/200.html'
|
|
137
|
+
}
|
|
138
|
+
})
|
|
130
139
|
});
|
|
131
140
|
}
|
|
132
141
|
|
|
@@ -137,6 +146,15 @@ export class NuxtStaticAppStack extends Stack {
|
|
|
137
146
|
* @private
|
|
138
147
|
*/
|
|
139
148
|
private createNuxtStaticAppRouteBehavior(): BehaviorOptions {
|
|
149
|
+
|
|
150
|
+
const redirectFunction = new Function(this, `${this.resourceIdPrefix}-redirect-to-index`, {
|
|
151
|
+
functionName: `${this.resourceIdPrefix}-redirect-to-index`,
|
|
152
|
+
comment: `Redirects incoming requests to the ${this.resourceIdPrefix} service to their corresponding S3 bucket file.`,
|
|
153
|
+
code: FunctionCode.fromFile({
|
|
154
|
+
filePath: path.join(__dirname, '../../functions/cloudfront/redirect-to-index.js')
|
|
155
|
+
}),
|
|
156
|
+
});
|
|
157
|
+
|
|
140
158
|
return {
|
|
141
159
|
origin: new S3Origin(this.staticAssetsBucket, {
|
|
142
160
|
connectionAttempts: 2,
|
|
@@ -147,7 +165,11 @@ export class NuxtStaticAppStack extends Stack {
|
|
|
147
165
|
allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
|
|
148
166
|
cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,
|
|
149
167
|
cachePolicy: CachePolicy.CACHING_OPTIMIZED,
|
|
150
|
-
viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS
|
|
168
|
+
viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
169
|
+
functionAssociations: [{
|
|
170
|
+
function: redirectFunction,
|
|
171
|
+
eventType: FunctionEventType.VIEWER_REQUEST,
|
|
172
|
+
}]
|
|
151
173
|
};
|
|
152
174
|
}
|
|
153
175
|
|