UNPKG

libphonenumber-js

Version:

A simpler (and smaller) rewrite of Google Android's libphonenumber library in javascript

180 lines (157 loc) 6.04 kB
import Metadata, { validateMetadata } from './metadata.js' import isPossibleNumber from './isPossible.js' import isValidNumber from './isValid.js' // import checkNumberLength from './helpers/checkNumberLength.js' import getNumberType from './helpers/getNumberType.js' import getPossibleCountriesForNumber from './helpers/getPossibleCountriesForNumber.js' import extractCountryCallingCode from './helpers/extractCountryCallingCode.js' import isObject from './helpers/isObject.js' import formatNumber from './format.js' const USE_NON_GEOGRAPHIC_COUNTRY_CODE = false export default class PhoneNumber { /** * @param {string} countryOrCountryCallingCode * @param {string} nationalNumber * @param {object} metadata — Metadata JSON * @return {PhoneNumber} */ constructor(countryOrCountryCallingCode, nationalNumber, metadata) { // Validate `countryOrCountryCallingCode` argument. if (!countryOrCountryCallingCode) { throw new TypeError('First argument is required') } if (typeof countryOrCountryCallingCode !== 'string') { throw new TypeError('First argument must be a string') } // In case of public API use: `constructor(number, metadata)`. // Transform the arguments from `constructor(number, metadata)` to // `constructor(countryOrCountryCallingCode, nationalNumber, metadata)`. if (typeof countryOrCountryCallingCode === 'string') { if (countryOrCountryCallingCode[0] === '+' && !nationalNumber) { throw new TypeError('`metadata` argument not passed') } if (isObject(nationalNumber) && isObject(nationalNumber.countries)) { metadata = nationalNumber const e164Number = countryOrCountryCallingCode if (!E164_NUMBER_REGEXP.test(e164Number)) { throw new Error('Invalid `number` argument passed: must consist of a "+" followed by digits') } const { countryCallingCode, number } = extractCountryCallingCode(e164Number, undefined, undefined, metadata) nationalNumber = number countryOrCountryCallingCode = countryCallingCode if (!nationalNumber) { throw new Error('Invalid `number` argument passed: too short') } } } // Validate `nationalNumber` argument. if (!nationalNumber) { throw new TypeError('`nationalNumber` argument is required') } if (typeof nationalNumber !== 'string') { throw new TypeError('`nationalNumber` argument must be a string') } // Validate `metadata` argument. validateMetadata(metadata) // Initialize properties. const { country, countryCallingCode } = getCountryAndCountryCallingCode( countryOrCountryCallingCode, metadata ) this.country = country this.countryCallingCode = countryCallingCode this.nationalNumber = nationalNumber this.number = '+' + this.countryCallingCode + this.nationalNumber // Exclude `metadata` property output from `PhoneNumber.toString()` // so that it doesn't clutter the console output of Node.js. // Previously, when Node.js did `console.log(new PhoneNumber(...))`, // it would output the whole internal structure of the `metadata` object. this.getMetadata = () => metadata } setExt(ext) { this.ext = ext } getPossibleCountries() { if (this.country) { return [this.country] } return getPossibleCountriesForNumber( this.countryCallingCode, this.nationalNumber, this.getMetadata() ) } isPossible() { return isPossibleNumber(this, { v2: true }, this.getMetadata()) } isValid() { return isValidNumber(this, { v2: true }, this.getMetadata()) } isNonGeographic() { const metadata = new Metadata(this.getMetadata()) return metadata.isNonGeographicCallingCode(this.countryCallingCode) } isEqual(phoneNumber) { return this.number === phoneNumber.number && this.ext === phoneNumber.ext } // This function was originally meant to be an equivalent for `validatePhoneNumberLength()`, // but later it was found out that it doesn't include the possible `TOO_SHORT` result // returned from `parsePhoneNumberWithError()` in the original `validatePhoneNumberLength()`, // so eventually I simply commented out this method from the `PhoneNumber` class // and just left the `validatePhoneNumberLength()` function, even though that one would require // and additional step to also validate the actual country / calling code of the phone number. // validateLength() { // const metadata = new Metadata(this.getMetadata()) // metadata.selectNumberingPlan(this.countryCallingCode) // const result = checkNumberLength(this.nationalNumber, metadata) // if (result !== 'IS_POSSIBLE') { // return result // } // } getType() { return getNumberType(this, { v2: true }, this.getMetadata()) } format(format, options) { return formatNumber( this, format, options ? { ...options, v2: true } : { v2: true }, this.getMetadata() ) } formatNational(options) { return this.format('NATIONAL', options) } formatInternational(options) { return this.format('INTERNATIONAL', options) } getURI(options) { return this.format('RFC3966', options) } } const isCountryCode = (value) => /^[A-Z]{2}$/.test(value) function getCountryAndCountryCallingCode(countryOrCountryCallingCode, metadataJson) { let country let countryCallingCode const metadata = new Metadata(metadataJson) // If country code is passed then derive `countryCallingCode` from it. // Also store the country code as `.country`. if (isCountryCode(countryOrCountryCallingCode)) { country = countryOrCountryCallingCode metadata.selectNumberingPlan(country) countryCallingCode = metadata.countryCallingCode() } else { countryCallingCode = countryOrCountryCallingCode /* istanbul ignore if */ if (USE_NON_GEOGRAPHIC_COUNTRY_CODE) { if (metadata.isNonGeographicCallingCode(countryCallingCode)) { country = '001' } } } return { country, countryCallingCode } } const E164_NUMBER_REGEXP = /^\+\d+$/