root / HServer / 00.Server / 00.Program / node_modules / brace-expansion / index.js
이력 | 보기 | 이력해설 | 다운로드 (4.68 KB)
1 |
var concatMap = require('concat-map'); |
---|---|
2 |
var balanced = require('balanced-match'); |
3 |
|
4 |
module.exports = expandTop; |
5 |
|
6 |
var escSlash = '\0SLASH'+Math.random()+'\0'; |
7 |
var escOpen = '\0OPEN'+Math.random()+'\0'; |
8 |
var escClose = '\0CLOSE'+Math.random()+'\0'; |
9 |
var escComma = '\0COMMA'+Math.random()+'\0'; |
10 |
var escPeriod = '\0PERIOD'+Math.random()+'\0'; |
11 |
|
12 |
function numeric(str) { |
13 |
return parseInt(str, 10) == str |
14 |
? parseInt(str, 10)
|
15 |
: str.charCodeAt(0);
|
16 |
} |
17 |
|
18 |
function escapeBraces(str) { |
19 |
return str.split('\\\\').join(escSlash) |
20 |
.split('\\{').join(escOpen)
|
21 |
.split('\\}').join(escClose)
|
22 |
.split('\\,').join(escComma)
|
23 |
.split('\\.').join(escPeriod);
|
24 |
} |
25 |
|
26 |
function unescapeBraces(str) { |
27 |
return str.split(escSlash).join('\\') |
28 |
.split(escOpen).join('{')
|
29 |
.split(escClose).join('}')
|
30 |
.split(escComma).join(',')
|
31 |
.split(escPeriod).join('.');
|
32 |
} |
33 |
|
34 |
|
35 |
// Basically just str.split(","), but handling cases
|
36 |
// where we have nested braced sections, which should be
|
37 |
// treated as individual members, like {a,{b,c},d}
|
38 |
function parseCommaParts(str) { |
39 |
if (!str)
|
40 |
return ['']; |
41 |
|
42 |
var parts = [];
|
43 |
var m = balanced('{', '}', str); |
44 |
|
45 |
if (!m)
|
46 |
return str.split(','); |
47 |
|
48 |
var pre = m.pre;
|
49 |
var body = m.body;
|
50 |
var post = m.post;
|
51 |
var p = pre.split(','); |
52 |
|
53 |
p[p.length-1] += '{' + body + '}'; |
54 |
var postParts = parseCommaParts(post);
|
55 |
if (post.length) {
|
56 |
p[p.length-1] += postParts.shift();
|
57 |
p.push.apply(p, postParts); |
58 |
} |
59 |
|
60 |
parts.push.apply(parts, p); |
61 |
|
62 |
return parts;
|
63 |
} |
64 |
|
65 |
function expandTop(str) { |
66 |
if (!str)
|
67 |
return [];
|
68 |
|
69 |
// I don't know why Bash 4.3 does this, but it does.
|
70 |
// Anything starting with {} will have the first two bytes preserved
|
71 |
// but *only* at the top level, so {},a}b will not expand to anything,
|
72 |
// but a{},b}c will be expanded to [a}c,abc].
|
73 |
// One could argue that this is a bug in Bash, but since the goal of
|
74 |
// this module is to match Bash's rules, we escape a leading {}
|
75 |
if (str.substr(0, 2) === '{}') { |
76 |
str = '\\{\\}' + str.substr(2); |
77 |
} |
78 |
|
79 |
return expand(escapeBraces(str), true).map(unescapeBraces); |
80 |
} |
81 |
|
82 |
function identity(e) { |
83 |
return e;
|
84 |
} |
85 |
|
86 |
function embrace(str) { |
87 |
return '{' + str + '}'; |
88 |
} |
89 |
function isPadded(el) { |
90 |
return /^-?0\d/.test(el); |
91 |
} |
92 |
|
93 |
function lte(i, y) { |
94 |
return i <= y;
|
95 |
} |
96 |
function gte(i, y) { |
97 |
return i >= y;
|
98 |
} |
99 |
|
100 |
function expand(str, isTop) { |
101 |
var expansions = [];
|
102 |
|
103 |
var m = balanced('{', '}', str); |
104 |
if (!m || /\$$/.test(m.pre)) return [str]; |
105 |
|
106 |
var isNumericSequence = /^-?\d+\.\.-?\d+(?:\.\.-?\d+)?$/.test(m.body); |
107 |
var isAlphaSequence = /^[a-zA-Z]\.\.[a-zA-Z](?:\.\.-?\d+)?$/.test(m.body); |
108 |
var isSequence = isNumericSequence || isAlphaSequence;
|
109 |
var isOptions = m.body.indexOf(',') >= 0; |
110 |
if (!isSequence && !isOptions) {
|
111 |
// {a},b}
|
112 |
if (m.post.match(/,.*\}/)) { |
113 |
str = m.pre + '{' + m.body + escClose + m.post;
|
114 |
return expand(str);
|
115 |
} |
116 |
return [str];
|
117 |
} |
118 |
|
119 |
var n;
|
120 |
if (isSequence) {
|
121 |
n = m.body.split(/\.\./);
|
122 |
} else {
|
123 |
n = parseCommaParts(m.body); |
124 |
if (n.length === 1) { |
125 |
// x{{a,b}}y ==> x{a}y x{b}y
|
126 |
n = expand(n[0], false).map(embrace); |
127 |
if (n.length === 1) { |
128 |
var post = m.post.length
|
129 |
? expand(m.post, false)
|
130 |
: [''];
|
131 |
return post.map(function(p) { |
132 |
return m.pre + n[0] + p; |
133 |
}); |
134 |
} |
135 |
} |
136 |
} |
137 |
|
138 |
// at this point, n is the parts, and we know it's not a comma set
|
139 |
// with a single entry.
|
140 |
|
141 |
// no need to expand pre, since it is guaranteed to be free of brace-sets
|
142 |
var pre = m.pre;
|
143 |
var post = m.post.length
|
144 |
? expand(m.post, false)
|
145 |
: [''];
|
146 |
|
147 |
var N;
|
148 |
|
149 |
if (isSequence) {
|
150 |
var x = numeric(n[0]); |
151 |
var y = numeric(n[1]); |
152 |
var width = Math.max(n[0].length, n[1].length) |
153 |
var incr = n.length == 3 |
154 |
? Math.abs(numeric(n[2]))
|
155 |
: 1;
|
156 |
var test = lte;
|
157 |
var reverse = y < x;
|
158 |
if (reverse) {
|
159 |
incr *= -1;
|
160 |
test = gte; |
161 |
} |
162 |
var pad = n.some(isPadded);
|
163 |
|
164 |
N = []; |
165 |
|
166 |
for (var i = x; test(i, y); i += incr) { |
167 |
var c;
|
168 |
if (isAlphaSequence) {
|
169 |
c = String.fromCharCode(i); |
170 |
if (c === '\\') |
171 |
c = '';
|
172 |
} else {
|
173 |
c = String(i); |
174 |
if (pad) {
|
175 |
var need = width - c.length;
|
176 |
if (need > 0) { |
177 |
var z = new Array(need + 1).join('0'); |
178 |
if (i < 0) |
179 |
c = '-' + z + c.slice(1); |
180 |
else
|
181 |
c = z + c; |
182 |
} |
183 |
} |
184 |
} |
185 |
N.push(c); |
186 |
} |
187 |
} else {
|
188 |
N = concatMap(n, function(el) { return expand(el, false) }); |
189 |
} |
190 |
|
191 |
for (var j = 0; j < N.length; j++) { |
192 |
for (var k = 0; k < post.length; k++) { |
193 |
var expansion = pre + N[j] + post[k];
|
194 |
if (!isTop || isSequence || expansion)
|
195 |
expansions.push(expansion); |
196 |
} |
197 |
} |
198 |
|
199 |
return expansions;
|
200 |
} |
201 |
|