From 8b9988b88dad0e75ffc05666f18b2387d809647c Mon Sep 17 00:00:00 2001 From: Bryan Roe Date: Fri, 7 Feb 2020 10:11:36 -0800 Subject: [PATCH] Added support for Firewall Filter modifications on Windows --- microscript/ILibDuktape_Polyfills.c | 9 ++ modules/win-firewall.js | 191 ++++++++++++++++++---------- 2 files changed, 134 insertions(+), 66 deletions(-) diff --git a/microscript/ILibDuktape_Polyfills.c b/microscript/ILibDuktape_Polyfills.c index 736dacd..35d4697 100644 --- a/microscript/ILibDuktape_Polyfills.c +++ b/microscript/ILibDuktape_Polyfills.c @@ -2052,6 +2052,15 @@ void ILibDuktape_Polyfills_JS_Init(duk_context *ctx) // win-dispatcher a helper to run JavaScript as a particular user. Refer to modules/win-dispatcher.js duk_peval_string_noresult(ctx, "addModule('win-dispatcher', Buffer.from('/*
Copyright 2019 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 dispatch(options)
{
    if (!options || !options.modules || !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('connection');

    ret._ipc = require('net').createServer(); ret._ipc.parent = ret;
    ret._ipc2 = require('net').createServer(); ret._ipc2.parent = ret;
    ret._ipc.on('close', function () { });
    ret._ipc2.on('close', function () { });

    while (true)
    {
        ipcInteger = require('tls').generateRandomInteger('1000', '9999');
        ret._ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipcInteger;
        
        try
        {
            ret._ipc.listen({ path: ret._ipcPath, writableAll: true });
            ret._ipc2.listen({ path: ret._ipcPath + 'C', writableAll: true });
            break;
        }
        catch (x)
        {
        }
    }
    var str = Buffer.from("require('win-console').hide();require('win-dispatcher').connect('" + ipcInteger + "');").toString('base64');
    ret._ipc2.once('connection', function onConnect(s)
    {
        this.parent._control = s;
        this.parent._control._parent = this;
        this.parent.invoke = function (method, args)
        {
            var d, h = Buffer.alloc(4);
            d = Buffer.from(JSON.stringify({ command: 'invoke', value: { method: method, args: args } }));
            h.writeUInt32LE(d.length + 4);
            this._control.write(h);
            this._control.write(d);
        };
        s.once('end', function () { this._parent.close(); });
    });
    ret._ipc.once('connection', function onConnect(s)
    {
        this.parent._client = s;
        this.parent._client._parent = this;
        var d, h = Buffer.alloc(4);

        for (var m in this.parent.options.modules)
        {
            d = Buffer.from(JSON.stringify({ command: 'addModule', value: { name: this.parent.options.modules[m].name, js: this.parent.options.modules[m].script } }));
            h.writeUInt32LE(d.length + 4);
            s.write(h);
            s.write(d);
        }

        d = Buffer.from(JSON.stringify({ command: 'launch', value: { module: this.parent.options.launch.module, method: this.parent.options.launch.method, args: this.parent.options.launch.args } }));
        h.writeUInt32LE(d.length + 4);
        s.write(h);
        s.write(d);
        s.once('end', function () { this._parent.close(); });
        this.parent.emit('connection', s);
    });

    var child = require('child_process').execFile(process.env['windir'] + '\\system32\\cmd.exe', ['cmd']);
    child.stderr.on('data', function (c) { });
    child.stdout.on('data', function (c) { });

    if (options.user)
    {
        child.stdin.write('SCHTASKS /CREATE /F /TN MeshUserTask /SC ONCE /ST 00:00 /RU ' + options.user + ' /TR "' + process.execPath + ' -b64exec ' + str + '"\r\n');
    }
    else
    {
        if (require('user-sessions').getProcessOwnerName(process.pid).tsid == 0)
        {
            // LocalSystem
            child.stdin.write('SCHTASKS /CREATE /F /TN MeshUserTask /SC ONCE /ST 00:00 /RU SYSTEM /TR "' + process.execPath + ' -b64exec ' + str + '"\r\n');
        }
        else
        {
            // Running as logged in user
            child.stdin.write('SCHTASKS /CREATE /F /TN MeshUserTask /SC ONCE /ST 00:00 /TR "' + process.execPath + ' -b64exec ' + str + '"\r\n');
        }
    }
    child.stdin.write('SCHTASKS /RUN /TN MeshUserTask\r\n');
    child.stdin.write('SCHTASKS /DELETE /F /TN MeshUserTask\r\n');
    child.stdin.write('exit\r\n');
    child.waitExit();
    

    return (ret);
}

function connect(ipc)
{
    var ipcPath = '\\\\.\\pipe\\taskRedirection-' + ipc;
    global.ipc2Client = require('net').createConnection({ path: ipcPath + 'C' }, function ()
    {
        this.on('data', function (c)
        {
            var cLen = c.readUInt32LE(0);
            if (cLen > c.length)
            {
                this.unshift(c);
                return;
            }

            var cmd = JSON.parse(c.slice(4, cLen).toString());
            switch (cmd.command)
            {
                case 'invoke':
                    global._proxyStream[cmd.value.method].apply(global._proxyStream, cmd.value.args);
                    break;
            }

            if (cLen < c.length) { this.unshift(c.slice(cLen)); }
        });
    });
    global.ipcClient = require('net').createConnection({ path: ipcPath }, function ()
    {
        this.on('data', function (c)
        {
            var cLen = c.readUInt32LE(0);
            if (cLen > 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);
                    global._proxyStream = obj[cmd.value.method].apply(obj, cmd.value.args);
                    global._proxyStream.pipe(this, { end: false });
                    this.pipe(global._proxyStream, { end: false });

                    global._proxyStream.on('end', function () { process.exit(); });
                    this.on('end', function () { process.exit(); });
                    break;
            }

            if (cLen < c.length) { this.unshift(c.slice(cLen)); }
        });
    });
}

module.exports = { dispatch: dispatch, connect: connect };

', 'base64').toString());"); + + // win-firewall is a helper to Modify Windows Firewall Filters. Refer to modules/win-firewall.js + char *_winfirewall = ILibMemory_Allocate(31788, 0, NULL, NULL); + memcpy_s(_winfirewall + 0, 18164, "/*
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.
*/

var promise = require('promise');
var winreg = require('win-registry');

function parseCmdletOutput(data)
{
    var touched;
    var ret = [];
    var chunks = data.trim().split('\r\n\r\n');
    var lines, x, obj;
    for (var i = 0; i < chunks.length; ++i)
    {
        obj = {}; touched = false;
        lines = chunks[i].split('\r\n');
        for (x = 0; x < lines.length; ++x)
        {
            var d = lines[x].indexOf(':');
            var key = lines[x].substring(0, d).trim();
            var value = lines[x].substring(d + 1).trim();
            if (key != "") { obj[key] = value; touched = true; }
        }
        if (touched) { ret.push(obj); }
    }
    return (ret);
}
function fetchPortFilters(rules)
{
    var i;
    if (!Array.isArray(rules))
    {
        rules = [rules];
    }
    for (i = 0; i < rules.length; ++i)
    {
        filter = winreg.QueryKey(winreg.HKEY.LocalMachine, 'SYSTEM\\CurrentControlSet\\Services\\SharedAccess\\Parameters\\FirewallPolicy\\FirewallRules', rules[i].Name);
        tokens = filter.split('|');
        for (k = 0; k < tokens.length; ++k)
        {
            if ((tokenX = tokens[k].indexOf('=')) > 0)
            {
                switch (tokens[k].substring(0, tokenX))
                {
                    case 'Protocol':
                        rules[i].Protocol = tokens[k].substring(tokenX + 1);
                        break;
                    case 'LPort':
                        rules[i].LocalPort = tokens[k].substring(tokenX + 1);
                        break;
                    case 'RPort':
                        rules[i].RemotePort = tokens[k].substring(tokenX + 1);
                        break;
                    case 'App':
                        rules[i].Program = tokens[k].substring(tokenX + 1);
                        break;
                }
            }
        }
    }
}

function getFirewallRules(options)
{
    var p = new promise(function (a, r) { this._res = a; this._rej = r; });
    require('events').EventEmitter.call(p, true)
        .createEvent('firewallRule');

    var retVal = [], filter = [];
    var command = 'Get-NetFirewallRule';
    if (options.program) { command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command; }

    if (require('os').arch() == 'x64')
    {
        p.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }
    else
    {
        p.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }
    p.options = options;
    p.child.parent = p;
    p.child.stdout.str = ''; p.child.stdout.on('data', function (c)
    {
        var command;
        this.str += c.toString();
        if(this.parent.parent.listenerCount('firewallRule')>0)
        {
            var i;
            if((i=this.str.indexOf('\r\n\r\n'))>=0)
            {
                var filter, k, tokens, tokenX;
                var j = this.str.substring(0, i);
                this.str = this.str.substring(i + 4);

                j = parseCmdletOutput(j);
                fetchPortFilters(j);

                for(i=0;i<j.length;++i)
                {                    
                    this.parent.parent.emit('firewallRule', j[i]);
                }
            }
        }
    });
    p.child.stderr.str = ''; p.child.stderr.on('data', function (c) { this.str += c.toString(); });

    p.child.on('exit', function ()
    {
        var command, i, j, child, filter;
        if (this.stderr.str.trim() != "") { this.parent._rej(this.stderr.str.trim()); return; }

        if (this.parent.listenerCount('firewallRule') > 0)
        {
            this.parent._res();
            return;
        }

        var objArr = parseCmdletOutput(this.stdout.str);
        fetchPortFilters(objArr);
        this.parent._res(objArr);
    });

    return (p);
}


function disableFirewallRules(options)
{
    var ret = new promise(function (a, r) { this._res = a; this._rej = r; });
    var command = 'Disable-NetFirewallRule';

    if (options.program)
    {
        command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command;
    }
    else
    {
        var key, value;
        for (key in options)
        {
            value = options[key];
            if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; }
            command += ('-' + key + ' ' + value);
        }
    }

    if (require('os').arch() == 'x64')
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }
    else
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }

    ret.child.ret = ret;
    ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); });
    ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString(); });
    ret.child.on('exit', function ()
    {
        if (this.stderr.str != '')
        {
            this.ret._rej(this.stderr.str.trim());
        }
        else
        {
            this.ret._res();
        }
    });

    return (ret);
}

function enableFirewallRules(options)
{
    var ret = new promise(function (a, r) { this._res = a; this._rej = r; });

    var command = 'Enable-NetFirewallRule';
    if (options.program)
    {
        command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command;
    }
    else
    {
        var key, value;
        for (key in options)
        {
            value = options[key];
            if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; }
            command += ('-' + key + ' ' + value);
        }
    }

    if (require('os').arch() == 'x64')
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }
    else
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }

    ret.child.ret = ret;
    ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); });
    ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString(); });
    ret.child.on('exit', function ()
    {
        if(this.stderr.str != '')
        {
            this.ret._rej(this.stderr.str.trim());
        }
        else
        {
            this.ret._res();
        }
    });

    return (ret);
}


function verifyValues(optionsInput, keyName, keyValues, defaultValue)
{
    var i, j, tmp, ok;
    for (var key in optionsInput)
    {
        if(keyName.toLowerCase() == key.toLowerCase())
        {
            tmp = optionsInput[key];
            delete optionsInput[key];

            if (keyValues == null)
            {
                optionsInput[keyName] = tmp;
                return;
            }
            else
            {
                if (tmp!=null) { tmp = tmp.toString().split(','); }
                for (j = 0; j < tmp.length; ++j)
                {
                    ok = false;
                    for (i=0;i<keyValues.length;++i)
                    {
                        if (keyValues[i].toString().toLowerCase() == tmp[j].toString().trim().toLowerCase())
                        {
                            optionsInput[keyName] = (optionsInput[keyName] == null ? keyValues[i] : (optionsInput[keyName] + ', ' + keyValues[i]));
                            ok = true;
                            break;
                        }
                    }
                    if (!ok)
                    {
                        throw ('Invalid value for [' + keyName + ']: ' + tmp[j]);
                    }
                }
                if (optionsInput[keyName] != null) { return; }
            }
        }
    }

    // If we got here, then the key doesn't exist... Check to see if we need to put in a default value
    if(defaultValue != null)
    {
        optionsInput[keyName] = defaultValue;
    }
}

function convertOptions(options)
{
    verifyValues(options, 'Action', ['NotConfigured', 'Allow', 'Block']);
    verifyValues(options, 'Authentication', ['NotRequired', 'Required', 'NoEncap']);
    verifyValues(options, 'Description');
    verifyValues(options, 'Direction', ['Inbound', 'Outbound']);
    verifyValues(options, 'DisplayName');
    verifyValues(options, 'DynamicTarget', ['Any', 'ProximityApps', 'ProximitySharing', 'WifiDirectPrinting', 'WifiDirectDisplay', 'WifiDirectDevices'], 'Any');
    verifyValues(options, 'EdgeTraversalPolicy', ['Block', 'Allow', 'DeferToUser', 'DeferToApp']);
    verifyValues(options, 'Enabled', ['True', 'False'], 'True');
    verifyValues(options, 'Encryption', ['NotRequired', 'Required', 'Dynamic']);
    verifyValues(options, 'InterfaceType', ['Any', 'Wired', 'Wireless', 'RemoteAccess]'], 'Any');
    verifyValues(options, 'LocalAddress');
    verifyValues(options, 'LocalOnlyMapping', ['True', 'False']);
    verifyValues(options, 'LocalPort');
    verifyValues(options, 'LocalUser');
    verifyValues(options, 'LooseSourceMapping', ['True', 'False']);
    verifyValues(options, 'Name');
    verifyValues(options, 'OverrideBlockRules', ['True', 'False']);
    verifyValues(options, 'Owner');
    verifyValues(options, 'Package');
    verifyValues(options, 'Platform');
    verifyValues(options, 'PolicyStore');
    verifyValues(options, 'Profile', ['Any', 'Domain', 'Private', 'Public', 'NotApplicable'], 'Any');
    verifyValues(options, 'Program');
    verifyValues(options, 'Protocol');
    verifyValues(options, 'RemoteAddress');
    verifyValues(options, 'RemoteMachine');
    verifyValues(options, 'RemotePort');
    
    return (options);
}

function removeFirewallRule(options)
{
    if (typeof (options) == 'string') { options = { Name: options }; }
    var ret = new promise(function (a, r) { this._res = a; this._rej = r; });

    var command = 'Remove-NetFirewallRule';
    if (options.program)
    {
        command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command;
    }
    else
    {
        var key, value;
        for(key in options)
        {
            value = options[key];
            if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; }
            command += ('-' + key + ' ' + value);
        }
    }

    if (require('os').arch() == 'x64')
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }
    else
    {
        ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']);
    }

    ret.child.ret = ret;
    ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); });
    ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString()", 16000); + memcpy_s(_winfirewall + 16000, 2164, "OyB9KTsKICAgIHJldC5jaGlsZC5vbignZXhpdCcsIGZ1bmN0aW9uICgpCiAgICB7DQogICAgICAgIGlmKHRoaXMuc3RkZXJyLnN0ciAhPSAnJykNCiAgICAgICAgew0KICAgICAgICAgICAgdGhpcy5yZXQuX3Jlaih0aGlzLnN0ZGVyci5zdHIudHJpbSgpKTsNCiAgICAgICAgfQ0KICAgICAgICBlbHNlDQogICAgICAgIHsNCiAgICAgICAgICAgIHRoaXMucmV0Ll9yZXMoKTsNCiAgICAgICAgfQ0KICAgIH0pOwogICAgcmV0dXJuIChyZXQpOw0KfQoKZnVuY3Rpb24gYWRkRmlyZXdhbGxSdWxlKG9wdGlvbnMpCnsKICAgIHZhciBjb21tYW5kID0gJ05ldy1OZXRGaXJld2FsbFJ1bGUnOwogICAgdmFyIHZhbCA9IGNvbnZlcnRPcHRpb25zKG9wdGlvbnMpOwogICAgdmFyIGtleTsKCiAgICBmb3IgKGtleSBpbiB2YWwpCiAgICB7CiAgICAgICAgaWYgKHZhbFtrZXldLnRvU3RyaW5nKCkuaW5kZXhPZignICcpID49IDApCiAgICAgICAgewogICAgICAgICAgICBjb21tYW5kICs9ICgnIC0nICsga2V5ICsgJyBcXCInICsgdmFsW2tleV0gKyAnXFwiJyk7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIGNvbW1hbmQgKz0gKCcgLScgKyBrZXkgKyAnICcgKyB2YWxba2V5XSArICcnKTsKICAgICAgICB9CiAgICB9CgogICAgdmFyIGNoaWxkOwogICAgaWYgKHJlcXVpcmUoJ29zJykuYXJjaCgpID09ICd4NjQnKQogICAgewogICAgICAgIGNoaWxkID0gcmVxdWlyZSgnY2hpbGRfcHJvY2VzcycpLmV4ZWNGaWxlKHByb2Nlc3MuZW52Wyd3aW5kaXInXSArICdcXFN5c3RlbTMyXFxXaW5kb3dzUG93ZXJTaGVsbFxcdjEuMFxccG93ZXJzaGVsbC5leGUnLCBbJy9DICInICsgY29tbWFuZCArICciJ10pOwogICAgfQogICAgZWxzZQogICAgewogICAgICAgIGNoaWxkID0gcmVxdWlyZSgnY2hpbGRfcHJvY2VzcycpLmV4ZWNGaWxlKHByb2Nlc3MuZW52Wyd3aW5kaXInXSArICdcXFN5c3RlbTMyXFxXaW5kb3dzUG93ZXJTaGVsbFxcdjEuMFxccG93ZXJzaGVsbC5leGUnLCBbJy9DICInICsgY29tbWFuZCArICciJ10pOwogICAgfQoKICAgIGNoaWxkLnN0ZG91dC5zdHIgPSAnJzsgY2hpbGQuc3Rkb3V0Lm9uKCdkYXRhJywgZnVuY3Rpb24gKGMpIHsgdGhpcy5zdHIgKz0gYy50b1N0cmluZygpOyB9KTsKICAgIGNoaWxkLnN0ZGVyci5zdHIgPSAnJzsgY2hpbGQuc3RkZXJyLm9uKCdkYXRhJywgZnVuY3Rpb24gKGMpIHsgdGhpcy5zdHIgKz0gYy50b1N0cmluZygpOyB9KTsKICAgIGNoaWxkLndhaXRFeGl0KCk7CgogICAgaWYoY2hpbGQuc3RkZXJyLnN0ci50cmltKCkgIT0gJycpCiAgICB7CiAgICAgICAgdGhyb3cgKGNoaWxkLnN0ZGVyci5zdHIudHJpbSgpKTsKICAgIH0KfQoKCm1vZHVsZS5leHBvcnRzID0KICAgIHsKICAgICAgICBnZXRGaXJld2FsbFJ1bGVzOiBnZXRGaXJld2FsbFJ1bGVzLAogICAgICAgIGRpc2FibGVGaXJld2FsbFJ1bGVzOiBkaXNhYmxlRmlyZXdhbGxSdWxlcywKICAgICAgICBlbmFibGVGaXJld2FsbFJ1bGVzOiBlbmFibGVGaXJld2FsbFJ1bGVzLAogICAgICAgIGFkZEZpcmV3YWxsUnVsZTogYWRkRmlyZXdhbGxSdWxlLAogICAgICAgIHJlbW92ZUZpcmV3YWxsUnVsZTogcmVtb3ZlRmlyZXdhbGxSdWxlCiAgICB9Owo=", 2164); + ILibBase64DecodeEx((unsigned char*)_winfirewall, 18164, (unsigned char*)_winfirewall + 18164); + duk_push_global_object(ctx); duk_get_prop_string(ctx, -1, "addModule"); duk_swap_top(ctx, -2); duk_push_string(ctx, "win-firewall"); duk_push_string(ctx, _winfirewall + 18164); + duk_pcall_method(ctx, 2); duk_pop(ctx); + free(_winfirewall); #endif #ifdef _FREEBSD diff --git a/modules/win-firewall.js b/modules/win-firewall.js index 436ecb8..6cd2f35 100644 --- a/modules/win-firewall.js +++ b/modules/win-firewall.js @@ -141,80 +141,94 @@ function getFirewallRules(options) function disableFirewallRules(options) { var ret = new promise(function (a, r) { this._res = a; this._rej = r; }); - var p = getFirewallRules(options).on('firewallRule', function (r) { if (this._count == null) { this._count = 0; } ++this._count; }); - p.options = options; - p.ret = ret; - p.then(function (a) + var command = 'Disable-NetFirewallRule'; + + if (options.program) { - if(this._count > 0) + command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command; + } + else + { + var key, value; + for (key in options) { - var command = 'Disable-NetFirewallRule'; - if (this.options.program) { command = 'Get-NetFirewallApplicationFilter -Program \\"' + this.options.program + '\\" | ' + command; } - - var child; - if (require('os').arch() == 'x64') - { - child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); - } - else - { - child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); - } - - 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.waitExit(); - - if (child.stderr.str.trim() != "") - { - this.ret._rej(child.stderr.str.trim()); - } - else - { - this.ret._res(); - } + value = options[key]; + if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; } + command += ('-' + key + ' ' + value); } - }, function (e) { this.ret._rej(e); }); + } + + if (require('os').arch() == 'x64') + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + else + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + + ret.child.ret = ret; + ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); }); + ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString(); }); + ret.child.on('exit', function () + { + if (this.stderr.str != '') + { + this.ret._rej(this.stderr.str.trim()); + } + else + { + this.ret._res(); + } + }); + return (ret); } function enableFirewallRules(options) { var ret = new promise(function (a, r) { this._res = a; this._rej = r; }); - var p = getFirewallRules(options).on('firewallRule', function (r) { if (this._count == null) { this._count = 0; } ++this._count; }); - p.options = options; - p.ret = ret; - p.then(function (a) + + var command = 'Enable-NetFirewallRule'; + if (options.program) { - if (this._count > 0) + command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command; + } + else + { + var key, value; + for (key in options) { - var command = 'Enable-NetFirewallRule'; - if (this.options.program) { command = 'Get-NetFirewallApplicationFilter -Program \\"' + this.options.program + '\\" | ' + command; } - - var child; - if (require('os').arch() == 'x64') - { - child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); - } - else - { - child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); - } - - 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.waitExit(); - - if (child.stderr.str.trim() != "") - { - this.ret._rej(child.stderr.str.trim()); - } - else - { - this.ret._res(); - } + value = options[key]; + if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; } + command += ('-' + key + ' ' + value); } - }, function (e) { this.ret._rej(e); }); + } + + if (require('os').arch() == 'x64') + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + else + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + + ret.child.ret = ret; + ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); }); + ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString(); }); + ret.child.on('exit', function () + { + if(this.stderr.str != '') + { + this.ret._rej(this.stderr.str.trim()); + } + else + { + this.ret._res(); + } + }); + return (ret); } @@ -299,12 +313,58 @@ function convertOptions(options) return (options); } +function removeFirewallRule(options) +{ + if (typeof (options) == 'string') { options = { Name: options }; } + var ret = new promise(function (a, r) { this._res = a; this._rej = r; }); + + var command = 'Remove-NetFirewallRule'; + if (options.program) + { + command = 'Get-NetFirewallApplicationFilter -Program \\"' + options.program + '\\" | ' + command; + } + else + { + var key, value; + for(key in options) + { + value = options[key]; + if (value.indexOf(' ') >= 0) { value = '\\"' + options[key] + '\\"'; } + command += ('-' + key + ' ' + value); + } + } + + if (require('os').arch() == 'x64') + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + else + { + ret.child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); + } + + ret.child.ret = ret; + ret.child.stdout.str = ''; ret.child.stdout.on('data', function (c) { this.str += c.toString(); }); + ret.child.stderr.str = ''; ret.child.stderr.on('data', function (c) { this.str += c.toString(); }); + ret.child.on('exit', function () + { + if(this.stderr.str != '') + { + this.ret._rej(this.stderr.str.trim()); + } + else + { + this.ret._res(); + } + }); + return (ret); +} + function addFirewallRule(options) { var command = 'New-NetFirewallRule'; var val = convertOptions(options); var key; - console.log(JSON.stringify(val, null, 1)); for (key in val) { @@ -318,9 +378,7 @@ function addFirewallRule(options) } } - console.log(command); var child; - if (require('os').arch() == 'x64') { child = require('child_process').execFile(process.env['windir'] + '\\System32\\WindowsPowerShell\\v1.0\\powershell.exe', ['/C "' + command + '"']); @@ -346,5 +404,6 @@ module.exports = getFirewallRules: getFirewallRules, disableFirewallRules: disableFirewallRules, enableFirewallRules: enableFirewallRules, - addFirewallRule: addFirewallRule + addFirewallRule: addFirewallRule, + removeFirewallRule: removeFirewallRule };