Add electron packaging for single exe server distribution

This commit is contained in:
2026-03-13 00:27:36 +08:00
parent decba25a08
commit 0d84793a48
2 changed files with 115 additions and 53 deletions

View File

@@ -1,68 +1,107 @@
const { app, BrowserWindow } = require('electron');
const { app, Tray, Menu, nativeImage } = require('electron');
const path = require('path');
const fs = require('fs');
const { spawn } = require('child_process');
let mainWindow = null;
let serverProcess = null;
let tray = null;
function startServer() {
console.log('Starting OpenChamber server...');
serverProcess = spawn('bun', ['run', 'start:web'], {
cwd: path.join(__dirname, '..'),
stdio: 'inherit',
shell: true,
detached: false
});
let serverPath;
let cwdPath;
let execPath;
serverProcess.on('error', (err) => {
console.error('Server error:', err);
});
if (app.isPackaged) {
serverPath = path.join(process.resourcesPath, 'server', 'index.js');
cwdPath = path.join(process.resourcesPath);
execPath = path.join(process.resourcesPath, 'nodejs', 'node.exe');
} else {
serverPath = path.join(__dirname, '..', 'server', 'index.js');
cwdPath = path.join(__dirname, '..');
execPath = 'node';
}
serverProcess.on('exit', (code) => {
console.log('Server exited with code:', code);
});
console.log('Server path:', serverPath);
console.log('CWD:', cwdPath);
console.log('Exec:', execPath);
const env = { ...process.env };
env.OPENCHAMBER_PORT = '3000';
env.OPENCODE_SKIP_START = 'true';
// 创建批处理脚本
const batContent = `"${execPath}" "${serverPath}"`;
const batPath = path.join(cwdPath, 'start-server.bat');
fs.writeFileSync(batPath, batContent);
console.log('Bat path:', batPath);
try {
const child = spawn('cmd.exe', ['/c', batPath], {
cwd: cwdPath,
stdio: 'inherit',
env: env,
detached: false
});
child.on('error', (err) => {
console.error('Server error:', err);
});
child.on('exit', (code) => {
console.log('Server exited:', code);
});
} catch (e) {
console.error('Failed to start:', e);
}
}
function createWindow() {
mainWindow = new BrowserWindow({
width: 800,
height: 600,
show: false,
webPreferences: {
nodeIntegration: false,
contextIsolation: true
function createTray() {
try {
let iconPath;
if (app.isPackaged) {
iconPath = path.join(process.resourcesPath, 'public', 'favicon.png');
} else {
iconPath = path.join(__dirname, '..', 'public', 'favicon.png');
}
});
mainWindow.loadURL('http://localhost:3000');
let icon;
try {
icon = nativeImage.createFromPath(iconPath);
if (icon.isEmpty()) {
icon = nativeImage.createEmpty();
}
} catch (e) {
icon = nativeImage.createEmpty();
}
mainWindow.once('ready-to-show', () => {
mainWindow.show();
});
tray = new Tray(icon);
mainWindow.on('closed', () => {
mainWindow = null;
});
const contextMenu = Menu.buildFromTemplate([
{ label: 'OpenChamber Server', enabled: false },
{ type: 'separator' },
{ label: 'Open http://localhost:3000', click: () => {
require('electron').shell.openExternal('http://localhost:3000');
}},
{ type: 'separator' },
{ label: 'Quit', click: () => {
app.quit();
}}
]);
tray.setToolTip('OpenChamber Server');
tray.setContextMenu(contextMenu);
} catch (e) {
console.error('Failed to create tray:', e);
}
}
app.whenReady().then(() => {
startServer();
setTimeout(() => {
createWindow();
}, 3000);
createTray();
});
app.on('window-all-closed', () => {
if (serverProcess) {
serverProcess.kill();
}
app.quit();
});
app.on('before-quit', () => {
if (serverProcess) {
serverProcess.kill();
}
});

View File

@@ -17,12 +17,35 @@
},
"files": [
"main.js",
"../dist/**/*",
"../server/**/*",
"../bin/**/*",
"../public/**/*",
"package.json"
],
"extraResources": [
{
"from": "../dist",
"to": "dist",
"filter": ["**/*"]
},
{
"from": "../server",
"to": "server",
"filter": ["**/*"]
},
{
"from": "../bin",
"to": "bin",
"filter": ["**/*"]
},
{
"from": "../public",
"to": "public",
"filter": ["**/*"]
},
{
"from": "D:\\Program Files\\nodejs",
"to": "nodejs",
"filter": ["node.exe"]
}
],
"win": {
"target": "portable"
},