@sitblueprint/website-construct 0.1.5 → 0.1.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/index.ts CHANGED
@@ -6,6 +6,10 @@ import * as certificatemanager from "aws-cdk-lib/aws-certificatemanager";
6
6
  import * as cdk from "aws-cdk-lib";
7
7
  import * as iam from "aws-cdk-lib/aws-iam";
8
8
  import * as route53 from "aws-cdk-lib/aws-route53";
9
+ import * as dynamodb from "aws-cdk-lib/aws-dynamodb";
10
+ import * as lambda from "aws-cdk-lib/aws-lambda";
11
+ import * as apigateway from "aws-cdk-lib/aws-apigateway";
12
+ import * as path from "path";
9
13
 
10
14
  export interface DomainConfig {
11
15
  /** The root domain name (e.g., example.com).
@@ -38,11 +42,43 @@ export interface WebsiteProps {
38
42
 
39
43
  /** Optional path to a custom 404 page. If not specified, the error file will be used. */
40
44
  notFoundResponsePagePath?: string;
45
+
46
+ /** Optional configuration for pull request preview environments. */
47
+ previewConfig?: PreviewConfig;
48
+ }
49
+
50
+ export interface PreviewConfig {
51
+ /** Prefix used to name preview buckets. Buckets are created as `${prefix}-0`, `${prefix}-1`, ... */
52
+ bucketPrefix: string;
53
+
54
+ /** Number of preview buckets to create.
55
+ * @default 2
56
+ */
57
+ bucketCount?: number;
58
+
59
+ /** If true, creates one CloudFront distribution per preview bucket.
60
+ * @default true
61
+ */
62
+ createDistributions?: boolean;
63
+
64
+ /** Maximum lease lifetime in hours before a slot is considered expired.
65
+ * @default 24
66
+ */
67
+ maxLeaseHours?: number;
68
+ }
69
+
70
+ export interface PreviewEnvironmentProps extends PreviewConfig {
71
+ /** Index document for preview buckets. */
72
+ indexFile: string;
73
+
74
+ /** Error document for preview buckets. */
75
+ errorFile: string;
41
76
  }
42
77
 
43
78
  export class Website extends Construct {
44
79
  public readonly bucket: s3.Bucket;
45
80
  public readonly distribution: cloudfont.Distribution;
81
+ public readonly previewEnvironment?: PreviewEnvironment;
46
82
 
47
83
  constructor(scope: Construct, id: string, props: WebsiteProps) {
48
84
  super(scope, id);
@@ -153,6 +189,18 @@ export class Website extends Construct {
153
189
  description: "Website URL",
154
190
  });
155
191
  }
192
+
193
+ if (props.previewConfig) {
194
+ this.previewEnvironment = new PreviewEnvironment(
195
+ this,
196
+ "PreviewEnvironment",
197
+ {
198
+ ...props.previewConfig,
199
+ indexFile: props.indexFile,
200
+ errorFile: props.errorFile,
201
+ },
202
+ );
203
+ }
156
204
  }
157
205
 
158
206
  private _getFullDomainName(domainConfig: DomainConfig): string {
@@ -169,3 +217,169 @@ export class Website extends Construct {
169
217
  );
170
218
  }
171
219
  }
220
+
221
+ export class PreviewEnvironment extends Construct {
222
+ public readonly buckets: s3.Bucket[];
223
+ public readonly distributions: cloudfont.Distribution[];
224
+ public readonly leaseTable: dynamodb.Table;
225
+ public readonly api: apigateway.RestApi;
226
+ public readonly claimEndpoint: string;
227
+ public readonly heartbeatEndpoint: string;
228
+ public readonly releaseEndpoint: string;
229
+
230
+ constructor(scope: Construct, id: string, props: PreviewEnvironmentProps) {
231
+ super(scope, id);
232
+
233
+ const bucketCount = props.bucketCount ?? 2;
234
+ if (bucketCount < 1) {
235
+ throw new Error("bucketCount must be greater than or equal to 1");
236
+ }
237
+
238
+ const indexFile = props.indexFile;
239
+ const errorFile = props.errorFile;
240
+ const createDistributions = props.createDistributions ?? true;
241
+ const maxLeaseHours = props.maxLeaseHours ?? 24;
242
+ const maxLeaseMs = cdk.Duration.hours(maxLeaseHours).toMilliseconds();
243
+
244
+ this.buckets = Array.from({ length: bucketCount }, (_, slotId) => {
245
+ const bucketName = `${props.bucketPrefix}-${slotId}`;
246
+ return new s3.Bucket(this, `PreviewBucket${slotId}`, {
247
+ bucketName,
248
+ websiteIndexDocument: indexFile,
249
+ websiteErrorDocument: errorFile,
250
+ publicReadAccess: true,
251
+ removalPolicy: cdk.RemovalPolicy.DESTROY,
252
+ autoDeleteObjects: true,
253
+ blockPublicAccess: s3.BlockPublicAccess.BLOCK_ACLS_ONLY,
254
+ accessControl: s3.BucketAccessControl.BUCKET_OWNER_FULL_CONTROL,
255
+ encryption: s3.BucketEncryption.S3_MANAGED,
256
+ });
257
+ });
258
+
259
+ this.distributions = createDistributions
260
+ ? this.buckets.map((bucket, slotId) => {
261
+ const oai = new cloudfont.OriginAccessIdentity(
262
+ this,
263
+ `PreviewOAI${slotId}`,
264
+ );
265
+ bucket.addToResourcePolicy(
266
+ new iam.PolicyStatement({
267
+ actions: ["s3:GetObject"],
268
+ resources: [bucket.arnForObjects("*")],
269
+ principals: [
270
+ new iam.CanonicalUserPrincipal(
271
+ oai.cloudFrontOriginAccessIdentityS3CanonicalUserId,
272
+ ),
273
+ ],
274
+ }),
275
+ );
276
+ return new cloudfont.Distribution(
277
+ this,
278
+ `PreviewDistribution${slotId}`,
279
+ {
280
+ defaultBehavior: {
281
+ origin: new origins.S3StaticWebsiteOrigin(bucket),
282
+ viewerProtocolPolicy:
283
+ cloudfont.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
284
+ },
285
+ priceClass: cloudfont.PriceClass.PRICE_CLASS_100,
286
+ },
287
+ );
288
+ })
289
+ : [];
290
+
291
+ this.leaseTable = new dynamodb.Table(this, "PreviewLeases", {
292
+ partitionKey: {
293
+ name: "slotId",
294
+ type: dynamodb.AttributeType.STRING,
295
+ },
296
+ billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
297
+ removalPolicy: cdk.RemovalPolicy.DESTROY,
298
+ timeToLiveAttribute: "ttlEpochSeconds",
299
+ });
300
+ this.leaseTable.addGlobalSecondaryIndex({
301
+ indexName: "RepoPrKeyIndex",
302
+ partitionKey: {
303
+ name: "repoPrKey",
304
+ type: dynamodb.AttributeType.STRING,
305
+ },
306
+ projectionType: dynamodb.ProjectionType.ALL,
307
+ });
308
+
309
+ const slotDefinitions = this.buckets.map((bucket, slotId) => ({
310
+ slotId,
311
+ bucketName: bucket.bucketName,
312
+ previewUrl: this.distributions[slotId]
313
+ ? `https://${this.distributions[slotId].distributionDomainName}`
314
+ : bucket.bucketWebsiteUrl,
315
+ }));
316
+
317
+ const leaseApiHandler = new lambda.Function(
318
+ this,
319
+ "PreviewLeaseApiHandler",
320
+ {
321
+ runtime: lambda.Runtime.NODEJS_20_X,
322
+ handler: "index.handler",
323
+ timeout: cdk.Duration.seconds(15),
324
+ code: lambda.Code.fromAsset(path.join(__dirname, "..", "lambda")),
325
+ environment: {
326
+ TABLE_NAME: this.leaseTable.tableName,
327
+ SLOT_DEFINITIONS: JSON.stringify(slotDefinitions),
328
+ MAX_LEASE_MS: String(maxLeaseMs),
329
+ },
330
+ },
331
+ );
332
+
333
+ this.leaseTable.grantReadWriteData(leaseApiHandler);
334
+
335
+ this.api = new apigateway.RestApi(this, "PreviewLeaseApi", {
336
+ restApiName: `${cdk.Names.uniqueId(this)}-preview-lease-api`,
337
+ description: "API for claiming/releasing preview slots",
338
+ });
339
+
340
+ const claimResource = this.api.root.addResource("claim");
341
+ claimResource.addMethod(
342
+ "POST",
343
+ new apigateway.LambdaIntegration(leaseApiHandler),
344
+ );
345
+ const heartbeatResource = this.api.root.addResource("heartbeat");
346
+ heartbeatResource.addMethod(
347
+ "POST",
348
+ new apigateway.LambdaIntegration(leaseApiHandler),
349
+ );
350
+ const releaseResource = this.api.root.addResource("release");
351
+ releaseResource.addMethod(
352
+ "POST",
353
+ new apigateway.LambdaIntegration(leaseApiHandler),
354
+ );
355
+
356
+ this.claimEndpoint = `${this.api.url}claim`;
357
+ this.heartbeatEndpoint = `${this.api.url}heartbeat`;
358
+ this.releaseEndpoint = `${this.api.url}release`;
359
+
360
+ new cdk.CfnOutput(this, "preview-claim-endpoint", {
361
+ value: this.claimEndpoint,
362
+ description: "POST endpoint used to claim a preview slot",
363
+ });
364
+ new cdk.CfnOutput(this, "preview-heartbeat-endpoint", {
365
+ value: this.heartbeatEndpoint,
366
+ description: "POST endpoint used to refresh a preview slot lease",
367
+ });
368
+ new cdk.CfnOutput(this, "preview-release-endpoint", {
369
+ value: this.releaseEndpoint,
370
+ description: "POST endpoint used to release a preview slot",
371
+ });
372
+ }
373
+
374
+ public grantDeploymentAccess(grantee: iam.IGrantable): void {
375
+ this.buckets.forEach((bucket) => bucket.grantReadWrite(grantee));
376
+ this.distributions.forEach((distribution) => {
377
+ grantee.grantPrincipal.addToPrincipalPolicy(
378
+ new iam.PolicyStatement({
379
+ actions: ["cloudfront:CreateInvalidation"],
380
+ resources: [distribution.distributionArn],
381
+ }),
382
+ );
383
+ });
384
+ }
385
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sitblueprint/website-construct",
3
- "version": "0.1.5",
3
+ "version": "0.1.6",
4
4
  "description": "A reusable AWS CDK construct for deploying static websites with optional custom domain support.",
5
5
  "author": "Miguel Merlin <mmerlin@stevens.edu>",
6
6
  "license": "MIT",
@@ -32,7 +32,7 @@
32
32
  "devDependencies": {
33
33
  "@types/jest": "^29.5.14",
34
34
  "@types/node": "24.10.1",
35
- "aws-cdk-lib": "2.235.0",
35
+ "aws-cdk-lib": "2.237.1",
36
36
  "constructs": "^10.0.0",
37
37
  "jest": "^29.7.0",
38
38
  "prettier": "^3.6.2",
@@ -40,7 +40,7 @@
40
40
  "typescript": "~5.9.3"
41
41
  },
42
42
  "peerDependencies": {
43
- "aws-cdk-lib": "2.235.0",
43
+ "aws-cdk-lib": "2.237.1",
44
44
  "constructs": "^10.0.0"
45
45
  }
46
46
  }
@@ -367,4 +367,130 @@ describe("Website", () => {
367
367
  });
368
368
  });
369
369
  });
370
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"website-construct.test.js","sourceRoot":"","sources":["website-construct.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Template, Match } from \"aws-cdk-lib/assertions\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { Website, WebsiteProps, DomainConfig } from \"../lib\";\n\ndescribe(\"Website\", () => {\n  let app: cdk.App;\n  let stack: cdk.Stack;\n\n  beforeEach(() => {\n    app = new cdk.App();\n    stack = new cdk.Stack(app, \"TestStack\", {\n      env: { account: \"123456789012\", region: \"us-east-1\" },\n    });\n    stack.node.setContext(\n      \"hosted-zone:account=123456789012:domainName=example.com:region=us-east-1\",\n      {\n        Id: \"/hostedzone/Z123456789012\",\n        Name: \"example.com.\",\n      },\n    );\n  });\n\n  describe(\"Basic functionality\", () => {\n    test(\"creates S3 bucket with basic configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::S3::Bucket\", {\n        WebsiteConfiguration: {\n          IndexDocument: \"index.html\",\n          ErrorDocument: \"error.html\",\n        },\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [\n            {\n              ServerSideEncryptionByDefault: {\n                SSEAlgorithm: \"AES256\",\n              },\n            },\n          ],\n        },\n      });\n    });\n\n    test(\"creates CloudFront Origin Access Identity\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\n        \"AWS::CloudFront::CloudFrontOriginAccessIdentity\",\n        {\n          CloudFrontOriginAccessIdentityConfig: {\n            Comment: Match.anyValue(),\n          },\n        },\n      );\n    });\n\n    test(\"creates CloudFront distribution with correct configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          DefaultCacheBehavior: {\n            ViewerProtocolPolicy: \"redirect-to-https\",\n          },\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n          PriceClass: \"PriceClass_100\",\n          Enabled: true,\n        },\n      });\n    });\n\n    test(\"exposes CloudFront distribution\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const website = new Website(stack, \"TestWebsite\", props);\n\n      expect(website.distribution).toBeDefined();\n    });\n  });\n\n  describe(\"Custom error page configuration\", () => {\n    test(\"uses custom not found response page path\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        notFoundResponsePagePath: \"/custom-404.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/custom-404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n        },\n      });\n    });\n\n    test(\"uses default 404.html when notFoundResponsePagePath is not provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n        },\n      });\n    });\n  });\n\n  describe(\"Domain configuration\", () => {\n    const domainConfig: DomainConfig = {\n      domainName: \"example.com\",\n      subdomainName: \"www\",\n      certificateArn:\n        \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n    };\n\n    test(\"configures CloudFront distribution with custom domain\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"www.example.com\"],\n          ViewerCertificate: {\n            AcmCertificateArn: domainConfig.certificateArn,\n            SslSupportMethod: \"sni-only\",\n          },\n        },\n      });\n    });\n\n    test(\"creates Route53 A record when domain config is provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Type: \"A\",\n        Name: \"www.example.com.\",\n        AliasTarget: {\n          DNSName: Match.anyValue(),\n          HostedZoneId: Match.anyValue(),\n        },\n      });\n    });\n\n    test(\"handles domain without subdomain\", () => {\n      const domainConfigWithoutSub: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: domainConfigWithoutSub,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"example.com\"],\n        },\n      });\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Type: \"A\",\n        Name: \"example.com.\",\n      });\n    });\n\n    test(\"configures CloudFront with both subdomain and root domain aliases when includeRootDomain is true\", () => {\n      const dualDomainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"www\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: dualDomainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"www.example.com\", \"example.com\"],\n        },\n      });\n    });\n\n    test(\"creates two Route53 A records when includeRootDomain is true\", () => {\n      const dualDomainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"www\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: dualDomainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 2);\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Name: \"www.example.com.\",\n        Type: \"A\",\n      });\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Name: \"example.com.\",\n        Type: \"A\",\n      });\n    });\n\n    test(\"ignores includeRootDomain if subdomain is empty to avoid duplicates\", () => {\n      const domainConfigWithoutSub: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: domainConfigWithoutSub,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 1);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"example.com\"],\n        },\n      });\n    });\n  });\n\n  describe(\"Private methods\", () => {\n    test(\"_getFullDomainName returns correct domain with subdomain\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const construct = new Website(stack, \"TestWebsite\", props);\n\n      // Access private method through bracket notation for testing\n      const getFullDomainName = (construct as any)._getFullDomainName;\n\n      const domainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"blog\",\n        certificateArn: \"arn:test\",\n      };\n\n      const result = getFullDomainName(domainConfig);\n      expect(result).toBe(\"blog.example.com\");\n    });\n\n    test(\"_getFullDomainName returns root domain when subdomain is empty\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const construct = new Website(stack, \"TestWebsite\", props);\n\n      const getFullDomainName = (construct as any)._getFullDomainName;\n\n      const domainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn: \"arn:test\",\n      };\n\n      const result = getFullDomainName(domainConfig);\n      expect(result).toBe(\"example.com\");\n    });\n  });\n\n  describe(\"Edge cases\", () => {\n    test(\"handles minimal configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"minimal-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      expect(() => {\n        new Website(stack, \"TestWebsite\", props);\n      }).not.toThrow();\n\n      const template = Template.fromStack(stack);\n      template.resourceCountIs(\"AWS::S3::Bucket\", 1);\n      template.resourceCountIs(\"AWS::CloudFront::Distribution\", 1);\n      template.resourceCountIs(\n        \"AWS::CloudFront::CloudFrontOriginAccessIdentity\",\n        1,\n      );\n    });\n\n    test(\"does not create Route53 resources when domain config is not provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 0);\n    });\n  });\n});\n"]}
370
+ describe("Preview config on Website", () => {
371
+ let app;
372
+ let stack;
373
+ beforeEach(() => {
374
+ app = new cdk.App();
375
+ stack = new cdk.Stack(app, "PreviewTestStack", {
376
+ env: { account: "123456789012", region: "us-east-1" },
377
+ });
378
+ });
379
+ test("creates two preview buckets by default when previewConfig is enabled", () => {
380
+ const website = new lib_1.Website(stack, "PreviewEnabledWebsite", {
381
+ bucketName: "website-bucket",
382
+ indexFile: "index.html",
383
+ errorFile: "error.html",
384
+ previewConfig: {
385
+ bucketPrefix: "preview-bucket",
386
+ },
387
+ });
388
+ const template = assertions_1.Template.fromStack(stack);
389
+ template.resourceCountIs("AWS::S3::Bucket", 3);
390
+ expect(website.previewEnvironment).toBeDefined();
391
+ });
392
+ test("creates requested number of preview buckets from previewConfig", () => {
393
+ new lib_1.Website(stack, "PreviewEnabledWebsite", {
394
+ bucketName: "website-bucket",
395
+ indexFile: "index.html",
396
+ errorFile: "error.html",
397
+ previewConfig: {
398
+ bucketPrefix: "preview-bucket",
399
+ bucketCount: 3,
400
+ },
401
+ });
402
+ const template = assertions_1.Template.fromStack(stack);
403
+ template.resourceCountIs("AWS::S3::Bucket", 4);
404
+ });
405
+ test("reuses website index and error files for preview buckets", () => {
406
+ new lib_1.Website(stack, "PreviewEnabledWebsite", {
407
+ bucketName: "website-bucket",
408
+ indexFile: "app.html",
409
+ errorFile: "fallback.html",
410
+ previewConfig: {
411
+ bucketPrefix: "preview-bucket",
412
+ },
413
+ });
414
+ const template = assertions_1.Template.fromStack(stack);
415
+ template.hasResourceProperties("AWS::S3::Bucket", {
416
+ BucketName: "preview-bucket-0",
417
+ WebsiteConfiguration: {
418
+ IndexDocument: "app.html",
419
+ ErrorDocument: "fallback.html",
420
+ },
421
+ });
422
+ template.hasResourceProperties("AWS::S3::Bucket", {
423
+ BucketName: "preview-bucket-1",
424
+ WebsiteConfiguration: {
425
+ IndexDocument: "app.html",
426
+ ErrorDocument: "fallback.html",
427
+ },
428
+ });
429
+ });
430
+ test("creates lease table with repo-pr lookup index when preview is enabled", () => {
431
+ new lib_1.Website(stack, "PreviewEnabledWebsite", {
432
+ bucketName: "website-bucket",
433
+ indexFile: "index.html",
434
+ errorFile: "error.html",
435
+ previewConfig: {
436
+ bucketPrefix: "preview-bucket",
437
+ },
438
+ });
439
+ const template = assertions_1.Template.fromStack(stack);
440
+ template.hasResourceProperties("AWS::DynamoDB::Table", {
441
+ KeySchema: [
442
+ {
443
+ AttributeName: "slotId",
444
+ KeyType: "HASH",
445
+ },
446
+ ],
447
+ GlobalSecondaryIndexes: [
448
+ {
449
+ IndexName: "RepoPrKeyIndex",
450
+ KeySchema: [
451
+ {
452
+ AttributeName: "repoPrKey",
453
+ KeyType: "HASH",
454
+ },
455
+ ],
456
+ Projection: {
457
+ ProjectionType: "ALL",
458
+ },
459
+ },
460
+ ],
461
+ });
462
+ });
463
+ test("creates lease API routes when preview is enabled", () => {
464
+ new lib_1.Website(stack, "PreviewEnabledWebsite", {
465
+ bucketName: "website-bucket",
466
+ indexFile: "index.html",
467
+ errorFile: "error.html",
468
+ previewConfig: {
469
+ bucketPrefix: "preview-bucket",
470
+ },
471
+ });
472
+ const template = assertions_1.Template.fromStack(stack);
473
+ template.hasResourceProperties("AWS::ApiGateway::Resource", {
474
+ PathPart: "claim",
475
+ });
476
+ template.hasResourceProperties("AWS::ApiGateway::Resource", {
477
+ PathPart: "heartbeat",
478
+ });
479
+ template.hasResourceProperties("AWS::ApiGateway::Resource", {
480
+ PathPart: "release",
481
+ });
482
+ template.resourceCountIs("AWS::ApiGateway::Method", 3);
483
+ });
484
+ test("does not create preview resources when previewConfig is omitted", () => {
485
+ const website = new lib_1.Website(stack, "WebsiteWithoutPreview", {
486
+ bucketName: "website-bucket",
487
+ indexFile: "index.html",
488
+ errorFile: "error.html",
489
+ });
490
+ const template = assertions_1.Template.fromStack(stack);
491
+ template.resourceCountIs("AWS::DynamoDB::Table", 0);
492
+ template.resourceCountIs("AWS::ApiGateway::RestApi", 0);
493
+ expect(website.previewEnvironment).toBeUndefined();
494
+ });
495
+ });
496
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"website-construct.test.js","sourceRoot":"","sources":["website-construct.test.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,uDAAyD;AACzD,iDAAmC;AACnC,gCAA6D;AAE7D,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;IACvB,IAAI,GAAY,CAAC;IACjB,IAAI,KAAgB,CAAC;IAErB,UAAU,CAAC,GAAG,EAAE;QACd,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,EAAE,CAAC;QACpB,KAAK,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,WAAW,EAAE;YACtC,GAAG,EAAE,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,WAAW,EAAE;SACtD,CAAC,CAAC;QACH,KAAK,CAAC,IAAI,CAAC,UAAU,CACnB,0EAA0E,EAC1E;YACE,EAAE,EAAE,2BAA2B;YAC/B,IAAI,EAAE,cAAc;SACrB,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;QACnC,IAAI,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACtD,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,iBAAiB,EAAE;gBAChD,oBAAoB,EAAE;oBACpB,aAAa,EAAE,YAAY;oBAC3B,aAAa,EAAE,YAAY;iBAC5B;gBACD,gBAAgB,EAAE;oBAChB,iCAAiC,EAAE;wBACjC;4BACE,6BAA6B,EAAE;gCAC7B,YAAY,EAAE,QAAQ;6BACvB;yBACF;qBACF;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,2CAA2C,EAAE,GAAG,EAAE;YACrD,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAC5B,iDAAiD,EACjD;gBACE,oCAAoC,EAAE;oBACpC,OAAO,EAAE,kBAAK,CAAC,QAAQ,EAAE;iBAC1B;aACF,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,4DAA4D,EAAE,GAAG,EAAE;YACtE,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,oBAAoB,EAAE;wBACpB,oBAAoB,EAAE,mBAAmB;qBAC1C;oBACD,oBAAoB,EAAE;wBACpB;4BACE,SAAS,EAAE,GAAG;4BACd,YAAY,EAAE,GAAG;4BACjB,gBAAgB,EAAE,WAAW;4BAC7B,kBAAkB,EAAE,IAAI;yBACzB;qBACF;oBACD,UAAU,EAAE,gBAAgB;oBAC5B,OAAO,EAAE,IAAI;iBACd;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,iCAAiC,EAAE,GAAG,EAAE;YAC3C,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,MAAM,OAAO,GAAG,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzD,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,WAAW,EAAE,CAAC;QAC7C,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iCAAiC,EAAE,GAAG,EAAE;QAC/C,IAAI,CAAC,0CAA0C,EAAE,GAAG,EAAE;YACpD,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,wBAAwB,EAAE,kBAAkB;aAC7C,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,oBAAoB,EAAE;wBACpB;4BACE,SAAS,EAAE,GAAG;4BACd,YAAY,EAAE,GAAG;4BACjB,gBAAgB,EAAE,kBAAkB;4BACpC,kBAAkB,EAAE,IAAI;yBACzB;qBACF;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,qEAAqE,EAAE,GAAG,EAAE;YAC/E,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,oBAAoB,EAAE;wBACpB;4BACE,SAAS,EAAE,GAAG;4BACd,YAAY,EAAE,GAAG;4BACjB,gBAAgB,EAAE,WAAW;4BAC7B,kBAAkB,EAAE,IAAI;yBACzB;qBACF;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;QACpC,MAAM,YAAY,GAAiB;YACjC,UAAU,EAAE,aAAa;YACzB,aAAa,EAAE,KAAK;YACpB,cAAc,EACZ,qFAAqF;SACxF,CAAC;QAEF,IAAI,CAAC,uDAAuD,EAAE,GAAG,EAAE;YACjE,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY;aACb,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,OAAO,EAAE,CAAC,iBAAiB,CAAC;oBAC5B,iBAAiB,EAAE;wBACjB,iBAAiB,EAAE,YAAY,CAAC,cAAc;wBAC9C,gBAAgB,EAAE,UAAU;qBAC7B;iBACF;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,yDAAyD,EAAE,GAAG,EAAE;YACnE,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY;aACb,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,yBAAyB,EAAE;gBACxD,IAAI,EAAE,GAAG;gBACT,IAAI,EAAE,kBAAkB;gBACxB,WAAW,EAAE;oBACX,OAAO,EAAE,kBAAK,CAAC,QAAQ,EAAE;oBACzB,YAAY,EAAE,kBAAK,CAAC,QAAQ,EAAE;iBAC/B;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,kCAAkC,EAAE,GAAG,EAAE;YAC5C,MAAM,sBAAsB,GAAiB;gBAC3C,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,EAAE;gBACjB,cAAc,EACZ,qFAAqF;aACxF,CAAC;YAEF,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY,EAAE,sBAAsB;aACrC,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,OAAO,EAAE,CAAC,aAAa,CAAC;iBACzB;aACF,CAAC,CAAC;YAEH,QAAQ,CAAC,qBAAqB,CAAC,yBAAyB,EAAE;gBACxD,IAAI,EAAE,GAAG;gBACT,IAAI,EAAE,cAAc;aACrB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,kGAAkG,EAAE,GAAG,EAAE;YAC5G,MAAM,gBAAgB,GAAiB;gBACrC,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,KAAK;gBACpB,cAAc,EACZ,qFAAqF;gBACvF,iBAAiB,EAAE,IAAI;aACxB,CAAC;YAEF,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY,EAAE,gBAAgB;aAC/B,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,OAAO,EAAE,CAAC,iBAAiB,EAAE,aAAa,CAAC;iBAC5C;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,8DAA8D,EAAE,GAAG,EAAE;YACxE,MAAM,gBAAgB,GAAiB;gBACrC,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,KAAK;gBACpB,cAAc,EACZ,qFAAqF;gBACvF,iBAAiB,EAAE,IAAI;aACxB,CAAC;YAEF,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY,EAAE,gBAAgB;aAC/B,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,eAAe,CAAC,yBAAyB,EAAE,CAAC,CAAC,CAAC;YAEvD,QAAQ,CAAC,qBAAqB,CAAC,yBAAyB,EAAE;gBACxD,IAAI,EAAE,kBAAkB;gBACxB,IAAI,EAAE,GAAG;aACV,CAAC,CAAC;YAEH,QAAQ,CAAC,qBAAqB,CAAC,yBAAyB,EAAE;gBACxD,IAAI,EAAE,cAAc;gBACpB,IAAI,EAAE,GAAG;aACV,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,qEAAqE,EAAE,GAAG,EAAE;YAC/E,MAAM,sBAAsB,GAAiB;gBAC3C,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,EAAE;gBACjB,cAAc,EACZ,qFAAqF;gBACvF,iBAAiB,EAAE,IAAI;aACxB,CAAC;YAEF,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;gBACvB,YAAY,EAAE,sBAAsB;aACrC,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE3C,QAAQ,CAAC,eAAe,CAAC,yBAAyB,EAAE,CAAC,CAAC,CAAC;YAEvD,QAAQ,CAAC,qBAAqB,CAAC,+BAA+B,EAAE;gBAC9D,kBAAkB,EAAE;oBAClB,OAAO,EAAE,CAAC,aAAa,CAAC;iBACzB;aACF,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;QAC/B,IAAI,CAAC,0DAA0D,EAAE,GAAG,EAAE;YACpE,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,aAAa;gBACzB,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,MAAM,SAAS,GAAG,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAE3D,6DAA6D;YAC7D,MAAM,iBAAiB,GAAI,SAAiB,CAAC,kBAAkB,CAAC;YAEhE,MAAM,YAAY,GAAiB;gBACjC,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,MAAM;gBACrB,cAAc,EAAE,UAAU;aAC3B,CAAC;YAEF,MAAM,MAAM,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;YAC/C,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,gEAAgE,EAAE,GAAG,EAAE;YAC1E,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,aAAa;gBACzB,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,MAAM,SAAS,GAAG,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAE3D,MAAM,iBAAiB,GAAI,SAAiB,CAAC,kBAAkB,CAAC;YAEhE,MAAM,YAAY,GAAiB;gBACjC,UAAU,EAAE,aAAa;gBACzB,aAAa,EAAE,EAAE;gBACjB,cAAc,EAAE,UAAU;aAC3B,CAAC;YAEF,MAAM,MAAM,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;YAC/C,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;QAC1B,IAAI,CAAC,+BAA+B,EAAE,GAAG,EAAE;YACzC,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,gBAAgB;gBAC5B,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,MAAM,CAAC,GAAG,EAAE;gBACV,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAC3C,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,CAAC;YAEjB,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAC3C,QAAQ,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC;YAC/C,QAAQ,CAAC,eAAe,CAAC,+BAA+B,EAAE,CAAC,CAAC,CAAC;YAC7D,QAAQ,CAAC,eAAe,CACtB,iDAAiD,EACjD,CAAC,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sEAAsE,EAAE,GAAG,EAAE;YAChF,MAAM,KAAK,GAAiB;gBAC1B,UAAU,EAAE,aAAa;gBACzB,SAAS,EAAE,YAAY;gBACvB,SAAS,EAAE,YAAY;aACxB,CAAC;YAEF,IAAI,aAAO,CAAC,KAAK,EAAE,aAAa,EAAE,KAAK,CAAC,CAAC;YAEzC,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAC3C,QAAQ,CAAC,eAAe,CAAC,yBAAyB,EAAE,CAAC,CAAC,CAAC;QACzD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;IACzC,IAAI,GAAY,CAAC;IACjB,IAAI,KAAgB,CAAC;IAErB,UAAU,CAAC,GAAG,EAAE;QACd,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,EAAE,CAAC;QACpB,KAAK,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,kBAAkB,EAAE;YAC7C,GAAG,EAAE,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,EAAE,WAAW,EAAE;SACtD,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,sEAAsE,EAAE,GAAG,EAAE;QAChF,MAAM,OAAO,GAAG,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1D,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,YAAY;YACvB,SAAS,EAAE,YAAY;YACvB,aAAa,EAAE;gBACb,YAAY,EAAE,gBAAgB;aAC/B;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC;QAC/C,MAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC,WAAW,EAAE,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,gEAAgE,EAAE,GAAG,EAAE;QAC1E,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1C,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,YAAY;YACvB,SAAS,EAAE,YAAY;YACvB,aAAa,EAAE;gBACb,YAAY,EAAE,gBAAgB;gBAC9B,WAAW,EAAE,CAAC;aACf;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,0DAA0D,EAAE,GAAG,EAAE;QACpE,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1C,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,UAAU;YACrB,SAAS,EAAE,eAAe;YAC1B,aAAa,EAAE;gBACb,YAAY,EAAE,gBAAgB;aAC/B;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,qBAAqB,CAAC,iBAAiB,EAAE;YAChD,UAAU,EAAE,kBAAkB;YAC9B,oBAAoB,EAAE;gBACpB,aAAa,EAAE,UAAU;gBACzB,aAAa,EAAE,eAAe;aAC/B;SACF,CAAC,CAAC;QACH,QAAQ,CAAC,qBAAqB,CAAC,iBAAiB,EAAE;YAChD,UAAU,EAAE,kBAAkB;YAC9B,oBAAoB,EAAE;gBACpB,aAAa,EAAE,UAAU;gBACzB,aAAa,EAAE,eAAe;aAC/B;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,uEAAuE,EAAE,GAAG,EAAE;QACjF,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1C,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,YAAY;YACvB,SAAS,EAAE,YAAY;YACvB,aAAa,EAAE;gBACb,YAAY,EAAE,gBAAgB;aAC/B;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,qBAAqB,CAAC,sBAAsB,EAAE;YACrD,SAAS,EAAE;gBACT;oBACE,aAAa,EAAE,QAAQ;oBACvB,OAAO,EAAE,MAAM;iBAChB;aACF;YACD,sBAAsB,EAAE;gBACtB;oBACE,SAAS,EAAE,gBAAgB;oBAC3B,SAAS,EAAE;wBACT;4BACE,aAAa,EAAE,WAAW;4BAC1B,OAAO,EAAE,MAAM;yBAChB;qBACF;oBACD,UAAU,EAAE;wBACV,cAAc,EAAE,KAAK;qBACtB;iBACF;aACF;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,kDAAkD,EAAE,GAAG,EAAE;QAC5D,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1C,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,YAAY;YACvB,SAAS,EAAE,YAAY;YACvB,aAAa,EAAE;gBACb,YAAY,EAAE,gBAAgB;aAC/B;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,qBAAqB,CAAC,2BAA2B,EAAE;YAC1D,QAAQ,EAAE,OAAO;SAClB,CAAC,CAAC;QACH,QAAQ,CAAC,qBAAqB,CAAC,2BAA2B,EAAE;YAC1D,QAAQ,EAAE,WAAW;SACtB,CAAC,CAAC;QACH,QAAQ,CAAC,qBAAqB,CAAC,2BAA2B,EAAE;YAC1D,QAAQ,EAAE,SAAS;SACpB,CAAC,CAAC;QACH,QAAQ,CAAC,eAAe,CAAC,yBAAyB,EAAE,CAAC,CAAC,CAAC;IACzD,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,iEAAiE,EAAE,GAAG,EAAE;QAC3E,MAAM,OAAO,GAAG,IAAI,aAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;YAC1D,UAAU,EAAE,gBAAgB;YAC5B,SAAS,EAAE,YAAY;YACvB,SAAS,EAAE,YAAY;SACxB,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,qBAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAC3C,QAAQ,CAAC,eAAe,CAAC,sBAAsB,EAAE,CAAC,CAAC,CAAC;QACpD,QAAQ,CAAC,eAAe,CAAC,0BAA0B,EAAE,CAAC,CAAC,CAAC;QACxD,MAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC,aAAa,EAAE,CAAC;IACrD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import { Template, Match } from \"aws-cdk-lib/assertions\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { Website, WebsiteProps, DomainConfig } from \"../lib\";\n\ndescribe(\"Website\", () => {\n  let app: cdk.App;\n  let stack: cdk.Stack;\n\n  beforeEach(() => {\n    app = new cdk.App();\n    stack = new cdk.Stack(app, \"TestStack\", {\n      env: { account: \"123456789012\", region: \"us-east-1\" },\n    });\n    stack.node.setContext(\n      \"hosted-zone:account=123456789012:domainName=example.com:region=us-east-1\",\n      {\n        Id: \"/hostedzone/Z123456789012\",\n        Name: \"example.com.\",\n      },\n    );\n  });\n\n  describe(\"Basic functionality\", () => {\n    test(\"creates S3 bucket with basic configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::S3::Bucket\", {\n        WebsiteConfiguration: {\n          IndexDocument: \"index.html\",\n          ErrorDocument: \"error.html\",\n        },\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [\n            {\n              ServerSideEncryptionByDefault: {\n                SSEAlgorithm: \"AES256\",\n              },\n            },\n          ],\n        },\n      });\n    });\n\n    test(\"creates CloudFront Origin Access Identity\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\n        \"AWS::CloudFront::CloudFrontOriginAccessIdentity\",\n        {\n          CloudFrontOriginAccessIdentityConfig: {\n            Comment: Match.anyValue(),\n          },\n        },\n      );\n    });\n\n    test(\"creates CloudFront distribution with correct configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          DefaultCacheBehavior: {\n            ViewerProtocolPolicy: \"redirect-to-https\",\n          },\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n          PriceClass: \"PriceClass_100\",\n          Enabled: true,\n        },\n      });\n    });\n\n    test(\"exposes CloudFront distribution\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const website = new Website(stack, \"TestWebsite\", props);\n\n      expect(website.distribution).toBeDefined();\n    });\n  });\n\n  describe(\"Custom error page configuration\", () => {\n    test(\"uses custom not found response page path\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        notFoundResponsePagePath: \"/custom-404.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/custom-404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n        },\n      });\n    });\n\n    test(\"uses default 404.html when notFoundResponsePagePath is not provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          CustomErrorResponses: [\n            {\n              ErrorCode: 404,\n              ResponseCode: 404,\n              ResponsePagePath: \"/404.html\",\n              ErrorCachingMinTTL: 1800,\n            },\n          ],\n        },\n      });\n    });\n  });\n\n  describe(\"Domain configuration\", () => {\n    const domainConfig: DomainConfig = {\n      domainName: \"example.com\",\n      subdomainName: \"www\",\n      certificateArn:\n        \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n    };\n\n    test(\"configures CloudFront distribution with custom domain\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"www.example.com\"],\n          ViewerCertificate: {\n            AcmCertificateArn: domainConfig.certificateArn,\n            SslSupportMethod: \"sni-only\",\n          },\n        },\n      });\n    });\n\n    test(\"creates Route53 A record when domain config is provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Type: \"A\",\n        Name: \"www.example.com.\",\n        AliasTarget: {\n          DNSName: Match.anyValue(),\n          HostedZoneId: Match.anyValue(),\n        },\n      });\n    });\n\n    test(\"handles domain without subdomain\", () => {\n      const domainConfigWithoutSub: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: domainConfigWithoutSub,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"example.com\"],\n        },\n      });\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Type: \"A\",\n        Name: \"example.com.\",\n      });\n    });\n\n    test(\"configures CloudFront with both subdomain and root domain aliases when includeRootDomain is true\", () => {\n      const dualDomainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"www\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: dualDomainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"www.example.com\", \"example.com\"],\n        },\n      });\n    });\n\n    test(\"creates two Route53 A records when includeRootDomain is true\", () => {\n      const dualDomainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"www\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: dualDomainConfig,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 2);\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Name: \"www.example.com.\",\n        Type: \"A\",\n      });\n\n      template.hasResourceProperties(\"AWS::Route53::RecordSet\", {\n        Name: \"example.com.\",\n        Type: \"A\",\n      });\n    });\n\n    test(\"ignores includeRootDomain if subdomain is empty to avoid duplicates\", () => {\n      const domainConfigWithoutSub: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn:\n          \"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012\",\n        includeRootDomain: true,\n      };\n\n      const props: WebsiteProps = {\n        bucketName: \"test-website-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n        domainConfig: domainConfigWithoutSub,\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 1);\n\n      template.hasResourceProperties(\"AWS::CloudFront::Distribution\", {\n        DistributionConfig: {\n          Aliases: [\"example.com\"],\n        },\n      });\n    });\n  });\n\n  describe(\"Private methods\", () => {\n    test(\"_getFullDomainName returns correct domain with subdomain\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const construct = new Website(stack, \"TestWebsite\", props);\n\n      // Access private method through bracket notation for testing\n      const getFullDomainName = (construct as any)._getFullDomainName;\n\n      const domainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"blog\",\n        certificateArn: \"arn:test\",\n      };\n\n      const result = getFullDomainName(domainConfig);\n      expect(result).toBe(\"blog.example.com\");\n    });\n\n    test(\"_getFullDomainName returns root domain when subdomain is empty\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      const construct = new Website(stack, \"TestWebsite\", props);\n\n      const getFullDomainName = (construct as any)._getFullDomainName;\n\n      const domainConfig: DomainConfig = {\n        domainName: \"example.com\",\n        subdomainName: \"\",\n        certificateArn: \"arn:test\",\n      };\n\n      const result = getFullDomainName(domainConfig);\n      expect(result).toBe(\"example.com\");\n    });\n  });\n\n  describe(\"Edge cases\", () => {\n    test(\"handles minimal configuration\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"minimal-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      expect(() => {\n        new Website(stack, \"TestWebsite\", props);\n      }).not.toThrow();\n\n      const template = Template.fromStack(stack);\n      template.resourceCountIs(\"AWS::S3::Bucket\", 1);\n      template.resourceCountIs(\"AWS::CloudFront::Distribution\", 1);\n      template.resourceCountIs(\n        \"AWS::CloudFront::CloudFrontOriginAccessIdentity\",\n        1,\n      );\n    });\n\n    test(\"does not create Route53 resources when domain config is not provided\", () => {\n      const props: WebsiteProps = {\n        bucketName: \"test-bucket\",\n        indexFile: \"index.html\",\n        errorFile: \"error.html\",\n      };\n\n      new Website(stack, \"TestWebsite\", props);\n\n      const template = Template.fromStack(stack);\n      template.resourceCountIs(\"AWS::Route53::RecordSet\", 0);\n    });\n  });\n});\n\ndescribe(\"Preview config on Website\", () => {\n  let app: cdk.App;\n  let stack: cdk.Stack;\n\n  beforeEach(() => {\n    app = new cdk.App();\n    stack = new cdk.Stack(app, \"PreviewTestStack\", {\n      env: { account: \"123456789012\", region: \"us-east-1\" },\n    });\n  });\n\n  test(\"creates two preview buckets by default when previewConfig is enabled\", () => {\n    const website = new Website(stack, \"PreviewEnabledWebsite\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"index.html\",\n      errorFile: \"error.html\",\n      previewConfig: {\n        bucketPrefix: \"preview-bucket\",\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs(\"AWS::S3::Bucket\", 3);\n    expect(website.previewEnvironment).toBeDefined();\n  });\n\n  test(\"creates requested number of preview buckets from previewConfig\", () => {\n    new Website(stack, \"PreviewEnabledWebsite\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"index.html\",\n      errorFile: \"error.html\",\n      previewConfig: {\n        bucketPrefix: \"preview-bucket\",\n        bucketCount: 3,\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs(\"AWS::S3::Bucket\", 4);\n  });\n\n  test(\"reuses website index and error files for preview buckets\", () => {\n    new Website(stack, \"PreviewEnabledWebsite\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"app.html\",\n      errorFile: \"fallback.html\",\n      previewConfig: {\n        bucketPrefix: \"preview-bucket\",\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties(\"AWS::S3::Bucket\", {\n      BucketName: \"preview-bucket-0\",\n      WebsiteConfiguration: {\n        IndexDocument: \"app.html\",\n        ErrorDocument: \"fallback.html\",\n      },\n    });\n    template.hasResourceProperties(\"AWS::S3::Bucket\", {\n      BucketName: \"preview-bucket-1\",\n      WebsiteConfiguration: {\n        IndexDocument: \"app.html\",\n        ErrorDocument: \"fallback.html\",\n      },\n    });\n  });\n\n  test(\"creates lease table with repo-pr lookup index when preview is enabled\", () => {\n    new Website(stack, \"PreviewEnabledWebsite\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"index.html\",\n      errorFile: \"error.html\",\n      previewConfig: {\n        bucketPrefix: \"preview-bucket\",\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties(\"AWS::DynamoDB::Table\", {\n      KeySchema: [\n        {\n          AttributeName: \"slotId\",\n          KeyType: \"HASH\",\n        },\n      ],\n      GlobalSecondaryIndexes: [\n        {\n          IndexName: \"RepoPrKeyIndex\",\n          KeySchema: [\n            {\n              AttributeName: \"repoPrKey\",\n              KeyType: \"HASH\",\n            },\n          ],\n          Projection: {\n            ProjectionType: \"ALL\",\n          },\n        },\n      ],\n    });\n  });\n\n  test(\"creates lease API routes when preview is enabled\", () => {\n    new Website(stack, \"PreviewEnabledWebsite\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"index.html\",\n      errorFile: \"error.html\",\n      previewConfig: {\n        bucketPrefix: \"preview-bucket\",\n      },\n    });\n\n    const template = Template.fromStack(stack);\n    template.hasResourceProperties(\"AWS::ApiGateway::Resource\", {\n      PathPart: \"claim\",\n    });\n    template.hasResourceProperties(\"AWS::ApiGateway::Resource\", {\n      PathPart: \"heartbeat\",\n    });\n    template.hasResourceProperties(\"AWS::ApiGateway::Resource\", {\n      PathPart: \"release\",\n    });\n    template.resourceCountIs(\"AWS::ApiGateway::Method\", 3);\n  });\n\n  test(\"does not create preview resources when previewConfig is omitted\", () => {\n    const website = new Website(stack, \"WebsiteWithoutPreview\", {\n      bucketName: \"website-bucket\",\n      indexFile: \"index.html\",\n      errorFile: \"error.html\",\n    });\n\n    const template = Template.fromStack(stack);\n    template.resourceCountIs(\"AWS::DynamoDB::Table\", 0);\n    template.resourceCountIs(\"AWS::ApiGateway::RestApi\", 0);\n    expect(website.previewEnvironment).toBeUndefined();\n  });\n});\n"]}