Permalink
Name already in use
A tag already exists with the provided branch name. Many Git commands accept both tag and branch names, so creating this branch may cause unexpected behavior. Are you sure you want to create this branch?
codeql-action/node_modules/ini/ini.js
Go to fileThis commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
206 lines (180 sloc)
4.86 KB
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
exports.parse = exports.decode = decode | |
exports.stringify = exports.encode = encode | |
exports.safe = safe | |
exports.unsafe = unsafe | |
var eol = typeof process !== 'undefined' && | |
process.platform === 'win32' ? '\r\n' : '\n' | |
function encode (obj, opt) { | |
var children = [] | |
var out = '' | |
if (typeof opt === 'string') { | |
opt = { | |
section: opt, | |
whitespace: false, | |
} | |
} else { | |
opt = opt || {} | |
opt.whitespace = opt.whitespace === true | |
} | |
var separator = opt.whitespace ? ' = ' : '=' | |
Object.keys(obj).forEach(function (k, _, __) { | |
var val = obj[k] | |
if (val && Array.isArray(val)) { | |
val.forEach(function (item) { | |
out += safe(k + '[]') + separator + safe(item) + '\n' | |
}) | |
} else if (val && typeof val === 'object') | |
children.push(k) | |
else | |
out += safe(k) + separator + safe(val) + eol | |
}) | |
if (opt.section && out.length) | |
out = '[' + safe(opt.section) + ']' + eol + out | |
children.forEach(function (k, _, __) { | |
var nk = dotSplit(k).join('\\.') | |
var section = (opt.section ? opt.section + '.' : '') + nk | |
var child = encode(obj[k], { | |
section: section, | |
whitespace: opt.whitespace, | |
}) | |
if (out.length && child.length) | |
out += eol | |
out += child | |
}) | |
return out | |
} | |
function dotSplit (str) { | |
return str.replace(/\1/g, '\u0002LITERAL\\1LITERAL\u0002') | |
.replace(/\\\./g, '\u0001') | |
.split(/\./).map(function (part) { | |
return part.replace(/\1/g, '\\.') | |
.replace(/\2LITERAL\\1LITERAL\2/g, '\u0001') | |
}) | |
} | |
function decode (str) { | |
var out = {} | |
var p = out | |
var section = null | |
// section |key = value | |
var re = /^\[([^\]]*)\]$|^([^=]+)(=(.*))?$/i | |
var lines = str.split(/[\r\n]+/g) | |
lines.forEach(function (line, _, __) { | |
if (!line || line.match(/^\s*[;#]/)) | |
return | |
var match = line.match(re) | |
if (!match) | |
return | |
if (match[1] !== undefined) { | |
section = unsafe(match[1]) | |
if (section === '__proto__') { | |
// not allowed | |
// keep parsing the section, but don't attach it. | |
p = {} | |
return | |
} | |
p = out[section] = out[section] || {} | |
return | |
} | |
var key = unsafe(match[2]) | |
if (key === '__proto__') | |
return | |
var value = match[3] ? unsafe(match[4]) : true | |
switch (value) { | |
case 'true': | |
case 'false': | |
case 'null': value = JSON.parse(value) | |
} | |
// Convert keys with '[]' suffix to an array | |
if (key.length > 2 && key.slice(-2) === '[]') { | |
key = key.substring(0, key.length - 2) | |
if (key === '__proto__') | |
return | |
if (!p[key]) | |
p[key] = [] | |
else if (!Array.isArray(p[key])) | |
p[key] = [p[key]] | |
} | |
// safeguard against resetting a previously defined | |
// array by accidentally forgetting the brackets | |
if (Array.isArray(p[key])) | |
p[key].push(value) | |
else | |
p[key] = value | |
}) | |
// {a:{y:1},"a.b":{x:2}} --> {a:{y:1,b:{x:2}}} | |
// use a filter to return the keys that have to be deleted. | |
Object.keys(out).filter(function (k, _, __) { | |
if (!out[k] || | |
typeof out[k] !== 'object' || | |
Array.isArray(out[k])) | |
return false | |
// see if the parent section is also an object. | |
// if so, add it to that, and mark this one for deletion | |
var parts = dotSplit(k) | |
var p = out | |
var l = parts.pop() | |
var nl = l.replace(/\\\./g, '.') | |
parts.forEach(function (part, _, __) { | |
if (part === '__proto__') | |
return | |
if (!p[part] || typeof p[part] !== 'object') | |
p[part] = {} | |
p = p[part] | |
}) | |
if (p === out && nl === l) | |
return false | |
p[nl] = out[k] | |
return true | |
}).forEach(function (del, _, __) { | |
delete out[del] | |
}) | |
return out | |
} | |
function isQuoted (val) { | |
return (val.charAt(0) === '"' && val.slice(-1) === '"') || | |
(val.charAt(0) === "'" && val.slice(-1) === "'") | |
} | |
function safe (val) { | |
return (typeof val !== 'string' || | |
val.match(/[=\r\n]/) || | |
val.match(/^\[/) || | |
(val.length > 1 && | |
isQuoted(val)) || | |
val !== val.trim()) | |
? JSON.stringify(val) | |
: val.replace(/;/g, '\\;').replace(/#/g, '\\#') | |
} | |
function unsafe (val, doUnesc) { | |
val = (val || '').trim() | |
if (isQuoted(val)) { | |
// remove the single quotes before calling JSON.parse | |
if (val.charAt(0) === "'") | |
val = val.substr(1, val.length - 2) | |
try { | |
val = JSON.parse(val) | |
} catch (_) {} | |
} else { | |
// walk the val to find the first not-escaped ; character | |
var esc = false | |
var unesc = '' | |
for (var i = 0, l = val.length; i < l; i++) { | |
var c = val.charAt(i) | |
if (esc) { | |
if ('\\;#'.indexOf(c) !== -1) | |
unesc += c | |
else | |
unesc += '\\' + c | |
esc = false | |
} else if (';#'.indexOf(c) !== -1) | |
break | |
else if (c === '\\') | |
esc = true | |
else | |
unesc += c | |
} | |
if (esc) | |
unesc += '\\' | |
return unesc.trim() | |
} | |
return val | |
} |