From 5aff4c172ebe522d5cda860bb5d2b1ecc6cdaf5c Mon Sep 17 00:00:00 2001 From: Bryan Roe Date: Wed, 10 Jun 2020 13:49:24 -0700 Subject: [PATCH] Added power-monitor.wakeDisplay() and sleepDisplay() --- microscript/ILibDuktape_Polyfills.c | 4 +- microscript/ILibDuktape_net.c | 11 +++ modules/child-container.js | 46 +++++++------ modules/power-monitor.js | 101 ++++++++++++++++++++++++++++ 4 files changed, 138 insertions(+), 24 deletions(-) diff --git a/microscript/ILibDuktape_Polyfills.c b/microscript/ILibDuktape_Polyfills.c index 00cfc5d..012936d 100644 --- a/microscript/ILibDuktape_Polyfills.c +++ b/microscript/ILibDuktape_Polyfills.c @@ -2158,7 +2158,7 @@ void ILibDuktape_Polyfills_JS_Init(duk_context *ctx) // power-monitor, refer to modules/power-monitor.js for details - duk_peval_string_noresult(ctx, "addModule('power-monitor', Buffer.from('/*
Copyright 2018-2020 Intel Corporation

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

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

function powerMonitor()
{
    this._ObjectID = 'power-monitor';
    require('events').EventEmitter.call(this, true)
        .createEvent('changed')
        .createEvent('sx')
        .createEvent('batteryLevel')
        .createEvent('acdc')
        .createEvent('display');

    this._ACState = 1;
    this._BatteryLevel = -1;

    if (process.platform == 'win32')
    {
        // These must be registered BEFORE newListener is hooked up
        this.on('batteryLevel', function (level) { this._BatteryLevel = level; });
        this.on('acdc', function (m) { this._ACState = (m == 'AC' ? 1 : 0); });
    }

    this.on('newListener', function (name, callback)
    {
        if (name == 'acdc') { callback.call(this, this._ACState == 1 ? 'AC' : 'BATTERY'); }
        if (name == 'batteryLevel') { if (this._BatteryLevel >= 0) { callback.call(this, this._BatteryLevel); } }
    });

    this._i = setImmediate(function (self)
    {
        require('user-sessions'); // This is needed because this is where the Windows Messages are processed for these events
        delete self._i;
    }, this);

    if (process.platform == 'linux')
    {
        this._ACPath = null;
        this._BatteryPath = [];

        var devices = require('fs').readdirSync('/sys/class/power_supply');
        for (var i in devices)
        {
            if (require('fs').readFileSync('/sys/class/power_supply/' + devices[i] + '/type').toString().trim() == 'Mains')
            {
                this._ACPath = '/sys/class/power_supply/' + devices[i] + '/';
                break;
            }
        }
        for (var i in devices)
        {
            if (require('fs').readFileSync('/sys/class/power_supply/' + devices[i] + '/type').toString().trim() == 'Battery')
            {
                this._BatteryPath.push('/sys/class/power_supply/' + devices[i] + '/');
            }
        }
        if (this._ACPath != null)
        {
            this._ACState = parseInt(require('fs').readFileSync(this._ACPath + 'online').toString().trim());
        }
        if (this._BatteryPath.length > 0)
        {
            this._getBatteryLevel = function _getBatteryLevel()
            {
                var sum = 0;
                var i;
                for (i in this._BatteryPath)
                {
                    sum += parseInt(require('fs').readFileSync(this._BatteryPath[i] + 'capacity').toString().trim());
                }
                sum = Math.floor(sum / this._BatteryPath.length);
                return (sum);
            }
            this._BatteryLevel = this._getBatteryLevel();

            // Since Battery Levels are not propagated with ACPI, we need to periodically check the battery level
            this._BatteryLevelCheck = function _BatteryLevelCheck()
            {
                var val = this._getBatteryLevel();
                if (val != this._BatteryLevel)
                {
                    this._BatteryLevel = val;
                    this.emit('batteryLevel', val);
                }
            };
            this._BattCheckInterval = setInterval(function (self)
            {
                self._BatteryLevelCheck.call(self);
            }, 300000, this);
        }
        this._acpiSink = function _acpiSink(acpiEvent)
        {
            if (acpiEvent.name == 'ac_adapter')
            {
                _acpiSink.self._ACState = acpiEvent.value;
                _acpiSink.self.emit('acdc', acpiEvent.value == 1 ? 'AC' : 'BATTERY');
                _acpiSink.self._BatteryLevelCheck();
            }
        };
        this._acpiSink.self = this;
        require('linux-acpi').on('acpi', this._acpiSink);
    }
    if (process.platform == 'darwin')
    {
        this._getBatteryLevel = function _getBatteryLevel()
        {
            var child = require('child_process').execFile('/bin/sh', ['sh']);
            child.stderr.str = ''; child.stderr.on('data', function (c) { this.str += c.toString(); });
            child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); });
            child.stdin.write("pmset -g batt | tr '\\n' '`' | awk -F'`' '");
            child.stdin.write('{');
            child.stdin.write('   power=split($1,pwr,"AC")>1?"1":"0";');
            child.stdin.write('   split($2, batt, " ");');
            child.stdin.write('   split(batt[2],chg,"%");');
            child.stdin.write('   printf "{\\"ac\\": %s,\\"level\\": %s}",power, chg[1]; ');
            child.stdin.write("}'\nexit\n");
            child.waitExit();

            try
            {
                var info = JSON.parse(child.stdout.str.trim());
                return (info);
            }
            catch (e)
            {
                return ({ ac: 1, level: -1 });
            }
        };
        this._batteryLevelCheck = function _batteryLevelCheck()
        {
            var newLevel = this._getBatteryLevel();
            if (newLevel.ac != this._ACState)
            {
                this._ACState = newLevel.ac;
                this.emit('acdc', this._ACState == 1 ? 'AC' : 'BATTERY');
            }
            if (newLevel.level != this._BatteryLevel)
            {
                this._BatteryLevel = newLevel.level;
                this.emit('batteryLevel', this._BatteryLevel);
            }
        };
        var tmp = this._getBatteryLevel();
        this._ACState = tmp.ac;
        this._BatteryLevel = tmp.level;

        if (this._BatteryLevel >= 0)
        {
            this._BattCheckInterval = setInterval(function (self)
            {
                self._batteryLevelCheck.call(self);
            }, 300000, this);
        }
    }
}

module.exports = new powerMonitor();
', 'base64').toString());"); + duk_peval_string_noresult(ctx, "addModule('power-monitor', Buffer.from('/*
Copyright 2018-2020 Intel Corporation

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

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

var WM_SYSCOMMAND = 0x0112;
var SC_MONITORPOWER = 0xF170;
var HWND_BROADCAST = 0xffff;
var ES_DISPLAY_REQUIRED = 0x00000002;

function powerMonitor()
{
    this._ObjectID = 'power-monitor';
    require('events').EventEmitter.call(this, true)
        .createEvent('changed')
        .createEvent('sx')
        .createEvent('batteryLevel')
        .createEvent('acdc')
        .createEvent('display');

    this._ACState = 1;
    this._BatteryLevel = -1;

    if (process.platform == 'win32')
    {
        // These must be registered BEFORE newListener is hooked up
        this.on('batteryLevel', function (level) { this._BatteryLevel = level; });
        this.on('acdc', function (m) { this._ACState = (m == 'AC' ? 1 : 0); });
    }

    this.on('newListener', function (name, callback)
    {
        if (name == 'acdc') { callback.call(this, this._ACState == 1 ? 'AC' : 'BATTERY'); }
        if (name == 'batteryLevel') { if (this._BatteryLevel >= 0) { callback.call(this, this._BatteryLevel); } }
    });

    this._i = setImmediate(function (self)
    {
        require('user-sessions'); // This is needed because this is where the Windows Messages are processed for these events
        delete self._i;
    }, this);

    if (process.platform == 'linux')
    {
        this._ACPath = null;
        this._BatteryPath = [];

        var devices = require('fs').readdirSync('/sys/class/power_supply');
        for (var i in devices)
        {
            if (require('fs').readFileSync('/sys/class/power_supply/' + devices[i] + '/type').toString().trim() == 'Mains')
            {
                this._ACPath = '/sys/class/power_supply/' + devices[i] + '/';
                break;
            }
        }
        for (var i in devices)
        {
            if (require('fs').readFileSync('/sys/class/power_supply/' + devices[i] + '/type').toString().trim() == 'Battery')
            {
                this._BatteryPath.push('/sys/class/power_supply/' + devices[i] + '/');
            }
        }
        if (this._ACPath != null)
        {
            this._ACState = parseInt(require('fs').readFileSync(this._ACPath + 'online').toString().trim());
        }
        if (this._BatteryPath.length > 0)
        {
            this._getBatteryLevel = function _getBatteryLevel()
            {
                var sum = 0;
                var i;
                for (i in this._BatteryPath)
                {
                    sum += parseInt(require('fs').readFileSync(this._BatteryPath[i] + 'capacity').toString().trim());
                }
                sum = Math.floor(sum / this._BatteryPath.length);
                return (sum);
            }
            this._BatteryLevel = this._getBatteryLevel();

            // Since Battery Levels are not propagated with ACPI, we need to periodically check the battery level
            this._BatteryLevelCheck = function _BatteryLevelCheck()
            {
                var val = this._getBatteryLevel();
                if (val != this._BatteryLevel)
                {
                    this._BatteryLevel = val;
                    this.emit('batteryLevel', val);
                }
            };
            this._BattCheckInterval = setInterval(function (self)
            {
                self._BatteryLevelCheck.call(self);
            }, 300000, this);
        }
        this._acpiSink = function _acpiSink(acpiEvent)
        {
            if (acpiEvent.name == 'ac_adapter')
            {
                _acpiSink.self._ACState = acpiEvent.value;
                _acpiSink.self.emit('acdc', acpiEvent.value == 1 ? 'AC' : 'BATTERY');
                _acpiSink.self._BatteryLevelCheck();
            }
        };
        this._acpiSink.self = this;
        require('linux-acpi').on('acpi', this._acpiSink);
    }
    if (process.platform == 'darwin')
    {
        Object.defineProperty(this, "_caffeinate", {
            value: (function ()
            {
                var child = require('child_process').execFile('/bin/sh', ['sh']);
                child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); });
                child.stdin.write('whereis caffeinate\nexit\n');
                child.waitExit();
                return (child.stdout.str.trim());
            })()
        });
        this._getBatteryLevel = function _getBatteryLevel()
        {
            var child = require('child_process').execFile('/bin/sh', ['sh']);
            child.stderr.str = ''; child.stderr.on('data', function (c) { this.str += c.toString(); });
            child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); });
            child.stdin.write("pmset -g batt | tr '\\n' '`' | awk -F'`' '");
            child.stdin.write('{');
            child.stdin.write('   power=split($1,pwr,"AC")>1?"1":"0";');
            child.stdin.write('   split($2, batt, " ");');
            child.stdin.write('   split(batt[2],chg,"%");');
            child.stdin.write('   printf "{\\"ac\\": %s,\\"level\\": %s}",power, chg[1]; ');
            child.stdin.write("}'\nexit\n");
            child.waitExit();

            try
            {
                var info = JSON.parse(child.stdout.str.trim());
                return (info);
            }
            catch (e)
            {
                return ({ ac: 1, level: -1 });
            }
        };
        this._batteryLevelCheck = function _batteryLevelCheck()
        {
            var newLevel = this._getBatteryLevel();
            if (newLevel.ac != this._ACState)
            {
                this._ACState = newLevel.ac;
                this.emit('acdc', this._ACState == 1 ? 'AC' : 'BATTERY');
            }
            if (newLevel.level != this._BatteryLevel)
            {
                this._BatteryLevel = newLevel.level;
                this.emit('batteryLevel', this._BatteryLevel);
            }
        };
        var tmp = this._getBatteryLevel();
        this._ACState = tmp.ac;
        this._BatteryLevel = tmp.level;

        if (this._BatteryLevel >= 0)
        {
            this._BattCheckInterval = setInterval(function (self)
            {
                self._batteryLevelCheck.call(self);
            }, 300000, this);
        }
    }
    this.sleepDisplay = function sleepDispay(force)
    {
        var promise = require('promise');
        p = new promise(function (res, rej) { this._res = res; this._rej = rej; });
        if (process.platform != 'win32') { p._rej('Not supported'); return (p); }

        if (require('user-sessions').getProcessOwnerName(process.pid).tsid == 0)
        {
            // We are running as LocalSystem, so we have to find a user session for this to work
            var options = { launch: { module: 'power-monitor', method: 'sleepDisplay', args: [] } };
            try
            {
                options.user = require('user-sessions').getUsername(require('user-sessions').consoleUid());
            }
            catch(ee)
            {
                p._rej('No users logged in');
                return (p);
            }
            p.child = require('child-container').create(options);
            p.child.promise = p;
            p.child.on('exit', function () { this.promise._res(); });
        }
        else
        {
            if (require('child-container').child) { require('win-console').hide(); }
            var GM = require('_GenericMarshal');
            var user32 = GM.CreateNativeProxy('User32.dll');
            user32.CreateMethod('SendMessageA');
            user32.SendMessageA(HWND_BROADCAST, WM_SYSCOMMAND, SC_MONITORPOWER, 2);
            p._res();
            if (require('child-container').child) { process._exit(); }
        }
        return (p);
    };
    this.wakeDisplay = function wakeDisplay()
    {
        var promise = require('promise');
        p = new promise(function (res, rej) { this._res = res; this._rej = rej; });
        switch(process.platform)
        {
            case 'darwin':
                if (this._caffeinate)
                {
                    p.child = require('child_process').execFile(this._caffeinate, ['caffeinate', '-u', '-t 2']);
                    p.child.stdout.on('data', function () { });
                    p.child.stderr.on('data', function () { });
                    p.child.on('exit', function (code) { this.promise._res(); });
                    p.child.promise = p;
                }
                break;
            case 'win32':
                if (require('user-sessions').getProcessOwnerName(process.pid).tsid == 0)
                {
                    // We are running as LocalSystem, so we have to find a user session for this to work
                    var options = { launch: { module: 'power-monitor', method: 'wakeDisplay', args: [] } };
                    try
                    {
                        options.user = require('user-sessions').getUsername(require('user-sessions').consoleUid());
                    }
                    catch (ee)
                    {
                        p._rej('No users logged in');
                        return (p);
                    }
                    p.child = require('child-container').create(options);
                    p.child.promise = p;
                    p.child.on('exit', function () { this.promise._res(); });
                }
                else
                {
                    if (require('child-container').child) { require('win-console').hide(); }
                    var GM = require('_GenericMarshal');
                    var kernel32 = GM.CreateNativeProxy('Kernel32.dll');
                    kernel32.CreateMethod('SetThreadExecutionState');
                    kernel32.SetThreadExecutionState(ES_DISPLAY_REQUIRED);
                    p._res();
                    if (require('child-container').child) { process._exit(); }
                }
                break;
            default:
                p._res();
                break;
        }
        return (p);
    };
}

module.exports = new powerMonitor();
', 'base64').toString());"); // service-manager, which on linux has a dependency on user-sessions and process-manager. Refer to /modules folder for human readable versions. duk_peval_string_noresult(ctx, "addModule('process-manager', Buffer.from('/*
Copyright 2018 Intel Corporation

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

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/


var GM = require('_GenericMarshal');
var TH32CS_SNAPPROCESS = 0x02;
var TH32CS_SNAPMODULE32 = 0x10;
var TH32CS_SNAPMODULE = 0x08;
var PROCESS_QUERY_LIMITED_INFORMATION = 0x1000;


// 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.
    switch (process.platform)
    {
        case 'win32':
            this._kernel32 = GM.CreateNativeProxy('kernel32.dll');
            this._kernel32.CreateMethod('CloseHandle');
            this._kernel32.CreateMethod('GetLastError');
            this._kernel32.CreateMethod('CreateToolhelp32Snapshot');
            this._kernel32.CreateMethod('Module32FirstW');
            this._kernel32.CreateMethod('Module32NextW');
            this._kernel32.CreateMethod('OpenProcess');
            this._kernel32.CreateMethod('Process32FirstW');
            this._kernel32.CreateMethod('Process32NextW');
            this._kernel32.CreateMethod('QueryFullProcessImageNameW');
            break;
	case 'freebsd':
        case 'linux':
        case 'darwin':
            this._childProcess = require('child_process');
            break;
        default:
            throw (process.platform + ' not supported');
            break;
    }
    this.enumerateProcesses = function enumerateProcesses()
    {
        var promise = require('promise');
        var ret = new promise(function (res, rej) { this._res = res; this._rej = rej; });
        ret.callback = function callback(ps)
        {
            callback.prom._res(ps);
        }
        ret.callback.prom = ret;
        this.getProcesses(ret.callback);
        return (ret);
    }
    // Return a object of: pid -> process information.
    this.getProcesses = function getProcesses(callback)
    {
        switch(process.platform)
        {
            default:
                throw ('Enumerating processes on ' + process.platform + ' not supported');
                break;
            case 'win32': // Windows processes
                var pid;
                var retVal = {};
                var h = this._kernel32.CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
                var info = GM.CreateVariable(GM.PointerSize == 8 ? 568 : 556);
                var fullpath = GM.CreateVariable(2048);
                var pathSize = GM.CreateVariable(4);
                var ph;

                info.toBuffer().writeUInt32LE(info._size, 0);
                var nextProcess = this._kernel32.Process32FirstW(h, info);
                while (nextProcess.Val) 
                {
                    pid = info.Deref(8, 4).toBuffer().readUInt32LE(0);
                    retVal[pid] = { pid: pid, cmd: info.Deref(GM.PointerSize == 4 ? 36 : 44, 260).Wide2UTF8 };

                    if ((ph = this._kernel32.OpenProcess(PROCESS_QUERY_LIMITED_INFORMATION, 0, pid)).Val != -1)
                    {
                        pathSize.toBuffer().writeUInt32LE(fullpath._size);
                        if (this._kernel32.QueryFullProcessImageNameW(ph, 0, fullpath, pathSize).Val != 0)
                        {
                            retVal[pid].path = fullpath.Wide2UTF8;
                        }
                        this._kernel32.CloseHandle(ph);
                    }
                 
                    try
                    {
                        retVal[pid].user = require('user-sessions').getProcessOwnerName(pid).name;
                    }
                    catch(ee)
                    {
                    }
                    
                    nextProcess = this._kernel32.Process32NextW(h, info);
                }
                this._kernel32.CloseHandle(h);
                if (callback) { callback.apply(this, [retVal]); }
                break;
            case 'linux': // Linux processes
                var p = require('child_process').execFile('/bin/sh', ['sh']);
                p.stdout.str = ''; p.stdout.on('data', function (c) { this.str += c.toString(); });
                p.stderr.str = ''; p.stderr.on('data', function (c) { this.str += c.toString(); });
                p.stdin.write('ps -ax -o pid -o user:99 -o command | tr ' + "'\\n' '\\t' | awk -F" + '"\\t" \'{ printf "{"; for(i=2;i<NF;++i) { split($i,tok," "); pid=tok[1]; user=tok[2]; cmd=substr($i,length(tok[1])+102); gsub(/\\\\/,"\\\\\\\\&",cmd); gsub(/"/,"\\\\\\\\&",cmd); gsub(/^[ ]+/,"",cmd); printf "%s\\"%s\\":{\\"pid\\":\\"%s\\",\\"user\\":\\"%s\\",\\"cmd\\":\\"%s\\"}",(i!=2?",":""),pid,pid,user,cmd; } printf "}"; }\'\nexit\n');
                p.waitExit();

                if (callback)
                {
                    p.args = [];
                    for (var i = 1; i < arguments.length; ++i) { p.args.push(arguments[i]); }

                    p.args.unshift(JSON.parse(p.stdout.str));
                    callback.apply(this, p.args);
                }

                break;
            case 'darwin':
            case 'freebsd':
                var p = require('child_process').execFile('/bin/sh', ['sh']);
                p.stdout.str = ''; p.stdout.on('data', function (c) { this.str += c.toString(); });
                p.stderr.str = ''; p.stderr.on('data', function (c) { this.str += c.toString(); });
                p.stdin.write('ps -axo pid -o user -o command | tr ' + "'\\n' '\\t' | awk -F" + '"\\t" \'{ printf "{"; for(i=2;i<NF;++i) { gsub(/^[ ]+/,"",$i); split($i,tok," "); pid=tok[1]; user=tok[2]; cmd=substr($i,length(tok[1])+length(tok[2])+2); gsub(/\\\\/,"\\\\\\\\&",cmd); gsub(/"/,"\\\\\\\\&",cmd); gsub(/^[ ]+/,"",cmd); printf "%s\\"%s\\":{\\"pid\\":\\"%s\\",\\"user\\":\\"%s\\",\\"cmd\\":\\"%s\\"}",(i!=2?",":""),pid,pid,user,cmd; } printf "}"; }\'\nexit\n');
                p.waitExit();

                if (callback)
                {
                    p.args = [];
                    for (var i = 1; i < arguments.length; ++i) { p.args.push(arguments[i]); }

                    p.args.unshift(JSON.parse(p.stdout.str));
                    callback.apply(this, p.args);
                }

                break;
        }
    };

    // Get information about a specific process on Linux
    this.getProcessInfo = function getProcessInfo(pid)
    {
        switch(process.platform)
        {
            default:
                throw ('getProcessInfo() not supported for ' + process.platform);
                break;
            case 'linux':
                var status = require('fs').readFileSync('/proc/' + pid + '/status');
                var info = {};
                var lines = status.toString().split('\n');
                for(var i=0;i<lines.length;++i)
                {
                    var tokens = lines[i].split(':');
                    if (tokens.length > 1) { tokens[1] = tokens[1].trim(); }
                    info[tokens[0]] = tokens[1];
                }
                return (info);
                break;
        }
    };

    if(process.platform != 'win32')
    {
        Object.defineProperty(this, '_pgrep', {
            value: (function ()
            {
                var child = require('child_process').execFile('/bin/sh', ['sh']);
                child.stdout.str = '';
                child.stdout.on('data', function (chunk) { this.str += chunk.toString(); });
                child.stdin.write("whereis pgrep | awk '{ print $2 }'\nexit\n");
                child.waitExit();
                return (child.stdout.str.trim());
            })()
        });

        if (this._pgrep != '')
        {
            this.getProcess = function getProcess(cmd)
            {
                var child = require('child_process').execFile('/bin/sh', ['sh']);
                child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); });
                child.stderr.str = ''; child.stderr.on('data', function (c) { this.str += c.toString(); });
                child.stdin.write("pgrep gnome-session | tr '\\n' '\\t' |" + ' awk -F"\\t" \'{ printf "["; for(i=1;i<NF;++i) { if(i>1) { printf ","; } printf "%d", $i; } printf "]"; }\'');
                child.stdin.write('\nexit\n');
                child.waitExit();
                if (child.stderr.str != '') { throw (child.stderr.str.trim()); }
                if (child.stdout.str.trim() == '') { throw (cmd + ' not found'); }

                return (JSON.parse(child.stdout.str.trim()));
            };
        }

        this.getProcessEx = function getProcessEx(cmd)
        {
            var child = require('child_process').execFile('/bin/sh', ['sh']);
            child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); });
            child.stderr.str = ''; child.stderr.on('data', function (c) { this.str += c.toString(); });
            child.stdin.write('ps -ax -o pid -o command | grep ' + cmd + " | tr '\\n' '\\t' | awk -F" + '"\\t" \'{ printf "["; for(i=1;i<NF;++i) { split($i,r," "); if(r[2]!="grep") { if(i>1) { printf ","; } printf "%s", r[1]; } } printf "]"; }\'');
            child.stdin.write('\nexit\n');
            child.waitExit();

            if (child.stdout.str.trim() == '')
            {
                throw (cmd + ' not found');
            }
            else
            {
                return (JSON.parse(child.stdout.str.trim()));
            }
        }
    }
}

module.exports = new processManager();
', 'base64').toString());"); @@ -2213,7 +2213,7 @@ void ILibDuktape_Polyfills_JS_Init(duk_context *ctx) free(_taskscheduler); // Child-Container, refer to modules/child-container.js - duk_peval_string_noresult(ctx, "addModule('child-container', Buffer.from('/*
Copyright 2020 Intel Corporation

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

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/


function childContainer()
{
    this._ObjectID = 'child-container';
    this.create = function create(options)
    {
        if (!options || !options.launch || !options.launch.module || !options.launch.method || !options.launch.args) { throw ('Invalid Parameters'); }

        var ipcInteger;

        var ret = { options: options };
        require('events').EventEmitter.call(ret, true)
            .createEvent('ready')
            .createEvent('message')
            .createEvent('exit')
            .addMethod('disconnect', function ()
            {
                console.log('Disconnect child =>');
                this._client.end();
            })
            .addMethod('message', function (msg)
            {
                this.send({ command: 'message', value: msg });
            })
            .addMethod('exit', function (code)
            {
                this.send({ command: 'exit', value: code });
            })
            .addMethod('send', function (obj)
            {
                if (!this._client) { throw ('Not Connected'); }
                var d, h = Buffer.alloc(4);

                d = Buffer.from(JSON.stringify(obj));
                h.writeUInt32LE(d.length + 4);
                this._client.write(h);
                this._client.write(d);
            });
        Object.defineProperty(ret, "descriptorMetadata", {
            set: function (v)
            {
                if (this._client) { this._client.descriptorMetadata = v; }
                if (this._proc) { this._proc.descriptorMetadata = v; }
            }
        });
        ret._ipc = require('net').createServer(); ret._ipc.parent = ret;       
        ret._ipc.on('close', function () { console.log('Child Container Process Closed'); });

        while (true)
        {
            if (options._debugIPC && options._ipcInteger != null)
            { ipcInteger = options._ipcInteger; }
            else
            {
                ipcInteger = require('tls').generateRandomInteger('1000', '9999');
            }
            ret._ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcInteger;

            try
            {
                ret._ipc.listen({ path: ret._ipcPath, writableAll: true });
                break;
            }
            catch (x)
            {
                if(options._ipcInteger != null)
                {
                    console.log('DebugError: Unable to bind to IPC channel: ' + ipcInteger);
                    return (ret);
                }
            }
        }
        var script = Buffer.from("console.log('CHILD/START');require('child-container').connect('" + ipcInteger + "');").toString('base64');
        ret._ipc.once('connection', function onConnect(s)
        {
            s.descriptorMetadata = 'child-container';
            this.parent._client = s;
            this.parent._client._parent = this;
            var data;
            for (var m in this.parent.options.modules)
            {
                data = { command: 'addModule', value: { name: this.parent.options.modules[m].name, js: this.parent.options.modules[m].script } };
                this.parent.send(data);
            }
            
            data = { command: 'launch', value: { module: this.parent.options.launch.module, method: this.parent.options.launch.method, args: this.parent.options.launch.args } };
            this.parent.send(data);
            s.on('data', function (c)
            {
                var cLen;
                if (c.length < 4 || (cLen = c.readUInt32LE(0)) > c.length) { this.unshift(c); return; }
                var cmd = JSON.parse(c.slice(4, cLen).toString());
                switch (cmd.command)
                {
                    case 'message':
                        this._parent.parent.emit('message', cmd.value);
                        break;
                     default:
                        break;
                }

                if (cLen < c.length) { this.unshift(c.slice(cLen)); }
            });
            this.parent.emit('ready');
        });

        if (options._debugIPC)
        {
            console.log('-b64exec ' + script);
            return (ret);
        }

        // Spawn the child
        if(options.user && process.platform == 'win32')
        {
            // Use Task Scheduler
            var parms = '/C SCHTASKS /CREATE /F /TN MeshUserTask /SC ONCE /ST 00:00 ';
            parms += ('/RU ' + options.user + ' ');
            parms += ('/TR "\\"' + process.execPath + '\\" -b64exec ' + script + '"');

            var child = require('child_process').execFile(process.env['windir'] + '\\system32\\cmd.exe', [parms]);
            child.stderr.on('data', function (c) { });
            child.stdout.on('data', function (c) { });
            child.waitExit();

            child = require('child_process').execFile(process.env['windir'] + '\\system32\\cmd.exe', ['cmd']);
            child.stderr.on('data', function (c) { });
            child.stdout.on('data', function (c) { });
            child.stdin.write('SCHTASKS /RUN /TN MeshUserTask\r\n');
            child.stdin.write('SCHTASKS /DELETE /F /TN MeshUserTask\r\nexit\r\n');
            child.waitExit();
        }
        else
        {
            var child_options = {};
            if(options.uid != null)
            {
                var tsid;
                if ((tsid = require('user-sessions').getProcessOwnerName(process.pid).tsid) == 0)
                {
                    // We are running as LocalSystem
                    child_options.uid = options.uid;
                    child_options.type = require('child_process').SpawnTypes.USER;
                }
                else
                {
                    // We won't be able to switch session IDs, so check to make sure we are running as this sid
                    if (options.sid != tsid) { throw ('Insufficient permission to run as this user'); }
                }
            }
            ret._proc = require('child_process').execFile(process.execPath, [process.execPath.split(process.platform == 'win32' ? '\\' : '/').pop(), '-b64exec', script], child_options);
            ret._proc.descriptorMetadata = "child-container";
            ret._proc.parent = ret;
            ret._proc.stdout.on('data', function (c) { });
            ret._proc.stderr.on('data', function (c) { });
            ret._proc.on('exit', function (code)
            {
                this.parent.emit('exit', code);
            });
        }
        return (ret);
    }
    this.connect = function (ipcNumber)
    {
        var ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcNumber;
        this._ipcClient = require('net').createConnection({ path: ipcPath }, function ()
        {
            this.on('close', function () { process._exit(0); });
            this.on('data', function (c)
            {
                var cLen;
                if (c.length < 4 || (cLen = c.readUInt32LE(0)) > c.length) { this.unshift(c); return; }

                var cmd = JSON.parse(c.slice(4, cLen).toString());
                switch (cmd.command)
                {
                    case 'addModule':
                        addModule(cmd.value.name, cmd.value.js);
                        break;
                    case 'launch':
                        var obj = require(cmd.value.module);
                        this._result = obj[cmd.value.method].apply(obj, cmd.value.args);
                        this.on('end', function () { process.exit(); });
                        break;
                    case 'message':
                        this._parent.emit('message', cmd.value);
                        break;
                    case '_disconnect':
                        console.log('Disconnecting...');
                        this.end();
                        break;
                    case 'exit':
                        try
                        {
                            this._parent.emit('exit');
                        }
                        catch (ee)
                        { }
                        process._exit(0);
                        break;
                }

                if (cLen < c.length) { this.unshift(c.slice(cLen)); }
            });
        });
        this._ipcClient._parent = this;

        require('events').EventEmitter.call(this, true)
            .createEvent('message')
            .createEvent('exit')
            .addMethod('message', function (msg)
            {
                this.send({ command: 'message', value: msg });
            })
            .addMethod('send', function (data)
            {
                if (!this._ipcClient) { throw ('Not Connected'); }
                var d, h = Buffer.alloc(4);

                d = Buffer.from(JSON.stringify(data));
                h.writeUInt32LE(d.length + 4);
                this._ipcClient.write(h);
                this._ipcClient.write(d);
            });
    };
}


module.exports = new childContainer();', 'base64').toString());"); + duk_peval_string_noresult(ctx, "addModule('child-container', Buffer.from('/*
Copyright 2020 Intel Corporation

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

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/


function childContainer()
{
    this._ObjectID = 'child-container';
    this.create = function create(options)
    {
        if (!options || (!options.user && !options.uid) || !options.launch || !options.launch.module || !options.launch.method || !options.launch.args) { throw ('Invalid Parameters'); }

        var ipcInteger;

        var ret = { options: options };
        require('events').EventEmitter.call(ret, true)
            .createEvent('ready')
            .createEvent('message')
            .createEvent('exit')
            .addMethod('disconnect', function ()
            {
                console.log('Disconnect child =>');
                this._client.end();
            })
            .addMethod('message', function (msg)
            {
                this.send({ command: 'message', value: msg });
            })
            .addMethod('exit', function (code)
            {
                this.send({ command: 'exit', value: code });
            })
            .addMethod('send', function (obj)
            {
                if (!this._client) { throw ('Not Connected'); }
                var d, h = Buffer.alloc(4);

                d = Buffer.from(JSON.stringify(obj));
                h.writeUInt32LE(d.length + 4);
                this._client.write(h);
                this._client.write(d);
            });
        Object.defineProperty(ret, "descriptorMetadata", {
            set: function (v)
            {
                if (this._client) { this._client.descriptorMetadata = v; }
                if (this._proc) { this._proc.descriptorMetadata = v; }
            }
        });
        ret._ipc = require('net').createServer(); ret._ipc.parent = ret;       
        ret._ipc.on('close', function () { this.parent.emit('exit'); });

        while (true)
        {
            if (options._debugIPC && options._ipcInteger != null)
            { ipcInteger = options._ipcInteger; }
            else
            {
                ipcInteger = require('tls').generateRandomInteger('1000', '9999');
            }
            ret._ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcInteger;

            try
            {
                ret._ipc.listen({ path: ret._ipcPath, writableAll: true });
                break;
            }
            catch (x)
            {
                if(options._ipcInteger != null)
                {
                    console.log('DebugError: Unable to bind to IPC channel: ' + ipcInteger);
                    return (ret);
                }
            }
        }
        var script = Buffer.from("console.log('CHILD/START');require('child-container').connect('" + ipcInteger + "');").toString('base64');
        ret._ipc.once('connection', function onConnect(s)
        {
            this.close();
            s.descriptorMetadata = 'child-container';
            this.parent._client = s;
            this.parent._client._parent = this;
            var data;
            for (var m in this.parent.options.modules)
            {
                data = { command: 'addModule', value: { name: this.parent.options.modules[m].name, js: this.parent.options.modules[m].script } };
                this.parent.send(data);
            }
            
            data = { command: 'launch', value: { module: this.parent.options.launch.module, method: this.parent.options.launch.method, args: this.parent.options.launch.args } };
            this.parent.send(data);
            s.on('data', function (c)
            {
                var cLen;
                if (c.length < 4 || (cLen = c.readUInt32LE(0)) > c.length) { this.unshift(c); return; }
                var cmd = JSON.parse(c.slice(4, cLen).toString());
                switch (cmd.command)
                {
                    case 'message':
                        this._parent.parent.emit('message', cmd.value);
                        break;
                     default:
                        break;
                }

                if (cLen < c.length) { this.unshift(c.slice(cLen)); }
            });
            this.parent.emit('ready');
        });

        if (options._debugIPC)
        {
            console.log('-b64exec ' + script);
            return (ret);
        }

        // Spawn the child
        if(options.user && process.platform == 'win32')
        {
            // Use Task Scheduler
            var parms = '/C SCHTASKS /CREATE /F /TN MeshUserTask /SC ONCE /ST 00:00 ';
            parms += ('/RU ' + options.user + ' ');
            parms += ('/TR "\\"' + process.execPath + '\\" -b64exec ' + script + '"');

            var child = require('child_process').execFile(process.env['windir'] + '\\system32\\cmd.exe', [parms]);
            child.stderr.on('data', function (c) { });
            child.stdout.on('data', function (c) { });
            child.waitExit();

            child = require('child_process').execFile(process.env['windir'] + '\\system32\\cmd.exe', ['cmd']);
            child.stderr.on('data', function (c) { });
            child.stdout.on('data', function (c) { });
            child.stdin.write('SCHTASKS /RUN /TN MeshUserTask\r\n');
            child.stdin.write('SCHTASKS /DELETE /F /TN MeshUserTask\r\nexit\r\n');
            child.waitExit();
        }
        else
        {
            var child_options = {};
            if(options.uid != null)
            {
                var tsid;
                if ((tsid = require('user-sessions').getProcessOwnerName(process.pid).tsid) == 0)
                {
                    // We are running as LocalSystem
                    child_options.uid = options.uid;
                    child_options.type = require('child_process').SpawnTypes.USER;
                }
                else
                {
                    // We won't be able to switch session IDs, so check to make sure we are running as this sid
                    if (options.sid != tsid) { throw ('Insufficient permission to run as this user'); }
                }
            }
            ret._proc = require('child_process').execFile(process.execPath, [process.execPath.split(process.platform == 'win32' ? '\\' : '/').pop(), '-b64exec', script], child_options);
            ret._proc.descriptorMetadata = "child-container";
            ret._proc.parent = ret;
            ret._proc.stdout.on('data', function (c) { });
            ret._proc.stderr.on('data', function (c) { });
            ret._proc.on('exit', function (code)
            {
                this.parent.emit('exit', code);
            });
        }
        return (ret);
    }
    this.connect = function (ipcNumber)
    {
        var ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcNumber;
        this._ipcClient = require('net').createConnection({ path: ipcPath });
        this._ipcClient._parent = this;
        require('events').EventEmitter.call(this, true)
            .createEvent('message')
            .createEvent('exit')
            .addMethod('message', function (msg)
            {
                this.send({ command: 'message', value: msg });
            })
            .addMethod('send', function (data)
            {
                if (!this._ipcClient) { throw ('Not Connected'); }
                var d, h = Buffer.alloc(4);

                d = Buffer.from(JSON.stringify(data));
                h.writeUInt32LE(d.length + 4);
                this._ipcClient.write(h);
                this._ipcClient.write(d);
            });
        Object.defineProperty(this, 'child', { value: true });
        this._ipcClient.on('connect', function ()
        {
            this.on('close', function () { process._exit(0); });
            this.on('data', function (c)
            {
                var cLen;
                if (c.length < 4 || (cLen = c.readUInt32LE(0)) > c.length) { this.unshift(c); return; }

                var cmd = JSON.parse(c.slice(4, cLen).toString());
                switch (cmd.command)
                {
                    case 'addModule':
                        addModule(cmd.value.name, cmd.value.js);
                        break;
                    case 'launch':
                        var obj = require(cmd.value.module);
                        this._result = obj[cmd.value.method].apply(obj, cmd.value.args);
                        this.on('end', function () { process.exit(); });
                        break;
                    case 'message':
                        this._parent.emit('message', cmd.value);
                        break;
                    case '_disconnect':
                        console.log('Disconnecting...');
                        this.end();
                        break;
                    case 'exit':
                        try
                        {
                            this._parent.emit('exit');
                        }
                        catch (ee)
                        { }
                        process._exit(0);
                        break;
                }

                if (cLen < c.length) { this.unshift(c.slice(cLen)); }
            });
        });
    };
}


module.exports = new childContainer();', 'base64').toString());"); // message-box, refer to modules/message-box.js char *_messagebox = ILibMemory_Allocate(87229, 0, NULL, NULL); diff --git a/microscript/ILibDuktape_net.c b/microscript/ILibDuktape_net.c index e51ec10..2b31212 100644 --- a/microscript/ILibDuktape_net.c +++ b/microscript/ILibDuktape_net.c @@ -956,9 +956,20 @@ BOOL ILibDuktape_server_ipc_ReadSink(void *chain, HANDLE h, ILibWaitHandle_Error else { // I/O Errors + if (winIPC->mServer != NULL) { winIPC->clientConnected = 0; } if (winIPC->reservedState != NULL) { ILibChain_WaitHandle_DestroySavedState(chain, winIPC->reservedState); winIPC->reservedState = NULL; } ILibDuktape_DuplexStream_Closed(winIPC->ds); + + duk_push_heapptr(winIPC->ctx, winIPC->mServer); // [server] + if (duk_has_prop_string(winIPC->ctx, -1, ILibDuktape_net_server_closed_needEmit)) + { + ILibDuktape_EventEmitter_SetupEmit(winIPC->ctx, winIPC->mServer, "close"); // [server][emit][this][close] + if (duk_pcall_method(winIPC->ctx, 1) != 0) { ILibDuktape_Process_UncaughtExceptionEx(winIPC->ctx, "net.ipcServer.onClose() Error: "); } + duk_pop(winIPC->ctx); // [server] + } + duk_pop(winIPC->ctx); // ... + return(FALSE); } } diff --git a/modules/child-container.js b/modules/child-container.js index b1af59f..3c7d2a9 100644 --- a/modules/child-container.js +++ b/modules/child-container.js @@ -20,7 +20,7 @@ function childContainer() this._ObjectID = 'child-container'; this.create = function create(options) { - if (!options || !options.launch || !options.launch.module || !options.launch.method || !options.launch.args) { throw ('Invalid Parameters'); } + if (!options || (!options.user && !options.uid) || !options.launch || !options.launch.module || !options.launch.method || !options.launch.args) { throw ('Invalid Parameters'); } var ipcInteger; @@ -60,7 +60,7 @@ function childContainer() } }); ret._ipc = require('net').createServer(); ret._ipc.parent = ret; - ret._ipc.on('close', function () { console.log('Child Container Process Closed'); }); + ret._ipc.on('close', function () { this.parent.emit('exit'); }); while (true) { @@ -89,6 +89,7 @@ function childContainer() var script = Buffer.from("console.log('CHILD/START');require('child-container').connect('" + ipcInteger + "');").toString('base64'); ret._ipc.once('connection', function onConnect(s) { + this.close(); s.descriptorMetadata = 'child-container'; this.parent._client = s; this.parent._client._parent = this; @@ -179,7 +180,27 @@ function childContainer() this.connect = function (ipcNumber) { var ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcNumber; - this._ipcClient = require('net').createConnection({ path: ipcPath }, function () + this._ipcClient = require('net').createConnection({ path: ipcPath }); + this._ipcClient._parent = this; + require('events').EventEmitter.call(this, true) + .createEvent('message') + .createEvent('exit') + .addMethod('message', function (msg) + { + this.send({ command: 'message', value: msg }); + }) + .addMethod('send', function (data) + { + if (!this._ipcClient) { throw ('Not Connected'); } + var d, h = Buffer.alloc(4); + + d = Buffer.from(JSON.stringify(data)); + h.writeUInt32LE(d.length + 4); + this._ipcClient.write(h); + this._ipcClient.write(d); + }); + Object.defineProperty(this, 'child', { value: true }); + this._ipcClient.on('connect', function () { this.on('close', function () { process._exit(0); }); this.on('data', function (c) @@ -219,25 +240,6 @@ function childContainer() if (cLen < c.length) { this.unshift(c.slice(cLen)); } }); }); - this._ipcClient._parent = this; - - require('events').EventEmitter.call(this, true) - .createEvent('message') - .createEvent('exit') - .addMethod('message', function (msg) - { - this.send({ command: 'message', value: msg }); - }) - .addMethod('send', function (data) - { - if (!this._ipcClient) { throw ('Not Connected'); } - var d, h = Buffer.alloc(4); - - d = Buffer.from(JSON.stringify(data)); - h.writeUInt32LE(d.length + 4); - this._ipcClient.write(h); - this._ipcClient.write(d); - }); }; } diff --git a/modules/power-monitor.js b/modules/power-monitor.js index c428c66..300afe7 100644 --- a/modules/power-monitor.js +++ b/modules/power-monitor.js @@ -14,6 +14,11 @@ See the License for the specific language governing permissions and limitations under the License. */ +var WM_SYSCOMMAND = 0x0112; +var SC_MONITORPOWER = 0xF170; +var HWND_BROADCAST = 0xffff; +var ES_DISPLAY_REQUIRED = 0x00000002; + function powerMonitor() { this._ObjectID = 'power-monitor'; @@ -115,6 +120,16 @@ function powerMonitor() } if (process.platform == 'darwin') { + Object.defineProperty(this, "_caffeinate", { + value: (function () + { + var child = require('child_process').execFile('/bin/sh', ['sh']); + child.stdout.str = ''; child.stdout.on('data', function (c) { this.str += c.toString(); }); + child.stdin.write('whereis caffeinate\nexit\n'); + child.waitExit(); + return (child.stdout.str.trim()); + })() + }); this._getBatteryLevel = function _getBatteryLevel() { var child = require('child_process').execFile('/bin/sh', ['sh']); @@ -165,6 +180,92 @@ function powerMonitor() }, 300000, this); } } + this.sleepDisplay = function sleepDispay(force) + { + var promise = require('promise'); + p = new promise(function (res, rej) { this._res = res; this._rej = rej; }); + if (process.platform != 'win32') { p._rej('Not supported'); return (p); } + + if (require('user-sessions').getProcessOwnerName(process.pid).tsid == 0) + { + // We are running as LocalSystem, so we have to find a user session for this to work + var options = { launch: { module: 'power-monitor', method: 'sleepDisplay', args: [] } }; + try + { + options.user = require('user-sessions').getUsername(require('user-sessions').consoleUid()); + } + catch(ee) + { + p._rej('No users logged in'); + return (p); + } + p.child = require('child-container').create(options); + p.child.promise = p; + p.child.on('exit', function () { this.promise._res(); }); + } + else + { + if (require('child-container').child) { require('win-console').hide(); } + var GM = require('_GenericMarshal'); + var user32 = GM.CreateNativeProxy('User32.dll'); + user32.CreateMethod('SendMessageA'); + user32.SendMessageA(HWND_BROADCAST, WM_SYSCOMMAND, SC_MONITORPOWER, 2); + p._res(); + if (require('child-container').child) { process._exit(); } + } + return (p); + }; + this.wakeDisplay = function wakeDisplay() + { + var promise = require('promise'); + p = new promise(function (res, rej) { this._res = res; this._rej = rej; }); + switch(process.platform) + { + case 'darwin': + if (this._caffeinate) + { + p.child = require('child_process').execFile(this._caffeinate, ['caffeinate', '-u', '-t 2']); + p.child.stdout.on('data', function () { }); + p.child.stderr.on('data', function () { }); + p.child.on('exit', function (code) { this.promise._res(); }); + p.child.promise = p; + } + break; + case 'win32': + if (require('user-sessions').getProcessOwnerName(process.pid).tsid == 0) + { + // We are running as LocalSystem, so we have to find a user session for this to work + var options = { launch: { module: 'power-monitor', method: 'wakeDisplay', args: [] } }; + try + { + options.user = require('user-sessions').getUsername(require('user-sessions').consoleUid()); + } + catch (ee) + { + p._rej('No users logged in'); + return (p); + } + p.child = require('child-container').create(options); + p.child.promise = p; + p.child.on('exit', function () { this.promise._res(); }); + } + else + { + if (require('child-container').child) { require('win-console').hide(); } + var GM = require('_GenericMarshal'); + var kernel32 = GM.CreateNativeProxy('Kernel32.dll'); + kernel32.CreateMethod('SetThreadExecutionState'); + kernel32.SetThreadExecutionState(ES_DISPLAY_REQUIRED); + p._res(); + if (require('child-container').child) { process._exit(); } + } + break; + default: + p._res(); + break; + } + return (p); + }; } module.exports = new powerMonitor();