'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 }