@sitblueprint/website-construct 0.1.4 → 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/README.md +85 -0
- package/docs/CHANGELOG.md +26 -0
- package/lambda/index.d.ts +11 -0
- package/lambda/index.js +238 -0
- package/lambda/index.ts +335 -0
- package/lib/index.d.ts +44 -0
- package/lib/index.js +173 -10
- package/lib/index.ts +250 -8
- package/package.json +5 -5
- package/test/website-construct.test.js +212 -8
- package/test/website-construct.test.ts +229 -0
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@sitblueprint/website-construct",
|
|
3
|
-
"version": "0.1.
|
|
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",
|
|
@@ -31,16 +31,16 @@
|
|
|
31
31
|
},
|
|
32
32
|
"devDependencies": {
|
|
33
33
|
"@types/jest": "^29.5.14",
|
|
34
|
-
"@types/node": "
|
|
35
|
-
"aws-cdk-lib": "2.
|
|
34
|
+
"@types/node": "24.10.1",
|
|
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",
|
|
39
39
|
"ts-jest": "^29.2.5",
|
|
40
|
-
"typescript": "~5.
|
|
40
|
+
"typescript": "~5.9.3"
|
|
41
41
|
},
|
|
42
42
|
"peerDependencies": {
|
|
43
|
-
"aws-cdk-lib": "2.
|
|
43
|
+
"aws-cdk-lib": "2.237.1",
|
|
44
44
|
"constructs": "^10.0.0"
|
|
45
45
|
}
|
|
46
46
|
}
|
|
@@ -15,13 +15,23 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
|
|
|
15
15
|
}) : function(o, v) {
|
|
16
16
|
o["default"] = v;
|
|
17
17
|
});
|
|
18
|
-
var __importStar = (this && this.__importStar) || function (
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
};
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
25
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
36
|
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
27
37
|
const cdk = __importStar(require("aws-cdk-lib"));
|
|
@@ -226,6 +236,74 @@ describe("Website", () => {
|
|
|
226
236
|
Name: "example.com.",
|
|
227
237
|
});
|
|
228
238
|
});
|
|
239
|
+
test("configures CloudFront with both subdomain and root domain aliases when includeRootDomain is true", () => {
|
|
240
|
+
const dualDomainConfig = {
|
|
241
|
+
domainName: "example.com",
|
|
242
|
+
subdomainName: "www",
|
|
243
|
+
certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
244
|
+
includeRootDomain: true,
|
|
245
|
+
};
|
|
246
|
+
const props = {
|
|
247
|
+
bucketName: "test-website-bucket",
|
|
248
|
+
indexFile: "index.html",
|
|
249
|
+
errorFile: "error.html",
|
|
250
|
+
domainConfig: dualDomainConfig,
|
|
251
|
+
};
|
|
252
|
+
new lib_1.Website(stack, "TestWebsite", props);
|
|
253
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
254
|
+
template.hasResourceProperties("AWS::CloudFront::Distribution", {
|
|
255
|
+
DistributionConfig: {
|
|
256
|
+
Aliases: ["www.example.com", "example.com"],
|
|
257
|
+
},
|
|
258
|
+
});
|
|
259
|
+
});
|
|
260
|
+
test("creates two Route53 A records when includeRootDomain is true", () => {
|
|
261
|
+
const dualDomainConfig = {
|
|
262
|
+
domainName: "example.com",
|
|
263
|
+
subdomainName: "www",
|
|
264
|
+
certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
265
|
+
includeRootDomain: true,
|
|
266
|
+
};
|
|
267
|
+
const props = {
|
|
268
|
+
bucketName: "test-website-bucket",
|
|
269
|
+
indexFile: "index.html",
|
|
270
|
+
errorFile: "error.html",
|
|
271
|
+
domainConfig: dualDomainConfig,
|
|
272
|
+
};
|
|
273
|
+
new lib_1.Website(stack, "TestWebsite", props);
|
|
274
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
275
|
+
template.resourceCountIs("AWS::Route53::RecordSet", 2);
|
|
276
|
+
template.hasResourceProperties("AWS::Route53::RecordSet", {
|
|
277
|
+
Name: "www.example.com.",
|
|
278
|
+
Type: "A",
|
|
279
|
+
});
|
|
280
|
+
template.hasResourceProperties("AWS::Route53::RecordSet", {
|
|
281
|
+
Name: "example.com.",
|
|
282
|
+
Type: "A",
|
|
283
|
+
});
|
|
284
|
+
});
|
|
285
|
+
test("ignores includeRootDomain if subdomain is empty to avoid duplicates", () => {
|
|
286
|
+
const domainConfigWithoutSub = {
|
|
287
|
+
domainName: "example.com",
|
|
288
|
+
subdomainName: "",
|
|
289
|
+
certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
290
|
+
includeRootDomain: true,
|
|
291
|
+
};
|
|
292
|
+
const props = {
|
|
293
|
+
bucketName: "test-website-bucket",
|
|
294
|
+
indexFile: "index.html",
|
|
295
|
+
errorFile: "error.html",
|
|
296
|
+
domainConfig: domainConfigWithoutSub,
|
|
297
|
+
};
|
|
298
|
+
new lib_1.Website(stack, "TestWebsite", props);
|
|
299
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
300
|
+
template.resourceCountIs("AWS::Route53::RecordSet", 1);
|
|
301
|
+
template.hasResourceProperties("AWS::CloudFront::Distribution", {
|
|
302
|
+
DistributionConfig: {
|
|
303
|
+
Aliases: ["example.com"],
|
|
304
|
+
},
|
|
305
|
+
});
|
|
306
|
+
});
|
|
229
307
|
});
|
|
230
308
|
describe("Private methods", () => {
|
|
231
309
|
test("_getFullDomainName returns correct domain with subdomain", () => {
|
|
@@ -289,4 +367,130 @@ describe("Website", () => {
|
|
|
289
367
|
});
|
|
290
368
|
});
|
|
291
369
|
});
|
|
292
|
-
//# 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\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"]}
|
|
@@ -249,6 +249,95 @@ describe("Website", () => {
|
|
|
249
249
|
Name: "example.com.",
|
|
250
250
|
});
|
|
251
251
|
});
|
|
252
|
+
|
|
253
|
+
test("configures CloudFront with both subdomain and root domain aliases when includeRootDomain is true", () => {
|
|
254
|
+
const dualDomainConfig: DomainConfig = {
|
|
255
|
+
domainName: "example.com",
|
|
256
|
+
subdomainName: "www",
|
|
257
|
+
certificateArn:
|
|
258
|
+
"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
259
|
+
includeRootDomain: true,
|
|
260
|
+
};
|
|
261
|
+
|
|
262
|
+
const props: WebsiteProps = {
|
|
263
|
+
bucketName: "test-website-bucket",
|
|
264
|
+
indexFile: "index.html",
|
|
265
|
+
errorFile: "error.html",
|
|
266
|
+
domainConfig: dualDomainConfig,
|
|
267
|
+
};
|
|
268
|
+
|
|
269
|
+
new Website(stack, "TestWebsite", props);
|
|
270
|
+
|
|
271
|
+
const template = Template.fromStack(stack);
|
|
272
|
+
|
|
273
|
+
template.hasResourceProperties("AWS::CloudFront::Distribution", {
|
|
274
|
+
DistributionConfig: {
|
|
275
|
+
Aliases: ["www.example.com", "example.com"],
|
|
276
|
+
},
|
|
277
|
+
});
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
test("creates two Route53 A records when includeRootDomain is true", () => {
|
|
281
|
+
const dualDomainConfig: DomainConfig = {
|
|
282
|
+
domainName: "example.com",
|
|
283
|
+
subdomainName: "www",
|
|
284
|
+
certificateArn:
|
|
285
|
+
"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
286
|
+
includeRootDomain: true,
|
|
287
|
+
};
|
|
288
|
+
|
|
289
|
+
const props: WebsiteProps = {
|
|
290
|
+
bucketName: "test-website-bucket",
|
|
291
|
+
indexFile: "index.html",
|
|
292
|
+
errorFile: "error.html",
|
|
293
|
+
domainConfig: dualDomainConfig,
|
|
294
|
+
};
|
|
295
|
+
|
|
296
|
+
new Website(stack, "TestWebsite", props);
|
|
297
|
+
|
|
298
|
+
const template = Template.fromStack(stack);
|
|
299
|
+
|
|
300
|
+
template.resourceCountIs("AWS::Route53::RecordSet", 2);
|
|
301
|
+
|
|
302
|
+
template.hasResourceProperties("AWS::Route53::RecordSet", {
|
|
303
|
+
Name: "www.example.com.",
|
|
304
|
+
Type: "A",
|
|
305
|
+
});
|
|
306
|
+
|
|
307
|
+
template.hasResourceProperties("AWS::Route53::RecordSet", {
|
|
308
|
+
Name: "example.com.",
|
|
309
|
+
Type: "A",
|
|
310
|
+
});
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
test("ignores includeRootDomain if subdomain is empty to avoid duplicates", () => {
|
|
314
|
+
const domainConfigWithoutSub: DomainConfig = {
|
|
315
|
+
domainName: "example.com",
|
|
316
|
+
subdomainName: "",
|
|
317
|
+
certificateArn:
|
|
318
|
+
"arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012",
|
|
319
|
+
includeRootDomain: true,
|
|
320
|
+
};
|
|
321
|
+
|
|
322
|
+
const props: WebsiteProps = {
|
|
323
|
+
bucketName: "test-website-bucket",
|
|
324
|
+
indexFile: "index.html",
|
|
325
|
+
errorFile: "error.html",
|
|
326
|
+
domainConfig: domainConfigWithoutSub,
|
|
327
|
+
};
|
|
328
|
+
|
|
329
|
+
new Website(stack, "TestWebsite", props);
|
|
330
|
+
|
|
331
|
+
const template = Template.fromStack(stack);
|
|
332
|
+
|
|
333
|
+
template.resourceCountIs("AWS::Route53::RecordSet", 1);
|
|
334
|
+
|
|
335
|
+
template.hasResourceProperties("AWS::CloudFront::Distribution", {
|
|
336
|
+
DistributionConfig: {
|
|
337
|
+
Aliases: ["example.com"],
|
|
338
|
+
},
|
|
339
|
+
});
|
|
340
|
+
});
|
|
252
341
|
});
|
|
253
342
|
|
|
254
343
|
describe("Private methods", () => {
|
|
@@ -331,3 +420,143 @@ describe("Website", () => {
|
|
|
331
420
|
});
|
|
332
421
|
});
|
|
333
422
|
});
|
|
423
|
+
|
|
424
|
+
describe("Preview config on Website", () => {
|
|
425
|
+
let app: cdk.App;
|
|
426
|
+
let stack: cdk.Stack;
|
|
427
|
+
|
|
428
|
+
beforeEach(() => {
|
|
429
|
+
app = new cdk.App();
|
|
430
|
+
stack = new cdk.Stack(app, "PreviewTestStack", {
|
|
431
|
+
env: { account: "123456789012", region: "us-east-1" },
|
|
432
|
+
});
|
|
433
|
+
});
|
|
434
|
+
|
|
435
|
+
test("creates two preview buckets by default when previewConfig is enabled", () => {
|
|
436
|
+
const website = new Website(stack, "PreviewEnabledWebsite", {
|
|
437
|
+
bucketName: "website-bucket",
|
|
438
|
+
indexFile: "index.html",
|
|
439
|
+
errorFile: "error.html",
|
|
440
|
+
previewConfig: {
|
|
441
|
+
bucketPrefix: "preview-bucket",
|
|
442
|
+
},
|
|
443
|
+
});
|
|
444
|
+
|
|
445
|
+
const template = Template.fromStack(stack);
|
|
446
|
+
template.resourceCountIs("AWS::S3::Bucket", 3);
|
|
447
|
+
expect(website.previewEnvironment).toBeDefined();
|
|
448
|
+
});
|
|
449
|
+
|
|
450
|
+
test("creates requested number of preview buckets from previewConfig", () => {
|
|
451
|
+
new Website(stack, "PreviewEnabledWebsite", {
|
|
452
|
+
bucketName: "website-bucket",
|
|
453
|
+
indexFile: "index.html",
|
|
454
|
+
errorFile: "error.html",
|
|
455
|
+
previewConfig: {
|
|
456
|
+
bucketPrefix: "preview-bucket",
|
|
457
|
+
bucketCount: 3,
|
|
458
|
+
},
|
|
459
|
+
});
|
|
460
|
+
|
|
461
|
+
const template = Template.fromStack(stack);
|
|
462
|
+
template.resourceCountIs("AWS::S3::Bucket", 4);
|
|
463
|
+
});
|
|
464
|
+
|
|
465
|
+
test("reuses website index and error files for preview buckets", () => {
|
|
466
|
+
new Website(stack, "PreviewEnabledWebsite", {
|
|
467
|
+
bucketName: "website-bucket",
|
|
468
|
+
indexFile: "app.html",
|
|
469
|
+
errorFile: "fallback.html",
|
|
470
|
+
previewConfig: {
|
|
471
|
+
bucketPrefix: "preview-bucket",
|
|
472
|
+
},
|
|
473
|
+
});
|
|
474
|
+
|
|
475
|
+
const template = Template.fromStack(stack);
|
|
476
|
+
template.hasResourceProperties("AWS::S3::Bucket", {
|
|
477
|
+
BucketName: "preview-bucket-0",
|
|
478
|
+
WebsiteConfiguration: {
|
|
479
|
+
IndexDocument: "app.html",
|
|
480
|
+
ErrorDocument: "fallback.html",
|
|
481
|
+
},
|
|
482
|
+
});
|
|
483
|
+
template.hasResourceProperties("AWS::S3::Bucket", {
|
|
484
|
+
BucketName: "preview-bucket-1",
|
|
485
|
+
WebsiteConfiguration: {
|
|
486
|
+
IndexDocument: "app.html",
|
|
487
|
+
ErrorDocument: "fallback.html",
|
|
488
|
+
},
|
|
489
|
+
});
|
|
490
|
+
});
|
|
491
|
+
|
|
492
|
+
test("creates lease table with repo-pr lookup index when preview is enabled", () => {
|
|
493
|
+
new Website(stack, "PreviewEnabledWebsite", {
|
|
494
|
+
bucketName: "website-bucket",
|
|
495
|
+
indexFile: "index.html",
|
|
496
|
+
errorFile: "error.html",
|
|
497
|
+
previewConfig: {
|
|
498
|
+
bucketPrefix: "preview-bucket",
|
|
499
|
+
},
|
|
500
|
+
});
|
|
501
|
+
|
|
502
|
+
const template = Template.fromStack(stack);
|
|
503
|
+
template.hasResourceProperties("AWS::DynamoDB::Table", {
|
|
504
|
+
KeySchema: [
|
|
505
|
+
{
|
|
506
|
+
AttributeName: "slotId",
|
|
507
|
+
KeyType: "HASH",
|
|
508
|
+
},
|
|
509
|
+
],
|
|
510
|
+
GlobalSecondaryIndexes: [
|
|
511
|
+
{
|
|
512
|
+
IndexName: "RepoPrKeyIndex",
|
|
513
|
+
KeySchema: [
|
|
514
|
+
{
|
|
515
|
+
AttributeName: "repoPrKey",
|
|
516
|
+
KeyType: "HASH",
|
|
517
|
+
},
|
|
518
|
+
],
|
|
519
|
+
Projection: {
|
|
520
|
+
ProjectionType: "ALL",
|
|
521
|
+
},
|
|
522
|
+
},
|
|
523
|
+
],
|
|
524
|
+
});
|
|
525
|
+
});
|
|
526
|
+
|
|
527
|
+
test("creates lease API routes when preview is enabled", () => {
|
|
528
|
+
new Website(stack, "PreviewEnabledWebsite", {
|
|
529
|
+
bucketName: "website-bucket",
|
|
530
|
+
indexFile: "index.html",
|
|
531
|
+
errorFile: "error.html",
|
|
532
|
+
previewConfig: {
|
|
533
|
+
bucketPrefix: "preview-bucket",
|
|
534
|
+
},
|
|
535
|
+
});
|
|
536
|
+
|
|
537
|
+
const template = Template.fromStack(stack);
|
|
538
|
+
template.hasResourceProperties("AWS::ApiGateway::Resource", {
|
|
539
|
+
PathPart: "claim",
|
|
540
|
+
});
|
|
541
|
+
template.hasResourceProperties("AWS::ApiGateway::Resource", {
|
|
542
|
+
PathPart: "heartbeat",
|
|
543
|
+
});
|
|
544
|
+
template.hasResourceProperties("AWS::ApiGateway::Resource", {
|
|
545
|
+
PathPart: "release",
|
|
546
|
+
});
|
|
547
|
+
template.resourceCountIs("AWS::ApiGateway::Method", 3);
|
|
548
|
+
});
|
|
549
|
+
|
|
550
|
+
test("does not create preview resources when previewConfig is omitted", () => {
|
|
551
|
+
const website = new Website(stack, "WebsiteWithoutPreview", {
|
|
552
|
+
bucketName: "website-bucket",
|
|
553
|
+
indexFile: "index.html",
|
|
554
|
+
errorFile: "error.html",
|
|
555
|
+
});
|
|
556
|
+
|
|
557
|
+
const template = Template.fromStack(stack);
|
|
558
|
+
template.resourceCountIs("AWS::DynamoDB::Table", 0);
|
|
559
|
+
template.resourceCountIs("AWS::ApiGateway::RestApi", 0);
|
|
560
|
+
expect(website.previewEnvironment).toBeUndefined();
|
|
561
|
+
});
|
|
562
|
+
});
|