lib/agent/control-panel/websockets/server.js
/* eslint-disable consistent-return */
const os = require('os');
const http = require('http');
const { exec } = require('child_process');
const websocket = require('./index');
const commands = require('../../commands');
const common = require('../../common');
const geo = require('../../providers/geo');
const logger = common.logger.prefix('server');
const gte = common.helpers.is_greater_or_equal;
let app;
const osName = os.platform().replace('darwin', 'mac').replace('win32', 'windows');
// eslint-disable-next-line consistent-return
const checkService = (cb) => {
if (osName === 'mac') return cb(true);
// eslint-disable-next-line global-require
const sysWin = require('../../../system/windows');
sysWin.get_winsvc_version((err, serviceVersion) => {
if (err) return cb(false);
if (serviceVersion && gte(serviceVersion, '1.2.0')) return cb(true);
cb(false);
});
};
const serverCommands = {
mac: {
scan: 'lsof -Pi :7738 | sed -n \'2 p\'| awk \'{print $2}\'',
kill: 'kill -9 ',
},
windows: {
scan: 'for /F "tokens=1,2,3,4,5" %A in (\'"netstat -ano | find "LISTENING" | find "127.0.0.1:7738""\') DO @echo %~E',
kill: 'taskkill /F /PID ',
},
};
const checkServerDown = (cb) => {
exec(serverCommands[osName].scan, (err, pid) => {
if (err || !pid) return cb(new Error('No other local server found, we are clear.'));
if (pid) {
logger.info('Localhost server on same port found.. killing it!');
exec(serverCommands[osName].kill + pid, cb);
} else return cb();
});
};
const createServer = (cb) => {
if (osName === 'linux') return cb(new Error('Service only available for mac and windows'));
checkService((valid) => {
if (!valid) return cb(new Error('Windows Service not compatible'));
checkServerDown((err) => {
if (err) logger.info(err.message);
app = http.createServer((req, res) => {
if (req.url.includes('/healthz')) {
// Change default location strategy if native is allowed on mac
// eslint-disable-next-line no-unused-expressions
if (osName === 'mac') req.url.includes('location=true') ? geo.set_default('native') : geo.set_default('wifi');
const isOk = websocket.check_timestamp();
if (isOk) {
res.writeHead(200);
} else {
logger.info('Client not responding, restarting...');
res.writeHead(400);
}
res.end();
}
// handles the information sent by macos app (location)
// but can be extended to new permissions
// example of required payload
// {
// "os": "macos",
// "permission": "location",
// "status": "rejected"
// }
if (req.url.includes('/permission')) {
logger.debug('permission status received');
req.on('data', (chunk) => {
const data = JSON.parse(chunk.toString());
logger.debug(data.permission);
logger.debug(data.os);
logger.debug(data.status);
if (data.status.localeCompare('Authorized') === 0 && data.permission.localeCompare('location') === 0) {
geo.get_location(() => {});
}
});
req.on('end', () => {
logger.debug('permission status finished');
});
res.writeHead(200);
res.end();
}
if (req.url.includes('/actions')) {
logger.debug('action command received');
req.on('data', (chunk) => {
logger.debug(`chunk is: ${chunk}`);
let parsedData;
try {
parsedData = JSON.parse(chunk.toString());
} catch (e) {
logger.debug(`error exception: ${e}`);
return;
}
commands.perform(parsedData);
});
req.on('end', () => {
logger.debug('action command finished');
});
res.writeHead(200);
res.end();
}
});
app.on('error', (errorApp) => {
logger.info(`Error creating check server: ${errorApp.message}`);
return cb(errorApp);
});
app.listen(7738, '127.0.0.1');
if (cb) return cb(null, app);
});
});
};
exports.close = () => {
if (app) app.close();
};
exports.create_server = createServer;
exports.check_server_down = checkServerDown;
exports.check_service = checkService;