mirror of
https://gitee.com/dromara/electron-egg.git
synced 2026-05-14 19:52:10 +08:00
addon java-server
This commit is contained in:
81
electron/addon/javaServer/index.js
Normal file
81
electron/addon/javaServer/index.js
Normal file
@@ -0,0 +1,81 @@
|
||||
const getPort = require('get-port');
|
||||
const server = require("./server");
|
||||
const electronApp = require('electron').app;
|
||||
|
||||
/**
|
||||
* java server插件
|
||||
* @class
|
||||
*/
|
||||
class JavaServerAddon {
|
||||
|
||||
constructor(app) {
|
||||
this.app = app;
|
||||
this.cfg = app.config.addons.javaServer;
|
||||
this.javaServer = null;
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建java服务
|
||||
*
|
||||
* @function
|
||||
* @since 1.0.0
|
||||
*/
|
||||
async createServer () {
|
||||
await this.createJavaPorts();
|
||||
|
||||
this.javaServer = new server(this.app);
|
||||
await this.javaServer.create();
|
||||
|
||||
// kill
|
||||
electronApp.on("before-quit", async () => {
|
||||
this.app.logger.info("[addon:javaServer] before-quit: kill-----------");
|
||||
await this.javaServer.kill();
|
||||
});
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
/**
|
||||
* todo 检查服务是否启动
|
||||
*
|
||||
* @function
|
||||
* @since 1.0.0
|
||||
*/
|
||||
async check () {
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建服务端口
|
||||
*
|
||||
* @function
|
||||
* @since 1.0.0
|
||||
*/
|
||||
async createJavaPorts() {
|
||||
if (!this.cfg.enable) {
|
||||
return;
|
||||
}
|
||||
const javaPort = await getPort({ port: this.cfg.port });
|
||||
process.env.EE_JAVA_PORT = javaPort;
|
||||
this.cfg.port = javaPort;
|
||||
|
||||
// 更新config配置
|
||||
this.app.getCoreDB().setItem("config", this.app.config);
|
||||
}
|
||||
|
||||
/**
|
||||
* 杀掉进程
|
||||
*
|
||||
* @function
|
||||
* @since 1.0.0
|
||||
*/
|
||||
async kill() {
|
||||
if (!this.cfg.enable) {
|
||||
return;
|
||||
}
|
||||
await this.javaServer.kill();
|
||||
}
|
||||
}
|
||||
|
||||
JavaServerAddon.toString = () => '[class JavaServerAddon]';
|
||||
module.exports = JavaServerAddon;
|
||||
257
electron/addon/javaServer/ps.js
Normal file
257
electron/addon/javaServer/ps.js
Normal file
@@ -0,0 +1,257 @@
|
||||
var ChildProcess = require("child_process");
|
||||
var IS_WIN = process.platform === "win32";
|
||||
var TableParser = require("table-parser");
|
||||
/**
|
||||
* End of line.
|
||||
* Basically, the EOL should be:
|
||||
* - windows: \r\n
|
||||
* - *nix: \n
|
||||
* But i'm trying to get every possibilities covered.
|
||||
*/
|
||||
var EOL = /(\r\n)|(\n\r)|\n|\r/;
|
||||
var SystemEOL = require("os").EOL;
|
||||
|
||||
/**
|
||||
* Execute child process
|
||||
* @type {Function}
|
||||
* @param {String[]} args
|
||||
* @param {String} where
|
||||
* @param {Function} callback
|
||||
* @param {Object=null} callback.err
|
||||
* @param {Object[]} callback.stdout
|
||||
*/
|
||||
var Exec = function (args, where) {
|
||||
var spawnSync = ChildProcess.spawnSync;
|
||||
var execSync = ChildProcess.execSync;
|
||||
|
||||
// on windows, if use ChildProcess.exec(`wmic process get`), the stdout will gives you nothing
|
||||
// that's why I use `cmd` instead
|
||||
if (IS_WIN) {
|
||||
const cmd = `wmic process where ${where} get ProcessId,ParentProcessId,CommandLine \n`;
|
||||
const result = execSync(cmd);
|
||||
if (!result) {
|
||||
throw new Error(result);
|
||||
}
|
||||
|
||||
var stdout = result.toString();
|
||||
|
||||
var beginRow;
|
||||
stdout = stdout.split(EOL);
|
||||
|
||||
// Find the line index for the titles
|
||||
stdout.forEach(function (out, index) {
|
||||
if (
|
||||
out &&
|
||||
typeof beginRow == "undefined" &&
|
||||
out.indexOf("CommandLine") === 0
|
||||
) {
|
||||
beginRow = index;
|
||||
}
|
||||
});
|
||||
|
||||
// get rid of the start (copyright) and the end (current pwd)
|
||||
stdout.splice(stdout.length - 1, 1);
|
||||
stdout.splice(0, beginRow);
|
||||
|
||||
return stdout.join(SystemEOL) || false;
|
||||
} else {
|
||||
if (typeof args === "string") {
|
||||
args = args.split(/\s+/);
|
||||
}
|
||||
const result = spawnSync("ps", args);
|
||||
if (result.stderr && !!result.stderr.toString()) {
|
||||
throw new Error(result.stderr);
|
||||
} else {
|
||||
return result.stdout.toString();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Query Process: Focus on pid & cmd
|
||||
* @param query
|
||||
* @param {String|String[]} query.pid
|
||||
* @param {String} query.command RegExp String
|
||||
* @param {String} query.arguments RegExp String
|
||||
* @param {String|array} query.psargs
|
||||
* @param {String|array} query.where where 条件
|
||||
* @param {Function} callback
|
||||
* @param {Object=null} callback.err
|
||||
* @param {Object[]} callback.processList
|
||||
* @return {Object}
|
||||
*/
|
||||
|
||||
exports.lookup = function (query) {
|
||||
/**
|
||||
* add 'lx' as default ps arguments, since the default ps output in linux like "ubuntu", wont include command arguments
|
||||
*/
|
||||
var exeArgs = query.psargs || ["lx"];
|
||||
var where = query.where || 'name="javaw.exe"';
|
||||
var filter = {};
|
||||
var idList;
|
||||
|
||||
// Lookup by PID
|
||||
if (query.pid) {
|
||||
if (Array.isArray(query.pid)) {
|
||||
idList = query.pid;
|
||||
} else {
|
||||
idList = [query.pid];
|
||||
}
|
||||
|
||||
// Cast all PIDs as Strings
|
||||
idList = idList.map(function (v) {
|
||||
return String(v);
|
||||
});
|
||||
}
|
||||
|
||||
if (query.command) {
|
||||
filter["command"] = new RegExp(query.command, "i");
|
||||
}
|
||||
|
||||
if (query.arguments) {
|
||||
filter["arguments"] = new RegExp(query.arguments, "i");
|
||||
}
|
||||
|
||||
if (query.ppid) {
|
||||
filter["ppid"] = new RegExp(query.ppid);
|
||||
}
|
||||
|
||||
const result = Exec(exeArgs, where);
|
||||
|
||||
var processList = parseGrid(result);
|
||||
var resultList = [];
|
||||
|
||||
processList.forEach(function (p) {
|
||||
var flt;
|
||||
var type;
|
||||
var result = true;
|
||||
|
||||
if (idList && idList.indexOf(String(p.pid)) < 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (type in filter) {
|
||||
flt = filter[type];
|
||||
result = flt.test(p[type]) ? result : false;
|
||||
}
|
||||
|
||||
if (result) {
|
||||
resultList.push(p);
|
||||
}
|
||||
});
|
||||
|
||||
return resultList;
|
||||
};
|
||||
|
||||
/**
|
||||
* Kill process
|
||||
* @param pid
|
||||
* @param {Object|String} signal
|
||||
* @param {String} signal.signal
|
||||
* @param {number} signal.timeout
|
||||
* @param next
|
||||
*/
|
||||
|
||||
exports.kill = function (pid, signal, next) {
|
||||
//opts are optional
|
||||
if (arguments.length == 2 && typeof signal == "function") {
|
||||
next = signal;
|
||||
signal = undefined;
|
||||
}
|
||||
|
||||
var checkTimeoutSeconds = (signal && signal.timeout) || 30;
|
||||
|
||||
if (typeof signal === "object") {
|
||||
signal = signal.signal;
|
||||
}
|
||||
|
||||
try {
|
||||
process.kill(pid, signal);
|
||||
} catch (e) {
|
||||
return next && next(e);
|
||||
}
|
||||
|
||||
var checkConfident = 0;
|
||||
var checkTimeoutTimer = null;
|
||||
var checkIsTimeout = false;
|
||||
|
||||
function checkKilled(finishCallback) {
|
||||
exports.lookup({ pid: pid }, function (err, list) {
|
||||
if (checkIsTimeout) return;
|
||||
|
||||
if (err) {
|
||||
clearTimeout(checkTimeoutTimer);
|
||||
finishCallback && finishCallback(err);
|
||||
} else if (list.length > 0) {
|
||||
checkConfident = checkConfident - 1 || 0;
|
||||
checkKilled(finishCallback);
|
||||
} else {
|
||||
checkConfident++;
|
||||
if (checkConfident === 5) {
|
||||
clearTimeout(checkTimeoutTimer);
|
||||
finishCallback && finishCallback();
|
||||
} else {
|
||||
checkKilled(finishCallback);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
next && checkKilled(next);
|
||||
|
||||
checkTimeoutTimer =
|
||||
next &&
|
||||
setTimeout(function () {
|
||||
checkIsTimeout = true;
|
||||
next(new Error("Kill process timeout"));
|
||||
}, checkTimeoutSeconds * 1000);
|
||||
};
|
||||
|
||||
/**
|
||||
* Parse the stdout into readable object.
|
||||
* @param {String} output
|
||||
*/
|
||||
|
||||
function parseGrid(output) {
|
||||
if (!output) {
|
||||
return [];
|
||||
}
|
||||
return formatOutput(TableParser.parse(output));
|
||||
}
|
||||
|
||||
/**
|
||||
* format the structure, extract pid, command, arguments, ppid
|
||||
* @param data
|
||||
* @return {Array}
|
||||
*/
|
||||
|
||||
function formatOutput(data) {
|
||||
var formatedData = [];
|
||||
data.forEach(function (d) {
|
||||
var pid =
|
||||
(d.PID && d.PID[0]) || (d.ProcessId && d.ProcessId[0]) || undefined;
|
||||
var cmd = d.CMD || d.CommandLine || d.COMMAND || undefined;
|
||||
var ppid =
|
||||
(d.PPID && d.PPID[0]) ||
|
||||
(d.ParentProcessId && d.ParentProcessId[0]) ||
|
||||
undefined;
|
||||
|
||||
if (pid && cmd) {
|
||||
var command = cmd[0];
|
||||
var args = "";
|
||||
|
||||
if (cmd.length > 1) {
|
||||
args = cmd.slice(1);
|
||||
}
|
||||
|
||||
formatedData.push({
|
||||
pid: pid,
|
||||
command: command,
|
||||
arguments: args,
|
||||
ppid: ppid,
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
return formatedData;
|
||||
}
|
||||
99
electron/addon/javaServer/server.js
Normal file
99
electron/addon/javaServer/server.js
Normal file
@@ -0,0 +1,99 @@
|
||||
const _ = require("lodash");
|
||||
const assert = require("assert");
|
||||
const fs = require("fs");
|
||||
const is = require('electron-is');
|
||||
const path = require("path");
|
||||
const { exec, execSync } = require("child_process");
|
||||
const Utils = require("ee-core").Utils;
|
||||
const ps = require("./ps");
|
||||
|
||||
/**
|
||||
* java server
|
||||
*/
|
||||
class JavaServer {
|
||||
constructor (app) {
|
||||
this.app = app;
|
||||
this.options = app.config.addons.javaServer;
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建服务
|
||||
*/
|
||||
async create () {
|
||||
if (!this.options.enable) {
|
||||
return;
|
||||
}
|
||||
|
||||
let port = process.env.EE_JAVA_PORT ? parseInt(process.env.EE_JAVA_PORT) : parseInt(this.options.port);
|
||||
assert(typeof port === "number", "java port required, and must be a number");
|
||||
|
||||
try {
|
||||
const jarName = this.options.name;
|
||||
let softwarePath = path.join(Utils.getExtraResourcesDir(), jarName);
|
||||
let javaOptStr = this.options.opt;
|
||||
let jrePath = path.join(Utils.getExtraResourcesDir(), this.options.jreVersion);
|
||||
let cmdStr = '';
|
||||
|
||||
this.app.console.info("[addon:javaServer] jar file path:", softwarePath);
|
||||
if (!fs.existsSync(softwarePath)) throw new Error('java program does not exist');
|
||||
|
||||
// 替换opt参数
|
||||
javaOptStr = _.replace(javaOptStr, "${port}", port);
|
||||
javaOptStr = _.replace(javaOptStr, "${path}", Utils.getLogDir());
|
||||
|
||||
if (is.windows()) {
|
||||
jrePath = path.join(jrePath, "bin", "javaw.exe");
|
||||
cmdStr = `start ${jrePath} -jar ${javaOptStr} ${softwarePath}`;
|
||||
} else if (is.macOS()) {
|
||||
// 如果提示:不受信任,请执行: sudo spctl --master-disable
|
||||
jrePath = path.join(jrePath, "Contents", "Home", "bin", "java");
|
||||
//cmdStr = `nohup ${jrePath} -jar ${javaOptStr} ${softwarePath} >/dev/null 2>&1 &`;
|
||||
cmdStr = `${jrePath} -jar ${javaOptStr} ${softwarePath}`;
|
||||
} else {
|
||||
// todo linux
|
||||
}
|
||||
|
||||
this.app.logger.info("[addon:javaServer] cmdStr:", cmdStr);
|
||||
exec(cmdStr);
|
||||
|
||||
} catch (err) {
|
||||
this.app.logger.error('[addon:javaServer] throw error:', err);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 关闭服务
|
||||
*/
|
||||
async kill () {
|
||||
const jarName = this.options.name;
|
||||
if (is.windows()) {
|
||||
const resultList = ps.lookup({
|
||||
command: "java",
|
||||
where: 'caption="javaw.exe"',
|
||||
arguments: jarName,
|
||||
});
|
||||
|
||||
//this.app.logger.info("[addon:javaServer] resultList:", resultList);
|
||||
resultList.forEach((item) => {
|
||||
ps.kill(item.pid, "SIGKILL", (err) => {
|
||||
if (err) {
|
||||
throw new Error(err);
|
||||
}
|
||||
console.info("[addon:javaServer] java程序退出 pid: ", item.pid);
|
||||
});
|
||||
});
|
||||
|
||||
// const cmd = `for /f "tokens=1-5" %i in ('netstat -ano ^| findstr ":${port}"') do taskkill /F /T /PID %m`;
|
||||
// const a = await execSync(cmd, {encoding: 'utf-8'});
|
||||
// app.logger.info("[javaServer] kill:", a);
|
||||
} else if (is.macOS()) {
|
||||
const cmd = `ps -ef | grep java | grep ${jarName} | grep -v grep | awk '{print $2}' | xargs kill -9`;
|
||||
const result = await execSync(cmd);
|
||||
this.app.logger.info("[addon:javaServer] kill:", result != null ? result.toString(): '');
|
||||
} else {
|
||||
// todo linux
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = JavaServer;
|
||||
Reference in New Issue
Block a user