Skip to content

Commit 813453c

Browse files
committed
add executableExists method to ProcessUtils
1 parent bca1383 commit 813453c

File tree

7 files changed

+252
-4
lines changed

7 files changed

+252
-4
lines changed

src/domains/processUtils-test.js

+48-3
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,57 @@
11
var ProcessUtils = require("./ProcessUtils");
2-
var pid = 5064;
32

3+
/*
4+
var pid = 5064;
45
ProcessUtils.getChildrenOfPid(pid, function (err, children) {
56
console.log(children);
6-
/*
77
children.push(pid);
88
children.forEach(function (pid) {
99
ProcessUtils.killSingleProcess(pid);
1010
});
11-
*/
1211
});
12+
*/
13+
14+
[
15+
"git",
16+
"C:\\Program Files (x86)\\Git\\cmd\\git.exe",
17+
"C:/Program Files (x86)/Git/cmd/git.exe",
18+
"C:/Program Files (x86)/Git/cmd/git2.exe",
19+
"C:/Program Files (x86)/Git/cmd/",
20+
"C:/Program Files (x86)/Git/cmd"
21+
].forEach(function (path) {
22+
23+
ProcessUtils.executableExists(path, function (err, exists, resolvedPath) {
24+
console.log("ProcessUtils.executableExists for: " + path);
25+
console.log(" - exists: " + exists);
26+
console.log(" - resolvedPath: " + resolvedPath);
27+
});
28+
29+
});
30+
31+
/*
32+
ProcessUtils.executableExists("git", function (err, result, resolvedPath) {
33+
console.log("git");
34+
console.log(result);
35+
});
36+
ProcessUtils.executableExists("C:\\Program Files (x86)\\Git\\cmd\\git.exe", function (err, result) {
37+
console.log("result for C:\\Program Files (x86)\\Git\\cmd\\git.exe");
38+
console.log(result);
39+
});
40+
ProcessUtils.executableExists("C:/Program Files (x86)/Git/cmd/git.exe", function (err, result) {
41+
console.log("result for C:/Program Files (x86)/Git/cmd/git.exe");
42+
console.log(result);
43+
});
44+
45+
ProcessUtils.executableExists("C:/Program Files (x86)/Git/cmd/git2.exe", function (err, result) {
46+
console.log("result for C:/Program Files (x86)/Git/cmd/git2.exe");
47+
console.log(result);
48+
});
49+
ProcessUtils.executableExists("C:/Program Files (x86)/Git/cmd/", function (err, result) {
50+
console.log("result for C:/Program Files (x86)/Git/cmd/");
51+
console.log(result);
52+
});
53+
ProcessUtils.executableExists("C:/Program Files (x86)/Git/cmd", function (err, result) {
54+
console.log("result for C:/Program Files (x86)/Git/cmd");
55+
console.log(result);
56+
});
57+
*/

src/domains/processUtils.js

+28-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,8 @@
1-
var exec = require("child_process").exec;
1+
var exec = require("child_process").exec,
2+
fs = require("fs"),
3+
Path = require("path"),
4+
which = require("../../thirdparty/which");
5+
26
var isWin = /^win/.test(process.platform);
37
var noop = function () {};
48

@@ -79,5 +83,28 @@ function getChildrenOfPid(pid, callback) {
7983
}
8084
}
8185

86+
function executableExists(filename, dir, callback) {
87+
if (typeof dir === "function") { callback = dir; dir = ""; }
88+
which(filename, function (err, path) {
89+
if (err) {
90+
return callback(err, false);
91+
}
92+
93+
path = Path.normalize(path);
94+
95+
fs.stat(path, function (err, stats) {
96+
if (err) {
97+
return callback(err, false);
98+
}
99+
100+
var exists = stats.isFile();
101+
if (!exists) { path = undefined; }
102+
103+
return callback(null, exists, path);
104+
});
105+
});
106+
}
107+
82108
exports.getChildrenOfPid = getChildrenOfPid;
83109
exports.killSingleProcess = killSingleProcess;
110+
exports.executableExists = executableExists;

thirdparty/which/LICENSE

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
Copyright 2009, 2010, 2011 Isaac Z. Schlueter.
2+
All rights reserved.
3+
4+
Permission is hereby granted, free of charge, to any person
5+
obtaining a copy of this software and associated documentation
6+
files (the "Software"), to deal in the Software without
7+
restriction, including without limitation the rights to use,
8+
copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the
10+
Software is furnished to do so, subject to the following
11+
conditions:
12+
13+
The above copyright notice and this permission notice shall be
14+
included in all copies or substantial portions of the Software.
15+
16+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
18+
OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
20+
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
21+
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22+
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
23+
OTHER DEALINGS IN THE SOFTWARE.

thirdparty/which/README.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
The "which" util from npm's guts.
2+
3+
Finds the first instance of a specified executable in the PATH
4+
environment variable. Does not cache the results, so `hash -r` is not
5+
needed when the PATH changes.

thirdparty/which/bin/which

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#!/usr/bin/env node
2+
var which = require("../")
3+
if (process.argv.length < 3) {
4+
console.error("Usage: which <thing>")
5+
process.exit(1)
6+
}
7+
8+
which(process.argv[2], function (er, thing) {
9+
if (er) {
10+
console.error(er.message)
11+
process.exit(er.errno || 127)
12+
}
13+
console.log(thing)
14+
})

thirdparty/which/package.json

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
{
2+
"author": {
3+
"name": "Isaac Z. Schlueter",
4+
"email": "[email protected]",
5+
"url": "http://blog.izs.me"
6+
},
7+
"name": "which",
8+
"description": "Like which(1) unix command. Find the first instance of an executable in the PATH.",
9+
"version": "1.0.5",
10+
"repository": {
11+
"type": "git",
12+
"url": "git://github.com/isaacs/node-which.git"
13+
},
14+
"main": "which.js",
15+
"bin": {
16+
"which": "./bin/which"
17+
},
18+
"engines": {
19+
"node": "*"
20+
},
21+
"dependencies": {},
22+
"devDependencies": {},
23+
"bugs": {
24+
"url": "https://github.com/isaacs/node-which/issues"
25+
},
26+
"readme": "ERROR: No README data found!",
27+
"homepage": "https://github.com/isaacs/node-which",
28+
29+
"_from": "which@"
30+
}

thirdparty/which/which.js

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
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

Comments
 (0)