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/brace-expansion/index.js
Go to fileThis commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

201 lines (169 sloc)
4.68 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
var concatMap = require('concat-map'); | |
var balanced = require('balanced-match'); | |
module.exports = expandTop; | |
var escSlash = '\0SLASH'+Math.random()+'\0'; | |
var escOpen = '\0OPEN'+Math.random()+'\0'; | |
var escClose = '\0CLOSE'+Math.random()+'\0'; | |
var escComma = '\0COMMA'+Math.random()+'\0'; | |
var escPeriod = '\0PERIOD'+Math.random()+'\0'; | |
function numeric(str) { | |
return parseInt(str, 10) == str | |
? parseInt(str, 10) | |
: str.charCodeAt(0); | |
} | |
function escapeBraces(str) { | |
return str.split('\\\\').join(escSlash) | |
.split('\\{').join(escOpen) | |
.split('\\}').join(escClose) | |
.split('\\,').join(escComma) | |
.split('\\.').join(escPeriod); | |
} | |
function unescapeBraces(str) { | |
return str.split(escSlash).join('\\') | |
.split(escOpen).join('{') | |
.split(escClose).join('}') | |
.split(escComma).join(',') | |
.split(escPeriod).join('.'); | |
} | |
// Basically just str.split(","), but handling cases | |
// where we have nested braced sections, which should be | |
// treated as individual members, like {a,{b,c},d} | |
function parseCommaParts(str) { | |
if (!str) | |
return ['']; | |
var parts = []; | |
var m = balanced('{', '}', str); | |
if (!m) | |
return str.split(','); | |
var pre = m.pre; | |
var body = m.body; | |
var post = m.post; | |
var p = pre.split(','); | |
p[p.length-1] += '{' + body + '}'; | |
var postParts = parseCommaParts(post); | |
if (post.length) { | |
p[p.length-1] += postParts.shift(); | |
p.push.apply(p, postParts); | |
} | |
parts.push.apply(parts, p); | |
return parts; | |
} | |
function expandTop(str) { | |
if (!str) | |
return []; | |
// I don't know why Bash 4.3 does this, but it does. | |
// Anything starting with {} will have the first two bytes preserved | |
// but *only* at the top level, so {},a}b will not expand to anything, | |
// but a{},b}c will be expanded to [a}c,abc]. | |
// One could argue that this is a bug in Bash, but since the goal of | |
// this module is to match Bash's rules, we escape a leading {} | |
if (str.substr(0, 2) === '{}') { | |
str = '\\{\\}' + str.substr(2); | |
} | |
return expand(escapeBraces(str), true).map(unescapeBraces); | |
} | |
function identity(e) { | |
return e; | |
} | |
function embrace(str) { | |
return '{' + str + '}'; | |
} | |
function isPadded(el) { | |
return /^-?0\d/.test(el); | |
} | |
function lte(i, y) { | |
return i <= y; | |
} | |
function gte(i, y) { | |
return i >= y; | |
} | |
function expand(str, isTop) { | |
var expansions = []; | |
var m = balanced('{', '}', str); | |
if (!m || /\$$/.test(m.pre)) return [str]; | |
var isNumericSequence = /^-?\d+\.\.-?\d+(?:\.\.-?\d+)?$/.test(m.body); | |
var isAlphaSequence = /^[a-zA-Z]\.\.[a-zA-Z](?:\.\.-?\d+)?$/.test(m.body); | |
var isSequence = isNumericSequence || isAlphaSequence; | |
var isOptions = m.body.indexOf(',') >= 0; | |
if (!isSequence && !isOptions) { | |
// {a},b} | |
if (m.post.match(/,.*\}/)) { | |
str = m.pre + '{' + m.body + escClose + m.post; | |
return expand(str); | |
} | |
return [str]; | |
} | |
var n; | |
if (isSequence) { | |
n = m.body.split(/\.\./); | |
} else { | |
n = parseCommaParts(m.body); | |
if (n.length === 1) { | |
// x{{a,b}}y ==> x{a}y x{b}y | |
n = expand(n[0], false).map(embrace); | |
if (n.length === 1) { | |
var post = m.post.length | |
? expand(m.post, false) | |
: ['']; | |
return post.map(function(p) { | |
return m.pre + n[0] + p; | |
}); | |
} | |
} | |
} | |
// at this point, n is the parts, and we know it's not a comma set | |
// with a single entry. | |
// no need to expand pre, since it is guaranteed to be free of brace-sets | |
var pre = m.pre; | |
var post = m.post.length | |
? expand(m.post, false) | |
: ['']; | |
var N; | |
if (isSequence) { | |
var x = numeric(n[0]); | |
var y = numeric(n[1]); | |
var width = Math.max(n[0].length, n[1].length) | |
var incr = n.length == 3 | |
? Math.abs(numeric(n[2])) | |
: 1; | |
var test = lte; | |
var reverse = y < x; | |
if (reverse) { | |
incr *= -1; | |
test = gte; | |
} | |
var pad = n.some(isPadded); | |
N = []; | |
for (var i = x; test(i, y); i += incr) { | |
var c; | |
if (isAlphaSequence) { | |
c = String.fromCharCode(i); | |
if (c === '\\') | |
c = ''; | |
} else { | |
c = String(i); | |
if (pad) { | |
var need = width - c.length; | |
if (need > 0) { | |
var z = new Array(need + 1).join('0'); | |
if (i < 0) | |
c = '-' + z + c.slice(1); | |
else | |
c = z + c; | |
} | |
} | |
} | |
N.push(c); | |
} | |
} else { | |
N = concatMap(n, function(el) { return expand(el, false) }); | |
} | |
for (var j = 0; j < N.length; j++) { | |
for (var k = 0; k < post.length; k++) { | |
var expansion = pre + N[j] + post[k]; | |
if (!isTop || isSequence || expansion) | |
expansions.push(expansion); | |
} | |
} | |
return expansions; | |
} |