123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- 'use strict'
- const { HEX } = require('./scopedChars')
- function normalizeIPv4 (host) {
- if (findToken(host, '.') < 3) { return { host, isIPV4: false } }
- const matches = host.match(/^(?:(?:25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9])\.){3}(?:25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]|[0-9])$/u) || []
- const [address] = matches
- if (address) {
- return { host: stripLeadingZeros(address, '.'), isIPV4: true }
- } else {
- return { host, isIPV4: false }
- }
- }
- /**
- * @param {string[]} input
- * @param {boolean} [keepZero=false]
- * @returns {string|undefined}
- */
- function stringArrayToHexStripped (input, keepZero = false) {
- let acc = ''
- let strip = true
- for (const c of input) {
- if (HEX[c] === undefined) return undefined
- if (c !== '0' && strip === true) strip = false
- if (!strip) acc += c
- }
- if (keepZero && acc.length === 0) acc = '0'
- return acc
- }
- function getIPV6 (input) {
- let tokenCount = 0
- const output = { error: false, address: '', zone: '' }
- const address = []
- const buffer = []
- let isZone = false
- let endipv6Encountered = false
- let endIpv6 = false
- function consume () {
- if (buffer.length) {
- if (isZone === false) {
- const hex = stringArrayToHexStripped(buffer)
- if (hex !== undefined) {
- address.push(hex)
- } else {
- output.error = true
- return false
- }
- }
- buffer.length = 0
- }
- return true
- }
- for (let i = 0; i < input.length; i++) {
- const cursor = input[i]
- if (cursor === '[' || cursor === ']') { continue }
- if (cursor === ':') {
- if (endipv6Encountered === true) {
- endIpv6 = true
- }
- if (!consume()) { break }
- tokenCount++
- address.push(':')
- if (tokenCount > 7) {
- // not valid
- output.error = true
- break
- }
- if (i - 1 >= 0 && input[i - 1] === ':') {
- endipv6Encountered = true
- }
- continue
- } else if (cursor === '%') {
- if (!consume()) { break }
- // switch to zone detection
- isZone = true
- } else {
- buffer.push(cursor)
- continue
- }
- }
- if (buffer.length) {
- if (isZone) {
- output.zone = buffer.join('')
- } else if (endIpv6) {
- address.push(buffer.join(''))
- } else {
- address.push(stringArrayToHexStripped(buffer))
- }
- }
- output.address = address.join('')
- return output
- }
- function normalizeIPv6 (host, opts = {}) {
- if (findToken(host, ':') < 2) { return { host, isIPV6: false } }
- const ipv6 = getIPV6(host)
- if (!ipv6.error) {
- let newHost = ipv6.address
- let escapedHost = ipv6.address
- if (ipv6.zone) {
- newHost += '%' + ipv6.zone
- escapedHost += '%25' + ipv6.zone
- }
- return { host: newHost, escapedHost, isIPV6: true }
- } else {
- return { host, isIPV6: false }
- }
- }
- function stripLeadingZeros (str, token) {
- let out = ''
- let skip = true
- const l = str.length
- for (let i = 0; i < l; i++) {
- const c = str[i]
- if (c === '0' && skip) {
- if ((i + 1 <= l && str[i + 1] === token) || i + 1 === l) {
- out += c
- skip = false
- }
- } else {
- if (c === token) {
- skip = true
- } else {
- skip = false
- }
- out += c
- }
- }
- return out
- }
- function findToken (str, token) {
- let ind = 0
- for (let i = 0; i < str.length; i++) {
- if (str[i] === token) ind++
- }
- return ind
- }
- const RDS1 = /^\.\.?\//u
- const RDS2 = /^\/\.(?:\/|$)/u
- const RDS3 = /^\/\.\.(?:\/|$)/u
- const RDS5 = /^\/?(?:.|\n)*?(?=\/|$)/u
- function removeDotSegments (input) {
- const output = []
- while (input.length) {
- if (input.match(RDS1)) {
- input = input.replace(RDS1, '')
- } else if (input.match(RDS2)) {
- input = input.replace(RDS2, '/')
- } else if (input.match(RDS3)) {
- input = input.replace(RDS3, '/')
- output.pop()
- } else if (input === '.' || input === '..') {
- input = ''
- } else {
- const im = input.match(RDS5)
- if (im) {
- const s = im[0]
- input = input.slice(s.length)
- output.push(s)
- } else {
- throw new Error('Unexpected dot segment condition')
- }
- }
- }
- return output.join('')
- }
- function normalizeComponentEncoding (components, esc) {
- const func = esc !== true ? escape : unescape
- if (components.scheme !== undefined) {
- components.scheme = func(components.scheme)
- }
- if (components.userinfo !== undefined) {
- components.userinfo = func(components.userinfo)
- }
- if (components.host !== undefined) {
- components.host = func(components.host)
- }
- if (components.path !== undefined) {
- components.path = func(components.path)
- }
- if (components.query !== undefined) {
- components.query = func(components.query)
- }
- if (components.fragment !== undefined) {
- components.fragment = func(components.fragment)
- }
- return components
- }
- function recomposeAuthority (components, options) {
- const uriTokens = []
- if (components.userinfo !== undefined) {
- uriTokens.push(components.userinfo)
- uriTokens.push('@')
- }
- if (components.host !== undefined) {
- let host = unescape(components.host)
- const ipV4res = normalizeIPv4(host)
- if (ipV4res.isIPV4) {
- host = ipV4res.host
- } else {
- const ipV6res = normalizeIPv6(ipV4res.host, { isIPV4: false })
- if (ipV6res.isIPV6 === true) {
- host = `[${ipV6res.escapedHost}]`
- } else {
- host = components.host
- }
- }
- uriTokens.push(host)
- }
- if (typeof components.port === 'number' || typeof components.port === 'string') {
- uriTokens.push(':')
- uriTokens.push(String(components.port))
- }
- return uriTokens.length ? uriTokens.join('') : undefined
- };
- module.exports = {
- recomposeAuthority,
- normalizeComponentEncoding,
- removeDotSegments,
- normalizeIPv4,
- normalizeIPv6,
- stringArrayToHexStripped
- }
|