source: imaps-frontend/node_modules/picomatch/lib/picomatch.js@ 79a0317

main
Last change on this file since 79a0317 was 0c6b92a, checked in by stefan toskovski <stefantoska84@…>, 6 weeks ago

Pred finalna verzija

  • Property mode set to 100644
File size: 9.7 KB
Line 
1'use strict';
2
3const path = require('path');
4const scan = require('./scan');
5const parse = require('./parse');
6const utils = require('./utils');
7const constants = require('./constants');
8const isObject = val => val && typeof val === 'object' && !Array.isArray(val);
9
10/**
11 * Creates a matcher function from one or more glob patterns. The
12 * returned function takes a string to match as its first argument,
13 * and returns true if the string is a match. The returned matcher
14 * function also takes a boolean as the second argument that, when true,
15 * returns an object with additional information.
16 *
17 * ```js
18 * const picomatch = require('picomatch');
19 * // picomatch(glob[, options]);
20 *
21 * const isMatch = picomatch('*.!(*a)');
22 * console.log(isMatch('a.a')); //=> false
23 * console.log(isMatch('a.b')); //=> true
24 * ```
25 * @name picomatch
26 * @param {String|Array} `globs` One or more glob patterns.
27 * @param {Object=} `options`
28 * @return {Function=} Returns a matcher function.
29 * @api public
30 */
31
32const picomatch = (glob, options, returnState = false) => {
33 if (Array.isArray(glob)) {
34 const fns = glob.map(input => picomatch(input, options, returnState));
35 const arrayMatcher = str => {
36 for (const isMatch of fns) {
37 const state = isMatch(str);
38 if (state) return state;
39 }
40 return false;
41 };
42 return arrayMatcher;
43 }
44
45 const isState = isObject(glob) && glob.tokens && glob.input;
46
47 if (glob === '' || (typeof glob !== 'string' && !isState)) {
48 throw new TypeError('Expected pattern to be a non-empty string');
49 }
50
51 const opts = options || {};
52 const posix = utils.isWindows(options);
53 const regex = isState
54 ? picomatch.compileRe(glob, options)
55 : picomatch.makeRe(glob, options, false, true);
56
57 const state = regex.state;
58 delete regex.state;
59
60 let isIgnored = () => false;
61 if (opts.ignore) {
62 const ignoreOpts = { ...options, ignore: null, onMatch: null, onResult: null };
63 isIgnored = picomatch(opts.ignore, ignoreOpts, returnState);
64 }
65
66 const matcher = (input, returnObject = false) => {
67 const { isMatch, match, output } = picomatch.test(input, regex, options, { glob, posix });
68 const result = { glob, state, regex, posix, input, output, match, isMatch };
69
70 if (typeof opts.onResult === 'function') {
71 opts.onResult(result);
72 }
73
74 if (isMatch === false) {
75 result.isMatch = false;
76 return returnObject ? result : false;
77 }
78
79 if (isIgnored(input)) {
80 if (typeof opts.onIgnore === 'function') {
81 opts.onIgnore(result);
82 }
83 result.isMatch = false;
84 return returnObject ? result : false;
85 }
86
87 if (typeof opts.onMatch === 'function') {
88 opts.onMatch(result);
89 }
90 return returnObject ? result : true;
91 };
92
93 if (returnState) {
94 matcher.state = state;
95 }
96
97 return matcher;
98};
99
100/**
101 * Test `input` with the given `regex`. This is used by the main
102 * `picomatch()` function to test the input string.
103 *
104 * ```js
105 * const picomatch = require('picomatch');
106 * // picomatch.test(input, regex[, options]);
107 *
108 * console.log(picomatch.test('foo/bar', /^(?:([^/]*?)\/([^/]*?))$/));
109 * // { isMatch: true, match: [ 'foo/', 'foo', 'bar' ], output: 'foo/bar' }
110 * ```
111 * @param {String} `input` String to test.
112 * @param {RegExp} `regex`
113 * @return {Object} Returns an object with matching info.
114 * @api public
115 */
116
117picomatch.test = (input, regex, options, { glob, posix } = {}) => {
118 if (typeof input !== 'string') {
119 throw new TypeError('Expected input to be a string');
120 }
121
122 if (input === '') {
123 return { isMatch: false, output: '' };
124 }
125
126 const opts = options || {};
127 const format = opts.format || (posix ? utils.toPosixSlashes : null);
128 let match = input === glob;
129 let output = (match && format) ? format(input) : input;
130
131 if (match === false) {
132 output = format ? format(input) : input;
133 match = output === glob;
134 }
135
136 if (match === false || opts.capture === true) {
137 if (opts.matchBase === true || opts.basename === true) {
138 match = picomatch.matchBase(input, regex, options, posix);
139 } else {
140 match = regex.exec(output);
141 }
142 }
143
144 return { isMatch: Boolean(match), match, output };
145};
146
147/**
148 * Match the basename of a filepath.
149 *
150 * ```js
151 * const picomatch = require('picomatch');
152 * // picomatch.matchBase(input, glob[, options]);
153 * console.log(picomatch.matchBase('foo/bar.js', '*.js'); // true
154 * ```
155 * @param {String} `input` String to test.
156 * @param {RegExp|String} `glob` Glob pattern or regex created by [.makeRe](#makeRe).
157 * @return {Boolean}
158 * @api public
159 */
160
161picomatch.matchBase = (input, glob, options, posix = utils.isWindows(options)) => {
162 const regex = glob instanceof RegExp ? glob : picomatch.makeRe(glob, options);
163 return regex.test(path.basename(input));
164};
165
166/**
167 * Returns true if **any** of the given glob `patterns` match the specified `string`.
168 *
169 * ```js
170 * const picomatch = require('picomatch');
171 * // picomatch.isMatch(string, patterns[, options]);
172 *
173 * console.log(picomatch.isMatch('a.a', ['b.*', '*.a'])); //=> true
174 * console.log(picomatch.isMatch('a.a', 'b.*')); //=> false
175 * ```
176 * @param {String|Array} str The string to test.
177 * @param {String|Array} patterns One or more glob patterns to use for matching.
178 * @param {Object} [options] See available [options](#options).
179 * @return {Boolean} Returns true if any patterns match `str`
180 * @api public
181 */
182
183picomatch.isMatch = (str, patterns, options) => picomatch(patterns, options)(str);
184
185/**
186 * Parse a glob pattern to create the source string for a regular
187 * expression.
188 *
189 * ```js
190 * const picomatch = require('picomatch');
191 * const result = picomatch.parse(pattern[, options]);
192 * ```
193 * @param {String} `pattern`
194 * @param {Object} `options`
195 * @return {Object} Returns an object with useful properties and output to be used as a regex source string.
196 * @api public
197 */
198
199picomatch.parse = (pattern, options) => {
200 if (Array.isArray(pattern)) return pattern.map(p => picomatch.parse(p, options));
201 return parse(pattern, { ...options, fastpaths: false });
202};
203
204/**
205 * Scan a glob pattern to separate the pattern into segments.
206 *
207 * ```js
208 * const picomatch = require('picomatch');
209 * // picomatch.scan(input[, options]);
210 *
211 * const result = picomatch.scan('!./foo/*.js');
212 * console.log(result);
213 * { prefix: '!./',
214 * input: '!./foo/*.js',
215 * start: 3,
216 * base: 'foo',
217 * glob: '*.js',
218 * isBrace: false,
219 * isBracket: false,
220 * isGlob: true,
221 * isExtglob: false,
222 * isGlobstar: false,
223 * negated: true }
224 * ```
225 * @param {String} `input` Glob pattern to scan.
226 * @param {Object} `options`
227 * @return {Object} Returns an object with
228 * @api public
229 */
230
231picomatch.scan = (input, options) => scan(input, options);
232
233/**
234 * Compile a regular expression from the `state` object returned by the
235 * [parse()](#parse) method.
236 *
237 * @param {Object} `state`
238 * @param {Object} `options`
239 * @param {Boolean} `returnOutput` Intended for implementors, this argument allows you to return the raw output from the parser.
240 * @param {Boolean} `returnState` Adds the state to a `state` property on the returned regex. Useful for implementors and debugging.
241 * @return {RegExp}
242 * @api public
243 */
244
245picomatch.compileRe = (state, options, returnOutput = false, returnState = false) => {
246 if (returnOutput === true) {
247 return state.output;
248 }
249
250 const opts = options || {};
251 const prepend = opts.contains ? '' : '^';
252 const append = opts.contains ? '' : '$';
253
254 let source = `${prepend}(?:${state.output})${append}`;
255 if (state && state.negated === true) {
256 source = `^(?!${source}).*$`;
257 }
258
259 const regex = picomatch.toRegex(source, options);
260 if (returnState === true) {
261 regex.state = state;
262 }
263
264 return regex;
265};
266
267/**
268 * Create a regular expression from a parsed glob pattern.
269 *
270 * ```js
271 * const picomatch = require('picomatch');
272 * const state = picomatch.parse('*.js');
273 * // picomatch.compileRe(state[, options]);
274 *
275 * console.log(picomatch.compileRe(state));
276 * //=> /^(?:(?!\.)(?=.)[^/]*?\.js)$/
277 * ```
278 * @param {String} `state` The object returned from the `.parse` method.
279 * @param {Object} `options`
280 * @param {Boolean} `returnOutput` Implementors may use this argument to return the compiled output, instead of a regular expression. This is not exposed on the options to prevent end-users from mutating the result.
281 * @param {Boolean} `returnState` Implementors may use this argument to return the state from the parsed glob with the returned regular expression.
282 * @return {RegExp} Returns a regex created from the given pattern.
283 * @api public
284 */
285
286picomatch.makeRe = (input, options = {}, returnOutput = false, returnState = false) => {
287 if (!input || typeof input !== 'string') {
288 throw new TypeError('Expected a non-empty string');
289 }
290
291 let parsed = { negated: false, fastpaths: true };
292
293 if (options.fastpaths !== false && (input[0] === '.' || input[0] === '*')) {
294 parsed.output = parse.fastpaths(input, options);
295 }
296
297 if (!parsed.output) {
298 parsed = parse(input, options);
299 }
300
301 return picomatch.compileRe(parsed, options, returnOutput, returnState);
302};
303
304/**
305 * Create a regular expression from the given regex source string.
306 *
307 * ```js
308 * const picomatch = require('picomatch');
309 * // picomatch.toRegex(source[, options]);
310 *
311 * const { output } = picomatch.parse('*.js');
312 * console.log(picomatch.toRegex(output));
313 * //=> /^(?:(?!\.)(?=.)[^/]*?\.js)$/
314 * ```
315 * @param {String} `source` Regular expression source string.
316 * @param {Object} `options`
317 * @return {RegExp}
318 * @api public
319 */
320
321picomatch.toRegex = (source, options) => {
322 try {
323 const opts = options || {};
324 return new RegExp(source, opts.flags || (opts.nocase ? 'i' : ''));
325 } catch (err) {
326 if (options && options.debug === true) throw err;
327 return /$^/;
328 }
329};
330
331/**
332 * Picomatch constants.
333 * @return {Object}
334 */
335
336picomatch.constants = constants;
337
338/**
339 * Expose "picomatch"
340 */
341
342module.exports = picomatch;
Note: See TracBrowser for help on using the repository browser.