cdk-nuxt 2.24.0 → 2.25.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -171,10 +171,11 @@ export declare class NuxtServerAppStack extends Stack {
171
171
  * Creates behaviors for the CloudFront distribution based on the consumer-supplied {@link NuxtCloudFrontBehavior} list.
172
172
  * Each behavior can override the cache policy, attach a CloudFront Function, or both.
173
173
  *
174
+ * When {@link NuxtCloudFrontBehavior.fnCode} is provided, this method creates the `cloudfront.Function`
175
+ * resource inside the stack itself so that it is always correctly scoped.
176
+ *
174
177
  * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the
175
178
  * default Nuxt app cache policy ({@link appCachePolicy}) is used.
176
- * The provided {@link NuxtCloudFrontBehavior.fn} is attached as a function association when specified;
177
- * otherwise no function association is added.
178
179
  */
179
180
  private createAdditionalBehaviors;
180
181
  /**
@@ -121,7 +121,7 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
121
121
  * @private
122
122
  */
123
123
  createAppLambdaFunction(props) {
124
- var _a, _b, _c, _d;
124
+ var _a, _b, _c, _d, _e;
125
125
  const funcName = `${this.resourceIdPrefix}-nuxt`;
126
126
  const appLogGroup = new aws_logs_1.LogGroup(this, `${funcName}-logs`, {
127
127
  logGroupName: `/aws/lambda/${funcName}`,
@@ -137,14 +137,14 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
137
137
  code: aws_lambda_1.Code.fromAsset(`${(_b = props.rootDir) !== null && _b !== void 0 ? _b : '.'}/.output/server`, {
138
138
  exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],
139
139
  }),
140
- timeout: aws_cdk_lib_1.Duration.seconds(10),
141
- memorySize: (_c = props.memorySize) !== null && _c !== void 0 ? _c : 1792,
140
+ timeout: (_c = props.timeout) !== null && _c !== void 0 ? _c : aws_cdk_lib_1.Duration.seconds(10),
141
+ memorySize: (_d = props.memorySize) !== null && _d !== void 0 ? _d : 1792,
142
142
  allowPublicSubnet: false,
143
143
  tracing: props.enableTracing ? aws_lambda_1.Tracing.ACTIVE : aws_lambda_1.Tracing.DISABLED,
144
144
  logGroup: appLogGroup,
145
145
  environment: {
146
146
  NODE_OPTIONS: '--enable-source-maps',
147
- ...JSON.parse((_d = props.entrypointEnv) !== null && _d !== void 0 ? _d : '{}'),
147
+ ...JSON.parse((_e = props.entrypointEnv) !== null && _e !== void 0 ? _e : '{}'),
148
148
  },
149
149
  });
150
150
  }
@@ -372,21 +372,27 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
372
372
  * Creates behaviors for the CloudFront distribution based on the consumer-supplied {@link NuxtCloudFrontBehavior} list.
373
373
  * Each behavior can override the cache policy, attach a CloudFront Function, or both.
374
374
  *
375
+ * When {@link NuxtCloudFrontBehavior.fnCode} is provided, this method creates the `cloudfront.Function`
376
+ * resource inside the stack itself so that it is always correctly scoped.
377
+ *
375
378
  * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the
376
379
  * default Nuxt app cache policy ({@link appCachePolicy}) is used.
377
- * The provided {@link NuxtCloudFrontBehavior.fn} is attached as a function association when specified;
378
- * otherwise no function association is added.
379
380
  */
380
381
  createAdditionalBehaviors(behaviors) {
381
382
  const rules = {};
382
- behaviors.forEach(behavior => {
383
+ behaviors.forEach((behavior, index) => {
383
384
  var _a, _b;
384
- const functionAssociations = behavior.fn
385
- ? [{
386
- function: behavior.fn,
385
+ let functionAssociations = [];
386
+ if (behavior.fnCode) {
387
+ // Create the CloudFront Function inside the stack so it is always correctly scoped.
388
+ const cfFunction = new aws_cloudfront_1.Function(this, `${this.resourceIdPrefix}-behavior-fn-${index}`, {
389
+ code: behavior.fnCode,
390
+ });
391
+ functionAssociations = [{
392
+ function: cfFunction,
387
393
  eventType: (_a = behavior.eventType) !== null && _a !== void 0 ? _a : aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST,
388
- }]
389
- : [];
394
+ }];
395
+ }
390
396
  rules[behavior.pathPattern] = {
391
397
  ...this.nuxtServerRouteBehavior,
392
398
  cachePolicy: (_b = behavior.cachePolicy) !== null && _b !== void 0 ? _b : this.appCachePolicy,
@@ -604,4 +610,4 @@ class NuxtServerAppStack extends aws_cdk_lib_1.Stack {
604
610
  }
605
611
  }
606
612
  exports.NuxtServerAppStack = NuxtServerAppStack;
607
- //# 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} 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    /**\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     * 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.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: 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     * 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     * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the\n     * default Nuxt app cache policy ({@link appCachePolicy}) is used.\n     * The provided {@link NuxtCloudFrontBehavior.fn} is attached as a function association when specified;\n     * otherwise no function association is added.\n     */\n    private createAdditionalBehaviors(behaviors: NuxtCloudFrontBehavior[]): Record<string, BehaviorOptions> {\n        const rules: Record<string, BehaviorOptions> = {};\n\n        behaviors.forEach(behavior => {\n            const functionAssociations: FunctionAssociation[] = behavior.fn\n                ? [{\n                    function: behavior.fn,\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        return this.staticAssetConfigs.filter(asset => existsSync(asset.source)).map((asset, assetIndex) => {\n            return 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: asset.invalidateOnChange ? [`/${asset.pattern}`] : 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    }\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"]}
613
+ //# 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    /**\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     * 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.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     * 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        return this.staticAssetConfigs.filter(asset => existsSync(asset.source)).map((asset, assetIndex) => {\n            return 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: asset.invalidateOnChange ? [`/${asset.pattern}`] : 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    }\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"]}
@@ -15,8 +15,9 @@ import {
15
15
  OriginProtocolPolicy, OriginRequestPolicy,
16
16
  PriceClass,
17
17
  SecurityPolicyProtocol,
18
- ViewerProtocolPolicy,OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior,
18
+ ViewerProtocolPolicy, OriginRequestCookieBehavior, OriginRequestHeaderBehavior, OriginRequestQueryStringBehavior,
19
19
  FunctionEventType, type FunctionAssociation,
20
+ Function as CloudFrontFunction,
20
21
  } from "aws-cdk-lib/aws-cloudfront";
21
22
  import {Architecture, Code, Function, Runtime, Tracing} from "aws-cdk-lib/aws-lambda";
22
23
  import {
@@ -275,7 +276,7 @@ export class NuxtServerAppStack extends Stack {
275
276
  code: Code.fromAsset(`${props.rootDir ?? '.' }/.output/server`, {
276
277
  exclude: ['**.svg', '**.ico', '**.png', '**.jpg', '**.js.map'],
277
278
  }),
278
- timeout: Duration.seconds(10),
279
+ timeout: props.timeout ?? Duration.seconds(10),
279
280
  memorySize: props.memorySize ?? 1792,
280
281
  allowPublicSubnet: false,
281
282
  tracing: props.enableTracing ? Tracing.ACTIVE : Tracing.DISABLED,
@@ -539,21 +540,28 @@ export class NuxtServerAppStack extends Stack {
539
540
  * Creates behaviors for the CloudFront distribution based on the consumer-supplied {@link NuxtCloudFrontBehavior} list.
540
541
  * Each behavior can override the cache policy, attach a CloudFront Function, or both.
541
542
  *
543
+ * When {@link NuxtCloudFrontBehavior.fnCode} is provided, this method creates the `cloudfront.Function`
544
+ * resource inside the stack itself so that it is always correctly scoped.
545
+ *
542
546
  * The provided {@link NuxtCloudFrontBehavior.cachePolicy} is used when specified; otherwise the
543
547
  * default Nuxt app cache policy ({@link appCachePolicy}) is used.
544
- * The provided {@link NuxtCloudFrontBehavior.fn} is attached as a function association when specified;
545
- * otherwise no function association is added.
546
548
  */
547
549
  private createAdditionalBehaviors(behaviors: NuxtCloudFrontBehavior[]): Record<string, BehaviorOptions> {
548
550
  const rules: Record<string, BehaviorOptions> = {};
549
551
 
550
- behaviors.forEach(behavior => {
551
- const functionAssociations: FunctionAssociation[] = behavior.fn
552
- ? [{
553
- function: behavior.fn,
552
+ behaviors.forEach((behavior, index) => {
553
+ let functionAssociations: FunctionAssociation[] = [];
554
+
555
+ if (behavior.fnCode) {
556
+ // Create the CloudFront Function inside the stack so it is always correctly scoped.
557
+ const cfFunction = new CloudFrontFunction(this, `${this.resourceIdPrefix}-behavior-fn-${index}`, {
558
+ code: behavior.fnCode,
559
+ });
560
+ functionAssociations = [{
561
+ function: cfFunction,
554
562
  eventType: behavior.eventType ?? FunctionEventType.VIEWER_REQUEST,
555
- }]
556
- : [];
563
+ }];
564
+ }
557
565
 
558
566
  rules[behavior.pathPattern] = {
559
567
  ...this.nuxtServerRouteBehavior,
@@ -1,6 +1,6 @@
1
1
  import { type NuxtAppStackProps } from "../NuxtAppStackProps";
2
2
  import { Duration } from "aws-cdk-lib";
3
- import { Function as CloudFrontFunction, FunctionEventType, ICachePolicy } from "aws-cdk-lib/aws-cloudfront";
3
+ import { FunctionCode, FunctionEventType, ICachePolicy } from "aws-cdk-lib/aws-cloudfront";
4
4
  /**
5
5
  * Defines a custom CloudFront behavior attached to a specific path pattern targeting the Nuxt app origin.
6
6
  * Use this to override the cache policy and/or attach a CloudFront Function for specific routes.
@@ -9,18 +9,32 @@ export interface NuxtCloudFrontBehavior {
9
9
  /**
10
10
  * The CloudFront distribution behavior path pattern to which this behavior shall be applied.
11
11
  *
12
- * @example '/users/*'
12
+ * @example '/posts/*'
13
13
  */
14
14
  readonly pathPattern: string;
15
15
  /**
16
- * An optional pre-instantiated CloudFront Function to associate with this behavior.
16
+ * The code of the CloudFront Function to associate with this behavior.
17
+ * The stack will create the CloudFront Function resource internally.
17
18
  * The function will be invoked on VIEWER_REQUEST events by default.
18
19
  * If omitted, no CloudFront Function is attached.
20
+ *
21
+ * @example
22
+ * ```typescript
23
+ * import { FunctionCode } from 'aws-cdk-lib/aws-cloudfront';
24
+ *
25
+ * fnCode: FunctionCode.fromInline(`
26
+ * function handler(event) {
27
+ * var request = event.request;
28
+ * request.uri = request.uri.replace(/\/+$/, '') || '/';
29
+ * return request;
30
+ * }
31
+ * `)
32
+ * ```
19
33
  */
20
- readonly fn?: CloudFrontFunction;
34
+ readonly fnCode?: FunctionCode;
21
35
  /**
22
36
  * The event type at which the CloudFront Function shall be invoked.
23
- * Only relevant when {@link fn} is specified.
37
+ * Only relevant when {@link fnCode} is specified.
24
38
  * Defaults to {@link FunctionEventType.VIEWER_REQUEST}.
25
39
  */
26
40
  readonly eventType?: FunctionEventType;
@@ -54,6 +68,14 @@ export interface NuxtServerAppStackProps extends NuxtAppStackProps {
54
68
  * Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).
55
69
  */
56
70
  readonly memorySize?: number;
71
+ /**
72
+ * The maximum execution duration for the Nuxt app's Lambda function.
73
+ * Increase this if your app performs long-running SSR operations (e.g. large data fetching).
74
+ * Note: The CloudFront read timeout is capped at 60 seconds, so values above 60 seconds
75
+ * only benefit direct invocations, not requests arriving via CloudFront.
76
+ * Defaults to {@link Duration.seconds(10)}.
77
+ */
78
+ readonly timeout?: Duration;
57
79
  /**
58
80
  * Whether to enable AWS X-Ray for the Nuxt Lambda function.
59
81
  */
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStackProps.js","sourceRoot":"","sources":["NuxtServerAppStackProps.ts"],"names":[],"mappings":"","sourcesContent":["import {type NuxtAppStackProps} from \"../NuxtAppStackProps\";\nimport {Duration} from \"aws-cdk-lib\";\nimport {Function as CloudFrontFunction, FunctionEventType, ICachePolicy} from \"aws-cdk-lib/aws-cloudfront\";\n\n/**\n * Defines a custom CloudFront behavior attached to a specific path pattern targeting the Nuxt app origin.\n * Use this to override the cache policy and/or attach a CloudFront Function for specific routes.\n */\nexport interface NuxtCloudFrontBehavior {\n\n    /**\n     * The CloudFront distribution behavior path pattern to which this behavior shall be applied.\n     *\n     * @example '/users/*'\n     */\n    readonly pathPattern: string;\n\n    /**\n     * An optional pre-instantiated CloudFront Function to associate with this behavior.\n     * The function will be invoked on VIEWER_REQUEST events by default.\n     * If omitted, no CloudFront Function is attached.\n     */\n    readonly fn?: CloudFrontFunction;\n\n    /**\n     * The event type at which the CloudFront Function shall be invoked.\n     * Only relevant when {@link fn} is specified.\n     * Defaults to {@link FunctionEventType.VIEWER_REQUEST}.\n     */\n    readonly eventType?: FunctionEventType;\n\n    /**\n     * An optional cache policy to use for this behavior.\n     * Falls back to the default Nuxt app cache policy if not specified.\n     */\n    readonly cachePolicy?: ICachePolicy;\n}\n\n/**\n * Defines the props required for the {@see NuxtServerAppStack}.\n */\nexport interface NuxtServerAppStackProps extends NuxtAppStackProps {\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The file name (without extension) of the Lambda entrypoint within the 'server' directory exporting a handler.\n     * Defaults to \"index\".\n     */\n    readonly entrypoint?: string;\n\n    /**\n     * A JSON serialized string of environment variables to pass to the Lambda function.\n     */\n    readonly entrypointEnv?: string;\n\n    /**\n     * The memory size to apply to the Nuxt app's Lambda.\n     * Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).\n     */\n    readonly memorySize?: number;\n\n    /**\n     * Whether to enable AWS X-Ray for the Nuxt Lambda function.\n     */\n    readonly enableTracing?: boolean;\n\n    /**\n     * Whether to enable a global Sitemap bucket which is permanently accessible through multiple deployments.\n     */\n    readonly enableSitemap?: boolean;\n\n    /**\n     * Whether to enable (HTTPS only) API access to the Nuxt app via the `/api` path which support all HTTP methods.\n     * See https://nuxt.com/docs/guide/directory-structure/server#recipes for details.\n     */\n    readonly enableApi?: boolean;\n\n    /**\n     * Whether to enable reporting of CloudFront access logs via Athena.\n     */\n    readonly enableAccessLogsAnalysis?: boolean;\n\n    /**\n     * Array of cookie names to include in the access logs (whitelist).\n     */\n    readonly accessLogCookies?: string[];\n\n    /**\n     * Whether to anonymize the client IP address in the access logs by replacing the last octet (IPv4)\n     * or the last group (IPv6) with 'xxx'.\n     *\n     * **DSGVO/Legal note:** IP addresses are considered personal data under the GDPR (cf. CJEU judgment C‑582/14).\n     * If you set this to `false`, you must ensure a legal basis under Art. 6 GDPR (e.g. legitimate interest),\n     * document it in your privacy policy, and limit the retention period to what is strictly necessary.\n     * When in doubt, consult a data protection officer or legal counsel before disabling this option.\n     *\n     * Defaults to `true`.\n     */\n    readonly anonymizeAccessLogClientIp?: boolean;\n\n    /**\n     * The duration after which raw (unprocessed) access logs are deleted from S3.\n     * Defaults to 7 days.\n     */\n    readonly accessLogsRawRetention?: Duration;\n\n    /**\n     * The duration after which intermediate (grouped-by-date) access logs are deleted from S3.\n     * Defaults to 7 days.\n     */\n    readonly accessLogsIntermediateRetention?: Duration;\n\n    /**\n     * The duration after which transformed (Parquet) access logs are deleted from S3.\n     * Defaults to 180 days.\n     */\n    readonly accessLogsTransformedRetention?: Duration;\n\n    /**\n     * The number of days to retain static assets of outdated deployments in the S3 bucket.\n     * Useful to allow users to still access old assets after a new deployment when they are still browsing on an old version.\n     * Defaults to 30 days.\n     */\n    readonly outdatedAssetsRetentionDays?: number;\n\n    /**\n     * An array of HTTP headers to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for headers that do not affect the response.\n     *\n     * Only the Cloudfront default headers are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardHeaders?: string[];\n\n    /**\n     * An array of HTTP headers to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for headers that might affect the response, e.g., 'Authorization'.\n     *\n     * Only the Cloudfront default headers are forwarded,\n     * but no headers are included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyHeaders?: string[];\n\n    /**\n     * An array of cookies to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for cookies that do not affect the response.\n     *\n     * No cookies are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardCookies?: string[];\n\n    /**\n     * An array of cookies to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for cookies that might affect the response, e.g., authentication cookies.\n     *\n     * No cookies are forwarded or included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyCookies?: string[];\n\n    /**\n     * An array of query params to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for query params that do not affect the response and are required on SSR requests.\n     *\n     * All query params are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardQueryParams?: string[];\n\n    /**\n     * An array of query params to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for query params that affect the response and are required on SSR requests, e.g., filters.\n     *\n     * All query params are forwarded and included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyQueryParams?: string[];\n\n    /**\n     * An array of query params to prevent forwarding to the Nuxt app and to not include in the cache key for objects that are cached at CloudFront edge locations.\n     * When set, all query params that are not specified in this array will be forwarded to the Nuxt app and included in the cache key.\n     * This should be used for query params that do not affect the response and are not required on SSR requests, e.g., 'fbclid' or 'utm_campaign'.\n     *\n     * If both {@see cacheKeyQueryParams} and {@see denyCacheKeyQueryParams} are specified, the {@see denyCacheKeyQueryParams} will be ignored.\n     * All query params are forwarded and included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly denyCacheKeyQueryParams?: string[];\n\n    /**\n     * An array of headers to pass to the Nuxt app on SSR requests.\n     * The more headers are passed, the weaker the cache performance will be, as the cache key\n     * is based on the headers.\n     * No headers are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyHeaders} instead.\n     */\n    readonly allowHeaders?: string[];\n\n    /**\n     * An array of cookies to pass to the Nuxt app on SSR requests.\n     * The more cookies are passed, the weaker the cache performance will be, as the cache key\n     * is based on the cookies.\n     * No cookies are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyCookies} instead.\n     */\n    readonly allowCookies?: string[];\n\n    /**\n     * An array of query param keys to pass to the Nuxt app on SSR requests.\n     * The more query params are passed, the weaker the cache performance will be, as the cache key\n     * is based on the query params.\n     * Note that this config can not be combined with {@see denyQueryParams}.\n     * If both are specified, the {@see denyQueryParams} will be ignored.\n     * All query params are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyQueryParams} instead.\n     */\n    readonly allowQueryParams?: string[];\n\n    /**\n     * An array of query param keys to deny passing to the Nuxt app on SSR requests.\n     * It might be useful to prevent specific external query params, e.g., fbclid, utm_campaign, ...,\n     * to improve cache performance, as the cache key is based on the specified query params.\n     * Note that this config can not be combined with {@see allowQueryParams}.\n     * If both are specified, the {@see denyQueryParams} will be ignored.\n     * All query params are passed by default.\n     *\n     * @deprecated Use {@see denyCacheKeyQueryParams} instead.\n     */\n    readonly denyQueryParams?: string[];\n\n    /**\n     * An array of path patterns for server endpoints that should be routed to the SSR origin (API Gateway → Lambda)\n     * instead of the default S3 \"file\" behavior.\n     * \n     * This is useful for server routes that generate dynamic content but use file-like URLs.\n     * For example, `@nuxtjs/sitemap` creates a `/sitemap.xml` endpoint that dynamically generates XML content,\n     * and `@nuxt/image` uses file-like URLs to serve dynamically processed images.\n     * \n     * Note: This is different from `enableSitemap` which serves pre-generated static sitemap files from S3.\n     * Use `serverRoutes` when you need the Lambda to handle requests and generate content on-the-fly.\n     * \n     * Examples: `['/sitemap.xml', '/robots.txt', '/__sitemap__/*', '/_ipx/*']`\n     */\n    readonly serverRoutes?: string[];\n\n    /**\n     * An array of custom CloudFront behaviors to inject into the CloudFront distribution for specific path patterns targeting the Nuxt app origin.\n     * Use this to override the cache policy and/or attach a CloudFront Function for specific routes.\n     *\n     * Each entry can independently define:\n     * - A custom cache policy (e.g., to disable caching for certain paths)\n     * - A CloudFront Function (e.g., for URI normalization at the edge)\n     * - Or both combined\n     *\n     * The behaviors are added to the distribution **before** the static-assets behaviors and in the order\n     * provided, so more specific patterns should be listed first.\n     *\n     * **Example – cache policy only:**\n     * ```typescript\n     * import { CachePolicy } from 'aws-cdk-lib/aws-cloudfront';\n     *\n     * additionalBehaviors: [\n     *   { pathPattern: '/api/realtime/*', cachePolicy: CachePolicy.CACHING_DISABLED },\n     * ]\n     * ```\n     *\n     * **Example – CloudFront Function only:**\n     * ```typescript\n     * import { Function, FunctionCode } from 'aws-cdk-lib/aws-cloudfront';\n     *\n     * const normalizeUri = new Function(this, 'NormalizeUri', {\n     *   code: FunctionCode.fromInline(`\n     *     function handler(event) {\n     *       var request = event.request;\n     *       request.uri = request.uri.replace(/\\\\/+$/, '') || '/';\n     *       return request;\n     *     }\n     *   `),\n     * });\n     *\n     * additionalBehaviors: [\n     *   { pathPattern: '/users/*', fn: normalizeUri },\n     * ]\n     * ```\n     *\n     * **Example – both combined:**\n     * ```typescript\n     * additionalBehaviors: [\n     *   { pathPattern: '/posts/*', fn: normalizeUri, cachePolicy: CachePolicy.CACHING_DISABLED },\n     * ]\n     * ```\n     */\n    readonly additionalBehaviors?: NuxtCloudFrontBehavior[];\n\n    /**\n     * The ARN of an existing AWS WAF Web ACL to associate with the CloudFront distribution.\n     * This should be used with a separate CloudFrontWafStack deployed in us-east-1.\n     *\n     * Example: 'arn:aws:wafv2:us-east-1:123456789012:global/webacl/my-web-acl/a1b2c3d4-5678-90ab-cdef-EXAMPLE11111'\n     */\n    readonly webAclArn?: string;\n}"]}
3
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NuxtServerAppStackProps.js","sourceRoot":"","sources":["NuxtServerAppStackProps.ts"],"names":[],"mappings":"","sourcesContent":["import {type NuxtAppStackProps} from \"../NuxtAppStackProps\";\nimport {Duration} from \"aws-cdk-lib\";\nimport {FunctionCode, FunctionEventType, ICachePolicy} from \"aws-cdk-lib/aws-cloudfront\";\n\n/**\n * Defines a custom CloudFront behavior attached to a specific path pattern targeting the Nuxt app origin.\n * Use this to override the cache policy and/or attach a CloudFront Function for specific routes.\n */\nexport interface NuxtCloudFrontBehavior {\n\n    /**\n     * The CloudFront distribution behavior path pattern to which this behavior shall be applied.\n     *\n     * @example '/posts/*'\n     */\n    readonly pathPattern: string;\n\n    /**\n     * The code of the CloudFront Function to associate with this behavior.\n     * The stack will create the CloudFront Function resource internally.\n     * The function will be invoked on VIEWER_REQUEST events by default.\n     * If omitted, no CloudFront Function is attached.\n     *\n     * @example\n     * ```typescript\n     * import { FunctionCode } from 'aws-cdk-lib/aws-cloudfront';\n     *\n     * fnCode: FunctionCode.fromInline(`\n     *   function handler(event) {\n     *     var request = event.request;\n     *     request.uri = request.uri.replace(/\\/+$/, '') || '/';\n     *     return request;\n     *   }\n     * `)\n     * ```\n     */\n    readonly fnCode?: FunctionCode;\n\n    /**\n     * The event type at which the CloudFront Function shall be invoked.\n     * Only relevant when {@link fnCode} is specified.\n     * Defaults to {@link FunctionEventType.VIEWER_REQUEST}.\n     */\n    readonly eventType?: FunctionEventType;\n\n    /**\n     * An optional cache policy to use for this behavior.\n     * Falls back to the default Nuxt app cache policy if not specified.\n     */\n    readonly cachePolicy?: ICachePolicy;\n}\n\n/**\n * Defines the props required for the {@see NuxtServerAppStack}.\n */\nexport interface NuxtServerAppStackProps extends NuxtAppStackProps {\n\n    /**\n     * The ARN of the certificate to use at the ApiGateway for the Nuxt app to make it accessible via the custom domain\n     * and to provide the custom domain to the Nuxt app via the 'Host' header for server side rendering use cases.\n     * The certificate must be issued in the same region as specified via 'env.region' as ApiGateway works regionally.\n     */\n    readonly regionalTlsCertificateArn: string;\n\n    /**\n     * The file name (without extension) of the Lambda entrypoint within the 'server' directory exporting a handler.\n     * Defaults to \"index\".\n     */\n    readonly entrypoint?: string;\n\n    /**\n     * A JSON serialized string of environment variables to pass to the Lambda function.\n     */\n    readonly entrypointEnv?: string;\n\n    /**\n     * The memory size to apply to the Nuxt app's Lambda.\n     * Defaults to 1792MB (optimized for costs and performance for standard Nuxt apps).\n     */\n    readonly memorySize?: number;\n\n    /**\n     * The maximum execution duration for the Nuxt app's Lambda function.\n     * Increase this if your app performs long-running SSR operations (e.g. large data fetching).\n     * Note: The CloudFront read timeout is capped at 60 seconds, so values above 60 seconds\n     * only benefit direct invocations, not requests arriving via CloudFront.\n     * Defaults to {@link Duration.seconds(10)}.\n     */\n    readonly timeout?: Duration;\n\n    /**\n     * Whether to enable AWS X-Ray for the Nuxt Lambda function.\n     */\n    readonly enableTracing?: boolean;\n\n    /**\n     * Whether to enable a global Sitemap bucket which is permanently accessible through multiple deployments.\n     */\n    readonly enableSitemap?: boolean;\n\n    /**\n     * Whether to enable (HTTPS only) API access to the Nuxt app via the `/api` path which support all HTTP methods.\n     * See https://nuxt.com/docs/guide/directory-structure/server#recipes for details.\n     */\n    readonly enableApi?: boolean;\n\n    /**\n     * Whether to enable reporting of CloudFront access logs via Athena.\n     */\n    readonly enableAccessLogsAnalysis?: boolean;\n\n    /**\n     * Array of cookie names to include in the access logs (whitelist).\n     */\n    readonly accessLogCookies?: string[];\n\n    /**\n     * Whether to anonymize the client IP address in the access logs by replacing the last octet (IPv4)\n     * or the last group (IPv6) with 'xxx'.\n     *\n     * **DSGVO/Legal note:** IP addresses are considered personal data under the GDPR (cf. CJEU judgment C‑582/14).\n     * If you set this to `false`, you must ensure a legal basis under Art. 6 GDPR (e.g. legitimate interest),\n     * document it in your privacy policy, and limit the retention period to what is strictly necessary.\n     * When in doubt, consult a data protection officer or legal counsel before disabling this option.\n     *\n     * Defaults to `true`.\n     */\n    readonly anonymizeAccessLogClientIp?: boolean;\n\n    /**\n     * The duration after which raw (unprocessed) access logs are deleted from S3.\n     * Defaults to 7 days.\n     */\n    readonly accessLogsRawRetention?: Duration;\n\n    /**\n     * The duration after which intermediate (grouped-by-date) access logs are deleted from S3.\n     * Defaults to 7 days.\n     */\n    readonly accessLogsIntermediateRetention?: Duration;\n\n    /**\n     * The duration after which transformed (Parquet) access logs are deleted from S3.\n     * Defaults to 180 days.\n     */\n    readonly accessLogsTransformedRetention?: Duration;\n\n    /**\n     * The number of days to retain static assets of outdated deployments in the S3 bucket.\n     * Useful to allow users to still access old assets after a new deployment when they are still browsing on an old version.\n     * Defaults to 30 days.\n     */\n    readonly outdatedAssetsRetentionDays?: number;\n\n    /**\n     * An array of HTTP headers to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for headers that do not affect the response.\n     *\n     * Only the Cloudfront default headers are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardHeaders?: string[];\n\n    /**\n     * An array of HTTP headers to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for headers that might affect the response, e.g., 'Authorization'.\n     *\n     * Only the Cloudfront default headers are forwarded,\n     * but no headers are included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyHeaders?: string[];\n\n    /**\n     * An array of cookies to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for cookies that do not affect the response.\n     *\n     * No cookies are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardCookies?: string[];\n\n    /**\n     * An array of cookies to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for cookies that might affect the response, e.g., authentication cookies.\n     *\n     * No cookies are forwarded or included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyCookies?: string[];\n\n    /**\n     * An array of query params to forward to the Nuxt app on origin requests without affecting the cache key at CloudFront edge locations.\n     * This should only be used for query params that do not affect the response and are required on SSR requests.\n     *\n     * All query params are forwarded by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-origin-requests.html}\n     */\n    readonly forwardQueryParams?: string[];\n\n    /**\n     * An array of query params to forward to the Nuxt app and to include in the cache key for objects that are cached at CloudFront edge locations.\n     * This should be used for query params that affect the response and are required on SSR requests, e.g., filters.\n     *\n     * All query params are forwarded and included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly cacheKeyQueryParams?: string[];\n\n    /**\n     * An array of query params to prevent forwarding to the Nuxt app and to not include in the cache key for objects that are cached at CloudFront edge locations.\n     * When set, all query params that are not specified in this array will be forwarded to the Nuxt app and included in the cache key.\n     * This should be used for query params that do not affect the response and are not required on SSR requests, e.g., 'fbclid' or 'utm_campaign'.\n     *\n     * If both {@see cacheKeyQueryParams} and {@see denyCacheKeyQueryParams} are specified, the {@see denyCacheKeyQueryParams} will be ignored.\n     * All query params are forwarded and included in the cache key by default.\n     *\n     * {@link https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/controlling-the-cache-key.html}\n     */\n    readonly denyCacheKeyQueryParams?: string[];\n\n    /**\n     * An array of headers to pass to the Nuxt app on SSR requests.\n     * The more headers are passed, the weaker the cache performance will be, as the cache key\n     * is based on the headers.\n     * No headers are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyHeaders} instead.\n     */\n    readonly allowHeaders?: string[];\n\n    /**\n     * An array of cookies to pass to the Nuxt app on SSR requests.\n     * The more cookies are passed, the weaker the cache performance will be, as the cache key\n     * is based on the cookies.\n     * No cookies are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyCookies} instead.\n     */\n    readonly allowCookies?: string[];\n\n    /**\n     * An array of query param keys to pass to the Nuxt app on SSR requests.\n     * The more query params are passed, the weaker the cache performance will be, as the cache key\n     * is based on the query params.\n     * Note that this config can not be combined with {@see denyQueryParams}.\n     * If both are specified, the {@see denyQueryParams} will be ignored.\n     * All query params are passed by default.\n     *\n     * @deprecated Use {@see cacheKeyQueryParams} instead.\n     */\n    readonly allowQueryParams?: string[];\n\n    /**\n     * An array of query param keys to deny passing to the Nuxt app on SSR requests.\n     * It might be useful to prevent specific external query params, e.g., fbclid, utm_campaign, ...,\n     * to improve cache performance, as the cache key is based on the specified query params.\n     * Note that this config can not be combined with {@see allowQueryParams}.\n     * If both are specified, the {@see denyQueryParams} will be ignored.\n     * All query params are passed by default.\n     *\n     * @deprecated Use {@see denyCacheKeyQueryParams} instead.\n     */\n    readonly denyQueryParams?: string[];\n\n    /**\n     * An array of path patterns for server endpoints that should be routed to the SSR origin (API Gateway → Lambda)\n     * instead of the default S3 \"file\" behavior.\n     * \n     * This is useful for server routes that generate dynamic content but use file-like URLs.\n     * For example, `@nuxtjs/sitemap` creates a `/sitemap.xml` endpoint that dynamically generates XML content,\n     * and `@nuxt/image` uses file-like URLs to serve dynamically processed images.\n     * \n     * Note: This is different from `enableSitemap` which serves pre-generated static sitemap files from S3.\n     * Use `serverRoutes` when you need the Lambda to handle requests and generate content on-the-fly.\n     * \n     * Examples: `['/sitemap.xml', '/robots.txt', '/__sitemap__/*', '/_ipx/*']`\n     */\n    readonly serverRoutes?: string[];\n\n    /**\n     * An array of custom CloudFront behaviors to inject into the CloudFront distribution for specific path patterns targeting the Nuxt app origin.\n     * Use this to override the cache policy and/or attach a CloudFront Function for specific routes.\n     *\n     * Each entry can independently define:\n     * - A custom cache policy (e.g., to disable caching for certain paths)\n     * - A CloudFront Function (e.g., for URI normalization at the edge)\n     * - Or both combined\n     *\n     * The behaviors are added to the distribution **before** the static-assets behaviors and in the order\n     * provided, so more specific patterns should be listed first.\n     *\n     * **Example – cache policy only:**\n     * ```typescript\n     * import { CachePolicy } from 'aws-cdk-lib/aws-cloudfront';\n     *\n     * additionalBehaviors: [\n     *   { pathPattern: '/api/realtime/*', cachePolicy: CachePolicy.CACHING_DISABLED },\n     * ]\n     * ```\n     *\n     * **Example – CloudFront Function only:**\n     * ```typescript\n     * import { Function, FunctionCode } from 'aws-cdk-lib/aws-cloudfront';\n     *\n     * const normalizeUri = new Function(this, 'NormalizeUri', {\n     *   code: FunctionCode.fromInline(`\n     *     function handler(event) {\n     *       var request = event.request;\n     *       request.uri = request.uri.replace(/\\\\/+$/, '') || '/';\n     *       return request;\n     *     }\n     *   `),\n     * });\n     *\n     * additionalBehaviors: [\n     *   { pathPattern: '/users/*', fn: normalizeUri },\n     * ]\n     * ```\n     *\n     * **Example – both combined:**\n     * ```typescript\n     * additionalBehaviors: [\n     *   { pathPattern: '/posts/*', fn: normalizeUri, cachePolicy: CachePolicy.CACHING_DISABLED },\n     * ]\n     * ```\n     */\n    readonly additionalBehaviors?: NuxtCloudFrontBehavior[];\n\n    /**\n     * The ARN of an existing AWS WAF Web ACL to associate with the CloudFront distribution.\n     * This should be used with a separate CloudFrontWafStack deployed in us-east-1.\n     *\n     * Example: 'arn:aws:wafv2:us-east-1:123456789012:global/webacl/my-web-acl/a1b2c3d4-5678-90ab-cdef-EXAMPLE11111'\n     */\n    readonly webAclArn?: string;\n}"]}
@@ -1,6 +1,6 @@
1
1
  import {type NuxtAppStackProps} from "../NuxtAppStackProps";
2
2
  import {Duration} from "aws-cdk-lib";
3
- import {Function as CloudFrontFunction, FunctionEventType, ICachePolicy} from "aws-cdk-lib/aws-cloudfront";
3
+ import {FunctionCode, FunctionEventType, ICachePolicy} from "aws-cdk-lib/aws-cloudfront";
4
4
 
5
5
  /**
6
6
  * Defines a custom CloudFront behavior attached to a specific path pattern targeting the Nuxt app origin.
@@ -11,20 +11,34 @@ export interface NuxtCloudFrontBehavior {
11
11
  /**
12
12
  * The CloudFront distribution behavior path pattern to which this behavior shall be applied.
13
13
  *
14
- * @example '/users/*'
14
+ * @example '/posts/*'
15
15
  */
16
16
  readonly pathPattern: string;
17
17
 
18
18
  /**
19
- * An optional pre-instantiated CloudFront Function to associate with this behavior.
19
+ * The code of the CloudFront Function to associate with this behavior.
20
+ * The stack will create the CloudFront Function resource internally.
20
21
  * The function will be invoked on VIEWER_REQUEST events by default.
21
22
  * If omitted, no CloudFront Function is attached.
23
+ *
24
+ * @example
25
+ * ```typescript
26
+ * import { FunctionCode } from 'aws-cdk-lib/aws-cloudfront';
27
+ *
28
+ * fnCode: FunctionCode.fromInline(`
29
+ * function handler(event) {
30
+ * var request = event.request;
31
+ * request.uri = request.uri.replace(/\/+$/, '') || '/';
32
+ * return request;
33
+ * }
34
+ * `)
35
+ * ```
22
36
  */
23
- readonly fn?: CloudFrontFunction;
37
+ readonly fnCode?: FunctionCode;
24
38
 
25
39
  /**
26
40
  * The event type at which the CloudFront Function shall be invoked.
27
- * Only relevant when {@link fn} is specified.
41
+ * Only relevant when {@link fnCode} is specified.
28
42
  * Defaults to {@link FunctionEventType.VIEWER_REQUEST}.
29
43
  */
30
44
  readonly eventType?: FunctionEventType;
@@ -65,6 +79,15 @@ export interface NuxtServerAppStackProps extends NuxtAppStackProps {
65
79
  */
66
80
  readonly memorySize?: number;
67
81
 
82
+ /**
83
+ * The maximum execution duration for the Nuxt app's Lambda function.
84
+ * Increase this if your app performs long-running SSR operations (e.g. large data fetching).
85
+ * Note: The CloudFront read timeout is capped at 60 seconds, so values above 60 seconds
86
+ * only benefit direct invocations, not requests arriving via CloudFront.
87
+ * Defaults to {@link Duration.seconds(10)}.
88
+ */
89
+ readonly timeout?: Duration;
90
+
68
91
  /**
69
92
  * Whether to enable AWS X-Ray for the Nuxt Lambda function.
70
93
  */
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cdk-nuxt",
3
- "version": "2.24.0",
3
+ "version": "2.25.0",
4
4
  "license": "MIT",
5
5
  "repository": {
6
6
  "type": "git",