| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360 | 'use strict';var Node = require('snapdragon-node');var utils = require('./utils');/** * Braces parsers */module.exports = function(braces, options) {  braces.parser    .set('bos', function() {      if (!this.parsed) {        this.ast = this.nodes[0] = new Node(this.ast);      }    })    /**     * Character parsers     */    .set('escape', function() {      var pos = this.position();      var m = this.match(/^(?:\\(.)|\$\{)/);      if (!m) return;      var prev = this.prev();      var last = utils.last(prev.nodes);      var node = pos(new Node({        type: 'text',        multiplier: 1,        val: m[0]      }));      if (node.val === '\\\\') {        return node;      }      if (node.val === '${') {        var str = this.input;        var idx = -1;        var ch;        while ((ch = str[++idx])) {          this.consume(1);          node.val += ch;          if (ch === '\\') {            node.val += str[++idx];            continue;          }          if (ch === '}') {            break;          }        }      }      if (this.options.unescape !== false) {        node.val = node.val.replace(/\\([{}])/g, '$1');      }      if (last.val === '"' && this.input.charAt(0) === '"') {        last.val = node.val;        this.consume(1);        return;      }      return concatNodes.call(this, pos, node, prev, options);    })    /**     * Brackets: "[...]" (basic, this is overridden by     * other parsers in more advanced implementations)     */    .set('bracket', function() {      var isInside = this.isInside('brace');      var pos = this.position();      var m = this.match(/^(?:\[([!^]?)([^\]]{2,}|\]-)(\]|[^*+?]+)|\[)/);      if (!m) return;      var prev = this.prev();      var val = m[0];      var negated = m[1] ? '^' : '';      var inner = m[2] || '';      var close = m[3] || '';      if (isInside && prev.type === 'brace') {        prev.text = prev.text || '';        prev.text += val;      }      var esc = this.input.slice(0, 2);      if (inner === '' && esc === '\\]') {        inner += esc;        this.consume(2);        var str = this.input;        var idx = -1;        var ch;        while ((ch = str[++idx])) {          this.consume(1);          if (ch === ']') {            close = ch;            break;          }          inner += ch;        }      }      return pos(new Node({        type: 'bracket',        val: val,        escaped: close !== ']',        negated: negated,        inner: inner,        close: close      }));    })    /**     * Empty braces (we capture these early to     * speed up processing in the compiler)     */    .set('multiplier', function() {      var isInside = this.isInside('brace');      var pos = this.position();      var m = this.match(/^\{((?:,|\{,+\})+)\}/);      if (!m) return;      this.multiplier = true;      var prev = this.prev();      var val = m[0];      if (isInside && prev.type === 'brace') {        prev.text = prev.text || '';        prev.text += val;      }      var node = pos(new Node({        type: 'text',        multiplier: 1,        match: m,        val: val      }));      return concatNodes.call(this, pos, node, prev, options);    })    /**     * Open     */    .set('brace.open', function() {      var pos = this.position();      var m = this.match(/^\{(?!(?:[^\\}]?|,+)\})/);      if (!m) return;      var prev = this.prev();      var last = utils.last(prev.nodes);      // if the last parsed character was an extglob character      // we need to _not optimize_ the brace pattern because      // it might be mistaken for an extglob by a downstream parser      if (last && last.val && isExtglobChar(last.val.slice(-1))) {        last.optimize = false;      }      var open = pos(new Node({        type: 'brace.open',        val: m[0]      }));      var node = pos(new Node({        type: 'brace',        nodes: []      }));      node.push(open);      prev.push(node);      this.push('brace', node);    })    /**     * Close     */    .set('brace.close', function() {      var pos = this.position();      var m = this.match(/^\}/);      if (!m || !m[0]) return;      var brace = this.pop('brace');      var node = pos(new Node({        type: 'brace.close',        val: m[0]      }));      if (!this.isType(brace, 'brace')) {        if (this.options.strict) {          throw new Error('missing opening "{"');        }        node.type = 'text';        node.multiplier = 0;        node.escaped = true;        return node;      }      var prev = this.prev();      var last = utils.last(prev.nodes);      if (last.text) {        var lastNode = utils.last(last.nodes);        if (lastNode.val === ')' && /[!@*?+]\(/.test(last.text)) {          var open = last.nodes[0];          var text = last.nodes[1];          if (open.type === 'brace.open' && text && text.type === 'text') {            text.optimize = false;          }        }      }      if (brace.nodes.length > 2) {        var first = brace.nodes[1];        if (first.type === 'text' && first.val === ',') {          brace.nodes.splice(1, 1);          brace.nodes.push(first);        }      }      brace.push(node);    })    /**     * Capture boundary characters     */    .set('boundary', function() {      var pos = this.position();      var m = this.match(/^[$^](?!\{)/);      if (!m) return;      return pos(new Node({        type: 'text',        val: m[0]      }));    })    /**     * One or zero, non-comma characters wrapped in braces     */    .set('nobrace', function() {      var isInside = this.isInside('brace');      var pos = this.position();      var m = this.match(/^\{[^,]?\}/);      if (!m) return;      var prev = this.prev();      var val = m[0];      if (isInside && prev.type === 'brace') {        prev.text = prev.text || '';        prev.text += val;      }      return pos(new Node({        type: 'text',        multiplier: 0,        val: val      }));    })    /**     * Text     */    .set('text', function() {      var isInside = this.isInside('brace');      var pos = this.position();      var m = this.match(/^((?!\\)[^${}[\]])+/);      if (!m) return;      var prev = this.prev();      var val = m[0];      if (isInside && prev.type === 'brace') {        prev.text = prev.text || '';        prev.text += val;      }      var node = pos(new Node({        type: 'text',        multiplier: 1,        val: val      }));      return concatNodes.call(this, pos, node, prev, options);    });};/** * Returns true if the character is an extglob character. */function isExtglobChar(ch) {  return ch === '!' || ch === '@' || ch === '*' || ch === '?' || ch === '+';}/** * Combine text nodes, and calculate empty sets (`{,,}`) * @param {Function} `pos` Function to calculate node position * @param {Object} `node` AST node * @return {Object} */function concatNodes(pos, node, parent, options) {  node.orig = node.val;  var prev = this.prev();  var last = utils.last(prev.nodes);  var isEscaped = false;  if (node.val.length > 1) {    var a = node.val.charAt(0);    var b = node.val.slice(-1);    isEscaped = (a === '"' && b === '"')      || (a === "'" && b === "'")      || (a === '`' && b === '`');  }  if (isEscaped && options.unescape !== false) {    node.val = node.val.slice(1, node.val.length - 1);    node.escaped = true;  }  if (node.match) {    var match = node.match[1];    if (!match || match.indexOf('}') === -1) {      match = node.match[0];    }    // replace each set with a single ","    var val = match.replace(/\{/g, ',').replace(/\}/g, '');    node.multiplier *= val.length;    node.val = '';  }  var simpleText = last.type === 'text'    && last.multiplier === 1    && node.multiplier === 1    && node.val;  if (simpleText) {    last.val += node.val;    return;  }  prev.push(node);}
 |