Spaces:
Sleeping
Sleeping
exports.setopts = setopts | |
exports.ownProp = ownProp | |
exports.makeAbs = makeAbs | |
exports.finish = finish | |
exports.mark = mark | |
exports.isIgnored = isIgnored | |
exports.childrenIgnored = childrenIgnored | |
function ownProp (obj, field) { | |
return Object.prototype.hasOwnProperty.call(obj, field) | |
} | |
var fs = require("fs") | |
var path = require("path") | |
var minimatch = require("minimatch") | |
var isAbsolute = require("path-is-absolute") | |
var Minimatch = minimatch.Minimatch | |
function alphasort (a, b) { | |
return a.localeCompare(b, 'en') | |
} | |
function setupIgnores (self, options) { | |
self.ignore = options.ignore || [] | |
if (!Array.isArray(self.ignore)) | |
self.ignore = [self.ignore] | |
if (self.ignore.length) { | |
self.ignore = self.ignore.map(ignoreMap) | |
} | |
} | |
// ignore patterns are always in dot:true mode. | |
function ignoreMap (pattern) { | |
var gmatcher = null | |
if (pattern.slice(-3) === '/**') { | |
var gpattern = pattern.replace(/(\/\*\*)+$/, '') | |
gmatcher = new Minimatch(gpattern, { dot: true }) | |
} | |
return { | |
matcher: new Minimatch(pattern, { dot: true }), | |
gmatcher: gmatcher | |
} | |
} | |
function setopts (self, pattern, options) { | |
if (!options) | |
options = {} | |
// base-matching: just use globstar for that. | |
if (options.matchBase && -1 === pattern.indexOf("/")) { | |
if (options.noglobstar) { | |
throw new Error("base matching requires globstar") | |
} | |
pattern = "**/" + pattern | |
} | |
self.silent = !!options.silent | |
self.pattern = pattern | |
self.strict = options.strict !== false | |
self.realpath = !!options.realpath | |
self.realpathCache = options.realpathCache || Object.create(null) | |
self.follow = !!options.follow | |
self.dot = !!options.dot | |
self.mark = !!options.mark | |
self.nodir = !!options.nodir | |
if (self.nodir) | |
self.mark = true | |
self.sync = !!options.sync | |
self.nounique = !!options.nounique | |
self.nonull = !!options.nonull | |
self.nosort = !!options.nosort | |
self.nocase = !!options.nocase | |
self.stat = !!options.stat | |
self.noprocess = !!options.noprocess | |
self.absolute = !!options.absolute | |
self.fs = options.fs || fs | |
self.maxLength = options.maxLength || Infinity | |
self.cache = options.cache || Object.create(null) | |
self.statCache = options.statCache || Object.create(null) | |
self.symlinks = options.symlinks || Object.create(null) | |
setupIgnores(self, options) | |
self.changedCwd = false | |
var cwd = process.cwd() | |
if (!ownProp(options, "cwd")) | |
self.cwd = cwd | |
else { | |
self.cwd = path.resolve(options.cwd) | |
self.changedCwd = self.cwd !== cwd | |
} | |
self.root = options.root || path.resolve(self.cwd, "/") | |
self.root = path.resolve(self.root) | |
if (process.platform === "win32") | |
self.root = self.root.replace(/\\/g, "/") | |
// TODO: is an absolute `cwd` supposed to be resolved against `root`? | |
// e.g. { cwd: '/test', root: __dirname } === path.join(__dirname, '/test') | |
self.cwdAbs = isAbsolute(self.cwd) ? self.cwd : makeAbs(self, self.cwd) | |
if (process.platform === "win32") | |
self.cwdAbs = self.cwdAbs.replace(/\\/g, "/") | |
self.nomount = !!options.nomount | |
// disable comments and negation in Minimatch. | |
// Note that they are not supported in Glob itself anyway. | |
options.nonegate = true | |
options.nocomment = true | |
self.minimatch = new Minimatch(pattern, options) | |
self.options = self.minimatch.options | |
} | |
function finish (self) { | |
var nou = self.nounique | |
var all = nou ? [] : Object.create(null) | |
for (var i = 0, l = self.matches.length; i < l; i ++) { | |
var matches = self.matches[i] | |
if (!matches || Object.keys(matches).length === 0) { | |
if (self.nonull) { | |
// do like the shell, and spit out the literal glob | |
var literal = self.minimatch.globSet[i] | |
if (nou) | |
all.push(literal) | |
else | |
all[literal] = true | |
} | |
} else { | |
// had matches | |
var m = Object.keys(matches) | |
if (nou) | |
all.push.apply(all, m) | |
else | |
m.forEach(function (m) { | |
all[m] = true | |
}) | |
} | |
} | |
if (!nou) | |
all = Object.keys(all) | |
if (!self.nosort) | |
all = all.sort(alphasort) | |
// at *some* point we statted all of these | |
if (self.mark) { | |
for (var i = 0; i < all.length; i++) { | |
all[i] = self._mark(all[i]) | |
} | |
if (self.nodir) { | |
all = all.filter(function (e) { | |
var notDir = !(/\/$/.test(e)) | |
var c = self.cache[e] || self.cache[makeAbs(self, e)] | |
if (notDir && c) | |
notDir = c !== 'DIR' && !Array.isArray(c) | |
return notDir | |
}) | |
} | |
} | |
if (self.ignore.length) | |
all = all.filter(function(m) { | |
return !isIgnored(self, m) | |
}) | |
self.found = all | |
} | |
function mark (self, p) { | |
var abs = makeAbs(self, p) | |
var c = self.cache[abs] | |
var m = p | |
if (c) { | |
var isDir = c === 'DIR' || Array.isArray(c) | |
var slash = p.slice(-1) === '/' | |
if (isDir && !slash) | |
m += '/' | |
else if (!isDir && slash) | |
m = m.slice(0, -1) | |
if (m !== p) { | |
var mabs = makeAbs(self, m) | |
self.statCache[mabs] = self.statCache[abs] | |
self.cache[mabs] = self.cache[abs] | |
} | |
} | |
return m | |
} | |
// lotta situps... | |
function makeAbs (self, f) { | |
var abs = f | |
if (f.charAt(0) === '/') { | |
abs = path.join(self.root, f) | |
} else if (isAbsolute(f) || f === '') { | |
abs = f | |
} else if (self.changedCwd) { | |
abs = path.resolve(self.cwd, f) | |
} else { | |
abs = path.resolve(f) | |
} | |
if (process.platform === 'win32') | |
abs = abs.replace(/\\/g, '/') | |
return abs | |
} | |
// Return true, if pattern ends with globstar '**', for the accompanying parent directory. | |
// Ex:- If node_modules/** is the pattern, add 'node_modules' to ignore list along with it's contents | |
function isIgnored (self, path) { | |
if (!self.ignore.length) | |
return false | |
return self.ignore.some(function(item) { | |
return item.matcher.match(path) || !!(item.gmatcher && item.gmatcher.match(path)) | |
}) | |
} | |
function childrenIgnored (self, path) { | |
if (!self.ignore.length) | |
return false | |
return self.ignore.some(function(item) { | |
return !!(item.gmatcher && item.gmatcher.match(path)) | |
}) | |
} | |