@sitblueprint/website-construct 0.1.4 → 0.1.5

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 CHANGED
@@ -6,6 +6,7 @@ A reusable [AWS CDK](https://docs.aws.amazon.com/cdk/) construct to deploy a web
6
6
 
7
7
  - CDN caching via CloudFont
8
8
  - Deployment via S3
9
+ - Dual domain support (e.g., deploy to both `www.example.com` and `example.com` simultaneously)
9
10
  - Hardened S3 bucket defaults with bucket-owner-only ACLs and automatic SSE
10
11
  - Direct access to the underlying S3 bucket and CloudFront distribution for advanced customization
11
12
 
@@ -39,6 +40,7 @@ export class MyWebsiteStack extends cdk.Stack {
39
40
  domainName: "example.com",
40
41
  subdomainName: "www",
41
42
  certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/abc123",
43
+ includeRootDomain: true, // Optional: also deploy to example.com
42
44
  },
43
45
  });
44
46
 
package/docs/CHANGELOG.md CHANGED
@@ -7,6 +7,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
7
7
 
8
8
  ---
9
9
 
10
+ ## [v0.1.5]
11
+
12
+ ### Added
13
+
14
+ - `includeRootDomain` option to `DomainConfig` to deploy to both subdomain and root domain simultaneously.
15
+
10
16
  ## [v0.1.4] - 2025-09-30
11
17
 
12
18
  ### Security
package/lib/index.d.ts CHANGED
@@ -10,6 +10,11 @@ export interface DomainConfig {
10
10
  subdomainName: string;
11
11
  /** The ARN of the SSL certificate to use for the domain. */
12
12
  certificateArn: string;
13
+ /**
14
+ * If true, creates an additional Route 53 record for the root domain pointing to the CloudFront distribution.
15
+ * @default false
16
+ */
17
+ includeRootDomain?: boolean;
13
18
  }
14
19
  export interface WebsiteProps {
15
20
  /** The name of the S3 bucket that will host the website content. */
package/lib/index.js CHANGED
@@ -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 (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
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
  exports.Website = void 0;
27
37
  const constructs_1 = require("constructs");
@@ -54,6 +64,14 @@ class Website extends constructs_1.Construct {
54
64
  new iam.CanonicalUserPrincipal(oai.cloudFrontOriginAccessIdentityS3CanonicalUserId),
55
65
  ],
56
66
  }));
67
+ const domainNames = [];
68
+ if (props.domainConfig) {
69
+ domainNames.push(this._getFullDomainName(props.domainConfig));
70
+ if (props.domainConfig.includeRootDomain &&
71
+ props.domainConfig.subdomainName) {
72
+ domainNames.push(props.domainConfig.domainName);
73
+ }
74
+ }
57
75
  this.distribution = new cloudfont.Distribution(this, `${props.bucketName}-distribution`, {
58
76
  defaultBehavior: {
59
77
  origin: new origins.S3StaticWebsiteOrigin(this.bucket),
@@ -70,7 +88,7 @@ class Website extends constructs_1.Construct {
70
88
  priceClass: cloudfont.PriceClass.PRICE_CLASS_100,
71
89
  ...(props.domainConfig
72
90
  ? {
73
- domainNames: [this._getFullDomainName(props.domainConfig)],
91
+ domainNames: domainNames,
74
92
  certificate: this._getCertificate(props.domainConfig.certificateArn),
75
93
  }
76
94
  : {}),
@@ -85,6 +103,14 @@ class Website extends constructs_1.Construct {
85
103
  target: cdk.aws_route53.RecordTarget.fromAlias(new cdk.aws_route53_targets.CloudFrontTarget(this.distribution)),
86
104
  });
87
105
  domainARecord.node.addDependency(this.distribution);
106
+ if (props.domainConfig.includeRootDomain &&
107
+ props.domainConfig.subdomainName) {
108
+ new route53.ARecord(this, "RootDomainARecord", {
109
+ zone: hostedZone,
110
+ recordName: props.domainConfig.domainName,
111
+ target: cdk.aws_route53.RecordTarget.fromAlias(new cdk.aws_route53_targets.CloudFrontTarget(this.distribution)),
112
+ }).node.addDependency(this.distribution);
113
+ }
88
114
  }
89
115
  new cdk.CfnOutput(this, "cloudfront-website-url", {
90
116
  value: this.distribution.distributionDomainName,
@@ -111,4 +137,4 @@ class Website extends constructs_1.Construct {
111
137
  }
112
138
  }
113
139
  exports.Website = Website;
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,2CAAuC;AACvC,uDAAyC;AACzC,sEAAwD;AACxD,4EAA8D;AAC9D,uFAAyE;AACzE,iDAAmC;AACnC,yDAA2C;AAC3C,iEAAmD;AA8BnD,MAAa,OAAQ,SAAQ,sBAAS;IACpB,MAAM,CAAY;IAClB,YAAY,CAAyB;IAErD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAmB;QAC3D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,UAAU,EAAE;YAClD,oBAAoB,EAAE,KAAK,CAAC,SAAS;YACrC,oBAAoB,EAAE,KAAK,CAAC,SAAS;YACrC,gBAAgB,EAAE,IAAI;YACtB,aAAa,EAAE,GAAG,CAAC,aAAa,CAAC,OAAO;YACxC,iBAAiB,EAAE,EAAE,CAAC,iBAAiB,CAAC,eAAe;YACvD,aAAa,EAAE,EAAE,CAAC,mBAAmB,CAAC,yBAAyB;YAC/D,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;SAC3C,CAAC,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,SAAS,CAAC,oBAAoB,CAC5C,IAAI,EACJ,GAAG,KAAK,CAAC,UAAU,MAAM,CAC1B,CAAC;QACF,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAC7B,IAAI,GAAG,CAAC,eAAe,CAAC;YACtB,OAAO,EAAE,CAAC,cAAc,CAAC;YACzB,SAAS,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;YAC3C,UAAU,EAAE;gBACV,IAAI,GAAG,CAAC,sBAAsB,CAC5B,GAAG,CAAC,+CAA+C,CACpD;aACF;SACF,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,YAAY,GAAG,IAAI,SAAS,CAAC,YAAY,CAC5C,IAAI,EACJ,GAAG,KAAK,CAAC,UAAU,eAAe,EAClC;YACE,eAAe,EAAE;gBACf,MAAM,EAAE,IAAI,OAAO,CAAC,qBAAqB,CAAC,IAAI,CAAC,MAAM,CAAC;gBACtD,oBAAoB,EAClB,SAAS,CAAC,oBAAoB,CAAC,iBAAiB;aACnD;YACD,cAAc,EAAE;gBACd;oBACE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,KAAK,CAAC,wBAAwB,IAAI,WAAW;oBAC/D,GAAG,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;iBAC9B;aACF;YACD,UAAU,EAAE,SAAS,CAAC,UAAU,CAAC,eAAe;YAChD,GAAG,CAAC,KAAK,CAAC,YAAY;gBACpB,CAAC,CAAC;oBACE,WAAW,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;oBAC1D,WAAW,EAAE,IAAI,CAAC,eAAe,CAC/B,KAAK,CAAC,YAAY,CAAC,cAAc,CAClC;iBACF;gBACH,CAAC,CAAC,EAAE,CAAC;SACR,CACF,CAAC;QAEF,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;YACvB,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,YAAY,EAAE;gBACnE,UAAU,EAAE,KAAK,CAAC,YAAY,CAAC,UAAU;aAC1C,CAAC,CAAC;YACH,MAAM,aAAa,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE,eAAe,EAAE;gBAC/D,IAAI,EAAE,UAAU;gBAChB,UAAU,EAAE,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,YAAY,CAAC;gBACvD,MAAM,EAAE,GAAG,CAAC,WAAW,CAAC,YAAY,CAAC,SAAS,CAC5C,IAAI,GAAG,CAAC,mBAAmB,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAChE;aACF,CAAC,CAAC;YACH,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAChD,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,sBAAsB;YAC/C,WAAW,EAAE,qCAAqC;SACnD,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACxC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,gBAAgB;YACnC,WAAW,EAAE,uBAAuB;SACrC,CAAC,CAAC;QAEH,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;YACvB,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,aAAa,EAAE;gBACrC,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,UAAU;gBACnC,WAAW,EAAE,aAAa;aAC3B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,kBAAkB,CAAC,YAA0B;QACnD,OAAO,YAAY,CAAC,aAAa;YAC/B,CAAC,CAAC,GAAG,YAAY,CAAC,aAAa,IAAI,YAAY,CAAC,UAAU,EAAE;YAC5D,CAAC,CAAC,YAAY,CAAC,UAAU,CAAC;IAC9B,CAAC;IAEO,eAAe,CAAC,GAAW;QACjC,OAAO,kBAAkB,CAAC,WAAW,CAAC,kBAAkB,CACtD,IAAI,EACJ,cAAc,EACd,GAAG,CACJ,CAAC;IACJ,CAAC;CACF;AAzGD,0BAyGC","sourcesContent":["import { Construct } from \"constructs\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport * as cloudfont from \"aws-cdk-lib/aws-cloudfront\";\nimport * as origins from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport * as certificatemanager from \"aws-cdk-lib/aws-certificatemanager\";\nimport * as cdk from \"aws-cdk-lib\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as route53 from \"aws-cdk-lib/aws-route53\";\n\nexport interface DomainConfig {\n  /** The root domain name (e.g., example.com).\n   * There must be an associated hosted zone in Route 53 for this domain.\n   */\n  domainName: string;\n  /** The subdomain name */\n  subdomainName: string;\n  /** The ARN of the SSL certificate to use for the domain. */\n  certificateArn: string;\n}\n\nexport interface WebsiteProps {\n  /** The name of the S3 bucket that will host the website content. */\n  bucketName: string;\n\n  /** The path to the index document that will be served as the default page. */\n  indexFile: string;\n\n  /** The path to the error document that will be served when an error occurs. */\n  errorFile: string;\n\n  /** Optional configuration for custom domain setup. */\n  domainConfig?: DomainConfig;\n\n  /** Optional path to a custom 404 page. If not specified, the error file will be used. */\n  notFoundResponsePagePath?: string;\n}\n\nexport class Website extends Construct {\n  public readonly bucket: s3.Bucket;\n  public readonly distribution: cloudfont.Distribution;\n\n  constructor(scope: Construct, id: string, props: WebsiteProps) {\n    super(scope, id);\n    this.bucket = new s3.Bucket(this, props.bucketName, {\n      websiteIndexDocument: props.indexFile,\n      websiteErrorDocument: props.errorFile,\n      publicReadAccess: true,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ACLS_ONLY,\n      accessControl: s3.BucketAccessControl.BUCKET_OWNER_FULL_CONTROL,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n    const oai = new cloudfont.OriginAccessIdentity(\n      this,\n      `${props.bucketName}-OAI`\n    );\n    this.bucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        actions: [\"s3:GetObject\"],\n        resources: [this.bucket.arnForObjects(\"*\")],\n        principals: [\n          new iam.CanonicalUserPrincipal(\n            oai.cloudFrontOriginAccessIdentityS3CanonicalUserId\n          ),\n        ],\n      })\n    );\n\n    this.distribution = new cloudfont.Distribution(\n      this,\n      `${props.bucketName}-distribution`,\n      {\n        defaultBehavior: {\n          origin: new origins.S3StaticWebsiteOrigin(this.bucket),\n          viewerProtocolPolicy:\n            cloudfont.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        },\n        errorResponses: [\n          {\n            httpStatus: 404,\n            responseHttpStatus: 404,\n            responsePagePath: props.notFoundResponsePagePath || `/404.html`,\n            ttl: cdk.Duration.minutes(30),\n          },\n        ],\n        priceClass: cloudfont.PriceClass.PRICE_CLASS_100,\n        ...(props.domainConfig\n          ? {\n              domainNames: [this._getFullDomainName(props.domainConfig)],\n              certificate: this._getCertificate(\n                props.domainConfig.certificateArn\n              ),\n            }\n          : {}),\n      }\n    );\n\n    if (props.domainConfig) {\n      const hostedZone = route53.HostedZone.fromLookup(this, \"HostedZone\", {\n        domainName: props.domainConfig.domainName,\n      });\n      const domainARecord = new route53.ARecord(this, \"DomainARecord\", {\n        zone: hostedZone,\n        recordName: this._getFullDomainName(props.domainConfig),\n        target: cdk.aws_route53.RecordTarget.fromAlias(\n          new cdk.aws_route53_targets.CloudFrontTarget(this.distribution)\n        ),\n      });\n      domainARecord.node.addDependency(this.distribution);\n    }\n\n    new cdk.CfnOutput(this, \"cloudfront-website-url\", {\n      value: this.distribution.distributionDomainName,\n      description: \"CloudFront Distribution Domain Name\",\n    });\n\n    new cdk.CfnOutput(this, \"s3-website-url\", {\n      value: this.bucket.bucketWebsiteUrl,\n      description: \"S3 Bucket Website URL\",\n    });\n\n    if (props.domainConfig) {\n      new cdk.CfnOutput(this, \"website-url\", {\n        value: this.distribution.domainName,\n        description: \"Website URL\",\n      });\n    }\n  }\n\n  private _getFullDomainName(domainConfig: DomainConfig): string {\n    return domainConfig.subdomainName\n      ? `${domainConfig.subdomainName}.${domainConfig.domainName}`\n      : domainConfig.domainName;\n  }\n\n  private _getCertificate(arn: string): certificatemanager.ICertificate {\n    return certificatemanager.Certificate.fromCertificateArn(\n      this,\n      `website-cert`,\n      arn\n    );\n  }\n}\n"]}
140
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from \"constructs\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport * as cloudfont from \"aws-cdk-lib/aws-cloudfront\";\nimport * as origins from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport * as certificatemanager from \"aws-cdk-lib/aws-certificatemanager\";\nimport * as cdk from \"aws-cdk-lib\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as route53 from \"aws-cdk-lib/aws-route53\";\n\nexport interface DomainConfig {\n  /** The root domain name (e.g., example.com).\n   * There must be an associated hosted zone in Route 53 for this domain.\n   */\n  domainName: string;\n  /** The subdomain name */\n  subdomainName: string;\n  /** The ARN of the SSL certificate to use for the domain. */\n  certificateArn: string;\n  /**\n   * If true, creates an additional Route 53 record for the root domain pointing to the CloudFront distribution.\n   * @default false\n   */\n  includeRootDomain?: boolean;\n}\n\nexport interface WebsiteProps {\n  /** The name of the S3 bucket that will host the website content. */\n  bucketName: string;\n\n  /** The path to the index document that will be served as the default page. */\n  indexFile: string;\n\n  /** The path to the error document that will be served when an error occurs. */\n  errorFile: string;\n\n  /** Optional configuration for custom domain setup. */\n  domainConfig?: DomainConfig;\n\n  /** Optional path to a custom 404 page. If not specified, the error file will be used. */\n  notFoundResponsePagePath?: string;\n}\n\nexport class Website extends Construct {\n  public readonly bucket: s3.Bucket;\n  public readonly distribution: cloudfont.Distribution;\n\n  constructor(scope: Construct, id: string, props: WebsiteProps) {\n    super(scope, id);\n    this.bucket = new s3.Bucket(this, props.bucketName, {\n      websiteIndexDocument: props.indexFile,\n      websiteErrorDocument: props.errorFile,\n      publicReadAccess: true,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ACLS_ONLY,\n      accessControl: s3.BucketAccessControl.BUCKET_OWNER_FULL_CONTROL,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n    const oai = new cloudfont.OriginAccessIdentity(\n      this,\n      `${props.bucketName}-OAI`,\n    );\n    this.bucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        actions: [\"s3:GetObject\"],\n        resources: [this.bucket.arnForObjects(\"*\")],\n        principals: [\n          new iam.CanonicalUserPrincipal(\n            oai.cloudFrontOriginAccessIdentityS3CanonicalUserId,\n          ),\n        ],\n      }),\n    );\n    const domainNames: string[] = [];\n    if (props.domainConfig) {\n      domainNames.push(this._getFullDomainName(props.domainConfig));\n      if (\n        props.domainConfig.includeRootDomain &&\n        props.domainConfig.subdomainName\n      ) {\n        domainNames.push(props.domainConfig.domainName);\n      }\n    }\n\n    this.distribution = new cloudfont.Distribution(\n      this,\n      `${props.bucketName}-distribution`,\n      {\n        defaultBehavior: {\n          origin: new origins.S3StaticWebsiteOrigin(this.bucket),\n          viewerProtocolPolicy:\n            cloudfont.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        },\n        errorResponses: [\n          {\n            httpStatus: 404,\n            responseHttpStatus: 404,\n            responsePagePath: props.notFoundResponsePagePath || `/404.html`,\n            ttl: cdk.Duration.minutes(30),\n          },\n        ],\n        priceClass: cloudfont.PriceClass.PRICE_CLASS_100,\n        ...(props.domainConfig\n          ? {\n              domainNames: domainNames,\n              certificate: this._getCertificate(\n                props.domainConfig.certificateArn,\n              ),\n            }\n          : {}),\n      },\n    );\n\n    if (props.domainConfig) {\n      const hostedZone = route53.HostedZone.fromLookup(this, \"HostedZone\", {\n        domainName: props.domainConfig.domainName,\n      });\n      const domainARecord = new route53.ARecord(this, \"DomainARecord\", {\n        zone: hostedZone,\n        recordName: this._getFullDomainName(props.domainConfig),\n        target: cdk.aws_route53.RecordTarget.fromAlias(\n          new cdk.aws_route53_targets.CloudFrontTarget(this.distribution),\n        ),\n      });\n      domainARecord.node.addDependency(this.distribution);\n\n      if (\n        props.domainConfig.includeRootDomain &&\n        props.domainConfig.subdomainName\n      ) {\n        new route53.ARecord(this, \"RootDomainARecord\", {\n          zone: hostedZone,\n          recordName: props.domainConfig.domainName,\n          target: cdk.aws_route53.RecordTarget.fromAlias(\n            new cdk.aws_route53_targets.CloudFrontTarget(this.distribution),\n          ),\n        }).node.addDependency(this.distribution);\n      }\n    }\n\n    new cdk.CfnOutput(this, \"cloudfront-website-url\", {\n      value: this.distribution.distributionDomainName,\n      description: \"CloudFront Distribution Domain Name\",\n    });\n\n    new cdk.CfnOutput(this, \"s3-website-url\", {\n      value: this.bucket.bucketWebsiteUrl,\n      description: \"S3 Bucket Website URL\",\n    });\n\n    if (props.domainConfig) {\n      new cdk.CfnOutput(this, \"website-url\", {\n        value: this.distribution.domainName,\n        description: \"Website URL\",\n      });\n    }\n  }\n\n  private _getFullDomainName(domainConfig: DomainConfig): string {\n    return domainConfig.subdomainName\n      ? `${domainConfig.subdomainName}.${domainConfig.domainName}`\n      : domainConfig.domainName;\n  }\n\n  private _getCertificate(arn: string): certificatemanager.ICertificate {\n    return certificatemanager.Certificate.fromCertificateArn(\n      this,\n      `website-cert`,\n      arn,\n    );\n  }\n}\n"]}
package/lib/index.ts CHANGED
@@ -16,6 +16,11 @@ export interface DomainConfig {
16
16
  subdomainName: string;
17
17
  /** The ARN of the SSL certificate to use for the domain. */
18
18
  certificateArn: string;
19
+ /**
20
+ * If true, creates an additional Route 53 record for the root domain pointing to the CloudFront distribution.
21
+ * @default false
22
+ */
23
+ includeRootDomain?: boolean;
19
24
  }
20
25
 
21
26
  export interface WebsiteProps {
@@ -52,7 +57,7 @@ export class Website extends Construct {
52
57
  });
53
58
  const oai = new cloudfont.OriginAccessIdentity(
54
59
  this,
55
- `${props.bucketName}-OAI`
60
+ `${props.bucketName}-OAI`,
56
61
  );
57
62
  this.bucket.addToResourcePolicy(
58
63
  new iam.PolicyStatement({
@@ -60,11 +65,21 @@ export class Website extends Construct {
60
65
  resources: [this.bucket.arnForObjects("*")],
61
66
  principals: [
62
67
  new iam.CanonicalUserPrincipal(
63
- oai.cloudFrontOriginAccessIdentityS3CanonicalUserId
68
+ oai.cloudFrontOriginAccessIdentityS3CanonicalUserId,
64
69
  ),
65
70
  ],
66
- })
71
+ }),
67
72
  );
73
+ const domainNames: string[] = [];
74
+ if (props.domainConfig) {
75
+ domainNames.push(this._getFullDomainName(props.domainConfig));
76
+ if (
77
+ props.domainConfig.includeRootDomain &&
78
+ props.domainConfig.subdomainName
79
+ ) {
80
+ domainNames.push(props.domainConfig.domainName);
81
+ }
82
+ }
68
83
 
69
84
  this.distribution = new cloudfont.Distribution(
70
85
  this,
@@ -86,13 +101,13 @@ export class Website extends Construct {
86
101
  priceClass: cloudfont.PriceClass.PRICE_CLASS_100,
87
102
  ...(props.domainConfig
88
103
  ? {
89
- domainNames: [this._getFullDomainName(props.domainConfig)],
104
+ domainNames: domainNames,
90
105
  certificate: this._getCertificate(
91
- props.domainConfig.certificateArn
106
+ props.domainConfig.certificateArn,
92
107
  ),
93
108
  }
94
109
  : {}),
95
- }
110
+ },
96
111
  );
97
112
 
98
113
  if (props.domainConfig) {
@@ -103,10 +118,23 @@ export class Website extends Construct {
103
118
  zone: hostedZone,
104
119
  recordName: this._getFullDomainName(props.domainConfig),
105
120
  target: cdk.aws_route53.RecordTarget.fromAlias(
106
- new cdk.aws_route53_targets.CloudFrontTarget(this.distribution)
121
+ new cdk.aws_route53_targets.CloudFrontTarget(this.distribution),
107
122
  ),
108
123
  });
109
124
  domainARecord.node.addDependency(this.distribution);
125
+
126
+ if (
127
+ props.domainConfig.includeRootDomain &&
128
+ props.domainConfig.subdomainName
129
+ ) {
130
+ new route53.ARecord(this, "RootDomainARecord", {
131
+ zone: hostedZone,
132
+ recordName: props.domainConfig.domainName,
133
+ target: cdk.aws_route53.RecordTarget.fromAlias(
134
+ new cdk.aws_route53_targets.CloudFrontTarget(this.distribution),
135
+ ),
136
+ }).node.addDependency(this.distribution);
137
+ }
110
138
  }
111
139
 
112
140
  new cdk.CfnOutput(this, "cloudfront-website-url", {
@@ -137,7 +165,7 @@ export class Website extends Construct {
137
165
  return certificatemanager.Certificate.fromCertificateArn(
138
166
  this,
139
167
  `website-cert`,
140
- arn
168
+ arn,
141
169
  );
142
170
  }
143
171
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sitblueprint/website-construct",
3
- "version": "0.1.4",
3
+ "version": "0.1.5",
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": "22.7.9",
35
- "aws-cdk-lib": "2.208.0",
34
+ "@types/node": "24.10.1",
35
+ "aws-cdk-lib": "2.235.0",
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.6.3"
40
+ "typescript": "~5.9.3"
41
41
  },
42
42
  "peerDependencies": {
43
- "aws-cdk-lib": "2.208.0",
43
+ "aws-cdk-lib": "2.235.0",
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 (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
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,4 @@ 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
+ //# 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"]}
@@ -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", () => {