mirror of
https://codeberg.org/tacerus/teddit.git
synced 2025-01-25 12:14:08 +01:00
188 lines
4.4 KiB
JavaScript
188 lines
4.4 KiB
JavaScript
|
var url = require('url')
|
||
|
var EMPTY = ''
|
||
|
var ZERO = '0'
|
||
|
var PERCENT = '%'
|
||
|
var STRING = 'string'
|
||
|
var AMPERSAND = '&'
|
||
|
var EQUALS = '='
|
||
|
var QUESTION_MARK = '?'
|
||
|
var stringify
|
||
|
var parse
|
||
|
|
||
|
/**
|
||
|
* Percent encode a character with given code.
|
||
|
*
|
||
|
* @param {Number} c - character code of the character to encode
|
||
|
* @returns {String} - percent encoding of given character
|
||
|
*/
|
||
|
var percentEncode = function (c) {
|
||
|
var hex = c.toString(16).toUpperCase()
|
||
|
hex.length === 1 && (hex = ZERO + hex)
|
||
|
return PERCENT + hex
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if character with given code is valid hexadecimal digit or not.
|
||
|
*
|
||
|
* @param {Number} byte
|
||
|
* @returns {Boolean}
|
||
|
*/
|
||
|
var isPreEncodedCharacter = function (byte) {
|
||
|
return (byte >= 0x30 && byte <= 0x39) || // 0-9
|
||
|
(byte >= 0x41 && byte <= 0x46) || // A-F
|
||
|
(byte >= 0x61 && byte <= 0x66) // a-f
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks if character at given index in the buffer is already percent encoded or not.
|
||
|
*
|
||
|
* @param {Buffer} buffer
|
||
|
* @param {Number} i
|
||
|
* @returns {Boolean}
|
||
|
*/
|
||
|
var isPreEncoded = function (buffer, i) {
|
||
|
// If it is % check next two bytes for percent encode characters
|
||
|
// looking for pattern %00 - %FF
|
||
|
return (buffer[i] === 0x25 &&
|
||
|
(isPreEncodedCharacter(buffer[i + 1]) &&
|
||
|
isPreEncodedCharacter(buffer[i + 2]))
|
||
|
)
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Checks whether given character should be percent encoded or not for fixture.
|
||
|
*
|
||
|
* @param {Number} byte
|
||
|
* @returns {Boolean}
|
||
|
*/
|
||
|
var charactersToPercentEncode = function (byte) {
|
||
|
return (byte < 0x23 || byte > 0x7E || // Below # and after ~
|
||
|
byte === 0x3C || byte === 0x3E || // > and <
|
||
|
byte === 0x28 || byte === 0x29 || // ( and )
|
||
|
byte === 0x25 || // %
|
||
|
byte === 0x27 || // '
|
||
|
byte === 0x2A // *
|
||
|
)
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Percent encode a query string according to RFC 3986.
|
||
|
* Note: This function is supposed to be used on top of node's inbuilt url encoding
|
||
|
* to solve issue https://github.com/nodejs/node/issues/8321
|
||
|
*
|
||
|
* @param {String} value
|
||
|
* @returns {String}
|
||
|
*/
|
||
|
var encode = function (value) {
|
||
|
if (!value) { return EMPTY }
|
||
|
|
||
|
var buffer = Buffer.from(value)
|
||
|
var ret = EMPTY
|
||
|
var i
|
||
|
var ii
|
||
|
|
||
|
for (i = 0, ii = buffer.length; i < ii; ++i) {
|
||
|
if (charactersToPercentEncode(buffer[i]) && !isPreEncoded(buffer, i)) {
|
||
|
ret += percentEncode(buffer[i])
|
||
|
} else {
|
||
|
ret += String.fromCodePoint(buffer[i]) // Only works in ES6 (available in Node v4+)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return ret
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parses a query string into an array, preserving parameter values
|
||
|
*
|
||
|
* @param string
|
||
|
* @returns {*}
|
||
|
*/
|
||
|
parse = function (string) {
|
||
|
var parts
|
||
|
if (typeof string === STRING) { // eslint-disable-line valid-typeof
|
||
|
parts = string.split(AMPERSAND)
|
||
|
return parts.map(function (param, idx) {
|
||
|
if (param === EMPTY && idx !== (parts.length - 1)) {
|
||
|
return { key: null, value: null }
|
||
|
}
|
||
|
|
||
|
var index = (typeof param === STRING) ? param.indexOf(EQUALS) : -1 // eslint-disable-line valid-typeof
|
||
|
var paramObj = {}
|
||
|
|
||
|
// this means that there was no value for this key (not even blank, so we store this info) and the value is set
|
||
|
// to null
|
||
|
if (index < 0) {
|
||
|
paramObj.key = param.substr(0, param.length)
|
||
|
paramObj.value = null
|
||
|
} else {
|
||
|
paramObj.key = param.substr(0, index)
|
||
|
paramObj.value = param.substr(index + 1)
|
||
|
}
|
||
|
|
||
|
return paramObj
|
||
|
})
|
||
|
}
|
||
|
return []
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Stringifies a query string, from an array of parameters
|
||
|
*
|
||
|
* @param parameters
|
||
|
* @returns {string}
|
||
|
*/
|
||
|
stringify = function (parameters) {
|
||
|
return parameters ? parameters.map(function (param) {
|
||
|
var key = param.key
|
||
|
var value = param.value
|
||
|
|
||
|
if (value === undefined) {
|
||
|
return EMPTY
|
||
|
}
|
||
|
|
||
|
if (key === null) {
|
||
|
key = EMPTY
|
||
|
}
|
||
|
|
||
|
if (value === null) {
|
||
|
return encode(key)
|
||
|
}
|
||
|
|
||
|
return encode(key) + EQUALS + encode(value)
|
||
|
}).join(AMPERSAND) : EMPTY
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Correctly URL encodes query parameters in a URL and returns the final parsed URL.
|
||
|
*
|
||
|
* @param str
|
||
|
*/
|
||
|
module.exports = function (str) {
|
||
|
var parsed = url.parse(str)
|
||
|
var rawQs
|
||
|
var search
|
||
|
var path
|
||
|
var qs
|
||
|
|
||
|
rawQs = parsed.query
|
||
|
|
||
|
if (rawQs && rawQs.length) {
|
||
|
qs = stringify(parse(parsed.query))
|
||
|
search = QUESTION_MARK + qs
|
||
|
path = parsed.pathname + search
|
||
|
|
||
|
parsed.query = qs
|
||
|
parsed.search = search
|
||
|
parsed.path = path
|
||
|
|
||
|
str = url.format(parsed)
|
||
|
}
|
||
|
|
||
|
// Parse again, because Node does not guarantee consistency of properties
|
||
|
return url.parse(str)
|
||
|
}
|
||
|
|
||
|
module.exports.parse = parse
|
||
|
module.exports.stringify = stringify
|