cdk-nextjs-standalone
Version:
Deploy a NextJS app to AWS using CDK and OpenNext.
106 lines • 18.2 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.NextjsDomain = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
const constructs_1 = require("constructs");
/**
* Use a custom domain with `Nextjs`. Requires a Route53 hosted zone to have been
* created within the same AWS account. For DNS setups where you cannot use a
* Route53 hosted zone in the same AWS account, use the `overrides.nextjsDistribution.distributionProps`
* prop of {@link NextjsProps}.
*
* See {@link NextjsDomainProps} TS Doc comments for detailed docs on how to customize.
* This construct is helpful to user to not have to worry about interdependencies
* between Route53 Hosted Zone, CloudFront Distribution, and Route53 Hosted Zone Records.
*
* Note, if you're using another service for domain name registration, you can
* still create a Route53 hosted zone. Please see [Configuring DNS Delegation from
* CloudFlare to AWS Route53](https://veducate.co.uk/dns-delegation-route53/)
* as an example.
*/
class NextjsDomain extends constructs_1.Construct {
/**
* Concatentation of {@link NextjsDomainProps.domainName} and {@link NextjsDomainProps.alternateNames}.
* Used in instantiation of CloudFront Distribution in NextjsDistribution
*/
get domainNames() {
const names = [this.props.domainName];
if (this.props.alternateNames?.length) {
names.push(...this.props.alternateNames);
}
return names;
}
constructor(scope, id, props) {
super(scope, id);
this.props = props;
this.hostedZone = this.getHostedZone();
this.certificate = this.getCertificate();
}
getHostedZone() {
if (!this.props.hostedZone) {
return aws_route53_1.HostedZone.fromLookup(this, 'HostedZone', {
domainName: this.props.domainName,
...this.props.overrides?.hostedZoneProviderProps,
});
}
else {
return this.props.hostedZone;
}
}
getCertificate() {
if (!this.props.certificate) {
return new aws_certificatemanager_1.Certificate(this, 'Certificate', {
domainName: this.props.certificateDomainName ?? this.props.domainName,
validation: aws_certificatemanager_1.CertificateValidation.fromDns(this.hostedZone),
...this.props.overrides?.certificateProps,
});
}
else {
return this.props.certificate;
}
}
/**
* Creates DNS records (A and AAAA) records for {@link NextjsDomainProps.domainName}
* and {@link NextjsDomainProps.alternateNames} if defined.
*/
createDnsRecords(distribution) {
// Create DNS record
const recordProps = {
recordName: this.props.domainName,
zone: this.hostedZone,
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(distribution)),
};
new aws_route53_1.ARecord(this, 'ARecordMain', {
...recordProps,
...this.props.overrides?.aRecordProps,
}); // IPv4
new aws_route53_1.AaaaRecord(this, 'AaaaRecordMain', {
...recordProps,
...this.props.overrides?.aaaaRecordProps,
}); // IPv6
if (this.props.alternateNames?.length) {
let i = 1;
for (const alternateName of this.props.alternateNames) {
new aws_route53_1.ARecord(this, 'ARecordAlt' + i, {
...recordProps,
recordName: `${alternateName}.`,
...this.props.overrides?.aRecordProps,
});
new aws_route53_1.AaaaRecord(this, 'AaaaRecordAlt' + i, {
...recordProps,
recordName: `${alternateName}.`,
...this.props.overrides?.aaaaRecordProps,
});
i++;
}
}
}
}
exports.NextjsDomain = NextjsDomain;
_a = JSII_RTTI_SYMBOL_1;
NextjsDomain[_a] = { fqn: "cdk-nextjs-standalone.NextjsDomain", version: "4.2.3" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsDomain.js","sourceRoot":"","sources":["../src/NextjsDomain.ts"],"names":[],"mappings":";;;;;AAAA,+EAAsG;AAEtG,yDAQiC;AACjC,yEAAmE;AACnE,2CAAuC;AA4EvC;;;;;;;;;;;;;;GAcG;AACH,MAAa,YAAa,SAAQ,sBAAS;IACzC;;;OAGG;IACH,IAAI,WAAW;QACb,MAAM,KAAK,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACtC,IAAI,IAAI,CAAC,KAAK,CAAC,cAAc,EAAE,MAAM,EAAE,CAAC;YACtC,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QAC3C,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAYD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAwB;QAChE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACvC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;IAC3C,CAAC;IAEO,aAAa;QACnB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC;YAC3B,OAAO,wBAAU,CAAC,UAAU,CAAC,IAAI,EAAE,YAAY,EAAE;gBAC/C,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU;gBACjC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,uBAAuB;aACjD,CAAC,CAAC;QACL,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC;QAC/B,CAAC;IACH,CAAC;IAEO,cAAc;QACpB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;YAC5B,OAAO,IAAI,oCAAW,CAAC,IAAI,EAAE,aAAa,EAAE;gBAC1C,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,qBAAqB,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU;gBACrE,UAAU,EAAE,8CAAqB,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC;gBAC1D,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,gBAAgB;aAC1C,CAAC,CAAC;QACL,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC;QAChC,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,gBAAgB,CAAC,YAA0B;QACzC,oBAAoB;QACpB,MAAM,WAAW,GAAmC;YAClD,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU;YACjC,IAAI,EAAE,IAAI,CAAC,UAAU;YACrB,MAAM,EAAE,0BAAY,CAAC,SAAS,CAAC,IAAI,sCAAgB,CAAC,YAAY,CAAC,CAAC;SACnE,CAAC;QACF,IAAI,qBAAO,CAAC,IAAI,EAAE,aAAa,EAAE;YAC/B,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,YAAY;SACtC,CAAC,CAAC,CAAC,OAAO;QACX,IAAI,wBAAU,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACrC,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,eAAe;SACzC,CAAC,CAAC,CAAC,OAAO;QACX,IAAI,IAAI,CAAC,KAAK,CAAC,cAAc,EAAE,MAAM,EAAE,CAAC;YACtC,IAAI,CAAC,GAAG,CAAC,CAAC;YACV,KAAK,MAAM,aAAa,IAAI,IAAI,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC;gBACtD,IAAI,qBAAO,CAAC,IAAI,EAAE,YAAY,GAAG,CAAC,EAAE;oBAClC,GAAG,WAAW;oBACd,UAAU,EAAE,GAAG,aAAa,GAAG;oBAC/B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,YAAY;iBACtC,CAAC,CAAC;gBACH,IAAI,wBAAU,CAAC,IAAI,EAAE,eAAe,GAAG,CAAC,EAAE;oBACxC,GAAG,WAAW;oBACd,UAAU,EAAE,GAAG,aAAa,GAAG;oBAC/B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,eAAe;iBACzC,CAAC,CAAC;gBACH,CAAC,EAAE,CAAC;YACN,CAAC;QACH,CAAC;IACH,CAAC;;AAxFH,oCAyFC","sourcesContent":["import { ICertificate, Certificate, CertificateValidation } from 'aws-cdk-lib/aws-certificatemanager';\nimport { Distribution } from 'aws-cdk-lib/aws-cloudfront';\nimport {\n  ARecord,\n  ARecordProps,\n  AaaaRecord,\n  AaaaRecordProps,\n  HostedZone,\n  IHostedZone,\n  RecordTarget,\n} from 'aws-cdk-lib/aws-route53';\nimport { CloudFrontTarget } from 'aws-cdk-lib/aws-route53-targets';\nimport { Construct } from 'constructs';\nimport { NextjsProps } from '.';\nimport {\n  OptionalAaaaRecordProps,\n  OptionalCertificateProps,\n  OptionalHostedZoneProviderProps,\n  OptionalARecordProps,\n} from './generated-structs';\n\nexport interface NextjsDomainOverrides {\n  readonly certificateProps?: OptionalCertificateProps;\n  readonly hostedZoneProviderProps?: OptionalHostedZoneProviderProps;\n  readonly aRecordProps?: OptionalARecordProps;\n  readonly aaaaRecordProps?: OptionalAaaaRecordProps;\n}\n\nexport interface NextjsDomainProps {\n  /**\n   * An easy to remember address of your website. Only supports domains hosted\n   * on [Route 53](https://aws.amazon.com/route53/). Used as `domainName` for\n   * ACM `Certificate` if {@link NextjsDomainProps.certificate} and\n   * {@link NextjsDomainProps.certificateDomainName} are `undefined`.\n   * @example \"example.com\"\n   */\n  readonly domainName: string;\n  /**\n   * Alternate domain names that should route to the Cloudfront Distribution.\n   * For example, if you specificied `\"example.com\"` as your {@link NextjsDomainProps.domainName},\n   * you could specify `[\"www.example.com\", \"api.example.com\"]`.\n   * Learn more about the [requirements](https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/CNAMEs.html#alternate-domain-names-requirements)\n   * and [restrictions](https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/CNAMEs.html#alternate-domain-names-restrictions)\n   * for using alternate domain names with CloudFront.\n   *\n   * Note, in order to use alternate domain names, they must be covered by your\n   * certificate. By default, the certificate created in this construct only covers\n   * the {@link NextjsDomainProps.domainName}. Therefore, you'll need to specify\n   * a wildcard domain name like `\"*.example.com\"` with {@link NextjsDomainProps.certificateDomainName}\n   * so that this construct will create the certificate the covers the alternate\n   * domain names. Otherwise, you can use {@link NextjsDomainProps.certificate}\n   * to create the certificate yourself where you'll need to ensure it has a\n   * wildcard or uses subject alternative names including the\n   * alternative names specified here.\n   * @example [\"www.example.com\", \"api.example.com\"]\n   */\n  readonly alternateNames?: string[];\n  /**\n   * You must create the hosted zone out-of-band.\n   * You can lookup the hosted zone outside this construct and pass it in via this prop.\n   * Alternatively if this prop is `undefined`, then the hosted zone will be\n   * **looked up** (not created) via `HostedZone.fromLookup` with {@link NextjsDomainProps.domainName}.\n   */\n  readonly hostedZone?: IHostedZone;\n  /**\n   * If this prop is `undefined` then an ACM `Certificate` will be created based on {@link NextjsDomainProps.domainName}\n   * with DNS Validation. This prop allows you to control the TLS/SSL\n   * certificate created. The certificate you create must be in the `us-east-1`\n   * (N. Virginia) region as required by AWS CloudFront.\n   *\n   * Set this option if you have an existing certificate in the `us-east-1` region in AWS Certificate Manager you want to use.\n   */\n  readonly certificate?: ICertificate;\n  /**\n   * The domain name used in this construct when creating an ACM `Certificate`. Useful\n   * when passing {@link NextjsDomainProps.alternateNames} and you need to specify\n   * a wildcard domain like \"*.example.com\". If `undefined`, then {@link NextjsDomainProps.domainName}\n   * will be used.\n   *\n   * If {@link NextjsDomainProps.certificate} is passed, then this prop is ignored.\n   */\n  readonly certificateDomainName?: string;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsDomainOverrides;\n}\n\n/**\n * Use a custom domain with `Nextjs`. Requires a Route53 hosted zone to have been\n * created within the same AWS account. For DNS setups where you cannot use a\n * Route53 hosted zone in the same AWS account, use the `overrides.nextjsDistribution.distributionProps`\n * prop of {@link NextjsProps}.\n *\n * See {@link NextjsDomainProps} TS Doc comments for detailed docs on how to customize.\n * This construct is helpful to user to not have to worry about interdependencies\n * between Route53 Hosted Zone, CloudFront Distribution, and Route53 Hosted Zone Records.\n *\n * Note, if you're using another service for domain name registration, you can\n * still create a Route53 hosted zone. Please see [Configuring DNS Delegation from\n * CloudFlare to AWS Route53](https://veducate.co.uk/dns-delegation-route53/)\n * as an example.\n */\nexport class NextjsDomain extends Construct {\n  /**\n   * Concatentation of {@link NextjsDomainProps.domainName} and {@link NextjsDomainProps.alternateNames}.\n   * Used in instantiation of CloudFront Distribution in NextjsDistribution\n   */\n  get domainNames(): string[] {\n    const names = [this.props.domainName];\n    if (this.props.alternateNames?.length) {\n      names.push(...this.props.alternateNames);\n    }\n    return names;\n  }\n  /**\n   * Route53 Hosted Zone.\n   */\n  hostedZone: IHostedZone;\n  /**\n   * ACM Certificate.\n   */\n  certificate: ICertificate;\n\n  private props: NextjsDomainProps;\n\n  constructor(scope: Construct, id: string, props: NextjsDomainProps) {\n    super(scope, id);\n    this.props = props;\n    this.hostedZone = this.getHostedZone();\n    this.certificate = this.getCertificate();\n  }\n\n  private getHostedZone(): IHostedZone {\n    if (!this.props.hostedZone) {\n      return HostedZone.fromLookup(this, 'HostedZone', {\n        domainName: this.props.domainName,\n        ...this.props.overrides?.hostedZoneProviderProps,\n      });\n    } else {\n      return this.props.hostedZone;\n    }\n  }\n\n  private getCertificate(): ICertificate {\n    if (!this.props.certificate) {\n      return new Certificate(this, 'Certificate', {\n        domainName: this.props.certificateDomainName ?? this.props.domainName,\n        validation: CertificateValidation.fromDns(this.hostedZone),\n        ...this.props.overrides?.certificateProps,\n      });\n    } else {\n      return this.props.certificate;\n    }\n  }\n\n  /**\n   * Creates DNS records (A and AAAA) records for {@link NextjsDomainProps.domainName}\n   * and {@link NextjsDomainProps.alternateNames} if defined.\n   */\n  createDnsRecords(distribution: Distribution): void {\n    // Create DNS record\n    const recordProps: ARecordProps & AaaaRecordProps = {\n      recordName: this.props.domainName,\n      zone: this.hostedZone,\n      target: RecordTarget.fromAlias(new CloudFrontTarget(distribution)),\n    };\n    new ARecord(this, 'ARecordMain', {\n      ...recordProps,\n      ...this.props.overrides?.aRecordProps,\n    }); // IPv4\n    new AaaaRecord(this, 'AaaaRecordMain', {\n      ...recordProps,\n      ...this.props.overrides?.aaaaRecordProps,\n    }); // IPv6\n    if (this.props.alternateNames?.length) {\n      let i = 1;\n      for (const alternateName of this.props.alternateNames) {\n        new ARecord(this, 'ARecordAlt' + i, {\n          ...recordProps,\n          recordName: `${alternateName}.`,\n          ...this.props.overrides?.aRecordProps,\n        });\n        new AaaaRecord(this, 'AaaaRecordAlt' + i, {\n          ...recordProps,\n          recordName: `${alternateName}.`,\n          ...this.props.overrides?.aaaaRecordProps,\n        });\n        i++;\n      }\n    }\n  }\n}\n"]}