cdk-nuxt 2.26.0 → 2.26.1

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.
@@ -27,10 +27,6 @@ export interface StaticAssetConfig {
27
27
  * The cache settings to use for the uploaded source files when accessing them on the target path with the specified pattern.
28
28
  */
29
29
  cacheControl: CacheControl[];
30
- /**
31
- * 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.
32
- */
33
- invalidateOnChange?: boolean;
34
30
  }
35
31
  /**
36
32
  * Retrieves the static assets of the Nuxt app that shall be publicly available.
@@ -28,8 +28,9 @@ const getNuxtAppStaticAssetConfigs = (rootDir = '.') => {
28
28
  },
29
29
  // Nuxt uses this file to detect that a newer deployment exists on the client.
30
30
  // It is not build-hashed and therefore must not be treated as immutable.
31
- // We also invalidate it on every deployment so Nuxt's outdated-build detection and any configured
32
- // shell HTML invalidations are switched only after the new Lambda and assets are in place.
31
+ // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge
32
+ // request triggers a conditional GET against S3 and the new ETag is picked up on the
33
+ // first request after the deployment.
33
34
  {
34
35
  pattern: latestBuildManifestPattern,
35
36
  source: customAssetsSourcePath,
@@ -40,7 +41,6 @@ const getNuxtAppStaticAssetConfigs = (rootDir = '.') => {
40
41
  aws_s3_deployment_1.CacheControl.sMaxAge(aws_cdk_lib_1.Duration.seconds(0)),
41
42
  aws_s3_deployment_1.CacheControl.fromString('must-revalidate'),
42
43
  ],
43
- invalidateOnChange: true,
44
44
  },
45
45
  // Build files
46
46
  {
@@ -55,6 +55,43 @@ const getNuxtAppStaticAssetConfigs = (rootDir = '.') => {
55
55
  aws_s3_deployment_1.CacheControl.fromString('immutable'),
56
56
  ],
57
57
  },
58
+ // Service worker bootstrap scripts emitted by PWA plugins (e.g. @vite-pwa/nuxt).
59
+ // These filenames are stable across deployments — they are NOT content-hashed — and
60
+ // they reference the hashed precache manifest baked into them. They MUST NOT be cached
61
+ // by the browser or CloudFront, otherwise:
62
+ // - browsers keep registering the old service worker for up to max-age, so users
63
+ // never receive new deployments until the cache expires,
64
+ // - registration.update() and periodic update checks revalidate against a stale
65
+ // edge copy and never observe new builds.
66
+ // The W3C service worker spec already caps browser caching of the SW script at 24 h,
67
+ // but explicit no-cache is the documented best practice (web.dev, MDN, Workbox docs).
68
+ // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge
69
+ // request triggers a conditional GET against S3 and the new ETag is picked up on the
70
+ // first request after the deployment.
71
+ // Two separate entries (one per file) because the pattern is also used as a CloudFront
72
+ // path pattern and CloudFront does not support brace expansion.
73
+ {
74
+ pattern: 'sw.js',
75
+ source: customAssetsSourcePath,
76
+ target: customAssetsTargetPath,
77
+ cacheControl: [
78
+ aws_s3_deployment_1.CacheControl.setPublic(),
79
+ aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.seconds(0)),
80
+ aws_s3_deployment_1.CacheControl.sMaxAge(aws_cdk_lib_1.Duration.seconds(0)),
81
+ aws_s3_deployment_1.CacheControl.fromString('must-revalidate'),
82
+ ],
83
+ },
84
+ {
85
+ pattern: 'registerSW.js',
86
+ source: customAssetsSourcePath,
87
+ target: customAssetsTargetPath,
88
+ cacheControl: [
89
+ aws_s3_deployment_1.CacheControl.setPublic(),
90
+ aws_s3_deployment_1.CacheControl.maxAge(aws_cdk_lib_1.Duration.seconds(0)),
91
+ aws_s3_deployment_1.CacheControl.sMaxAge(aws_cdk_lib_1.Duration.seconds(0)),
92
+ aws_s3_deployment_1.CacheControl.fromString('must-revalidate'),
93
+ ],
94
+ },
58
95
  // Files for native app links
59
96
  {
60
97
  pattern: '.well-known/*',
@@ -85,4 +122,4 @@ const getNuxtAppStaticAssetConfigs = (rootDir = '.') => {
85
122
  return configs;
86
123
  };
87
124
  exports.getNuxtAppStaticAssetConfigs = getNuxtAppStaticAssetConfigs;
88
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtAppStaticAssets.js","sourceRoot":"","sources":["NuxtAppStaticAssets.ts"],"names":[],"mappings":";;;AAAA,qEAA2D;AAC3D,6CAAqC;AA2CrC;;;;GAIG;AACI,MAAM,4BAA4B,GAAG,CAAC,UAAkB,GAAG,EAAuB,EAAE;IACvF,MAAM,sBAAsB,GAAG,GAAG,OAAO,iBAAiB,CAAC;IAC3D,MAAM,sBAAsB,GAAG,GAAG,CAAC;IACnC,MAAM,0BAA0B,GAAG,0BAA0B,CAAC;IAE9D,MAAM,OAAO,GAAwB;QAEjC,yFAAyF;QACzF;YACI,OAAO,EAAE,cAAc;YACvB,MAAM,EAAE,sBAAsB;YAC9B,MAAM,EAAE,sBAAsB;YAC9B,WAAW,EAAE,2BAA2B;YACxC,qGAAqG;YACrG,YAAY,EAAE;gBACV,gCAAY,CAAC,SAAS,EAAE;gBACxB,gCAAY,CAAC,MAAM,CAAC,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACzC,gCAAY,CAAC,OAAO,CAAC,sBAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;aAC1C;SACJ;QAED,8EAA8E;QAC9E,yEAAyE;QACzE,kGAAkG;QAClG,2FAA2F;QAC3F;YACI,OAAO,EAAE,0BAA0B;YACnC,MAAM,EAAE,sBAAsB;YAC9B,MAAM,EAAE,sBAAsB;YAC9B,YAAY,EAAE;gBACV,gCAAY,CAAC,SAAS,EAAE;gBACxB,gCAAY,CAAC,MAAM,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACxC,gCAAY,CAAC,OAAO,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACzC,gCAAY,CAAC,UAAU,CAAC,iBAAiB,CAAC;aAC7C;YACD,kBAAkB,EAAE,IAAI;SAC3B;QAED,cAAc;QACd;YACI,OAAO,EAAE,SAAS;YAClB,MAAM,EAAE,sBAAsB;YAC9B,MAAM,EAAE,sBAAsB;YAC9B,OAAO,EAAE,CAAC,0BAA0B,CAAC;YAErC,uFAAuF;YACvF,YAAY,EAAE;gBACV,gCAAY,CAAC,SAAS,EAAE;gBACxB,gCAAY,CAAC,MAAM,CAAC,sBAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBACvC,gCAAY,CAAC,UAAU,CAAC,WAAW,CAAC;aACvC;SACJ;QAED,6BAA6B;QAC7B;YACI,OAAO,EAAE,eAAe;YACxB,MAAM,EAAE,sBAAsB;YAC9B,MAAM,EAAE,sBAAsB;YAC9B,WAAW,EAAE,iCAAiC,EAAE,8DAA8D;YAC9G,YAAY,EAAE;gBACV,gCAAY,CAAC,SAAS,EAAE;gBACxB,gCAAY,CAAC,MAAM,CAAC,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBACrC,gCAAY,CAAC,OAAO,CAAC,sBAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;aAC1C;SACJ;KACJ,CAAC;IAEF,iHAAiH;IACjH,OAAO,CAAC,IAAI,CAAC;QACT,OAAO,EAAE,MAAM,EAAE,4CAA4C;QAC7D,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,+BAA+B;QAC/E,MAAM,EAAE,sBAAsB;QAC9B,MAAM,EAAE,sBAAsB;QAE9B,wGAAwG;QACxG,uDAAuD;QACvD,YAAY,EAAE;YACV,gCAAY,CAAC,SAAS,EAAE;YACxB,gCAAY,CAAC,MAAM,CAAC,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YACrC,gCAAY,CAAC,OAAO,CAAC,sBAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SAC1C;KACJ,CAAC,CAAA;IAEF,OAAO,OAAO,CAAC;AACnB,CAAC,CAAC;AApFW,QAAA,4BAA4B,gCAoFvC","sourcesContent":["import {CacheControl} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {Duration} from \"aws-cdk-lib\";\n\nexport interface StaticAssetConfig {\n\n    /**\n     * The local directory to upload the files from.\n     */\n    source: string,\n\n    /**\n     * The file pattern that matches files to upload (recursively) from the source prop directory.\n     * Also defines the pattern for the incoming requests that should be forwarded to the target path in the\n     * static assets S3 bucket with the appropriate cache and content settings defined in the same object.\n     */\n    pattern: string,\n\n    /**\n     * An array of file patterns to exclude from the upload.\n     */\n    exclude?: 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     * Useful to override force content types for specific files.\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 static assets of the Nuxt app that shall be publicly available.\n *\n * @param rootDir The path to the root directory of the Nuxt app at which the `.output` build folder is located.\n */\nexport const getNuxtAppStaticAssetConfigs = (rootDir: string = '.'): StaticAssetConfig[] => {\n    const customAssetsSourcePath = `${rootDir}/.output/public`;\n    const customAssetsTargetPath = '/';\n    const latestBuildManifestPattern = '_nuxt/builds/latest.json';\n\n    const configs: StaticAssetConfig[] = [\n\n        // File to detect current deployment revision to delete outdated files of old deployments\n        {\n            pattern: 'app-revision',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'text/plain; charset=UTF-8',\n            // Internal deployment marker used by the cleanup job to identify the newest uploaded asset revision.\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(10)),\n                CacheControl.sMaxAge(Duration.days(14)),\n            ],\n        },\n\n        // Nuxt uses this file to detect that a newer deployment exists on the client.\n        // It is not build-hashed and therefore must not be treated as immutable.\n        // We also invalidate it on every deployment so Nuxt's outdated-build detection and any configured\n        // shell HTML invalidations are switched only after the new Lambda and assets are in place.\n        {\n            pattern: latestBuildManifestPattern,\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(0)),\n                CacheControl.sMaxAge(Duration.seconds(0)),\n                CacheControl.fromString('must-revalidate'),\n            ],\n            invalidateOnChange: true,\n        },\n\n        // Build files\n        {\n            pattern: '_nuxt/*',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            exclude: [latestBuildManifestPattern],\n\n            // Build assets are hashed whereby they are immutable and can be cached for a long time\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.days(365)),\n                CacheControl.fromString('immutable'),\n            ],\n        },\n\n        // Files for native app links\n        {\n            pattern: '.well-known/*',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'application/json; charset=UTF-8', // Explicitly provided as these file usually have no extension\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.days(1)),\n                CacheControl.sMaxAge(Duration.hours(1)),\n            ],\n        }\n    ];\n\n    // All custom files from the public dir, e.g., robots.txt, ads.txt, sitemap.xml, *.js, manifest.webmanifest, etc.\n    configs.push({\n        pattern: '?*.*', // exclude .gitignore and other hidden files\n        exclude: configs.map(config => config.pattern), // exclude our specific configs\n        source: customAssetsSourcePath,\n        target: customAssetsTargetPath,\n\n        // Custom assets might not be versioned whereby we want to prevent any caching issues when updating them\n        // -> cache for only 1 day on CDN and 1 hour on browser\n        cacheControl: [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(1)),\n            CacheControl.sMaxAge(Duration.hours(1)),\n        ],\n    })\n\n    return configs;\n};\n"]}
125
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtAppStaticAssets.js","sourceRoot":"","sources":["NuxtAppStaticAssets.ts"],"names":[],"mappings":"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","sourcesContent":["import {CacheControl} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {Duration} from \"aws-cdk-lib\";\n\nexport interface StaticAssetConfig {\n\n    /**\n     * The local directory to upload the files from.\n     */\n    source: string,\n\n    /**\n     * The file pattern that matches files to upload (recursively) from the source prop directory.\n     * Also defines the pattern for the incoming requests that should be forwarded to the target path in the\n     * static assets S3 bucket with the appropriate cache and content settings defined in the same object.\n     */\n    pattern: string,\n\n    /**\n     * An array of file patterns to exclude from the upload.\n     */\n    exclude?: 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     * Useful to override force content types for specific files.\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/**\n * Retrieves the static assets of the Nuxt app that shall be publicly available.\n *\n * @param rootDir The path to the root directory of the Nuxt app at which the `.output` build folder is located.\n */\nexport const getNuxtAppStaticAssetConfigs = (rootDir: string = '.'): StaticAssetConfig[] => {\n    const customAssetsSourcePath = `${rootDir}/.output/public`;\n    const customAssetsTargetPath = '/';\n    const latestBuildManifestPattern = '_nuxt/builds/latest.json';\n\n    const configs: StaticAssetConfig[] = [\n\n        // File to detect current deployment revision to delete outdated files of old deployments\n        {\n            pattern: 'app-revision',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'text/plain; charset=UTF-8',\n            // Internal deployment marker used by the cleanup job to identify the newest uploaded asset revision.\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(10)),\n                CacheControl.sMaxAge(Duration.days(14)),\n            ],\n        },\n\n        // Nuxt uses this file to detect that a newer deployment exists on the client.\n        // It is not build-hashed and therefore must not be treated as immutable.\n        // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge\n        // request triggers a conditional GET against S3 and the new ETag is picked up on the\n        // first request after the deployment.\n        {\n            pattern: latestBuildManifestPattern,\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(0)),\n                CacheControl.sMaxAge(Duration.seconds(0)),\n                CacheControl.fromString('must-revalidate'),\n            ],\n        },\n\n        // Build files\n        {\n            pattern: '_nuxt/*',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            exclude: [latestBuildManifestPattern],\n\n            // Build assets are hashed whereby they are immutable and can be cached for a long time\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.days(365)),\n                CacheControl.fromString('immutable'),\n            ],\n        },\n\n        // Service worker bootstrap scripts emitted by PWA plugins (e.g. @vite-pwa/nuxt).\n        // These filenames are stable across deployments — they are NOT content-hashed — and\n        // they reference the hashed precache manifest baked into them. They MUST NOT be cached\n        // by the browser or CloudFront, otherwise:\n        //   - browsers keep registering the old service worker for up to max-age, so users\n        //     never receive new deployments until the cache expires,\n        //   - registration.update() and periodic update checks revalidate against a stale\n        //     edge copy and never observe new builds.\n        // The W3C service worker spec already caps browser caching of the SW script at 24 h,\n        // but explicit no-cache is the documented best practice (web.dev, MDN, Workbox docs).\n        // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge\n        // request triggers a conditional GET against S3 and the new ETag is picked up on the\n        // first request after the deployment.\n        // Two separate entries (one per file) because the pattern is also used as a CloudFront\n        // path pattern and CloudFront does not support brace expansion.\n        {\n            pattern: 'sw.js',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(0)),\n                CacheControl.sMaxAge(Duration.seconds(0)),\n                CacheControl.fromString('must-revalidate'),\n            ],\n        },\n        {\n            pattern: 'registerSW.js',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.seconds(0)),\n                CacheControl.sMaxAge(Duration.seconds(0)),\n                CacheControl.fromString('must-revalidate'),\n            ],\n        },\n\n        // Files for native app links\n        {\n            pattern: '.well-known/*',\n            source: customAssetsSourcePath,\n            target: customAssetsTargetPath,\n            contentType: 'application/json; charset=UTF-8', // Explicitly provided as these file usually have no extension\n            cacheControl: [\n                CacheControl.setPublic(),\n                CacheControl.maxAge(Duration.days(1)),\n                CacheControl.sMaxAge(Duration.hours(1)),\n            ],\n        }\n    ];\n\n    // All custom files from the public dir, e.g., robots.txt, ads.txt, sitemap.xml, *.js, manifest.webmanifest, etc.\n    configs.push({\n        pattern: '?*.*', // exclude .gitignore and other hidden files\n        exclude: configs.map(config => config.pattern), // exclude our specific configs\n        source: customAssetsSourcePath,\n        target: customAssetsTargetPath,\n\n        // Custom assets might not be versioned whereby we want to prevent any caching issues when updating them\n        // -> cache for only 1 day on CDN and 1 hour on browser\n        cacheControl: [\n            CacheControl.setPublic(),\n            CacheControl.maxAge(Duration.days(1)),\n            CacheControl.sMaxAge(Duration.hours(1)),\n        ],\n    })\n\n    return configs;\n};\n"]}
@@ -35,11 +35,6 @@ export interface StaticAssetConfig {
35
35
  * The cache settings to use for the uploaded source files when accessing them on the target path with the specified pattern.
36
36
  */
37
37
  cacheControl: CacheControl[],
38
-
39
- /**
40
- * 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.
41
- */
42
- invalidateOnChange?: boolean;
43
38
  }
44
39
 
45
40
  /**
@@ -70,8 +65,9 @@ export const getNuxtAppStaticAssetConfigs = (rootDir: string = '.'): StaticAsset
70
65
 
71
66
  // Nuxt uses this file to detect that a newer deployment exists on the client.
72
67
  // It is not build-hashed and therefore must not be treated as immutable.
73
- // We also invalidate it on every deployment so Nuxt's outdated-build detection and any configured
74
- // shell HTML invalidations are switched only after the new Lambda and assets are in place.
68
+ // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge
69
+ // request triggers a conditional GET against S3 and the new ETag is picked up on the
70
+ // first request after the deployment.
75
71
  {
76
72
  pattern: latestBuildManifestPattern,
77
73
  source: customAssetsSourcePath,
@@ -82,7 +78,6 @@ export const getNuxtAppStaticAssetConfigs = (rootDir: string = '.'): StaticAsset
82
78
  CacheControl.sMaxAge(Duration.seconds(0)),
83
79
  CacheControl.fromString('must-revalidate'),
84
80
  ],
85
- invalidateOnChange: true,
86
81
  },
87
82
 
88
83
  // Build files
@@ -100,6 +95,44 @@ export const getNuxtAppStaticAssetConfigs = (rootDir: string = '.'): StaticAsset
100
95
  ],
101
96
  },
102
97
 
98
+ // Service worker bootstrap scripts emitted by PWA plugins (e.g. @vite-pwa/nuxt).
99
+ // These filenames are stable across deployments — they are NOT content-hashed — and
100
+ // they reference the hashed precache manifest baked into them. They MUST NOT be cached
101
+ // by the browser or CloudFront, otherwise:
102
+ // - browsers keep registering the old service worker for up to max-age, so users
103
+ // never receive new deployments until the cache expires,
104
+ // - registration.update() and periodic update checks revalidate against a stale
105
+ // edge copy and never observe new builds.
106
+ // The W3C service worker spec already caps browser caching of the SW script at 24 h,
107
+ // but explicit no-cache is the documented best practice (web.dev, MDN, Workbox docs).
108
+ // No CloudFront invalidation is needed: with s-maxage=0 + must-revalidate, every edge
109
+ // request triggers a conditional GET against S3 and the new ETag is picked up on the
110
+ // first request after the deployment.
111
+ // Two separate entries (one per file) because the pattern is also used as a CloudFront
112
+ // path pattern and CloudFront does not support brace expansion.
113
+ {
114
+ pattern: 'sw.js',
115
+ source: customAssetsSourcePath,
116
+ target: customAssetsTargetPath,
117
+ cacheControl: [
118
+ CacheControl.setPublic(),
119
+ CacheControl.maxAge(Duration.seconds(0)),
120
+ CacheControl.sMaxAge(Duration.seconds(0)),
121
+ CacheControl.fromString('must-revalidate'),
122
+ ],
123
+ },
124
+ {
125
+ pattern: 'registerSW.js',
126
+ source: customAssetsSourcePath,
127
+ target: customAssetsTargetPath,
128
+ cacheControl: [
129
+ CacheControl.setPublic(),
130
+ CacheControl.maxAge(Duration.seconds(0)),
131
+ CacheControl.sMaxAge(Duration.seconds(0)),
132
+ CacheControl.fromString('must-revalidate'),
133
+ ],
134
+ },
135
+
103
136
  // Files for native app links
104
137
  {
105
138
  pattern: '.well-known/*',
@@ -6,7 +6,18 @@ import { type NuxtServerAppStackProps } from "./NuxtServerAppStackProps";
6
6
  * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.
7
7
  */
8
8
  export declare class NuxtServerAppStack extends Stack {
9
- private static readonly LATEST_BUILD_MANIFEST_INVALIDATION_PATH;
9
+ /**
10
+ * The static asset pattern whose post-Lambda {@link BucketDeployment} acts as the carrier for
11
+ * consumer-supplied CloudFront invalidations (via {@link NuxtServerAppStackProps.invalidatePathsOnDeploy}).
12
+ *
13
+ * We attach {@link BucketDeployment.distributionPaths} to this specific deployment so that:
14
+ * - CDK reuses its built-in CloudFront invalidation mechanism (no new construct types),
15
+ * - the invalidation fires AFTER the new Lambda and hashed assets are in place, since this
16
+ * deployment is already part of the post-Lambda deployment phase in {@link configureDeployments},
17
+ * - the files managed by {@link StaticAssetConfig} entries themselves are never invalidated
18
+ * (they are either build-hashed immutable assets or served with `s-maxage=0, must-revalidate`).
19
+ */
20
+ private static readonly CONSUMER_INVALIDATION_CARRIER_PATTERN;
10
21
  /**
11
22
  * The identifier prefix of the resources created by the stack.
12
23
  *
@@ -474,20 +474,20 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
474
474
  * but gets cleaned up after a specified period of time via the cleanup Lambda function.
475
475
  */
476
476
  configureDeployments() {
477
- var _a;
478
477
  const logGroup = new aws_logs_1.LogGroup(this, `${this.resourceIdPrefix}-assets-deployment-logs`, {
479
478
  logGroupName: `/aws/lambda/${this.resourceIdPrefix}-assets-deployment`,
480
479
  retention: aws_logs_1.RetentionDays.ONE_DAY,
481
480
  removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
482
481
  });
483
- // Returns a deployment for every configured static asset type to respect the different cache settings
482
+ // Returns a deployment for every configured static asset type to respect the different cache settings.
483
+ // Consumer-supplied CloudFront invalidation paths (props.invalidatePathsOnDeploy) are attached to a
484
+ // single, specific deployment — the "carrier" identified by CONSUMER_INVALIDATION_CARRIER_PATTERN —
485
+ // so the invalidation piggybacks on CDK's existing BucketDeployment invalidation mechanism while
486
+ // firing at the correct point in the deployment phase (see the post-Lambda logic below).
487
+ const hasConsumerInvalidations = this.deployInvalidationPaths.length > 0;
484
488
  const deployments = this.staticAssetConfigs.filter(asset => (0, fs_1.existsSync)(asset.source)).map((asset, assetIndex) => {
485
- const distributionPaths = asset.invalidateOnChange
486
- ? this.normalizeInvalidationPaths([
487
- NuxtServerAppStack.LATEST_BUILD_MANIFEST_INVALIDATION_PATH,
488
- ...this.deployInvalidationPaths,
489
- ])
490
- : undefined;
489
+ const isInvalidationCarrier = asset.pattern === NuxtServerAppStack.CONSUMER_INVALIDATION_CARRIER_PATTERN;
490
+ const attachConsumerInvalidations = isInvalidationCarrier && hasConsumerInvalidations;
491
491
  const deployment = new aws_s3_deployment_1.BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {
492
492
  sources: [aws_s3_deployment_1.Source.asset(asset.source, {
493
493
  exclude: asset.exclude,
@@ -500,8 +500,8 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
500
500
  include: [asset.pattern],
501
501
  cacheControl: asset.cacheControl,
502
502
  contentType: asset.contentType,
503
- distribution: asset.invalidateOnChange ? this.cdn : undefined,
504
- distributionPaths: distributionPaths,
503
+ distribution: attachConsumerInvalidations ? this.cdn : undefined,
504
+ distributionPaths: attachConsumerInvalidations ? this.deployInvalidationPaths : undefined,
505
505
  logGroup: logGroup,
506
506
  metadata: {
507
507
  // Store build revision on every asset to allow cleanup of outdated assets
@@ -513,19 +513,29 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
513
513
  });
514
514
  return { asset, deployment };
515
515
  });
516
- const latestBuildDeployment = (_a = deployments.find(({ asset }) => asset.pattern === '_nuxt/builds/latest.json')) === null || _a === void 0 ? void 0 : _a.deployment;
516
+ // Patterns that must be uploaded AFTER the new Lambda + hashed assets are in place.
517
+ // These are non-hashed "switchover" files that flip clients onto the new build:
518
+ // - latest.json: Nuxt's outdated-build manifest pointer
519
+ // - sw.js / registerSW.js: PWA service worker bootstrap; once a client picks up the
520
+ // new sw.js it will try to precache the new chunks, which must already exist
521
+ const postLambdaPatterns = new Set([
522
+ '_nuxt/builds/latest.json',
523
+ 'sw.js',
524
+ 'registerSW.js',
525
+ ]);
526
+ const postLambdaDeployments = deployments.filter(({ asset }) => postLambdaPatterns.has(asset.pattern));
517
527
  const preLambdaDeployments = deployments
518
- .filter(({ asset }) => asset.pattern !== '_nuxt/builds/latest.json')
528
+ .filter(({ asset }) => !postLambdaPatterns.has(asset.pattern))
519
529
  .map(({ deployment }) => deployment);
520
530
  preLambdaDeployments.forEach(deployment => {
521
531
  this.appLambdaFunction.node.addDependency(deployment);
522
532
  });
523
- if (latestBuildDeployment) {
524
- preLambdaDeployments.forEach(deployment => {
525
- latestBuildDeployment.node.addDependency(deployment);
533
+ postLambdaDeployments.forEach(({ deployment }) => {
534
+ preLambdaDeployments.forEach(preLambda => {
535
+ deployment.node.addDependency(preLambda);
526
536
  });
527
- latestBuildDeployment.node.addDependency(this.appLambdaFunction);
528
- }
537
+ deployment.node.addDependency(this.appLambdaFunction);
538
+ });
529
539
  return deployments.map(({ deployment }) => deployment);
530
540
  }
531
541
  /**
@@ -644,5 +654,16 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
644
654
  }
645
655
  }
646
656
  exports.NuxtServerAppStack = NuxtServerAppStack;
647
- NuxtServerAppStack.LATEST_BUILD_MANIFEST_INVALIDATION_PATH = '/_nuxt/builds/latest.json';
648
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    type BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution, HttpVersion,\n    type IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy, OriginRequestPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy, OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior,\n    FunctionEventType, type FunctionAssociation,\n    Function as CloudFrontFunction,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, Runtime, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {\n    BlockPublicAccess,\n    Bucket,\n    BucketAccessControl,\n    BucketEncryption,\n    ObjectOwnership\n} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, type IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3BucketOrigin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, type StaticAssetConfig} from \"../NuxtAppStaticAssets\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\nimport {writeFileSync, mkdirSync, existsSync} from \"fs\";\nimport {type NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {type NuxtCloudFrontBehavior} from \"./NuxtServerAppStackProps\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, HttpMethod, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack extends Stack {\n    private static readonly LATEST_BUILD_MANIFEST_INVALIDATION_PATH = '/_nuxt/builds/latest.json';\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used to tag the static assets of the deployment\n     * to later be able to clean up outdated assets.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * Additional CloudFront paths to invalidate after a deployment.\n     *\n     * @private\n     */\n    private readonly deployInvalidationPaths: string[];\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: Bucket;\n\n    /**\n     * The S3 bucket where the access logs of the CloudFront distribution gets stored.\n     */\n    public accessLogsBucket: Bucket|undefined;\n\n    /**\n     * The S3 bucket where the sitemap assets gets stored.\n     */\n    public sitemapBucket: Bucket|undefined;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: Function;\n\n    /**\n     * The API gateway to make the Lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution origin for the API gateway to route incoming requests to the Nuxt Lambda function.\n     */\n    private httpOrigin: HttpOrigin;\n\n    /**\n     * The cache policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app and uses to generate a cache key.\n     */\n    private appCachePolicy: CachePolicy;\n\n    /**\n     * The origin request policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app without affecting the cache key.\n     */\n    private appRequestPolicy: OriginRequestPolicy | undefined;\n\n    /**\n     * The behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt Lambda function (via API gateway).\n     */\n    private nuxtServerRouteBehavior: BehaviorOptions;\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtServerAppStackProps) {\n        super(scope, id, {\n            ...props,\n\n            // Force cross-region references if a WAF ACL is used outside us-east-1\n            crossRegionReferences: props.webAclArn !== undefined && props.env?.region !== 'us-east-1' ? true : props.crossRegionReferences,\n        });\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = this.createDeploymentRevision(props);\n        this.deployInvalidationPaths = this.normalizeInvalidationPaths(props.invalidatePathsOnDeploy);\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.rootDir ?? '.');\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n\n        if (props.enableAccessLogsAnalysis) {\n            this.accessLogsBucket = this.createAccessLogsBucket();\n            this.createAccessLogsAnalysis(props);\n        }\n\n        if (props.enableSitemap) {\n            this.sitemapBucket = this.createSitemapBucket();\n        }\n\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.httpOrigin = this.createNuxtAppHttpOrigin();\n        this.appCachePolicy = this.createNuxtAppCachePolicy(props)\n        this.appRequestPolicy = this.createNuxtAppRequestPolicy(props)\n        this.nuxtServerRouteBehavior = this.createNuxtServerRouteBehavior()\n\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the current deployment revision file in the public folder of the Nuxt app to be accessible\n     * and returns the current revision.\n     */\n    private createDeploymentRevision(props: NuxtServerAppStackProps): string {\n        const appRevision = new Date().toISOString();\n\n        const dir = path.join(props.rootDir ?? '.', '.output', 'public');\n        mkdirSync(dir, { recursive: true });\n        writeFileSync(path.join(dir, 'app-revision'), appRevision, { encoding: 'utf-8' });\n\n        return appRevision;\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the bucket to store the sitemap assets of the Nuxt app.\n     *\n     * @private\n     */\n    private createSitemapBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-sitemap`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            accessControl: BucketAccessControl.PRIVATE,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            encryption: BucketEncryption.S3_MANAGED,\n            enforceSSL: true,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-nuxt`;\n\n        const appLogGroup = new LogGroup(this, `${funcName}-logs`, {\n            logGroupName: `/aws/lambda/${funcName}`,\n            retention: RetentionDays.ONE_MONTH,\n        });\n        appLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: `${props.entrypoint ?? 'index'}.handler`,\n            code: Code.fromAsset(`${props.rootDir ?? '.' }/.output/server`, {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: props.timeout ?? Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            logGroup: appLogGroup,\n            environment: {\n                NODE_OPTIONS: '--enable-source-maps',\n                ...JSON.parse(props.entrypointEnv ?? '{}'),\n            },\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     * Note that we use the bundled AWS SDK for Node to avoid the need for a custom layer\n     * which restricts the consumer to a specific yarn or npm version.\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup`;\n        const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');\n\n        const cleanupLogGroup = new LogGroup(this, `${functionName}-logs`, {\n            logGroupName: `/aws/lambda/${functionName}`,\n            retention: RetentionDays.TWO_WEEKS,\n        });\n        cleanupLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: 'index.handler',\n            code: Code.fromAsset(`${functionDirPath}/build/app`, {\n                exclude: ['*.d.ts']\n            }),\n            timeout: Duration.minutes(15),\n            memorySize: 512,\n            environment: {\n                STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,\n                OUTDATED_ASSETS_RETENTION_DAYS: `${props.outdatedAssetsRetentionDays ?? 30}`,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n            logGroup: cleanupLogGroup,\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render Lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [\n                HttpMethod.GET,\n                HttpMethod.HEAD,\n                HttpMethod.OPTIONS,\n                HttpMethod.POST,\n                HttpMethod.PUT,\n                HttpMethod.PATCH,\n                HttpMethod.DELETE,\n            ],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            httpVersion: HttpVersion.HTTP2_AND_3,\n            defaultBehavior: this.nuxtServerRouteBehavior,\n            additionalBehaviors: this.setupCloudFrontRouting(props),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n            logBucket: this.accessLogsBucket,\n            logFilePrefix: props.enableAccessLogsAnalysis ? 'unprocessed' : undefined,\n            logIncludesCookies: props.enableAccessLogsAnalysis,\n            webAclId: props.webAclArn,\n        });\n    }\n\n    /**\n     * Normalizes CloudFront invalidation paths by trimming whitespace, ensuring a leading slash, and removing duplicates.\n     *\n     * @private\n     */\n    private normalizeInvalidationPaths(paths: string[] | undefined): string[] {\n        return Array.from(\n            new Set(\n                (paths ?? [])\n                    .map(path => path.trim())\n                    .filter((path): path is string => path.length > 0)\n                    .map(path => path.startsWith('/') ? path : `/${path}`)\n            )\n        );\n    }\n\n    /**\n     * Creates the CloudFront distribution behavior origin to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     */\n    private createNuxtAppHttpOrigin(): HttpOrigin {\n        return new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n            connectionAttempts: 2,\n            connectionTimeout: Duration.seconds(2),\n            readTimeout: Duration.seconds(10),\n            protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     */\n    private createNuxtServerRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: this.appRequestPolicy,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {\n\n            // Nuxt I18n files are served via a server route\n            '/_i18n/*': this.nuxtServerRouteBehavior,\n        };\n\n        // Specific ones first\n        if (props.enableApi) {\n            routingBehaviours = {...routingBehaviours, ...this.createApiRouteBehavior()};\n        }\n        if (props.enableSitemap) {\n            routingBehaviours = {...routingBehaviours, ...this.createSitemapRouteBehavior()};\n        }\n\n        // Add custom server routes before static assets to ensure they take precedence\n        if (props.serverRoutes && props.serverRoutes.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createServerRouteBehavior(props.serverRoutes)};\n        }\n\n        // Inject custom behaviors (cache policy overrides and/or CloudFront Functions) before static asset behaviors\n        if (props.additionalBehaviors && props.additionalBehaviors.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createAdditionalBehaviors(props.additionalBehaviors)};\n        }\n\n        routingBehaviours = {...routingBehaviours, ...this.createStaticAssetsRouteBehavior()};\n\n        return routingBehaviours;\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     */\n    private createNuxtAppCachePolicy(props: NuxtServerAppStackProps): CachePolicy {\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin and how the cache key is calculated.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.days(365),\n            queryStringBehavior: props.cacheKeyQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.cacheKeyQueryParams) : (props.denyCacheKeyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyCacheKeyQueryParams) : (props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()))),\n            headerBehavior: props.cacheKeyHeaders?.length ? CacheHeaderBehavior.allowList(...props.cacheKeyHeaders) : (props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none()),\n            cookieBehavior: props.cacheKeyCookies?.length ? CacheCookieBehavior.allowList(...props.cacheKeyCookies) : (props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none()),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates an origin request policy for the Nuxt app route behavior of the CloudFront distribution.\n     * No policy is created if no explicit config is provided.\n     */\n    private createNuxtAppRequestPolicy(props: NuxtServerAppStackProps): OriginRequestPolicy|undefined {\n\n        // If no explicit config is provided, we want to use the default from Cloudfront\n        const hasAnyConfig = props.forwardQueryParams?.length || props.forwardHeaders?.length || props.forwardCookies?.length;\n        if (!hasAnyConfig) {\n            return undefined;\n        }\n\n        return new OriginRequestPolicy(this, `${this.resourceIdPrefix}-request-policy`, {\n            originRequestPolicyName: `${this.resourceIdPrefix}-cdn-request-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin without affecting the cache key.`,\n            queryStringBehavior: props.forwardQueryParams?.length ? OriginRequestQueryStringBehavior.allowList(...props.forwardQueryParams) : OriginRequestQueryStringBehavior.all(),\n            headerBehavior: props.forwardHeaders?.length ? OriginRequestHeaderBehavior.allowList(...props.forwardHeaders) : OriginRequestHeaderBehavior.none(),\n            cookieBehavior: props.forwardCookies?.length ? OriginRequestCookieBehavior.allowList(...props.forwardCookies) : OriginRequestCookieBehavior.none(),\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching Nuxt app API requests to the API gateway.\n     */\n    private createApiRouteBehavior(): Record<string, BehaviorOptions> {\n        const apiBehavior: BehaviorOptions = {\n            origin: this.httpOrigin,\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_ALL,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: this.appCachePolicy,\n            originRequestPolicy: this.appRequestPolicy,\n            viewerProtocolPolicy: ViewerProtocolPolicy.HTTPS_ONLY\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['/api/*'] = apiBehavior;\n\n        return rules;\n    }\n\n    /**\n     * Creates behaviors for the CloudFront distribution to route specified path patterns to the SSR origin.\n     * This allows server endpoints that use file-like URLs (e.g., /sitemap.xml from @nuxtjs/sitemap) \n     * to be handled by the Lambda function for dynamic content generation.\n     */\n    private createServerRouteBehavior(serverRoutes: string[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n        \n        serverRoutes.forEach(route => {\n            rules[route] = this.nuxtServerRouteBehavior;\n        });\n\n        return rules;\n    }\n\n    /**\n     * Creates behaviors for the CloudFront distribution based on the consumer-supplied {@link NuxtCloudFrontBehavior} list.\n     * Each behavior can override the cache policy, attach a CloudFront Function, or both.\n     *\n     * When {@link NuxtCloudFrontBehavior.fnCode} is provided, this method creates the `cloudfront.Function`\n     * resource inside the stack itself so that it is always correctly scoped.\n     *\n     * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the\n     * default Nuxt app cache policy ({@link appCachePolicy}) is used.\n     */\n    private createAdditionalBehaviors(behaviors: NuxtCloudFrontBehavior[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n\n        behaviors.forEach((behavior, index) => {\n            let functionAssociations: FunctionAssociation[] = [];\n\n            if (behavior.fnCode) {\n                // Create the CloudFront Function inside the stack so it is always correctly scoped.\n                const cfFunction = new CloudFrontFunction(this, `${this.resourceIdPrefix}-behavior-fn-${index}`, {\n                    code: behavior.fnCode,\n                });\n                functionAssociations = [{\n                    function: cfFunction,\n                    eventType: behavior.eventType ?? FunctionEventType.VIEWER_REQUEST,\n                }];\n            }\n\n            rules[behavior.pathPattern] = {\n                ...this.nuxtServerRouteBehavior,\n                cachePolicy: behavior.cachePolicy ?? this.appCachePolicy,\n                ...(functionAssociations.length > 0 ? {functionAssociations} : {}),\n            };\n        });\n\n        return rules;\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: S3BucketOrigin.withOriginAccessIdentity(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the sitemap assets\n     * to the S3 bucket that holds these sitemap assets.\n     *\n     * @private\n     */\n    private createSitemapRouteBehavior(): Record<string, BehaviorOptions> {\n        if (!this.sitemapBucket) {\n            throw new Error(\"Sitemap bucket must exist before creating sitemap route behavior.\");\n        }\n\n        const sitemapCacheConfig: BehaviorOptions = {\n            origin: S3BucketOrigin.withOriginAccessIdentity(this.sitemapBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['*sitemap.xml'] = sitemapCacheConfig;\n        rules['*sitemap-gone.xml'] = sitemapCacheConfig;\n        rules['/sitemaps/*'] = sitemapCacheConfig;\n\n        return rules;\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment with minimal storage load,\n     * we deploy the static assets of every deployment into the same folder but mark them with a deployment revision.\n     * By doing so, the files of previous deployments are retained to allow clients to continue to work with an older revision\n     * but gets cleaned up after a specified period of time via the cleanup Lambda function.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const logGroup = new LogGroup(this, `${this.resourceIdPrefix}-assets-deployment-logs`, {\n            logGroupName: `/aws/lambda/${this.resourceIdPrefix}-assets-deployment`,\n            retention: RetentionDays.ONE_DAY,\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings\n        const deployments = this.staticAssetConfigs.filter(asset => existsSync(asset.source)).map((asset, assetIndex) => {\n            const distributionPaths = asset.invalidateOnChange\n                ? this.normalizeInvalidationPaths([\n                    NuxtServerAppStack.LATEST_BUILD_MANIFEST_INVALIDATION_PATH,\n                    ...this.deployInvalidationPaths,\n                ])\n                : undefined;\n\n            const deployment = new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source, {\n                    exclude: asset.exclude,\n                })],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target.replace(/^\\/+/g, ''), // Remove leading slash\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl,\n                contentType: asset.contentType,\n                distribution: asset.invalidateOnChange ? this.cdn : undefined,\n                distributionPaths: distributionPaths,\n                logGroup: logGroup,\n\n                metadata: {\n                    // Store build revision on every asset to allow cleanup of outdated assets\n                    revision: this.deploymentRevision,\n                },\n\n                // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory.\n                // Additionally, a high memory limit might speed up deployments.\n                memoryLimit: 1792\n            });\n\n            return {asset, deployment};\n        });\n\n        const latestBuildDeployment = deployments.find(({asset}) => asset.pattern === '_nuxt/builds/latest.json')?.deployment;\n        const preLambdaDeployments = deployments\n            .filter(({asset}) => asset.pattern !== '_nuxt/builds/latest.json')\n            .map(({deployment}) => deployment);\n\n        preLambdaDeployments.forEach(deployment => {\n            this.appLambdaFunction.node.addDependency(deployment);\n        });\n\n        if (latestBuildDeployment) {\n            preLambdaDeployments.forEach(deployment => {\n                latestBuildDeployment.node.addDependency(deployment);\n            });\n            latestBuildDeployment.node.addDependency(this.appLambdaFunction);\n        }\n\n        return deployments.map(({deployment}) => deployment);\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtServerAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtServerAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every Tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '2', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n\n    /**\n     * Creates a S3 bucket to store the access logs of the CloudFront distribution.\n     */\n    private createAccessLogsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-access-logs`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_PREFERRED,\n            // When the stack is destroyed, we expect everything to be deleted\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): void {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n\n        const { CloudFrontAccessLogsAnalysis } = require('../access-logs-analysis/CloudFrontAccessLogsAnalysis');\n\n        new CloudFrontAccessLogsAnalysis(this, `${this.resourceIdPrefix}-access-logs-analysis`, {\n            bucket: this.accessLogsBucket,\n            resourcePrefix: `${this.resourceIdPrefix}-access-logs`,\n            accessLogCookies: props.accessLogCookies,\n            anonymizeClientIp: props.anonymizeAccessLogClientIp ?? true,\n            expireRawLogsAfter: props.accessLogsRawRetention,\n            expireIntermediateLogsAfter: props.accessLogsIntermediateRetention,\n            expireTransformedLogsAfter: props.accessLogsTransformedRetention,\n        });\n    }\n}\n"]}
657
+ /**
658
+ * The static asset pattern whose post-Lambda {@link BucketDeployment} acts as the carrier for
659
+ * consumer-supplied CloudFront invalidations (via {@link NuxtServerAppStackProps.invalidatePathsOnDeploy}).
660
+ *
661
+ * We attach {@link BucketDeployment.distributionPaths} to this specific deployment so that:
662
+ * - CDK reuses its built-in CloudFront invalidation mechanism (no new construct types),
663
+ * - the invalidation fires AFTER the new Lambda and hashed assets are in place, since this
664
+ * deployment is already part of the post-Lambda deployment phase in {@link configureDeployments},
665
+ * - the files managed by {@link StaticAssetConfig} entries themselves are never invalidated
666
+ * (they are either build-hashed immutable assets or served with `s-maxage=0, must-revalidate`).
667
+ */
668
+ NuxtServerAppStack.CONSUMER_INVALIDATION_CARRIER_PATTERN = '_nuxt/builds/latest.json';
669
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStack.js","sourceRoot":"","sources":["NuxtServerAppStack.ts"],"names":[],"mappings":"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","sourcesContent":["import {Duration, RemovalPolicy, Stack} from 'aws-cdk-lib';\nimport {Construct} from 'constructs';\nimport {Certificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n    AllowedMethods,\n    type BehaviorOptions,\n    CacheCookieBehavior,\n    CachedMethods,\n    CacheHeaderBehavior,\n    CachePolicy,\n    CacheQueryStringBehavior,\n    Distribution, HttpVersion,\n    type IOriginAccessIdentity,\n    OriginAccessIdentity,\n    OriginProtocolPolicy, OriginRequestPolicy,\n    PriceClass,\n    SecurityPolicyProtocol,\n    ViewerProtocolPolicy, OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior,\n    FunctionEventType, type FunctionAssociation,\n    Function as CloudFrontFunction,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {Architecture, Code, Function, Runtime, Tracing} from \"aws-cdk-lib/aws-lambda\";\nimport {\n    BlockPublicAccess,\n    Bucket,\n    BucketAccessControl,\n    BucketEncryption,\n    ObjectOwnership\n} from \"aws-cdk-lib/aws-s3\";\nimport {AaaaRecord, ARecord, HostedZone, type IHostedZone, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {BucketDeployment, Source, StorageClass} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {HttpOrigin, S3BucketOrigin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport {getNuxtAppStaticAssetConfigs, type StaticAssetConfig} from \"../NuxtAppStaticAssets\";\nimport {Rule, RuleTargetInput, Schedule} from \"aws-cdk-lib/aws-events\";\nimport {LambdaFunction} from \"aws-cdk-lib/aws-events-targets\";\nimport * as path from \"path\";\nimport {writeFileSync, mkdirSync, existsSync} from \"fs\";\nimport {type NuxtServerAppStackProps} from \"./NuxtServerAppStackProps\";\nimport {type NuxtCloudFrontBehavior} from \"./NuxtServerAppStackProps\";\nimport {HttpLambdaIntegration} from \"aws-cdk-lib/aws-apigatewayv2-integrations\";\nimport {DomainName, EndpointType, HttpApi, HttpMethod, SecurityPolicy} from \"aws-cdk-lib/aws-apigatewayv2\";\n\n/**\n * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.\n */\nexport class NuxtServerAppStack extends Stack {\n    /**\n     * The static asset pattern whose post-Lambda {@link BucketDeployment} acts as the carrier for\n     * consumer-supplied CloudFront invalidations (via {@link NuxtServerAppStackProps.invalidatePathsOnDeploy}).\n     *\n     * We attach {@link BucketDeployment.distributionPaths} to this specific deployment so that:\n     *   - CDK reuses its built-in CloudFront invalidation mechanism (no new construct types),\n     *   - the invalidation fires AFTER the new Lambda and hashed assets are in place, since this\n     *     deployment is already part of the post-Lambda deployment phase in {@link configureDeployments},\n     *   - the files managed by {@link StaticAssetConfig} entries themselves are never invalidated\n     *     (they are either build-hashed immutable assets or served with `s-maxage=0, must-revalidate`).\n     */\n    private static readonly CONSUMER_INVALIDATION_CARRIER_PATTERN = '_nuxt/builds/latest.json';\n\n    /**\n     * The identifier prefix of the resources created by the stack.\n     *\n     * @private\n     */\n    private readonly resourceIdPrefix: string;\n\n    /**\n     * The identifier for the current deployment that is used to tag the static assets of the deployment\n     * to later be able to clean up outdated assets.\n     *\n     * @private\n     */\n    private readonly deploymentRevision: string;\n\n    /**\n     * Additional CloudFront paths to invalidate after a deployment.\n     *\n     * @private\n     */\n    private readonly deployInvalidationPaths: string[];\n\n    /**\n     * The identity to use for accessing the deployment assets on S3.\n     *\n     * @private\n     */\n    private readonly cdnAccessIdentity: IOriginAccessIdentity;\n\n    /**\n     * The S3 bucket where the deployment assets gets stored.\n     */\n    public staticAssetsBucket: Bucket;\n\n    /**\n     * The S3 bucket where the access logs of the CloudFront distribution gets stored.\n     */\n    public accessLogsBucket: Bucket|undefined;\n\n    /**\n     * The S3 bucket where the sitemap assets gets stored.\n     */\n    public sitemapBucket: Bucket|undefined;\n\n    /**\n     * The Lambda function to render the Nuxt app on the server side.\n     *\n     * @private\n     */\n    private readonly appLambdaFunction: Function;\n\n    /**\n     * The Lambda function that cleanups the outdated static assets of the Nuxt app.\n     *\n     * @private\n     */\n    private readonly cleanupLambdaFunction: Function;\n\n    /**\n     * The API gateway to make the Lambda function to render the Nuxt app publicly available.\n     *\n     * @private\n     */\n    private apiGateway: HttpApi;\n\n    /**\n     * The configs for the static assets of the Nuxt app that shall be publicly available.\n     *\n     * @private\n     */\n    private staticAssetConfigs: StaticAssetConfig[];\n\n    /**\n     * The CloudFront distribution origin for the API gateway to route incoming requests to the Nuxt Lambda function.\n     */\n    private httpOrigin: HttpOrigin;\n\n    /**\n     * The cache policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app and uses to generate a cache key.\n     */\n    private appCachePolicy: CachePolicy;\n\n    /**\n     * The origin request policy that specifies which HTTP headers, cookies, and query strings\n     * CloudFront forwards to the Nuxt app without affecting the cache key.\n     */\n    private appRequestPolicy: OriginRequestPolicy | undefined;\n\n    /**\n     * The behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt Lambda function (via API gateway).\n     */\n    private nuxtServerRouteBehavior: BehaviorOptions;\n\n    /**\n     * The CloudFront distribution to route incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @private\n     */\n    private readonly cdn: Distribution;\n\n    constructor(scope: Construct, id: string, props: NuxtServerAppStackProps) {\n        super(scope, id, {\n            ...props,\n\n            // Force cross-region references if a WAF ACL is used outside us-east-1\n            crossRegionReferences: props.webAclArn !== undefined && props.env?.region !== 'us-east-1' ? true : props.crossRegionReferences,\n        });\n\n        this.resourceIdPrefix = `${props.project}-${props.service}-${props.environment}`;\n\n        // Nuxt app resources\n        this.deploymentRevision = this.createDeploymentRevision(props);\n        this.deployInvalidationPaths = this.normalizeInvalidationPaths(props.invalidatePathsOnDeploy);\n        this.staticAssetConfigs = getNuxtAppStaticAssetConfigs(props.rootDir ?? '.');\n        this.cdnAccessIdentity = this.createCdnAccessIdentity();\n        this.staticAssetsBucket = this.createStaticAssetsBucket();\n\n        if (props.enableAccessLogsAnalysis) {\n            this.accessLogsBucket = this.createAccessLogsBucket();\n            this.createAccessLogsAnalysis(props);\n        }\n\n        if (props.enableSitemap) {\n            this.sitemapBucket = this.createSitemapBucket();\n        }\n\n        this.appLambdaFunction = this.createAppLambdaFunction(props);\n        this.apiGateway = this.createApiGateway(props);\n        this.httpOrigin = this.createNuxtAppHttpOrigin();\n        this.appCachePolicy = this.createNuxtAppCachePolicy(props)\n        this.appRequestPolicy = this.createNuxtAppRequestPolicy(props)\n        this.nuxtServerRouteBehavior = this.createNuxtServerRouteBehavior()\n\n        this.cdn = this.createCloudFrontDistribution(props);\n        this.configureDeployments();\n        this.createDnsRecords(props);\n        this.createAppPingRule(props);\n\n        // Static assets cleanup resources\n        this.cleanupLambdaFunction = this.createCleanupLambdaFunction(props);\n        this.createCleanupTriggerRule();\n    }\n\n    /**\n     * Creates the current deployment revision file in the public folder of the Nuxt app to be accessible\n     * and returns the current revision.\n     */\n    private createDeploymentRevision(props: NuxtServerAppStackProps): string {\n        const appRevision = new Date().toISOString();\n\n        const dir = path.join(props.rootDir ?? '.', '.output', 'public');\n        mkdirSync(dir, { recursive: true });\n        writeFileSync(path.join(dir, 'app-revision'), appRevision, { encoding: 'utf-8' });\n\n        return appRevision;\n    }\n\n    /**\n     * Creates the identity to access the S3 deployment asset files via the CloudFront distribution.\n     *\n     * @private\n     */\n    private createCdnAccessIdentity(): IOriginAccessIdentity {\n        const originAccessIdentityName = `${this.resourceIdPrefix}-cdn-s3-access`;\n        return new OriginAccessIdentity(this, originAccessIdentityName);\n    }\n\n    /**\n     * Creates the bucket to store the static deployment asset files of the Nuxt app.\n     *\n     * @private\n     */\n    private createStaticAssetsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-assets`;\n        const bucket = new Bucket(this, bucketName, {\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            bucketName,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the bucket to store the sitemap assets of the Nuxt app.\n     *\n     * @private\n     */\n    private createSitemapBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-sitemap`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            accessControl: BucketAccessControl.PRIVATE,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            encryption: BucketEncryption.S3_MANAGED,\n            enforceSSL: true,\n            // The bucket and all of its objects can be deleted, because all the content is managed in this project\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n    /**\n     * Creates the Lambda function to render the Nuxt app.\n     *\n     * @private\n     */\n    private createAppLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const funcName = `${this.resourceIdPrefix}-nuxt`;\n\n        const appLogGroup = new LogGroup(this, `${funcName}-logs`, {\n            logGroupName: `/aws/lambda/${funcName}`,\n            retention: RetentionDays.ONE_MONTH,\n        });\n        appLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\n\n        return new Function(this, funcName, {\n            functionName: funcName,\n            description: `Renders the ${this.resourceIdPrefix} Nuxt app.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: `${props.entrypoint ?? 'index'}.handler`,\n            code: Code.fromAsset(`${props.rootDir ?? '.' }/.output/server`, {\n                exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],\n            }),\n            timeout: props.timeout ?? Duration.seconds(10),\n            memorySize: props.memorySize ?? 1792,\n            allowPublicSubnet: false,\n            tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,\n            logGroup: appLogGroup,\n            environment: {\n                NODE_OPTIONS: '--enable-source-maps',\n                ...JSON.parse(props.entrypointEnv ?? '{}'),\n            },\n        });\n    }\n\n    /**\n     * Creates the Lambda function that cleanups the outdated static assets of the Nuxt app.\n     * Note that we use the bundled AWS SDK for Node to avoid the need for a custom layer\n     * which restricts the consumer to a specific yarn or npm version.\n     */\n    private createCleanupLambdaFunction(props: NuxtServerAppStackProps): Function {\n        const functionName: string = `${this.resourceIdPrefix}-cleanup`;\n        const functionDirPath = path.join(__dirname, '../../functions/assets-cleanup');\n\n        const cleanupLogGroup = new LogGroup(this, `${functionName}-logs`, {\n            logGroupName: `/aws/lambda/${functionName}`,\n            retention: RetentionDays.TWO_WEEKS,\n        });\n        cleanupLogGroup.applyRemovalPolicy(RemovalPolicy.DESTROY);\n\n        const result: Function = new Function(this, functionName, {\n            functionName: functionName,\n            description: `Auto-deletes the outdated static assets in the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            runtime: Runtime.NODEJS_20_X,\n            architecture: Architecture.ARM_64,\n            handler: 'index.handler',\n            code: Code.fromAsset(`${functionDirPath}/build/app`, {\n                exclude: ['*.d.ts']\n            }),\n            timeout: Duration.minutes(15),\n            memorySize: 512,\n            environment: {\n                STATIC_ASSETS_BUCKET: this.staticAssetsBucket.bucketName,\n                OUTDATED_ASSETS_RETENTION_DAYS: `${props.outdatedAssetsRetentionDays ?? 30}`,\n                ENVIRONMENT: props.environment,\n                AWS_NODEJS_CONNECTION_REUSE_ENABLED: '1',\n                NODE_OPTIONS: '--enable-source-maps',\n            },\n            logGroup: cleanupLogGroup,\n        });\n\n        // grant function access to S3 bucket\n        this.staticAssetsBucket.grantRead(result);\n        this.staticAssetsBucket.grantDelete(result);\n\n        return result;\n    }\n\n    /**\n     * Creates the API gateway to make the Nuxt app render Lambda function publicly available.\n     *\n     * @private\n     */\n    private createApiGateway(props: NuxtServerAppStackProps): HttpApi {\n        const apiName = `${this.resourceIdPrefix}-api`;\n        const lambdaIntegration = new HttpLambdaIntegration(`${this.resourceIdPrefix}-lambda-integration`, this.appLambdaFunction);\n\n        // We want the API gateway to be accessible by the custom domain name.\n        // Even though we access the gateway via CloudFront (for auto http to https redirects), this is required\n        // to be able to redirect the original 'Host' header to the Nuxt application, if requested.\n        const domainName = new DomainName(this, `${this.resourceIdPrefix}-api-domain`, {\n            domainName: props.domain,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-regional-certificate`, props.regionalTlsCertificateArn),\n            endpointType: EndpointType.REGIONAL,\n            securityPolicy: SecurityPolicy.TLS_1_2\n        });\n\n        const apiGateway = new HttpApi(this, apiName, {\n            apiName,\n            description: `Connects the ${this.resourceIdPrefix} CloudFront distribution with the ${this.resourceIdPrefix} Lambda function to make it publicly available.`,\n            // The app does not allow any cross-origin access by purpose: the app should not be embeddable anywhere\n            corsPreflight: undefined,\n            defaultIntegration: lambdaIntegration,\n            defaultDomainMapping: {\n                domainName: domainName\n            }\n        });\n\n        apiGateway.addRoutes({\n            integration: lambdaIntegration,\n            path: '/{proxy+}',\n            methods: [\n                HttpMethod.GET,\n                HttpMethod.HEAD,\n                HttpMethod.OPTIONS,\n                HttpMethod.POST,\n                HttpMethod.PUT,\n                HttpMethod.PATCH,\n                HttpMethod.DELETE,\n            ],\n        });\n\n        return apiGateway;\n    }\n\n    /**\n     * Creates the CloudFront distribution that routes incoming requests to the Nuxt Lambda function (via the API gateway)\n     * or the S3 assets folder (with caching).\n     *\n     * @param props\n     * @private\n     */\n    private createCloudFrontDistribution(props: NuxtServerAppStackProps): Distribution {\n        const cdnName = `${this.resourceIdPrefix}-cdn`;\n\n        return new Distribution(this, cdnName, {\n            domainNames: [props.domain],\n            comment: cdnName,\n            minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2_2018,\n            certificate: Certificate.fromCertificateArn(this, `${this.resourceIdPrefix}-global-certificate`, props.globalTlsCertificateArn),\n            httpVersion: HttpVersion.HTTP2_AND_3,\n            defaultBehavior: this.nuxtServerRouteBehavior,\n            additionalBehaviors: this.setupCloudFrontRouting(props),\n            priceClass: PriceClass.PRICE_CLASS_100, // Use only North America and Europe\n            logBucket: this.accessLogsBucket,\n            logFilePrefix: props.enableAccessLogsAnalysis ? 'unprocessed' : undefined,\n            logIncludesCookies: props.enableAccessLogsAnalysis,\n            webAclId: props.webAclArn,\n        });\n    }\n\n    /**\n     * Normalizes CloudFront invalidation paths by trimming whitespace, ensuring a leading slash, and removing duplicates.\n     *\n     * @private\n     */\n    private normalizeInvalidationPaths(paths: string[] | undefined): string[] {\n        return Array.from(\n            new Set(\n                (paths ?? [])\n                    .map(path => path.trim())\n                    .filter((path): path is string => path.length > 0)\n                    .map(path => path.startsWith('/') ? path : `/${path}`)\n            )\n        );\n    }\n\n    /**\n     * Creates the CloudFront distribution behavior origin to route incoming requests to the Nuxt render Lambda function (via API gateway).\n     */\n    private createNuxtAppHttpOrigin(): HttpOrigin {\n        return new HttpOrigin(`${this.apiGateway.httpApiId}.execute-api.${this.region}.amazonaws.com`, {\n            connectionAttempts: 2,\n            connectionTimeout: Duration.seconds(2),\n            readTimeout: Duration.seconds(10),\n            protocolPolicy: OriginProtocolPolicy.HTTPS_ONLY,\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route incoming web requests\n     * to the Nuxt render Lambda function (via API gateway).\n     * Additionally, this automatically redirects HTTP requests to HTTPS.\n     */\n    private createNuxtServerRouteBehavior(): BehaviorOptions {\n        return {\n            origin: this.httpOrigin,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD,\n            compress: true,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n            originRequestPolicy: this.appRequestPolicy,\n            cachePolicy: this.appCachePolicy\n        };\n    }\n\n    private setupCloudFrontRouting(props: NuxtServerAppStackProps): Record<string, BehaviorOptions> {\n        let routingBehaviours: Record<string, BehaviorOptions> = {\n\n            // Nuxt I18n files are served via a server route\n            '/_i18n/*': this.nuxtServerRouteBehavior,\n        };\n\n        // Specific ones first\n        if (props.enableApi) {\n            routingBehaviours = {...routingBehaviours, ...this.createApiRouteBehavior()};\n        }\n        if (props.enableSitemap) {\n            routingBehaviours = {...routingBehaviours, ...this.createSitemapRouteBehavior()};\n        }\n\n        // Add custom server routes before static assets to ensure they take precedence\n        if (props.serverRoutes && props.serverRoutes.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createServerRouteBehavior(props.serverRoutes)};\n        }\n\n        // Inject custom behaviors (cache policy overrides and/or CloudFront Functions) before static asset behaviors\n        if (props.additionalBehaviors && props.additionalBehaviors.length > 0) {\n            routingBehaviours = {...routingBehaviours, ...this.createAdditionalBehaviors(props.additionalBehaviors)};\n        }\n\n        routingBehaviours = {...routingBehaviours, ...this.createStaticAssetsRouteBehavior()};\n\n        return routingBehaviours;\n    }\n\n    /**\n     * Creates a cache policy for the Nuxt app route behavior of the CloudFront distribution.\n     */\n    private createNuxtAppCachePolicy(props: NuxtServerAppStackProps): CachePolicy {\n        return new CachePolicy(this, `${this.resourceIdPrefix}-cache-policy`, {\n            cachePolicyName: `${this.resourceIdPrefix}-cdn-cache-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin and how the cache key is calculated.`,\n            defaultTtl: Duration.seconds(0),\n            minTtl: Duration.seconds(0),\n            maxTtl: Duration.days(365),\n            queryStringBehavior: props.cacheKeyQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.cacheKeyQueryParams) : (props.denyCacheKeyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyCacheKeyQueryParams) : (props.allowQueryParams?.length ? CacheQueryStringBehavior.allowList(...props.allowQueryParams) : (props.denyQueryParams?.length ? CacheQueryStringBehavior.denyList(...props.denyQueryParams) : CacheQueryStringBehavior.all()))),\n            headerBehavior: props.cacheKeyHeaders?.length ? CacheHeaderBehavior.allowList(...props.cacheKeyHeaders) : (props.allowHeaders?.length ? CacheHeaderBehavior.allowList(...props.allowHeaders) : CacheHeaderBehavior.none()),\n            cookieBehavior: props.cacheKeyCookies?.length ? CacheCookieBehavior.allowList(...props.cacheKeyCookies) : (props.allowCookies?.length ? CacheCookieBehavior.allowList(...props.allowCookies) : CacheCookieBehavior.none()),\n            enableAcceptEncodingBrotli: true,\n            enableAcceptEncodingGzip: true,\n        });\n    }\n\n    /**\n     * Creates an origin request policy for the Nuxt app route behavior of the CloudFront distribution.\n     * No policy is created if no explicit config is provided.\n     */\n    private createNuxtAppRequestPolicy(props: NuxtServerAppStackProps): OriginRequestPolicy|undefined {\n\n        // If no explicit config is provided, we want to use the default from Cloudfront\n        const hasAnyConfig = props.forwardQueryParams?.length || props.forwardHeaders?.length || props.forwardCookies?.length;\n        if (!hasAnyConfig) {\n            return undefined;\n        }\n\n        return new OriginRequestPolicy(this, `${this.resourceIdPrefix}-request-policy`, {\n            originRequestPolicyName: `${this.resourceIdPrefix}-cdn-request-policy`,\n            comment: `Defines which request data to pass to the ${this.resourceIdPrefix} origin without affecting the cache key.`,\n            queryStringBehavior: props.forwardQueryParams?.length ? OriginRequestQueryStringBehavior.allowList(...props.forwardQueryParams) : OriginRequestQueryStringBehavior.all(),\n            headerBehavior: props.forwardHeaders?.length ? OriginRequestHeaderBehavior.allowList(...props.forwardHeaders) : OriginRequestHeaderBehavior.none(),\n            cookieBehavior: props.forwardCookies?.length ? OriginRequestCookieBehavior.allowList(...props.forwardCookies) : OriginRequestCookieBehavior.none(),\n        });\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching Nuxt app API requests to the API gateway.\n     */\n    private createApiRouteBehavior(): Record<string, BehaviorOptions> {\n        const apiBehavior: BehaviorOptions = {\n            origin: this.httpOrigin,\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_ALL,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: this.appCachePolicy,\n            originRequestPolicy: this.appRequestPolicy,\n            viewerProtocolPolicy: ViewerProtocolPolicy.HTTPS_ONLY\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['/api/*'] = apiBehavior;\n\n        return rules;\n    }\n\n    /**\n     * Creates behaviors for the CloudFront distribution to route specified path patterns to the SSR origin.\n     * This allows server endpoints that use file-like URLs (e.g., /sitemap.xml from @nuxtjs/sitemap) \n     * to be handled by the Lambda function for dynamic content generation.\n     */\n    private createServerRouteBehavior(serverRoutes: string[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n        \n        serverRoutes.forEach(route => {\n            rules[route] = this.nuxtServerRouteBehavior;\n        });\n\n        return rules;\n    }\n\n    /**\n     * Creates behaviors for the CloudFront distribution based on the consumer-supplied {@link NuxtCloudFrontBehavior} list.\n     * Each behavior can override the cache policy, attach a CloudFront Function, or both.\n     *\n     * When {@link NuxtCloudFrontBehavior.fnCode} is provided, this method creates the `cloudfront.Function`\n     * resource inside the stack itself so that it is always correctly scoped.\n     *\n     * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the\n     * default Nuxt app cache policy ({@link appCachePolicy}) is used.\n     */\n    private createAdditionalBehaviors(behaviors: NuxtCloudFrontBehavior[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n\n        behaviors.forEach((behavior, index) => {\n            let functionAssociations: FunctionAssociation[] = [];\n\n            if (behavior.fnCode) {\n                // Create the CloudFront Function inside the stack so it is always correctly scoped.\n                const cfFunction = new CloudFrontFunction(this, `${this.resourceIdPrefix}-behavior-fn-${index}`, {\n                    code: behavior.fnCode,\n                });\n                functionAssociations = [{\n                    function: cfFunction,\n                    eventType: behavior.eventType ?? FunctionEventType.VIEWER_REQUEST,\n                }];\n            }\n\n            rules[behavior.pathPattern] = {\n                ...this.nuxtServerRouteBehavior,\n                cachePolicy: behavior.cachePolicy ?? this.appCachePolicy,\n                ...(functionAssociations.length > 0 ? {functionAssociations} : {}),\n            };\n        });\n\n        return rules;\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the static assets\n     * to the S3 bucket that holds these static assets.\n     *\n     * @private\n     */\n    private createStaticAssetsRouteBehavior(): Record<string, BehaviorOptions> {\n        const staticAssetsCacheConfig: BehaviorOptions = {\n            origin: S3BucketOrigin.withOriginAccessIdentity(this.staticAssetsBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        this.staticAssetConfigs.forEach(asset => {\n            rules[`${asset.target}${asset.pattern}`] = staticAssetsCacheConfig\n        })\n\n        return rules\n    }\n\n    /**\n     * Creates a behavior for the CloudFront distribution to route matching incoming requests for the sitemap assets\n     * to the S3 bucket that holds these sitemap assets.\n     *\n     * @private\n     */\n    private createSitemapRouteBehavior(): Record<string, BehaviorOptions> {\n        if (!this.sitemapBucket) {\n            throw new Error(\"Sitemap bucket must exist before creating sitemap route behavior.\");\n        }\n\n        const sitemapCacheConfig: BehaviorOptions = {\n            origin: S3BucketOrigin.withOriginAccessIdentity(this.sitemapBucket, {\n                connectionAttempts: 2,\n                connectionTimeout: Duration.seconds(3),\n                originAccessIdentity: this.cdnAccessIdentity,\n            }),\n            compress: true,\n            allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n            cachedMethods: CachedMethods.CACHE_GET_HEAD_OPTIONS,\n            cachePolicy: CachePolicy.CACHING_OPTIMIZED,\n            viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS\n        };\n\n        const rules: Record<string, BehaviorOptions> = {};\n        rules['*sitemap.xml'] = sitemapCacheConfig;\n        rules['*sitemap-gone.xml'] = sitemapCacheConfig;\n        rules['/sitemaps/*'] = sitemapCacheConfig;\n\n        return rules;\n    }\n\n    /**\n     * Uploads the static assets of the Nuxt app as defined in {@see getNuxtAppStaticAssetConfigs} to the static assets S3 bucket.\n     * In order to enable a zero-downtime deployment with minimal storage load,\n     * we deploy the static assets of every deployment into the same folder but mark them with a deployment revision.\n     * By doing so, the files of previous deployments are retained to allow clients to continue to work with an older revision\n     * but gets cleaned up after a specified period of time via the cleanup Lambda function.\n     */\n    private configureDeployments(): BucketDeployment[] {\n        const logGroup = new LogGroup(this, `${this.resourceIdPrefix}-assets-deployment-logs`, {\n            logGroupName: `/aws/lambda/${this.resourceIdPrefix}-assets-deployment`,\n            retention: RetentionDays.ONE_DAY,\n            removalPolicy: RemovalPolicy.DESTROY,\n        });\n\n        // Returns a deployment for every configured static asset type to respect the different cache settings.\n        // Consumer-supplied CloudFront invalidation paths (props.invalidatePathsOnDeploy) are attached to a\n        // single, specific deployment — the \"carrier\" identified by CONSUMER_INVALIDATION_CARRIER_PATTERN —\n        // so the invalidation piggybacks on CDK's existing BucketDeployment invalidation mechanism while\n        // firing at the correct point in the deployment phase (see the post-Lambda logic below).\n        const hasConsumerInvalidations = this.deployInvalidationPaths.length > 0;\n        const deployments = this.staticAssetConfigs.filter(asset => existsSync(asset.source)).map((asset, assetIndex) => {\n            const isInvalidationCarrier = asset.pattern === NuxtServerAppStack.CONSUMER_INVALIDATION_CARRIER_PATTERN;\n            const attachConsumerInvalidations = isInvalidationCarrier && hasConsumerInvalidations;\n\n            const deployment = new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {\n                sources: [Source.asset(asset.source, {\n                    exclude: asset.exclude,\n                })],\n                destinationBucket: this.staticAssetsBucket,\n                destinationKeyPrefix: asset.target.replace(/^\\/+/g, ''), // Remove leading slash\n                prune: false,\n                storageClass: StorageClass.STANDARD,\n                exclude: ['*'],\n                include: [asset.pattern],\n                cacheControl: asset.cacheControl,\n                contentType: asset.contentType,\n                distribution: attachConsumerInvalidations ? this.cdn : undefined,\n                distributionPaths: attachConsumerInvalidations ? this.deployInvalidationPaths : undefined,\n                logGroup: logGroup,\n\n                metadata: {\n                    // Store build revision on every asset to allow cleanup of outdated assets\n                    revision: this.deploymentRevision,\n                },\n\n                // Some Nuxt applications have a lot of assets to deploy whereby the function might run out of memory.\n                // Additionally, a high memory limit might speed up deployments.\n                memoryLimit: 1792\n            });\n\n            return {asset, deployment};\n        });\n\n        // Patterns that must be uploaded AFTER the new Lambda + hashed assets are in place.\n        // These are non-hashed \"switchover\" files that flip clients onto the new build:\n        //   - latest.json: Nuxt's outdated-build manifest pointer\n        //   - sw.js / registerSW.js: PWA service worker bootstrap; once a client picks up the\n        //     new sw.js it will try to precache the new chunks, which must already exist\n        const postLambdaPatterns = new Set([\n            '_nuxt/builds/latest.json',\n            'sw.js',\n            'registerSW.js',\n        ]);\n\n        const postLambdaDeployments = deployments.filter(({asset}) => postLambdaPatterns.has(asset.pattern));\n        const preLambdaDeployments = deployments\n            .filter(({asset}) => !postLambdaPatterns.has(asset.pattern))\n            .map(({deployment}) => deployment);\n\n        preLambdaDeployments.forEach(deployment => {\n            this.appLambdaFunction.node.addDependency(deployment);\n        });\n\n        postLambdaDeployments.forEach(({deployment}) => {\n            preLambdaDeployments.forEach(preLambda => {\n                deployment.node.addDependency(preLambda);\n            });\n            deployment.node.addDependency(this.appLambdaFunction);\n        });\n\n        return deployments.map(({deployment}) => deployment);\n    }\n\n    /**\n     * Resolves the hosted zone at which the DNS records shall be created to access the Nuxt app on the internet.\n     *\n     * @param props\n     * @private\n     */\n    private findHostedZone(props: NuxtServerAppStackProps): IHostedZone {\n        const domainParts = props.domain.split('.');\n\n        return HostedZone.fromHostedZoneAttributes(this, `${this.resourceIdPrefix}-hosted-zone`, {\n            hostedZoneId: props.hostedZoneId,\n            zoneName: domainParts[domainParts.length - 1], // Support subdomains\n        });\n    }\n\n    /**\n     * Creates the DNS records to access the Nuxt app on the internet via the custom domain.\n     *\n     * @param props\n     * @private\n     */\n    private createDnsRecords(props: NuxtServerAppStackProps): void {\n        const hostedZone = this.findHostedZone(props);\n        const dnsTarget = RecordTarget.fromAlias(new CloudFrontTarget(this.cdn));\n\n        // Create a record for IPv4\n        new ARecord(this, `${this.resourceIdPrefix}-ipv4-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n\n        // Create a record for IPv6\n        new AaaaRecord(this, `${this.resourceIdPrefix}-ipv6-record`, {\n            recordName: props.domain,\n            zone: hostedZone,\n            target: dnsTarget,\n        });\n    }\n\n    /**\n     * Creates a scheduled rule to ping the Nuxt app Lambda function every 5 minutes in order to keep it warm\n     * and speed up initial SSR requests.\n     *\n     * @private\n     */\n    private createAppPingRule(props: NuxtServerAppStackProps): void {\n        const fakeApiGatewayEventData = {\n            \"version\": \"2.0\",\n            \"routeKey\": \"GET /{proxy+}\",\n            \"rawPath\": \"/\",\n            \"rawQueryString\": \"\",\n            \"headers\": {},\n            \"requestContext\": {\n                \"domainName\": props.domain,\n                \"http\": {\n                    \"method\": \"GET\",\n                    \"path\": \"/\",\n                    \"protocol\": \"HTTP/1.1\"\n                }\n            }\n        };\n\n        new Rule(this, `${this.resourceIdPrefix}-pinger-rule`, {\n            ruleName: `${this.resourceIdPrefix}-pinger`,\n            description: `Pings the Lambda function of the ${this.resourceIdPrefix} app every 5 minutes to keep it warm.`,\n            enabled: true,\n            schedule: Schedule.rate(Duration.minutes(5)),\n            targets: [new LambdaFunction(this.appLambdaFunction, {\n                event: RuleTargetInput.fromObject(fakeApiGatewayEventData)\n            })],\n        });\n    }\n\n\n    /**\n     * Creates a scheduled rule that runs every Tuesday at 03:30 AM GMT to trigger\n     * our cleanup Lambda function.\n     *\n     * @private\n     */\n    private createCleanupTriggerRule(): void {\n        new Rule(this, `${this.resourceIdPrefix}-scheduler-rule`, {\n            ruleName: `${this.resourceIdPrefix}-scheduler`,\n            description: `Triggers a cleanup of the outdated static assets at the ${this.staticAssetsBucket.bucketName} S3 bucket.`,\n            enabled: true,\n            schedule: Schedule.cron({weekDay: '2', hour: '3', minute: '30'}),\n            targets: [new LambdaFunction(this.cleanupLambdaFunction)],\n        });\n    }\n\n    /**\n     * Creates a S3 bucket to store the access logs of the CloudFront distribution.\n     */\n    private createAccessLogsBucket(): Bucket {\n        const bucketName = `${this.resourceIdPrefix}-access-logs`;\n        const bucket = new Bucket(this, bucketName, {\n            bucketName,\n            blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n            objectOwnership: ObjectOwnership.BUCKET_OWNER_PREFERRED,\n            // When the stack is destroyed, we expect everything to be deleted\n            removalPolicy: RemovalPolicy.DESTROY,\n            autoDeleteObjects: true,\n        });\n\n        bucket.grantReadWrite(this.cdnAccessIdentity);\n\n        return bucket;\n    }\n\n\n    private createAccessLogsAnalysis(props: NuxtServerAppStackProps): void {\n        if (!this.accessLogsBucket) {\n            throw new Error('Access bucket not set');\n        }\n\n        const { CloudFrontAccessLogsAnalysis } = require('../access-logs-analysis/CloudFrontAccessLogsAnalysis');\n\n        new CloudFrontAccessLogsAnalysis(this, `${this.resourceIdPrefix}-access-logs-analysis`, {\n            bucket: this.accessLogsBucket,\n            resourcePrefix: `${this.resourceIdPrefix}-access-logs`,\n            accessLogCookies: props.accessLogCookies,\n            anonymizeClientIp: props.anonymizeAccessLogClientIp ?? true,\n            expireRawLogsAfter: props.accessLogsRawRetention,\n            expireIntermediateLogsAfter: props.accessLogsIntermediateRetention,\n            expireTransformedLogsAfter: props.accessLogsTransformedRetention,\n        });\n    }\n}\n"]}
@@ -46,7 +46,18 @@ import {DomainName, EndpointType, HttpApi, HttpMethod, SecurityPolicy} from "aws
46
46
  * CDK stack to deploy a dynamic Nuxt app (target=server) on AWS with Lambda, ApiGateway, S3 and CloudFront.
47
47
  */
48
48
  export class NuxtServerAppStack extends Stack {
49
- private static readonly LATEST_BUILD_MANIFEST_INVALIDATION_PATH = '/_nuxt/builds/latest.json';
49
+ /**
50
+ * The static asset pattern whose post-Lambda {@link BucketDeployment} acts as the carrier for
51
+ * consumer-supplied CloudFront invalidations (via {@link NuxtServerAppStackProps.invalidatePathsOnDeploy}).
52
+ *
53
+ * We attach {@link BucketDeployment.distributionPaths} to this specific deployment so that:
54
+ * - CDK reuses its built-in CloudFront invalidation mechanism (no new construct types),
55
+ * - the invalidation fires AFTER the new Lambda and hashed assets are in place, since this
56
+ * deployment is already part of the post-Lambda deployment phase in {@link configureDeployments},
57
+ * - the files managed by {@link StaticAssetConfig} entries themselves are never invalidated
58
+ * (they are either build-hashed immutable assets or served with `s-maxage=0, must-revalidate`).
59
+ */
60
+ private static readonly CONSUMER_INVALIDATION_CARRIER_PATTERN = '_nuxt/builds/latest.json';
50
61
 
51
62
  /**
52
63
  * The identifier prefix of the resources created by the stack.
@@ -672,14 +683,15 @@ export class NuxtServerAppStack extends Stack {
672
683
  removalPolicy: RemovalPolicy.DESTROY,
673
684
  });
674
685
 
675
- // Returns a deployment for every configured static asset type to respect the different cache settings
686
+ // Returns a deployment for every configured static asset type to respect the different cache settings.
687
+ // Consumer-supplied CloudFront invalidation paths (props.invalidatePathsOnDeploy) are attached to a
688
+ // single, specific deployment — the "carrier" identified by CONSUMER_INVALIDATION_CARRIER_PATTERN —
689
+ // so the invalidation piggybacks on CDK's existing BucketDeployment invalidation mechanism while
690
+ // firing at the correct point in the deployment phase (see the post-Lambda logic below).
691
+ const hasConsumerInvalidations = this.deployInvalidationPaths.length > 0;
676
692
  const deployments = this.staticAssetConfigs.filter(asset => existsSync(asset.source)).map((asset, assetIndex) => {
677
- const distributionPaths = asset.invalidateOnChange
678
- ? this.normalizeInvalidationPaths([
679
- NuxtServerAppStack.LATEST_BUILD_MANIFEST_INVALIDATION_PATH,
680
- ...this.deployInvalidationPaths,
681
- ])
682
- : undefined;
693
+ const isInvalidationCarrier = asset.pattern === NuxtServerAppStack.CONSUMER_INVALIDATION_CARRIER_PATTERN;
694
+ const attachConsumerInvalidations = isInvalidationCarrier && hasConsumerInvalidations;
683
695
 
684
696
  const deployment = new BucketDeployment(this, `${this.resourceIdPrefix}-assets-deployment-${assetIndex}`, {
685
697
  sources: [Source.asset(asset.source, {
@@ -693,8 +705,8 @@ export class NuxtServerAppStack extends Stack {
693
705
  include: [asset.pattern],
694
706
  cacheControl: asset.cacheControl,
695
707
  contentType: asset.contentType,
696
- distribution: asset.invalidateOnChange ? this.cdn : undefined,
697
- distributionPaths: distributionPaths,
708
+ distribution: attachConsumerInvalidations ? this.cdn : undefined,
709
+ distributionPaths: attachConsumerInvalidations ? this.deployInvalidationPaths : undefined,
698
710
  logGroup: logGroup,
699
711
 
700
712
  metadata: {
@@ -710,21 +722,32 @@ export class NuxtServerAppStack extends Stack {
710
722
  return {asset, deployment};
711
723
  });
712
724
 
713
- const latestBuildDeployment = deployments.find(({asset}) => asset.pattern === '_nuxt/builds/latest.json')?.deployment;
725
+ // Patterns that must be uploaded AFTER the new Lambda + hashed assets are in place.
726
+ // These are non-hashed "switchover" files that flip clients onto the new build:
727
+ // - latest.json: Nuxt's outdated-build manifest pointer
728
+ // - sw.js / registerSW.js: PWA service worker bootstrap; once a client picks up the
729
+ // new sw.js it will try to precache the new chunks, which must already exist
730
+ const postLambdaPatterns = new Set([
731
+ '_nuxt/builds/latest.json',
732
+ 'sw.js',
733
+ 'registerSW.js',
734
+ ]);
735
+
736
+ const postLambdaDeployments = deployments.filter(({asset}) => postLambdaPatterns.has(asset.pattern));
714
737
  const preLambdaDeployments = deployments
715
- .filter(({asset}) => asset.pattern !== '_nuxt/builds/latest.json')
738
+ .filter(({asset}) => !postLambdaPatterns.has(asset.pattern))
716
739
  .map(({deployment}) => deployment);
717
740
 
718
741
  preLambdaDeployments.forEach(deployment => {
719
742
  this.appLambdaFunction.node.addDependency(deployment);
720
743
  });
721
744
 
722
- if (latestBuildDeployment) {
723
- preLambdaDeployments.forEach(deployment => {
724
- latestBuildDeployment.node.addDependency(deployment);
745
+ postLambdaDeployments.forEach(({deployment}) => {
746
+ preLambdaDeployments.forEach(preLambda => {
747
+ deployment.node.addDependency(preLambda);
725
748
  });
726
- latestBuildDeployment.node.addDependency(this.appLambdaFunction);
727
- }
749
+ deployment.node.addDependency(this.appLambdaFunction);
750
+ });
728
751
 
729
752
  return deployments.map(({deployment}) => deployment);
730
753
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cdk-nuxt",
3
- "version": "2.26.0",
3
+ "version": "2.26.1",
4
4
  "license": "MIT",
5
5
  "repository": {
6
6
  "type": "git",