frpc-desktop/electron/api/frpc.ts

241 lines
6.9 KiB
TypeScript
Raw Normal View History

2023-12-01 15:16:16 +08:00
import {app, ipcMain, Notification} from "electron";
import {Config, getConfig} from "../storage/config";
import {listProxy} from "../storage/proxy";
import {getVersionById} from "../storage/version";
2023-12-01 14:17:14 +08:00
import treeKill from "tree-kill";
2023-11-27 15:03:25 +08:00
const fs = require("fs");
const path = require("path");
const {exec, spawn} = require("child_process");
2023-11-27 15:03:25 +08:00
export let frpcProcess = null;
const runningCmd = {
commandPath: null,
configPath: null
2023-11-27 15:03:25 +08:00
};
2023-12-01 15:16:16 +08:00
let frpcStatusListener = null;
2023-11-27 15:03:25 +08:00
/**
*
* @param versionId ID
* @param callback
*/
const getFrpcVersionWorkerPath = (
versionId: string,
callback: (workerPath: string) => void
2023-11-27 15:03:25 +08:00
) => {
getVersionById(versionId, (err2, version) => {
if (!err2) {
callback(version["frpcVersionPath"]);
}
});
2023-11-27 15:03:25 +08:00
};
/**
*
*/
export const generateConfig = (
config: Config,
callback: (configPath: string) => void
2023-11-27 15:03:25 +08:00
) => {
listProxy((err3, proxys) => {
if (!err3) {
const proxyToml = proxys.map(m => {
let toml = `
2023-11-27 15:03:25 +08:00
[[proxies]]
name = "${m.name}"
type = "${m.type}"
localIP = "${m.localIp}"
localPort = ${m.localPort}
`;
switch (m.type) {
case "tcp":
toml += `remotePort = ${m.remotePort}`;
break;
case "http":
case "https":
toml += `customDomains=[${m.customDomains.map(m => `"${m}"`)}]`;
break;
default:
break;
}
2023-11-27 15:03:25 +08:00
return toml;
});
let toml = `
2023-11-27 15:03:25 +08:00
serverAddr = "${config.serverAddr}"
serverPort = ${config.serverPort}
auth.method = "${config.authMethod}"
auth.token = "${config.authToken}"
log.to = "frpc.log"
log.level = "${config.logLevel}"
log.maxDays = ${config.logMaxDays}
2023-11-27 15:03:25 +08:00
webServer.addr = "127.0.0.1"
webServer.port = 57400
2023-12-01 14:17:14 +08:00
transport.tls.enable = ${config.tlsConfigEnable}
${config.tlsConfigEnable ? `
transport.tls.certFile = "${config.tlsConfigCertFile}"
transport.tls.keyFile = "${config.tlsConfigKeyFile}"
transport.tls.trustedCaFile = "${config.tlsConfigTrustedCaFile}"
transport.tls.serverName = "${config.tlsConfigServerName}"
` : ""}
2023-11-27 15:03:25 +08:00
2023-12-01 14:17:14 +08:00
${proxyToml.join("")}
2023-11-27 15:03:25 +08:00
`;
// const configPath = path.join("frp.toml");
const filename = "frp.toml";
2023-12-01 14:17:14 +08:00
const configPath = path.join(app.getPath("userData"), filename)
console.debug("生成配置成功", configPath)
fs.writeFile(
2023-12-01 14:17:14 +08:00
configPath, // 配置文件目录
toml, // 配置文件内容
{flag: "w"},
err => {
if (!err) {
callback(filename);
}
}
);
2023-11-27 15:03:25 +08:00
}
});
2023-11-27 15:03:25 +08:00
};
/**
* frpc子进程
* @param cwd
* @param commandPath
* @param configPath
*/
const startFrpcProcess = (commandPath: string, configPath: string) => {
const command = `${commandPath} -c ${configPath}`;
2023-12-01 14:17:14 +08:00
console.info("启动", command)
frpcProcess = spawn(command, {
cwd: app.getPath("userData"),
shell: true
});
runningCmd.commandPath = commandPath;
runningCmd.configPath = configPath;
frpcProcess.stdout.on("data", data => {
2023-12-01 14:17:14 +08:00
console.debug(`命令输出: ${data}`);
});
frpcProcess.stdout.on("error", data => {
2023-12-01 14:17:14 +08:00
console.log("启动错误", data)
stopFrpcProcess()
});
2023-12-01 15:16:16 +08:00
frpcStatusListener = setInterval(() => {
const status = frpcProcessStatus()
if (!status) {
console.log("连接已断开")
new Notification({
title: "Frpc Desktop",
body: "连接已断开,请前往日志查看原因"
}).show()
clearInterval(frpcStatusListener)
}
}, 3000)
2023-11-27 15:03:25 +08:00
};
2023-12-01 15:16:16 +08:00
/**
* frpc配置
*/
2023-11-27 15:03:25 +08:00
export const reloadFrpcProcess = () => {
if (frpcProcess && !frpcProcess.killed) {
getConfig((err1, config) => {
if (!err1) {
if (config) {
generateConfig(config, configPath => {
const command = `${runningCmd.commandPath} reload -c ${configPath}`;
2023-12-01 14:17:14 +08:00
console.info("重启", command);
exec(command, {
cwd: app.getPath("userData"),
shell: true
});
});
}
}
});
}
2023-11-27 15:03:25 +08:00
};
2023-12-01 15:16:16 +08:00
/**
* frpc子进程
*/
export const stopFrpcProcess = (callback?:() => void) => {
2023-12-01 14:17:14 +08:00
if (frpcProcess) {
treeKill(frpcProcess.pid, (error: Error) => {
if (error) {
console.log("关闭失败", frpcProcess.pid, error)
} else {
console.log('关闭成功')
2023-12-01 14:17:14 +08:00
frpcProcess = null
2023-12-01 15:16:16 +08:00
clearInterval(frpcStatusListener)
2023-12-01 14:17:14 +08:00
}
callback()
2023-12-01 14:17:14 +08:00
})
}
}
2023-12-01 15:16:16 +08:00
/**
* frpc子进程状态
*/
export const frpcProcessStatus = () => {
if (!frpcProcess) {
return false;
}
try {
// 发送信号给进程,如果进程存在,会正常返回
process.kill(frpcProcess.pid, 0);
return true;
} catch (error) {
// 进程不存在,抛出异常
return false;
}
}
2023-11-27 15:03:25 +08:00
export const initFrpcApi = () => {
ipcMain.handle("frpc.running", async (event, args) => {
2023-12-01 15:16:16 +08:00
return frpcProcessStatus()
});
2023-11-27 15:03:25 +08:00
ipcMain.on("frpc.start", async (event, args) => {
getConfig((err1, config) => {
if (!err1) {
if (config) {
getFrpcVersionWorkerPath(
config.currentVersion,
(frpcVersionPath: string) => {
generateConfig(config, configPath => {
const platform = process.platform;
if (platform === 'win32') {
startFrpcProcess(
path.join(frpcVersionPath, "frpc.exe"),
configPath
);
} else {
startFrpcProcess(
path.join(frpcVersionPath, "frpc"),
configPath
);
}
});
}
);
} else {
event.reply(
"Home.frpc.start.error.hook",
"请先前往设置页面,修改配置后再启动"
);
2023-11-27 17:54:09 +08:00
}
2023-11-27 15:03:25 +08:00
}
});
2023-11-27 15:03:25 +08:00
});
ipcMain.on("frpc.stop", () => {
if (frpcProcess && !frpcProcess.killed) {
2023-12-01 14:17:14 +08:00
stopFrpcProcess()
}
});
2023-11-27 15:03:25 +08:00
};