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;
							 | 
						|
								}
							 | 
						|
								
							 |