| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 | 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 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) return [str];  // 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)    : [''];  if (/\$$/.test(m.pre)) {        for (var k = 0; k < post.length; k++) {      var expansion = pre+ '{' + m.body + '}' + post[k];      expansions.push(expansion);    }  } else {    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) {          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.    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 = [];      for (var j = 0; j < n.length; j++) {        N.push.apply(N, expand(n[j], 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;}
 |