You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							237 lines
						
					
					
						
							5.5 KiB
						
					
					
				
			
		
		
	
	
							237 lines
						
					
					
						
							5.5 KiB
						
					
					
				| 'use strict'
 | |
| 
 | |
| const { HEX } = require('./scopedChars')
 | |
| 
 | |
| function normalizeIPv4 (host) {
 | |
|   if (findToken(host, '.') < 3) { return { host, isIPV4: false } }
 | |
|   const matches = host.match(/^(\b25[0-5]|\b2[0-4][0-9]|\b[01]?[0-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/) || []
 | |
|   const [address] = matches
 | |
|   if (address) {
 | |
|     return { host: stripLeadingZeros(address, '.'), isIPV4: true }
 | |
|   } else {
 | |
|     return { host, isIPV4: false }
 | |
|   }
 | |
| }
 | |
| 
 | |
| function stringToHexStripped (input) {
 | |
|   let acc = ''
 | |
|   let strip = true
 | |
|   for (const c of input) {
 | |
|     if (c !== '0' && strip === true) strip = false
 | |
|     if (HEX[c] === undefined) return undefined
 | |
|     if (!strip) acc += c
 | |
|   }
 | |
|   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 = stringToHexStripped(buffer.join(''))
 | |
|         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(stringToHexStripped(buffer.join('')))
 | |
|     }
 | |
|   }
 | |
|   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 = /^\.\.?\//
 | |
| const RDS2 = /^\/\.(\/|$)/
 | |
| const RDS3 = /^\/\.\.(\/|$)/
 | |
| const RDS5 = /^\/?(?:.|\n)*?(?=\/|$)/
 | |
| 
 | |
| 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,
 | |
|   stringToHexStripped
 | |
| }
 |