| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 | 
							- 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 = /^(.*,)+(.+)?$/.test(m.body);
 
-   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;
 
- }
 
 
  |