@kubb/agent 4.27.4 → 4.28.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (26) hide show
  1. package/.output/nitro.json +1 -1
  2. package/.output/server/chunks/nitro/nitro.mjs +274 -140
  3. package/.output/server/chunks/nitro/nitro.mjs.map +1 -1
  4. package/.output/server/chunks/routes/api/health.get.mjs +5 -0
  5. package/.output/server/chunks/routes/api/health.get.mjs.map +1 -1
  6. package/.output/server/index.mjs +6 -1
  7. package/.output/server/index.mjs.map +1 -1
  8. package/.output/server/node_modules/anymatch/index.js +104 -0
  9. package/.output/server/node_modules/anymatch/package.json +48 -0
  10. package/.output/server/node_modules/chokidar/handler.js +632 -0
  11. package/.output/server/node_modules/chokidar/index.js +822 -0
  12. package/.output/server/node_modules/chokidar/package.json +63 -0
  13. package/.output/server/node_modules/normalize-path/index.js +35 -0
  14. package/.output/server/node_modules/normalize-path/package.json +77 -0
  15. package/.output/server/node_modules/picomatch/index.js +3 -0
  16. package/.output/server/node_modules/picomatch/lib/constants.js +179 -0
  17. package/.output/server/node_modules/picomatch/lib/parse.js +1091 -0
  18. package/.output/server/node_modules/picomatch/lib/picomatch.js +342 -0
  19. package/.output/server/node_modules/picomatch/lib/scan.js +391 -0
  20. package/.output/server/node_modules/picomatch/lib/utils.js +64 -0
  21. package/.output/server/node_modules/picomatch/package.json +81 -0
  22. package/.output/server/node_modules/readdirp/index.js +272 -0
  23. package/.output/server/node_modules/readdirp/package.json +66 -0
  24. package/.output/server/package.json +6 -1
  25. package/README.md +98 -42
  26. package/package.json +16 -23
@@ -0,0 +1,1091 @@
1
+ 'use strict';
2
+
3
+ const constants = require('./constants');
4
+ const utils = require('./utils');
5
+
6
+ /**
7
+ * Constants
8
+ */
9
+
10
+ const {
11
+ MAX_LENGTH,
12
+ POSIX_REGEX_SOURCE,
13
+ REGEX_NON_SPECIAL_CHARS,
14
+ REGEX_SPECIAL_CHARS_BACKREF,
15
+ REPLACEMENTS
16
+ } = constants;
17
+
18
+ /**
19
+ * Helpers
20
+ */
21
+
22
+ const expandRange = (args, options) => {
23
+ if (typeof options.expandRange === 'function') {
24
+ return options.expandRange(...args, options);
25
+ }
26
+
27
+ args.sort();
28
+ const value = `[${args.join('-')}]`;
29
+
30
+ try {
31
+ /* eslint-disable-next-line no-new */
32
+ new RegExp(value);
33
+ } catch (ex) {
34
+ return args.map(v => utils.escapeRegex(v)).join('..');
35
+ }
36
+
37
+ return value;
38
+ };
39
+
40
+ /**
41
+ * Create the message for a syntax error
42
+ */
43
+
44
+ const syntaxError = (type, char) => {
45
+ return `Missing ${type}: "${char}" - use "\\\\${char}" to match literal characters`;
46
+ };
47
+
48
+ /**
49
+ * Parse the given input string.
50
+ * @param {String} input
51
+ * @param {Object} options
52
+ * @return {Object}
53
+ */
54
+
55
+ const parse = (input, options) => {
56
+ if (typeof input !== 'string') {
57
+ throw new TypeError('Expected a string');
58
+ }
59
+
60
+ input = REPLACEMENTS[input] || input;
61
+
62
+ const opts = { ...options };
63
+ const max = typeof opts.maxLength === 'number' ? Math.min(MAX_LENGTH, opts.maxLength) : MAX_LENGTH;
64
+
65
+ let len = input.length;
66
+ if (len > max) {
67
+ throw new SyntaxError(`Input length: ${len}, exceeds maximum allowed length: ${max}`);
68
+ }
69
+
70
+ const bos = { type: 'bos', value: '', output: opts.prepend || '' };
71
+ const tokens = [bos];
72
+
73
+ const capture = opts.capture ? '' : '?:';
74
+ const win32 = utils.isWindows(options);
75
+
76
+ // create constants based on platform, for windows or posix
77
+ const PLATFORM_CHARS = constants.globChars(win32);
78
+ const EXTGLOB_CHARS = constants.extglobChars(PLATFORM_CHARS);
79
+
80
+ const {
81
+ DOT_LITERAL,
82
+ PLUS_LITERAL,
83
+ SLASH_LITERAL,
84
+ ONE_CHAR,
85
+ DOTS_SLASH,
86
+ NO_DOT,
87
+ NO_DOT_SLASH,
88
+ NO_DOTS_SLASH,
89
+ QMARK,
90
+ QMARK_NO_DOT,
91
+ STAR,
92
+ START_ANCHOR
93
+ } = PLATFORM_CHARS;
94
+
95
+ const globstar = opts => {
96
+ return `(${capture}(?:(?!${START_ANCHOR}${opts.dot ? DOTS_SLASH : DOT_LITERAL}).)*?)`;
97
+ };
98
+
99
+ const nodot = opts.dot ? '' : NO_DOT;
100
+ const qmarkNoDot = opts.dot ? QMARK : QMARK_NO_DOT;
101
+ let star = opts.bash === true ? globstar(opts) : STAR;
102
+
103
+ if (opts.capture) {
104
+ star = `(${star})`;
105
+ }
106
+
107
+ // minimatch options support
108
+ if (typeof opts.noext === 'boolean') {
109
+ opts.noextglob = opts.noext;
110
+ }
111
+
112
+ const state = {
113
+ input,
114
+ index: -1,
115
+ start: 0,
116
+ dot: opts.dot === true,
117
+ consumed: '',
118
+ output: '',
119
+ prefix: '',
120
+ backtrack: false,
121
+ negated: false,
122
+ brackets: 0,
123
+ braces: 0,
124
+ parens: 0,
125
+ quotes: 0,
126
+ globstar: false,
127
+ tokens
128
+ };
129
+
130
+ input = utils.removePrefix(input, state);
131
+ len = input.length;
132
+
133
+ const extglobs = [];
134
+ const braces = [];
135
+ const stack = [];
136
+ let prev = bos;
137
+ let value;
138
+
139
+ /**
140
+ * Tokenizing helpers
141
+ */
142
+
143
+ const eos = () => state.index === len - 1;
144
+ const peek = state.peek = (n = 1) => input[state.index + n];
145
+ const advance = state.advance = () => input[++state.index] || '';
146
+ const remaining = () => input.slice(state.index + 1);
147
+ const consume = (value = '', num = 0) => {
148
+ state.consumed += value;
149
+ state.index += num;
150
+ };
151
+
152
+ const append = token => {
153
+ state.output += token.output != null ? token.output : token.value;
154
+ consume(token.value);
155
+ };
156
+
157
+ const negate = () => {
158
+ let count = 1;
159
+
160
+ while (peek() === '!' && (peek(2) !== '(' || peek(3) === '?')) {
161
+ advance();
162
+ state.start++;
163
+ count++;
164
+ }
165
+
166
+ if (count % 2 === 0) {
167
+ return false;
168
+ }
169
+
170
+ state.negated = true;
171
+ state.start++;
172
+ return true;
173
+ };
174
+
175
+ const increment = type => {
176
+ state[type]++;
177
+ stack.push(type);
178
+ };
179
+
180
+ const decrement = type => {
181
+ state[type]--;
182
+ stack.pop();
183
+ };
184
+
185
+ /**
186
+ * Push tokens onto the tokens array. This helper speeds up
187
+ * tokenizing by 1) helping us avoid backtracking as much as possible,
188
+ * and 2) helping us avoid creating extra tokens when consecutive
189
+ * characters are plain text. This improves performance and simplifies
190
+ * lookbehinds.
191
+ */
192
+
193
+ const push = tok => {
194
+ if (prev.type === 'globstar') {
195
+ const isBrace = state.braces > 0 && (tok.type === 'comma' || tok.type === 'brace');
196
+ const isExtglob = tok.extglob === true || (extglobs.length && (tok.type === 'pipe' || tok.type === 'paren'));
197
+
198
+ if (tok.type !== 'slash' && tok.type !== 'paren' && !isBrace && !isExtglob) {
199
+ state.output = state.output.slice(0, -prev.output.length);
200
+ prev.type = 'star';
201
+ prev.value = '*';
202
+ prev.output = star;
203
+ state.output += prev.output;
204
+ }
205
+ }
206
+
207
+ if (extglobs.length && tok.type !== 'paren') {
208
+ extglobs[extglobs.length - 1].inner += tok.value;
209
+ }
210
+
211
+ if (tok.value || tok.output) append(tok);
212
+ if (prev && prev.type === 'text' && tok.type === 'text') {
213
+ prev.value += tok.value;
214
+ prev.output = (prev.output || '') + tok.value;
215
+ return;
216
+ }
217
+
218
+ tok.prev = prev;
219
+ tokens.push(tok);
220
+ prev = tok;
221
+ };
222
+
223
+ const extglobOpen = (type, value) => {
224
+ const token = { ...EXTGLOB_CHARS[value], conditions: 1, inner: '' };
225
+
226
+ token.prev = prev;
227
+ token.parens = state.parens;
228
+ token.output = state.output;
229
+ const output = (opts.capture ? '(' : '') + token.open;
230
+
231
+ increment('parens');
232
+ push({ type, value, output: state.output ? '' : ONE_CHAR });
233
+ push({ type: 'paren', extglob: true, value: advance(), output });
234
+ extglobs.push(token);
235
+ };
236
+
237
+ const extglobClose = token => {
238
+ let output = token.close + (opts.capture ? ')' : '');
239
+ let rest;
240
+
241
+ if (token.type === 'negate') {
242
+ let extglobStar = star;
243
+
244
+ if (token.inner && token.inner.length > 1 && token.inner.includes('/')) {
245
+ extglobStar = globstar(opts);
246
+ }
247
+
248
+ if (extglobStar !== star || eos() || /^\)+$/.test(remaining())) {
249
+ output = token.close = `)$))${extglobStar}`;
250
+ }
251
+
252
+ if (token.inner.includes('*') && (rest = remaining()) && /^\.[^\\/.]+$/.test(rest)) {
253
+ // Any non-magical string (`.ts`) or even nested expression (`.{ts,tsx}`) can follow after the closing parenthesis.
254
+ // In this case, we need to parse the string and use it in the output of the original pattern.
255
+ // Suitable patterns: `/!(*.d).ts`, `/!(*.d).{ts,tsx}`, `**/!(*-dbg).@(js)`.
256
+ //
257
+ // Disabling the `fastpaths` option due to a problem with parsing strings as `.ts` in the pattern like `**/!(*.d).ts`.
258
+ const expression = parse(rest, { ...options, fastpaths: false }).output;
259
+
260
+ output = token.close = `)${expression})${extglobStar})`;
261
+ }
262
+
263
+ if (token.prev.type === 'bos') {
264
+ state.negatedExtglob = true;
265
+ }
266
+ }
267
+
268
+ push({ type: 'paren', extglob: true, value, output });
269
+ decrement('parens');
270
+ };
271
+
272
+ /**
273
+ * Fast paths
274
+ */
275
+
276
+ if (opts.fastpaths !== false && !/(^[*!]|[/()[\]{}"])/.test(input)) {
277
+ let backslashes = false;
278
+
279
+ let output = input.replace(REGEX_SPECIAL_CHARS_BACKREF, (m, esc, chars, first, rest, index) => {
280
+ if (first === '\\') {
281
+ backslashes = true;
282
+ return m;
283
+ }
284
+
285
+ if (first === '?') {
286
+ if (esc) {
287
+ return esc + first + (rest ? QMARK.repeat(rest.length) : '');
288
+ }
289
+ if (index === 0) {
290
+ return qmarkNoDot + (rest ? QMARK.repeat(rest.length) : '');
291
+ }
292
+ return QMARK.repeat(chars.length);
293
+ }
294
+
295
+ if (first === '.') {
296
+ return DOT_LITERAL.repeat(chars.length);
297
+ }
298
+
299
+ if (first === '*') {
300
+ if (esc) {
301
+ return esc + first + (rest ? star : '');
302
+ }
303
+ return star;
304
+ }
305
+ return esc ? m : `\\${m}`;
306
+ });
307
+
308
+ if (backslashes === true) {
309
+ if (opts.unescape === true) {
310
+ output = output.replace(/\\/g, '');
311
+ } else {
312
+ output = output.replace(/\\+/g, m => {
313
+ return m.length % 2 === 0 ? '\\\\' : (m ? '\\' : '');
314
+ });
315
+ }
316
+ }
317
+
318
+ if (output === input && opts.contains === true) {
319
+ state.output = input;
320
+ return state;
321
+ }
322
+
323
+ state.output = utils.wrapOutput(output, state, options);
324
+ return state;
325
+ }
326
+
327
+ /**
328
+ * Tokenize input until we reach end-of-string
329
+ */
330
+
331
+ while (!eos()) {
332
+ value = advance();
333
+
334
+ if (value === '\u0000') {
335
+ continue;
336
+ }
337
+
338
+ /**
339
+ * Escaped characters
340
+ */
341
+
342
+ if (value === '\\') {
343
+ const next = peek();
344
+
345
+ if (next === '/' && opts.bash !== true) {
346
+ continue;
347
+ }
348
+
349
+ if (next === '.' || next === ';') {
350
+ continue;
351
+ }
352
+
353
+ if (!next) {
354
+ value += '\\';
355
+ push({ type: 'text', value });
356
+ continue;
357
+ }
358
+
359
+ // collapse slashes to reduce potential for exploits
360
+ const match = /^\\+/.exec(remaining());
361
+ let slashes = 0;
362
+
363
+ if (match && match[0].length > 2) {
364
+ slashes = match[0].length;
365
+ state.index += slashes;
366
+ if (slashes % 2 !== 0) {
367
+ value += '\\';
368
+ }
369
+ }
370
+
371
+ if (opts.unescape === true) {
372
+ value = advance();
373
+ } else {
374
+ value += advance();
375
+ }
376
+
377
+ if (state.brackets === 0) {
378
+ push({ type: 'text', value });
379
+ continue;
380
+ }
381
+ }
382
+
383
+ /**
384
+ * If we're inside a regex character class, continue
385
+ * until we reach the closing bracket.
386
+ */
387
+
388
+ if (state.brackets > 0 && (value !== ']' || prev.value === '[' || prev.value === '[^')) {
389
+ if (opts.posix !== false && value === ':') {
390
+ const inner = prev.value.slice(1);
391
+ if (inner.includes('[')) {
392
+ prev.posix = true;
393
+
394
+ if (inner.includes(':')) {
395
+ const idx = prev.value.lastIndexOf('[');
396
+ const pre = prev.value.slice(0, idx);
397
+ const rest = prev.value.slice(idx + 2);
398
+ const posix = POSIX_REGEX_SOURCE[rest];
399
+ if (posix) {
400
+ prev.value = pre + posix;
401
+ state.backtrack = true;
402
+ advance();
403
+
404
+ if (!bos.output && tokens.indexOf(prev) === 1) {
405
+ bos.output = ONE_CHAR;
406
+ }
407
+ continue;
408
+ }
409
+ }
410
+ }
411
+ }
412
+
413
+ if ((value === '[' && peek() !== ':') || (value === '-' && peek() === ']')) {
414
+ value = `\\${value}`;
415
+ }
416
+
417
+ if (value === ']' && (prev.value === '[' || prev.value === '[^')) {
418
+ value = `\\${value}`;
419
+ }
420
+
421
+ if (opts.posix === true && value === '!' && prev.value === '[') {
422
+ value = '^';
423
+ }
424
+
425
+ prev.value += value;
426
+ append({ value });
427
+ continue;
428
+ }
429
+
430
+ /**
431
+ * If we're inside a quoted string, continue
432
+ * until we reach the closing double quote.
433
+ */
434
+
435
+ if (state.quotes === 1 && value !== '"') {
436
+ value = utils.escapeRegex(value);
437
+ prev.value += value;
438
+ append({ value });
439
+ continue;
440
+ }
441
+
442
+ /**
443
+ * Double quotes
444
+ */
445
+
446
+ if (value === '"') {
447
+ state.quotes = state.quotes === 1 ? 0 : 1;
448
+ if (opts.keepQuotes === true) {
449
+ push({ type: 'text', value });
450
+ }
451
+ continue;
452
+ }
453
+
454
+ /**
455
+ * Parentheses
456
+ */
457
+
458
+ if (value === '(') {
459
+ increment('parens');
460
+ push({ type: 'paren', value });
461
+ continue;
462
+ }
463
+
464
+ if (value === ')') {
465
+ if (state.parens === 0 && opts.strictBrackets === true) {
466
+ throw new SyntaxError(syntaxError('opening', '('));
467
+ }
468
+
469
+ const extglob = extglobs[extglobs.length - 1];
470
+ if (extglob && state.parens === extglob.parens + 1) {
471
+ extglobClose(extglobs.pop());
472
+ continue;
473
+ }
474
+
475
+ push({ type: 'paren', value, output: state.parens ? ')' : '\\)' });
476
+ decrement('parens');
477
+ continue;
478
+ }
479
+
480
+ /**
481
+ * Square brackets
482
+ */
483
+
484
+ if (value === '[') {
485
+ if (opts.nobracket === true || !remaining().includes(']')) {
486
+ if (opts.nobracket !== true && opts.strictBrackets === true) {
487
+ throw new SyntaxError(syntaxError('closing', ']'));
488
+ }
489
+
490
+ value = `\\${value}`;
491
+ } else {
492
+ increment('brackets');
493
+ }
494
+
495
+ push({ type: 'bracket', value });
496
+ continue;
497
+ }
498
+
499
+ if (value === ']') {
500
+ if (opts.nobracket === true || (prev && prev.type === 'bracket' && prev.value.length === 1)) {
501
+ push({ type: 'text', value, output: `\\${value}` });
502
+ continue;
503
+ }
504
+
505
+ if (state.brackets === 0) {
506
+ if (opts.strictBrackets === true) {
507
+ throw new SyntaxError(syntaxError('opening', '['));
508
+ }
509
+
510
+ push({ type: 'text', value, output: `\\${value}` });
511
+ continue;
512
+ }
513
+
514
+ decrement('brackets');
515
+
516
+ const prevValue = prev.value.slice(1);
517
+ if (prev.posix !== true && prevValue[0] === '^' && !prevValue.includes('/')) {
518
+ value = `/${value}`;
519
+ }
520
+
521
+ prev.value += value;
522
+ append({ value });
523
+
524
+ // when literal brackets are explicitly disabled
525
+ // assume we should match with a regex character class
526
+ if (opts.literalBrackets === false || utils.hasRegexChars(prevValue)) {
527
+ continue;
528
+ }
529
+
530
+ const escaped = utils.escapeRegex(prev.value);
531
+ state.output = state.output.slice(0, -prev.value.length);
532
+
533
+ // when literal brackets are explicitly enabled
534
+ // assume we should escape the brackets to match literal characters
535
+ if (opts.literalBrackets === true) {
536
+ state.output += escaped;
537
+ prev.value = escaped;
538
+ continue;
539
+ }
540
+
541
+ // when the user specifies nothing, try to match both
542
+ prev.value = `(${capture}${escaped}|${prev.value})`;
543
+ state.output += prev.value;
544
+ continue;
545
+ }
546
+
547
+ /**
548
+ * Braces
549
+ */
550
+
551
+ if (value === '{' && opts.nobrace !== true) {
552
+ increment('braces');
553
+
554
+ const open = {
555
+ type: 'brace',
556
+ value,
557
+ output: '(',
558
+ outputIndex: state.output.length,
559
+ tokensIndex: state.tokens.length
560
+ };
561
+
562
+ braces.push(open);
563
+ push(open);
564
+ continue;
565
+ }
566
+
567
+ if (value === '}') {
568
+ const brace = braces[braces.length - 1];
569
+
570
+ if (opts.nobrace === true || !brace) {
571
+ push({ type: 'text', value, output: value });
572
+ continue;
573
+ }
574
+
575
+ let output = ')';
576
+
577
+ if (brace.dots === true) {
578
+ const arr = tokens.slice();
579
+ const range = [];
580
+
581
+ for (let i = arr.length - 1; i >= 0; i--) {
582
+ tokens.pop();
583
+ if (arr[i].type === 'brace') {
584
+ break;
585
+ }
586
+ if (arr[i].type !== 'dots') {
587
+ range.unshift(arr[i].value);
588
+ }
589
+ }
590
+
591
+ output = expandRange(range, opts);
592
+ state.backtrack = true;
593
+ }
594
+
595
+ if (brace.comma !== true && brace.dots !== true) {
596
+ const out = state.output.slice(0, brace.outputIndex);
597
+ const toks = state.tokens.slice(brace.tokensIndex);
598
+ brace.value = brace.output = '\\{';
599
+ value = output = '\\}';
600
+ state.output = out;
601
+ for (const t of toks) {
602
+ state.output += (t.output || t.value);
603
+ }
604
+ }
605
+
606
+ push({ type: 'brace', value, output });
607
+ decrement('braces');
608
+ braces.pop();
609
+ continue;
610
+ }
611
+
612
+ /**
613
+ * Pipes
614
+ */
615
+
616
+ if (value === '|') {
617
+ if (extglobs.length > 0) {
618
+ extglobs[extglobs.length - 1].conditions++;
619
+ }
620
+ push({ type: 'text', value });
621
+ continue;
622
+ }
623
+
624
+ /**
625
+ * Commas
626
+ */
627
+
628
+ if (value === ',') {
629
+ let output = value;
630
+
631
+ const brace = braces[braces.length - 1];
632
+ if (brace && stack[stack.length - 1] === 'braces') {
633
+ brace.comma = true;
634
+ output = '|';
635
+ }
636
+
637
+ push({ type: 'comma', value, output });
638
+ continue;
639
+ }
640
+
641
+ /**
642
+ * Slashes
643
+ */
644
+
645
+ if (value === '/') {
646
+ // if the beginning of the glob is "./", advance the start
647
+ // to the current index, and don't add the "./" characters
648
+ // to the state. This greatly simplifies lookbehinds when
649
+ // checking for BOS characters like "!" and "." (not "./")
650
+ if (prev.type === 'dot' && state.index === state.start + 1) {
651
+ state.start = state.index + 1;
652
+ state.consumed = '';
653
+ state.output = '';
654
+ tokens.pop();
655
+ prev = bos; // reset "prev" to the first token
656
+ continue;
657
+ }
658
+
659
+ push({ type: 'slash', value, output: SLASH_LITERAL });
660
+ continue;
661
+ }
662
+
663
+ /**
664
+ * Dots
665
+ */
666
+
667
+ if (value === '.') {
668
+ if (state.braces > 0 && prev.type === 'dot') {
669
+ if (prev.value === '.') prev.output = DOT_LITERAL;
670
+ const brace = braces[braces.length - 1];
671
+ prev.type = 'dots';
672
+ prev.output += value;
673
+ prev.value += value;
674
+ brace.dots = true;
675
+ continue;
676
+ }
677
+
678
+ if ((state.braces + state.parens) === 0 && prev.type !== 'bos' && prev.type !== 'slash') {
679
+ push({ type: 'text', value, output: DOT_LITERAL });
680
+ continue;
681
+ }
682
+
683
+ push({ type: 'dot', value, output: DOT_LITERAL });
684
+ continue;
685
+ }
686
+
687
+ /**
688
+ * Question marks
689
+ */
690
+
691
+ if (value === '?') {
692
+ const isGroup = prev && prev.value === '(';
693
+ if (!isGroup && opts.noextglob !== true && peek() === '(' && peek(2) !== '?') {
694
+ extglobOpen('qmark', value);
695
+ continue;
696
+ }
697
+
698
+ if (prev && prev.type === 'paren') {
699
+ const next = peek();
700
+ let output = value;
701
+
702
+ if (next === '<' && !utils.supportsLookbehinds()) {
703
+ throw new Error('Node.js v10 or higher is required for regex lookbehinds');
704
+ }
705
+
706
+ if ((prev.value === '(' && !/[!=<:]/.test(next)) || (next === '<' && !/<([!=]|\w+>)/.test(remaining()))) {
707
+ output = `\\${value}`;
708
+ }
709
+
710
+ push({ type: 'text', value, output });
711
+ continue;
712
+ }
713
+
714
+ if (opts.dot !== true && (prev.type === 'slash' || prev.type === 'bos')) {
715
+ push({ type: 'qmark', value, output: QMARK_NO_DOT });
716
+ continue;
717
+ }
718
+
719
+ push({ type: 'qmark', value, output: QMARK });
720
+ continue;
721
+ }
722
+
723
+ /**
724
+ * Exclamation
725
+ */
726
+
727
+ if (value === '!') {
728
+ if (opts.noextglob !== true && peek() === '(') {
729
+ if (peek(2) !== '?' || !/[!=<:]/.test(peek(3))) {
730
+ extglobOpen('negate', value);
731
+ continue;
732
+ }
733
+ }
734
+
735
+ if (opts.nonegate !== true && state.index === 0) {
736
+ negate();
737
+ continue;
738
+ }
739
+ }
740
+
741
+ /**
742
+ * Plus
743
+ */
744
+
745
+ if (value === '+') {
746
+ if (opts.noextglob !== true && peek() === '(' && peek(2) !== '?') {
747
+ extglobOpen('plus', value);
748
+ continue;
749
+ }
750
+
751
+ if ((prev && prev.value === '(') || opts.regex === false) {
752
+ push({ type: 'plus', value, output: PLUS_LITERAL });
753
+ continue;
754
+ }
755
+
756
+ if ((prev && (prev.type === 'bracket' || prev.type === 'paren' || prev.type === 'brace')) || state.parens > 0) {
757
+ push({ type: 'plus', value });
758
+ continue;
759
+ }
760
+
761
+ push({ type: 'plus', value: PLUS_LITERAL });
762
+ continue;
763
+ }
764
+
765
+ /**
766
+ * Plain text
767
+ */
768
+
769
+ if (value === '@') {
770
+ if (opts.noextglob !== true && peek() === '(' && peek(2) !== '?') {
771
+ push({ type: 'at', extglob: true, value, output: '' });
772
+ continue;
773
+ }
774
+
775
+ push({ type: 'text', value });
776
+ continue;
777
+ }
778
+
779
+ /**
780
+ * Plain text
781
+ */
782
+
783
+ if (value !== '*') {
784
+ if (value === '$' || value === '^') {
785
+ value = `\\${value}`;
786
+ }
787
+
788
+ const match = REGEX_NON_SPECIAL_CHARS.exec(remaining());
789
+ if (match) {
790
+ value += match[0];
791
+ state.index += match[0].length;
792
+ }
793
+
794
+ push({ type: 'text', value });
795
+ continue;
796
+ }
797
+
798
+ /**
799
+ * Stars
800
+ */
801
+
802
+ if (prev && (prev.type === 'globstar' || prev.star === true)) {
803
+ prev.type = 'star';
804
+ prev.star = true;
805
+ prev.value += value;
806
+ prev.output = star;
807
+ state.backtrack = true;
808
+ state.globstar = true;
809
+ consume(value);
810
+ continue;
811
+ }
812
+
813
+ let rest = remaining();
814
+ if (opts.noextglob !== true && /^\([^?]/.test(rest)) {
815
+ extglobOpen('star', value);
816
+ continue;
817
+ }
818
+
819
+ if (prev.type === 'star') {
820
+ if (opts.noglobstar === true) {
821
+ consume(value);
822
+ continue;
823
+ }
824
+
825
+ const prior = prev.prev;
826
+ const before = prior.prev;
827
+ const isStart = prior.type === 'slash' || prior.type === 'bos';
828
+ const afterStar = before && (before.type === 'star' || before.type === 'globstar');
829
+
830
+ if (opts.bash === true && (!isStart || (rest[0] && rest[0] !== '/'))) {
831
+ push({ type: 'star', value, output: '' });
832
+ continue;
833
+ }
834
+
835
+ const isBrace = state.braces > 0 && (prior.type === 'comma' || prior.type === 'brace');
836
+ const isExtglob = extglobs.length && (prior.type === 'pipe' || prior.type === 'paren');
837
+ if (!isStart && prior.type !== 'paren' && !isBrace && !isExtglob) {
838
+ push({ type: 'star', value, output: '' });
839
+ continue;
840
+ }
841
+
842
+ // strip consecutive `/**/`
843
+ while (rest.slice(0, 3) === '/**') {
844
+ const after = input[state.index + 4];
845
+ if (after && after !== '/') {
846
+ break;
847
+ }
848
+ rest = rest.slice(3);
849
+ consume('/**', 3);
850
+ }
851
+
852
+ if (prior.type === 'bos' && eos()) {
853
+ prev.type = 'globstar';
854
+ prev.value += value;
855
+ prev.output = globstar(opts);
856
+ state.output = prev.output;
857
+ state.globstar = true;
858
+ consume(value);
859
+ continue;
860
+ }
861
+
862
+ if (prior.type === 'slash' && prior.prev.type !== 'bos' && !afterStar && eos()) {
863
+ state.output = state.output.slice(0, -(prior.output + prev.output).length);
864
+ prior.output = `(?:${prior.output}`;
865
+
866
+ prev.type = 'globstar';
867
+ prev.output = globstar(opts) + (opts.strictSlashes ? ')' : '|$)');
868
+ prev.value += value;
869
+ state.globstar = true;
870
+ state.output += prior.output + prev.output;
871
+ consume(value);
872
+ continue;
873
+ }
874
+
875
+ if (prior.type === 'slash' && prior.prev.type !== 'bos' && rest[0] === '/') {
876
+ const end = rest[1] !== void 0 ? '|$' : '';
877
+
878
+ state.output = state.output.slice(0, -(prior.output + prev.output).length);
879
+ prior.output = `(?:${prior.output}`;
880
+
881
+ prev.type = 'globstar';
882
+ prev.output = `${globstar(opts)}${SLASH_LITERAL}|${SLASH_LITERAL}${end})`;
883
+ prev.value += value;
884
+
885
+ state.output += prior.output + prev.output;
886
+ state.globstar = true;
887
+
888
+ consume(value + advance());
889
+
890
+ push({ type: 'slash', value: '/', output: '' });
891
+ continue;
892
+ }
893
+
894
+ if (prior.type === 'bos' && rest[0] === '/') {
895
+ prev.type = 'globstar';
896
+ prev.value += value;
897
+ prev.output = `(?:^|${SLASH_LITERAL}|${globstar(opts)}${SLASH_LITERAL})`;
898
+ state.output = prev.output;
899
+ state.globstar = true;
900
+ consume(value + advance());
901
+ push({ type: 'slash', value: '/', output: '' });
902
+ continue;
903
+ }
904
+
905
+ // remove single star from output
906
+ state.output = state.output.slice(0, -prev.output.length);
907
+
908
+ // reset previous token to globstar
909
+ prev.type = 'globstar';
910
+ prev.output = globstar(opts);
911
+ prev.value += value;
912
+
913
+ // reset output with globstar
914
+ state.output += prev.output;
915
+ state.globstar = true;
916
+ consume(value);
917
+ continue;
918
+ }
919
+
920
+ const token = { type: 'star', value, output: star };
921
+
922
+ if (opts.bash === true) {
923
+ token.output = '.*?';
924
+ if (prev.type === 'bos' || prev.type === 'slash') {
925
+ token.output = nodot + token.output;
926
+ }
927
+ push(token);
928
+ continue;
929
+ }
930
+
931
+ if (prev && (prev.type === 'bracket' || prev.type === 'paren') && opts.regex === true) {
932
+ token.output = value;
933
+ push(token);
934
+ continue;
935
+ }
936
+
937
+ if (state.index === state.start || prev.type === 'slash' || prev.type === 'dot') {
938
+ if (prev.type === 'dot') {
939
+ state.output += NO_DOT_SLASH;
940
+ prev.output += NO_DOT_SLASH;
941
+
942
+ } else if (opts.dot === true) {
943
+ state.output += NO_DOTS_SLASH;
944
+ prev.output += NO_DOTS_SLASH;
945
+
946
+ } else {
947
+ state.output += nodot;
948
+ prev.output += nodot;
949
+ }
950
+
951
+ if (peek() !== '*') {
952
+ state.output += ONE_CHAR;
953
+ prev.output += ONE_CHAR;
954
+ }
955
+ }
956
+
957
+ push(token);
958
+ }
959
+
960
+ while (state.brackets > 0) {
961
+ if (opts.strictBrackets === true) throw new SyntaxError(syntaxError('closing', ']'));
962
+ state.output = utils.escapeLast(state.output, '[');
963
+ decrement('brackets');
964
+ }
965
+
966
+ while (state.parens > 0) {
967
+ if (opts.strictBrackets === true) throw new SyntaxError(syntaxError('closing', ')'));
968
+ state.output = utils.escapeLast(state.output, '(');
969
+ decrement('parens');
970
+ }
971
+
972
+ while (state.braces > 0) {
973
+ if (opts.strictBrackets === true) throw new SyntaxError(syntaxError('closing', '}'));
974
+ state.output = utils.escapeLast(state.output, '{');
975
+ decrement('braces');
976
+ }
977
+
978
+ if (opts.strictSlashes !== true && (prev.type === 'star' || prev.type === 'bracket')) {
979
+ push({ type: 'maybe_slash', value: '', output: `${SLASH_LITERAL}?` });
980
+ }
981
+
982
+ // rebuild the output if we had to backtrack at any point
983
+ if (state.backtrack === true) {
984
+ state.output = '';
985
+
986
+ for (const token of state.tokens) {
987
+ state.output += token.output != null ? token.output : token.value;
988
+
989
+ if (token.suffix) {
990
+ state.output += token.suffix;
991
+ }
992
+ }
993
+ }
994
+
995
+ return state;
996
+ };
997
+
998
+ /**
999
+ * Fast paths for creating regular expressions for common glob patterns.
1000
+ * This can significantly speed up processing and has very little downside
1001
+ * impact when none of the fast paths match.
1002
+ */
1003
+
1004
+ parse.fastpaths = (input, options) => {
1005
+ const opts = { ...options };
1006
+ const max = typeof opts.maxLength === 'number' ? Math.min(MAX_LENGTH, opts.maxLength) : MAX_LENGTH;
1007
+ const len = input.length;
1008
+ if (len > max) {
1009
+ throw new SyntaxError(`Input length: ${len}, exceeds maximum allowed length: ${max}`);
1010
+ }
1011
+
1012
+ input = REPLACEMENTS[input] || input;
1013
+ const win32 = utils.isWindows(options);
1014
+
1015
+ // create constants based on platform, for windows or posix
1016
+ const {
1017
+ DOT_LITERAL,
1018
+ SLASH_LITERAL,
1019
+ ONE_CHAR,
1020
+ DOTS_SLASH,
1021
+ NO_DOT,
1022
+ NO_DOTS,
1023
+ NO_DOTS_SLASH,
1024
+ STAR,
1025
+ START_ANCHOR
1026
+ } = constants.globChars(win32);
1027
+
1028
+ const nodot = opts.dot ? NO_DOTS : NO_DOT;
1029
+ const slashDot = opts.dot ? NO_DOTS_SLASH : NO_DOT;
1030
+ const capture = opts.capture ? '' : '?:';
1031
+ const state = { negated: false, prefix: '' };
1032
+ let star = opts.bash === true ? '.*?' : STAR;
1033
+
1034
+ if (opts.capture) {
1035
+ star = `(${star})`;
1036
+ }
1037
+
1038
+ const globstar = opts => {
1039
+ if (opts.noglobstar === true) return star;
1040
+ return `(${capture}(?:(?!${START_ANCHOR}${opts.dot ? DOTS_SLASH : DOT_LITERAL}).)*?)`;
1041
+ };
1042
+
1043
+ const create = str => {
1044
+ switch (str) {
1045
+ case '*':
1046
+ return `${nodot}${ONE_CHAR}${star}`;
1047
+
1048
+ case '.*':
1049
+ return `${DOT_LITERAL}${ONE_CHAR}${star}`;
1050
+
1051
+ case '*.*':
1052
+ return `${nodot}${star}${DOT_LITERAL}${ONE_CHAR}${star}`;
1053
+
1054
+ case '*/*':
1055
+ return `${nodot}${star}${SLASH_LITERAL}${ONE_CHAR}${slashDot}${star}`;
1056
+
1057
+ case '**':
1058
+ return nodot + globstar(opts);
1059
+
1060
+ case '**/*':
1061
+ return `(?:${nodot}${globstar(opts)}${SLASH_LITERAL})?${slashDot}${ONE_CHAR}${star}`;
1062
+
1063
+ case '**/*.*':
1064
+ return `(?:${nodot}${globstar(opts)}${SLASH_LITERAL})?${slashDot}${star}${DOT_LITERAL}${ONE_CHAR}${star}`;
1065
+
1066
+ case '**/.*':
1067
+ return `(?:${nodot}${globstar(opts)}${SLASH_LITERAL})?${DOT_LITERAL}${ONE_CHAR}${star}`;
1068
+
1069
+ default: {
1070
+ const match = /^(.*?)\.(\w+)$/.exec(str);
1071
+ if (!match) return;
1072
+
1073
+ const source = create(match[1]);
1074
+ if (!source) return;
1075
+
1076
+ return source + DOT_LITERAL + match[2];
1077
+ }
1078
+ }
1079
+ };
1080
+
1081
+ const output = utils.removePrefix(input, state);
1082
+ let source = create(output);
1083
+
1084
+ if (source && opts.strictSlashes !== true) {
1085
+ source += `${SLASH_LITERAL}?`;
1086
+ }
1087
+
1088
+ return source;
1089
+ };
1090
+
1091
+ module.exports = parse;