domain.js 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. 'use strict';
  2. const Url = require('url');
  3. const Errors = require('./errors');
  4. const internals = {
  5. minDomainSegments: 2,
  6. nonAsciiRx: /[^\x00-\x7f]/,
  7. domainControlRx: /[\x00-\x20@\:\/\\#!\$&\'\(\)\*\+,;=\?]/, // Control + space + separators
  8. tldSegmentRx: /^[a-zA-Z](?:[a-zA-Z0-9\-]*[a-zA-Z0-9])?$/,
  9. domainSegmentRx: /^[a-zA-Z0-9](?:[a-zA-Z0-9\-]*[a-zA-Z0-9])?$/,
  10. URL: Url.URL || URL // $lab:coverage:ignore$
  11. };
  12. exports.analyze = function (domain, options = {}) {
  13. if (!domain) { // Catch null / undefined
  14. return Errors.code('DOMAIN_NON_EMPTY_STRING');
  15. }
  16. if (typeof domain !== 'string') {
  17. throw new Error('Invalid input: domain must be a string');
  18. }
  19. if (domain.length > 256) {
  20. return Errors.code('DOMAIN_TOO_LONG');
  21. }
  22. const ascii = !internals.nonAsciiRx.test(domain);
  23. if (!ascii) {
  24. if (options.allowUnicode === false) { // Defaults to true
  25. return Errors.code('DOMAIN_INVALID_UNICODE_CHARS');
  26. }
  27. domain = domain.normalize('NFC');
  28. }
  29. if (internals.domainControlRx.test(domain)) {
  30. return Errors.code('DOMAIN_INVALID_CHARS');
  31. }
  32. domain = internals.punycode(domain);
  33. // https://tools.ietf.org/html/rfc1035 section 2.3.1
  34. if (options.allowFullyQualified &&
  35. domain[domain.length - 1] === '.') {
  36. domain = domain.slice(0, -1);
  37. }
  38. const minDomainSegments = options.minDomainSegments || internals.minDomainSegments;
  39. const segments = domain.split('.');
  40. if (segments.length < minDomainSegments) {
  41. return Errors.code('DOMAIN_SEGMENTS_COUNT');
  42. }
  43. if (options.maxDomainSegments) {
  44. if (segments.length > options.maxDomainSegments) {
  45. return Errors.code('DOMAIN_SEGMENTS_COUNT_MAX');
  46. }
  47. }
  48. const tlds = options.tlds;
  49. if (tlds) {
  50. const tld = segments[segments.length - 1].toLowerCase();
  51. if (tlds.deny && tlds.deny.has(tld) ||
  52. tlds.allow && !tlds.allow.has(tld)) {
  53. return Errors.code('DOMAIN_FORBIDDEN_TLDS');
  54. }
  55. }
  56. for (let i = 0; i < segments.length; ++i) {
  57. const segment = segments[i];
  58. if (!segment.length) {
  59. return Errors.code('DOMAIN_EMPTY_SEGMENT');
  60. }
  61. if (segment.length > 63) {
  62. return Errors.code('DOMAIN_LONG_SEGMENT');
  63. }
  64. if (i < segments.length - 1) {
  65. if (!internals.domainSegmentRx.test(segment)) {
  66. return Errors.code('DOMAIN_INVALID_CHARS');
  67. }
  68. }
  69. else {
  70. if (!internals.tldSegmentRx.test(segment)) {
  71. return Errors.code('DOMAIN_INVALID_TLDS_CHARS');
  72. }
  73. }
  74. }
  75. return null;
  76. };
  77. exports.isValid = function (domain, options) {
  78. return !exports.analyze(domain, options);
  79. };
  80. internals.punycode = function (domain) {
  81. if (domain.includes('%')) {
  82. domain = domain.replace(/%/g, '%25');
  83. }
  84. try {
  85. return new internals.URL(`http://${domain}`).host;
  86. }
  87. catch (err) {
  88. return domain;
  89. }
  90. };