背景
之前写了一篇Electron通信的方式,讲述了一下三者之间的通信机制,比较恶心,后来发现有个@electron/remote,
可以让渲染进程直接访问主进程,但遗憾的是,preload.js无法使用这个库,webview更是不可能直接触达主进程,总要通过preload.js绕一圈才能找到渲染进程。为此,我们需要一个新的方式,来解决所有通信链路的问题。
思路
1. 就像vuex的诞生一样,没有vuex的时候,只能通过父子通信,后来层级太多,有了子子孙孙,一层层汇报显得不可能,同理多webview,多子进程,多渲染进程相互之间如何通信,只要有个中转站即可。
2. 上文已经写明,通过websocket对所有节点进行互联,无论你是子进程(用可执行文件创建的子进程,本身可以发送fetch或request请求,甚至有些可以实现websocket连接),渲染进程和webview自然支持websocket,但是需要修改CSP,CSP怎么修改,我们留在后面的内容讲解,https中访问http需要去除安全机制。
具体实现
消息中转websocket服务
1. 主进程实现websocket服务, `"ws": "6.2.2",` 这个版本比较稳,其他版本会报错,这个大致结构已经给了,就是每个接入的client都要带着自己的名字进行握手,一般通过url的path进行获取,不起名字后续无法实现消息中转。
const WebSocket = require('ws')
let isBusy = 0
const clients = new Map()
function wssInit() {
// 创建 WebSocket 服务器
const wss = new WebSocket.Server({ port: 59296 })
wss.on('listening', () => {
console.log('WebSocket Server start')
})
wss.on('error', (error) => {
console.error('WebSocket Server Error:', error.message)
// 服务器发生错误时,关闭服务器并重启
wss.close(() => {
setTimeout(wssInit, 1000) // 设置1秒的延迟,避免过于频繁的重启
})
})
// 当有新的 WebSocket 连接时
wss.on('connection', (ws) => {
console.log('WebSocket connection established with Webview')
const clientId = getClientName()
clients.set(clientId, ws)
// 监听从 Webview 收到的消息
ws.on('message', (message) => {
})
// 连接关闭时的处理
ws.on('close', () => {
clients.delete(clientId)
console.log('WebSocket connection with Webview closed')
});
})
// 监听服务器关闭事件,并自动重启
wss.on('close', () => {
console.log('WebSocket Server closed')
setTimeout(wssInit, 1000) // 设置1秒的延迟,避免过于频繁的重启
})
}
本地数据CURD的Http服务
可以引入Express框架,实现本地化的Http服务,这样所有的客户端都可以直接操作数据库,而不必绕一大堆通信
这样渲染进程就可以访问主进程中的sqlite,且还可以阻塞式请求,相当Nice,通信链路大大降低,开发复杂度大大降低,产出bug率大大降低
未来展望
1. websocket & http 服务可以进行合并为一体,通过python的fastapi框架,可直接打包成一个可执行文件,windows下就是一个大概6M左右的exe文件,直接通过子进程运行,会将大量代码从主进程抽离,来减轻主进程未知异常和内存溢出的问题,说实话Electron做得是真柔弱,一不小心就内存溢出,崩溃给你看,对于纯前端开发,上手容易,深入太难!