124 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| const Url = require('url');
 | |
| 
 | |
| const Errors = require('./errors');
 | |
| 
 | |
| 
 | |
| const internals = {
 | |
|     minDomainSegments: 2,
 | |
|     nonAsciiRx: /[^\x00-\x7f]/,
 | |
|     domainControlRx: /[\x00-\x20@\:\/\\#!\$&\'\(\)\*\+,;=\?]/,                          // Control + space + separators
 | |
|     tldSegmentRx: /^[a-zA-Z](?:[a-zA-Z0-9\-]*[a-zA-Z0-9])?$/,
 | |
|     domainSegmentRx: /^[a-zA-Z0-9](?:[a-zA-Z0-9\-]*[a-zA-Z0-9])?$/,
 | |
|     URL: Url.URL || URL                                                                 // $lab:coverage:ignore$
 | |
| };
 | |
| 
 | |
| 
 | |
| exports.analyze = function (domain, options = {}) {
 | |
| 
 | |
|     if (!domain) {                                                                      // Catch null / undefined
 | |
|         return Errors.code('DOMAIN_NON_EMPTY_STRING');
 | |
|     }
 | |
| 
 | |
|     if (typeof domain !== 'string') {
 | |
|         throw new Error('Invalid input: domain must be a string');
 | |
|     }
 | |
| 
 | |
|     if (domain.length > 256) {
 | |
|         return Errors.code('DOMAIN_TOO_LONG');
 | |
|     }
 | |
| 
 | |
|     const ascii = !internals.nonAsciiRx.test(domain);
 | |
|     if (!ascii) {
 | |
|         if (options.allowUnicode === false) {                                           // Defaults to true
 | |
|             return Errors.code('DOMAIN_INVALID_UNICODE_CHARS');
 | |
|         }
 | |
| 
 | |
|         domain = domain.normalize('NFC');
 | |
|     }
 | |
| 
 | |
|     if (internals.domainControlRx.test(domain)) {
 | |
|         return Errors.code('DOMAIN_INVALID_CHARS');
 | |
|     }
 | |
| 
 | |
|     domain = internals.punycode(domain);
 | |
| 
 | |
|     // https://tools.ietf.org/html/rfc1035 section 2.3.1
 | |
| 
 | |
|     if (options.allowFullyQualified &&
 | |
|         domain[domain.length - 1] === '.') {
 | |
| 
 | |
|         domain = domain.slice(0, -1);
 | |
|     }
 | |
| 
 | |
|     const minDomainSegments = options.minDomainSegments || internals.minDomainSegments;
 | |
| 
 | |
|     const segments = domain.split('.');
 | |
|     if (segments.length < minDomainSegments) {
 | |
|         return Errors.code('DOMAIN_SEGMENTS_COUNT');
 | |
|     }
 | |
| 
 | |
|     if (options.maxDomainSegments) {
 | |
|         if (segments.length > options.maxDomainSegments) {
 | |
|             return Errors.code('DOMAIN_SEGMENTS_COUNT_MAX');
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const tlds = options.tlds;
 | |
|     if (tlds) {
 | |
|         const tld = segments[segments.length - 1].toLowerCase();
 | |
|         if (tlds.deny && tlds.deny.has(tld) ||
 | |
|             tlds.allow && !tlds.allow.has(tld)) {
 | |
| 
 | |
|             return Errors.code('DOMAIN_FORBIDDEN_TLDS');
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     for (let i = 0; i < segments.length; ++i) {
 | |
|         const segment = segments[i];
 | |
| 
 | |
|         if (!segment.length) {
 | |
|             return Errors.code('DOMAIN_EMPTY_SEGMENT');
 | |
|         }
 | |
| 
 | |
|         if (segment.length > 63) {
 | |
|             return Errors.code('DOMAIN_LONG_SEGMENT');
 | |
|         }
 | |
| 
 | |
|         if (i < segments.length - 1) {
 | |
|             if (!internals.domainSegmentRx.test(segment)) {
 | |
|                 return Errors.code('DOMAIN_INVALID_CHARS');
 | |
|             }
 | |
|         }
 | |
|         else {
 | |
|             if (!internals.tldSegmentRx.test(segment)) {
 | |
|                 return Errors.code('DOMAIN_INVALID_TLDS_CHARS');
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return null;
 | |
| };
 | |
| 
 | |
| 
 | |
| exports.isValid = function (domain, options) {
 | |
| 
 | |
|     return !exports.analyze(domain, options);
 | |
| };
 | |
| 
 | |
| 
 | |
| internals.punycode = function (domain) {
 | |
| 
 | |
|     if (domain.includes('%')) {
 | |
|         domain = domain.replace(/%/g, '%25');
 | |
|     }
 | |
| 
 | |
|     try {
 | |
|         return new internals.URL(`http://${domain}`).host;
 | |
|     }
 | |
|     catch (err) {
 | |
|         return domain;
 | |
|     }
 | |
| };
 |