module.exports = which which.sync = whichSync var isWindows = process.platform === 'win32' || process.env.OSTYPE === 'cygwin' || process.env.OSTYPE === 'msys' var path = require('path') var COLON = isWindows ? ';' : ':' var isExe var fs = require('fs') var isAbsolute = require('is-absolute') var G = parseInt('0010', 8) var U = parseInt('0100', 8) var UG = parseInt('0110', 8) if (isWindows) { // On windows, there is no good way to check that a file is executable isExe = function isExe () { return true } } else { isExe = function isExe (mod, uid, gid) { var ret = (mod & 1) || (mod & U) && process.getgid && gid === process.getgid() || (mod & G) && process.getuid && uid === process.getuid() || (mod & UG) && process.getuid && 0 === process.getuid() if (!ret && process.getgroups && (mod & G)) { var groups = process.getgroups() for (var g = 0; g < groups.length; g++) { if (groups[g] === gid) return true } } return ret } } function getPathInfo(cmd, opt) { var colon = opt.colon || COLON var pathEnv = opt.path || process.env.PATH || '' var pathExt = [''] pathEnv = pathEnv.split(colon) if (isWindows) { pathEnv.unshift(process.cwd()) pathExt = (opt.pathExt || process.env.PATHEXT || '.EXE').split(colon) if (cmd.indexOf('.') !== -1 && pathExt[0] !== '') pathExt.unshift('') } // If it's absolute, then we don't bother searching the pathenv. // just check the file itself, and that's it. if (isAbsolute(cmd)) pathEnv = [''] return {env: pathEnv, ext: pathExt} } function which (cmd, opt, cb) { if (typeof opt === 'function') { cb = opt opt = {} } var info = getPathInfo(cmd, opt) var pathEnv = info.env var pathExt = info.ext var found = [] ;(function F (i, l) { if (i === l) { if (opt.all && found.length) return cb(null, found) else return cb(new Error('not found: '+cmd)) } var p = path.resolve(pathEnv[i], cmd) ;(function E (ii, ll) { if (ii === ll) return F(i + 1, l) var ext = pathExt[ii] fs.stat(p + ext, function (er, stat) { if (!er && stat.isFile() && isExe(stat.mode, stat.uid, stat.gid)) { if (opt.all) found.push(p + ext) else return cb(null, p + ext) } return E(ii + 1, ll) }) })(0, pathExt.length) })(0, pathEnv.length) } function whichSync (cmd, opt) { opt = opt || {} var info = getPathInfo(cmd, opt) var pathEnv = info.env var pathExt = info.ext var found = [] for (var i = 0, l = pathEnv.length; i < l; i ++) { var p = path.join(pathEnv[i], cmd) for (var j = 0, ll = pathExt.length; j < ll; j ++) { var cur = p + pathExt[j] var stat try { stat = fs.statSync(cur) if (stat.isFile() && isExe(stat.mode, stat.uid, stat.gid)) { if (opt.all) found.push(cur) else return cur } } catch (ex) {} } } if (opt.all && found.length) return found throw new Error('not found: '+cmd) }