2018-04-11 23:49:05 +03:00
|
|
|
/*
|
2019-01-04 03:22:15 +03:00
|
|
|
Copyright 2018-2019 Intel Corporation
|
2018-04-11 23:49:05 +03:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
2018-04-20 04:19:15 +03:00
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
2018-04-11 23:49:05 +03:00
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
2018-04-20 04:19:15 +03:00
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
2018-04-11 23:49:05 +03:00
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2018-09-19 05:41:59 +03:00
|
|
|
|
2018-04-11 23:49:05 +03:00
|
|
|
var GM = require('_GenericMarshal');
|
|
|
|
|
2018-09-12 00:28:47 +03:00
|
|
|
// Used on Windows and Linux to get information about running processes
|
|
|
|
function processManager() {
|
|
|
|
this._ObjectID = 'process-manager'; // Used for debugging, allows you to get the object type at runtime.
|
|
|
|
|
|
|
|
// Setup the platform specific calls.
|
2018-09-19 05:41:59 +03:00
|
|
|
switch (process.platform)
|
|
|
|
{
|
2018-04-11 23:49:05 +03:00
|
|
|
case 'win32':
|
|
|
|
this._kernel32 = GM.CreateNativeProxy('kernel32.dll');
|
|
|
|
this._kernel32.CreateMethod('GetLastError');
|
|
|
|
this._kernel32.CreateMethod('CreateToolhelp32Snapshot');
|
|
|
|
this._kernel32.CreateMethod('Process32First');
|
|
|
|
this._kernel32.CreateMethod('Process32Next');
|
|
|
|
break;
|
|
|
|
case 'linux':
|
2018-09-19 05:41:59 +03:00
|
|
|
case 'darwin':
|
2018-04-11 23:49:05 +03:00
|
|
|
this._childProcess = require('child_process');
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw (process.platform + ' not supported');
|
2018-08-30 04:47:22 +03:00
|
|
|
break;
|
2018-04-11 23:49:05 +03:00
|
|
|
}
|
2018-09-19 05:41:59 +03:00
|
|
|
this.enumerateProcesses = function enumerateProcesses()
|
|
|
|
{
|
|
|
|
var promise = require('promise');
|
|
|
|
var ret = new promise(function (res, rej) { this._res = res; this._rej = rej; });
|
|
|
|
this.getProcesses(function (ps, prom) { prom._res(ps); }, ret);
|
|
|
|
return (ret);
|
|
|
|
}
|
2018-09-12 00:28:47 +03:00
|
|
|
// Return a object of: pid -> process information.
|
2018-09-19 05:41:59 +03:00
|
|
|
this.getProcesses = function getProcesses(callback)
|
|
|
|
{
|
|
|
|
switch(process.platform)
|
|
|
|
{
|
|
|
|
default:
|
2018-04-20 04:19:15 +03:00
|
|
|
throw ('Enumerating processes on ' + process.platform + ' not supported');
|
2018-08-30 04:47:22 +03:00
|
|
|
break;
|
2018-09-12 00:28:47 +03:00
|
|
|
case 'win32': // Windows processes
|
|
|
|
var retVal = {};
|
2018-04-20 04:19:15 +03:00
|
|
|
var h = this._kernel32.CreateToolhelp32Snapshot(2, 0);
|
|
|
|
var info = GM.CreateVariable(304);
|
2018-04-11 23:49:05 +03:00
|
|
|
info.toBuffer().writeUInt32LE(304, 0);
|
|
|
|
var nextProcess = this._kernel32.Process32First(h, info);
|
2018-09-19 05:41:59 +03:00
|
|
|
while (nextProcess.Val)
|
|
|
|
{
|
|
|
|
retVal[info.Deref(8, 4).toBuffer().readUInt32LE(0)] = { pid: info.Deref(8, 4).toBuffer().readUInt32LE(0), cmd: info.Deref(GM.PointerSize == 4 ? 36 : 44, 260).String };
|
2018-04-11 23:49:05 +03:00
|
|
|
nextProcess = this._kernel32.Process32Next(h, info);
|
|
|
|
}
|
|
|
|
if (callback) { callback.apply(this, [retVal]); }
|
|
|
|
break;
|
2018-09-12 00:28:47 +03:00
|
|
|
case 'linux': // Linux processes
|
2018-04-11 23:49:05 +03:00
|
|
|
if (!this._psp) { this._psp = {}; }
|
|
|
|
var p = this._childProcess.execFile("/bin/ps", ["ps", "-uxa"], { type: this._childProcess.SpawnTypes.TERM });
|
|
|
|
this._psp[p.pid] = p;
|
|
|
|
p.Parent = this;
|
|
|
|
p.ps = '';
|
|
|
|
p.callback = callback;
|
|
|
|
p.args = [];
|
|
|
|
for (var i = 1; i < arguments.length; ++i) { p.args.push(arguments[i]); }
|
2018-09-19 05:41:59 +03:00
|
|
|
p.on('exit', function onGetProcesses()
|
|
|
|
{
|
|
|
|
delete this.Parent._psp[this.pid];
|
2018-09-12 00:28:47 +03:00
|
|
|
var retVal = {}, lines = this.ps.split('\x0D\x0A'), key = {}, keyi = 0;
|
2018-09-19 05:41:59 +03:00
|
|
|
for (var i in lines)
|
|
|
|
{
|
2018-04-20 04:19:15 +03:00
|
|
|
var tokens = lines[i].split(' ');
|
|
|
|
var tokenList = [];
|
2018-09-19 05:41:59 +03:00
|
|
|
for(var x in tokens)
|
|
|
|
{
|
2018-04-11 23:49:05 +03:00
|
|
|
if (i == 0 && tokens[x]) { key[tokens[x]] = keyi++; }
|
2018-09-19 05:41:59 +03:00
|
|
|
if (i > 0 && tokens[x]) { tokenList.push(tokens[x]);}
|
2018-04-11 23:49:05 +03:00
|
|
|
}
|
2018-09-12 00:28:47 +03:00
|
|
|
if (i > 0) {
|
2018-09-19 05:41:59 +03:00
|
|
|
if (tokenList[key.PID]) { retVal[tokenList[key.PID]] = { pid: key.PID, user: tokenList[key.USER], cmd: tokenList[key.COMMAND] }; }
|
2018-04-11 23:49:05 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-19 05:41:59 +03:00
|
|
|
if (this.callback)
|
|
|
|
{
|
2018-04-11 23:49:05 +03:00
|
|
|
this.args.unshift(retVal);
|
|
|
|
this.callback.apply(this.parent, this.args);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
p.stdout.on('data', function (chunk) { this.parent.ps += chunk.toString(); });
|
|
|
|
break;
|
2018-09-19 05:41:59 +03:00
|
|
|
case 'darwin':
|
|
|
|
var promise = require('promise');
|
|
|
|
var p = new promise(function (res, rej) { this._res = res; this._rej = rej; });
|
|
|
|
p.pm = this;
|
|
|
|
p.callback = callback;
|
|
|
|
p.args = [];
|
|
|
|
for (var i = 1; i < arguments.length; ++i) { p.args.push(arguments[i]); }
|
|
|
|
p.child = this._childProcess.execFile("/bin/ps", ["ps", "-xa"]);
|
|
|
|
p.child.promise = p;
|
|
|
|
p.child.stdout.ps = '';
|
|
|
|
p.child.stdout.on('data', function (chunk) { this.ps += chunk.toString(); });
|
|
|
|
p.child.on('exit', function ()
|
|
|
|
{
|
|
|
|
var lines = this.stdout.ps.split('\n');
|
|
|
|
var pidX = lines[0].split('PID')[0].length + 3;
|
|
|
|
var cmdX = lines[0].split('CMD')[0].length;
|
|
|
|
var ret = {};
|
|
|
|
for (var i = 1; i < lines.length; ++i)
|
|
|
|
{
|
|
|
|
if (lines[i].length > 0)
|
|
|
|
{
|
|
|
|
ret[lines[i].substring(0, pidX).trim()] = { pid: lines[i].substring(0, pidX).trim(), cmd: lines[i].substring(cmdX) };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.promise._res(ret);
|
|
|
|
});
|
|
|
|
p.then(function (ps)
|
|
|
|
{
|
|
|
|
this.args.unshift(ps);
|
|
|
|
this.callback.apply(this.pm, this.args);
|
|
|
|
});
|
|
|
|
break;
|
2018-04-11 23:49:05 +03:00
|
|
|
}
|
|
|
|
};
|
2018-09-19 05:41:59 +03:00
|
|
|
|
2018-09-12 00:28:47 +03:00
|
|
|
// Get information about a specific process on Linux
|
2018-09-19 05:41:59 +03:00
|
|
|
this.getProcessInfo = function getProcessInfo(pid)
|
|
|
|
{
|
|
|
|
switch(process.platform)
|
|
|
|
{
|
2018-04-20 04:19:15 +03:00
|
|
|
default:
|
|
|
|
throw ('getProcessInfo() not supported for ' + process.platform);
|
2018-08-30 04:47:22 +03:00
|
|
|
break;
|
2018-04-11 23:49:05 +03:00
|
|
|
case 'linux':
|
2018-08-30 04:47:22 +03:00
|
|
|
var status = require('fs').readFileSync('/proc/' + pid + '/status');
|
2018-09-19 05:41:59 +03:00
|
|
|
var info = {};
|
|
|
|
var lines = status.toString().split('\n');
|
|
|
|
for(var i in lines)
|
|
|
|
{
|
2018-04-11 23:49:05 +03:00
|
|
|
var tokens = lines[i].split(':');
|
|
|
|
if (tokens.length > 1) { tokens[1] = tokens[1].trim(); }
|
|
|
|
info[tokens[0]] = tokens[1];
|
|
|
|
}
|
2018-04-20 04:19:15 +03:00
|
|
|
return (info);
|
2018-08-30 04:47:22 +03:00
|
|
|
break;
|
2018-04-11 23:49:05 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = new processManager();
|