| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226 | 'use strict';// '<(' is process substitution operator and// can be parsed the same as control operatorvar CONTROL = '(?:' + [	'\\|\\|',	'\\&\\&',	';;',	'\\|\\&',	'\\<\\(',	'\\<\\<\\<',	'>>',	'>\\&',	'<\\&',	'[&;()|<>]'].join('|') + ')';var controlRE = new RegExp('^' + CONTROL + '$');var META = '|&;()<> \\t';var SINGLE_QUOTE = '"((\\\\"|[^"])*?)"';var DOUBLE_QUOTE = '\'((\\\\\'|[^\'])*?)\'';var hash = /^#$/;var SQ = "'";var DQ = '"';var DS = '$';var TOKEN = '';var mult = 0x100000000; // Math.pow(16, 8);for (var i = 0; i < 4; i++) {	TOKEN += (mult * Math.random()).toString(16);}var startsWithToken = new RegExp('^' + TOKEN);function matchAll(s, r) {	var origIndex = r.lastIndex;	var matches = [];	var matchObj;	while ((matchObj = r.exec(s))) {		matches.push(matchObj);		if (r.lastIndex === matchObj.index) {			r.lastIndex += 1;		}	}	r.lastIndex = origIndex;	return matches;}function getVar(env, pre, key) {	var r = typeof env === 'function' ? env(key) : env[key];	if (typeof r === 'undefined' && key != '') {		r = '';	} else if (typeof r === 'undefined') {		r = '$';	}	if (typeof r === 'object') {		return pre + TOKEN + JSON.stringify(r) + TOKEN;	}	return pre + r;}function parseInternal(string, env, opts) {	if (!opts) {		opts = {};	}	var BS = opts.escape || '\\';	var BAREWORD = '(\\' + BS + '[\'"' + META + ']|[^\\s\'"' + META + '])+';	var chunker = new RegExp([		'(' + CONTROL + ')', // control chars		'(' + BAREWORD + '|' + SINGLE_QUOTE + '|' + DOUBLE_QUOTE + ')+'	].join('|'), 'g');	var matches = matchAll(string, chunker);	if (matches.length === 0) {		return [];	}	if (!env) {		env = {};	}	var commented = false;	return matches.map(function (match) {		var s = match[0];		if (!s || commented) {			return void undefined;		}		if (controlRE.test(s)) {			return { op: s };		}		// Hand-written scanner/parser for Bash quoting rules:		//		// 1. inside single quotes, all characters are printed literally.		// 2. inside double quotes, all characters are printed literally		//    except variables prefixed by '$' and backslashes followed by		//    either a double quote or another backslash.		// 3. outside of any quotes, backslashes are treated as escape		//    characters and not printed (unless they are themselves escaped)		// 4. quote context can switch mid-token if there is no whitespace		//     between the two quote contexts (e.g. all'one'"token" parses as		//     "allonetoken")		var quote = false;		var esc = false;		var out = '';		var isGlob = false;		var i;		function parseEnvVar() {			i += 1;			var varend;			var varname;			var char = s.charAt(i);			if (char === '{') {				i += 1;				if (s.charAt(i) === '}') {					throw new Error('Bad substitution: ' + s.slice(i - 2, i + 1));				}				varend = s.indexOf('}', i);				if (varend < 0) {					throw new Error('Bad substitution: ' + s.slice(i));				}				varname = s.slice(i, varend);				i = varend;			} else if ((/[*@#?$!_-]/).test(char)) {				varname = char;				i += 1;			} else {				var slicedFromI = s.slice(i);				varend = slicedFromI.match(/[^\w\d_]/);				if (!varend) {					varname = slicedFromI;					i = s.length;				} else {					varname = slicedFromI.slice(0, varend.index);					i += varend.index - 1;				}			}			return getVar(env, '', varname);		}		for (i = 0; i < s.length; i++) {			var c = s.charAt(i);			isGlob = isGlob || (!quote && (c === '*' || c === '?'));			if (esc) {				out += c;				esc = false;			} else if (quote) {				if (c === quote) {					quote = false;				} else if (quote == SQ) {					out += c;				} else { // Double quote					if (c === BS) {						i += 1;						c = s.charAt(i);						if (c === DQ || c === BS || c === DS) {							out += c;						} else {							out += BS + c;						}					} else if (c === DS) {						out += parseEnvVar();					} else {						out += c;					}				}			} else if (c === DQ || c === SQ) {				quote = c;			} else if (controlRE.test(c)) {				return { op: s };			} else if (hash.test(c)) {				commented = true;				var commentObj = { comment: string.slice(match.index + i + 1) };				if (out.length) {					return [out, commentObj];				}				return [commentObj];			} else if (c === BS) {				esc = true;			} else if (c === DS) {				out += parseEnvVar();			} else {				out += c;			}		}		if (isGlob) {			return { op: 'glob', pattern: out };		}		return out;	}).reduce(function (prev, arg) { // finalize parsed arguments		// TODO: replace this whole reduce with a concat		return typeof arg === 'undefined' ? prev : prev.concat(arg);	}, []);}module.exports = function parse(s, env, opts) {	var mapped = parseInternal(s, env, opts);	if (typeof env !== 'function') {		return mapped;	}	return mapped.reduce(function (acc, s) {		if (typeof s === 'object') {			return acc.concat(s);		}		var xs = s.split(RegExp('(' + TOKEN + '.*?' + TOKEN + ')', 'g'));		if (xs.length === 1) {			return acc.concat(xs[0]);		}		return acc.concat(xs.filter(Boolean).map(function (x) {			if (startsWithToken.test(x)) {				return JSON.parse(x.split(TOKEN)[1]);			}			return x;		}));	}, []);};
 |