|
| 1 | +module.exports = which |
| 2 | +which.sync = whichSync |
| 3 | + |
| 4 | +var path = require("path") |
| 5 | + , fs |
| 6 | + , COLON = process.platform === "win32" ? ";" : ":" |
| 7 | + , isExe |
| 8 | + |
| 9 | +try { |
| 10 | + fs = require("graceful-fs") |
| 11 | +} catch (ex) { |
| 12 | + fs = require("fs") |
| 13 | +} |
| 14 | + |
| 15 | +if (process.platform == "win32") { |
| 16 | + // On windows, there is no good way to check that a file is executable |
| 17 | + isExe = function isExe () { return true } |
| 18 | +} else { |
| 19 | + isExe = function isExe (mod, uid, gid) { |
| 20 | + //console.error(mod, uid, gid); |
| 21 | + //console.error("isExe?", (mod & 0111).toString(8)) |
| 22 | + var ret = (mod & 0001) |
| 23 | + || (mod & 0010) && process.getgid && gid === process.getgid() |
| 24 | + || (mod & 0100) && process.getuid && uid === process.getuid() |
| 25 | + //console.error("isExe?", ret) |
| 26 | + return ret |
| 27 | + } |
| 28 | +} |
| 29 | + |
| 30 | + |
| 31 | + |
| 32 | +function which (cmd, cb) { |
| 33 | + if (isAbsolute(cmd)) return cb(null, cmd) |
| 34 | + var pathEnv = (process.env.PATH || "").split(COLON) |
| 35 | + , pathExt = [""] |
| 36 | + if (process.platform === "win32") { |
| 37 | + pathEnv.push(process.cwd()) |
| 38 | + pathExt = (process.env.PATHEXT || ".EXE").split(COLON) |
| 39 | + if (cmd.indexOf(".") !== -1) pathExt.unshift("") |
| 40 | + } |
| 41 | + //console.error("pathEnv", pathEnv) |
| 42 | + ;(function F (i, l) { |
| 43 | + if (i === l) return cb(new Error("not found: "+cmd)) |
| 44 | + var p = path.resolve(pathEnv[i], cmd) |
| 45 | + ;(function E (ii, ll) { |
| 46 | + if (ii === ll) return F(i + 1, l) |
| 47 | + var ext = pathExt[ii] |
| 48 | + //console.error(p + ext) |
| 49 | + fs.stat(p + ext, function (er, stat) { |
| 50 | + if (!er && |
| 51 | + stat && |
| 52 | + stat.isFile() && |
| 53 | + isExe(stat.mode, stat.uid, stat.gid)) { |
| 54 | + //console.error("yes, exe!", p + ext) |
| 55 | + return cb(null, p + ext) |
| 56 | + } |
| 57 | + return E(ii + 1, ll) |
| 58 | + }) |
| 59 | + })(0, pathExt.length) |
| 60 | + })(0, pathEnv.length) |
| 61 | +} |
| 62 | + |
| 63 | +function whichSync (cmd) { |
| 64 | + if (isAbsolute(cmd)) return cmd |
| 65 | + var pathEnv = (process.env.PATH || "").split(COLON) |
| 66 | + , pathExt = [""] |
| 67 | + if (process.platform === "win32") { |
| 68 | + pathEnv.push(process.cwd()) |
| 69 | + pathExt = (process.env.PATHEXT || ".EXE").split(COLON) |
| 70 | + if (cmd.indexOf(".") !== -1) pathExt.unshift("") |
| 71 | + } |
| 72 | + for (var i = 0, l = pathEnv.length; i < l; i ++) { |
| 73 | + var p = path.join(pathEnv[i], cmd) |
| 74 | + for (var j = 0, ll = pathExt.length; j < ll; j ++) { |
| 75 | + var cur = p + pathExt[j] |
| 76 | + var stat |
| 77 | + try { stat = fs.statSync(cur) } catch (ex) {} |
| 78 | + if (stat && |
| 79 | + stat.isFile() && |
| 80 | + isExe(stat.mode, stat.uid, stat.gid)) return cur |
| 81 | + } |
| 82 | + } |
| 83 | + throw new Error("not found: "+cmd) |
| 84 | +} |
| 85 | + |
| 86 | +var isAbsolute = process.platform === "win32" ? absWin : absUnix |
| 87 | + |
| 88 | +function absWin (p) { |
| 89 | + if (absUnix(p)) return true |
| 90 | + // pull off the device/UNC bit from a windows path. |
| 91 | + // from node's lib/path.js |
| 92 | + var splitDeviceRe = |
| 93 | + /^([a-zA-Z]:|[\\\/]{2}[^\\\/]+[\\\/][^\\\/]+)?([\\\/])?/ |
| 94 | + , result = splitDeviceRe.exec(p) |
| 95 | + , device = result[1] || '' |
| 96 | + , isUnc = device && device.charAt(1) !== ':' |
| 97 | + , isAbsolute = !!result[2] || isUnc // UNC paths are always absolute |
| 98 | + |
| 99 | + return isAbsolute |
| 100 | +} |
| 101 | + |
| 102 | +function absUnix (p) { |
| 103 | + return p.charAt(0) === "/" || p === "" |
| 104 | +} |
0 commit comments