fix windows desktop bridge startup (#1196)

This commit is contained in:
ekko
2026-06-01 11:30:04 +08:00
committed by GitHub
parent 4516502705
commit 15d358f602
4 changed files with 118 additions and 9 deletions
+9 -3
View File
@@ -28,16 +28,22 @@ function quitApp() {
app.quit()
}
function loginItemOptions() {
return {
path: process.execPath,
args: ['--hidden'],
}
}
function getOpenAtLogin(): boolean {
return app.getLoginItemSettings().openAtLogin
return app.getLoginItemSettings(loginItemOptions()).openAtLogin
}
function setOpenAtLogin(openAtLogin: boolean) {
app.setLoginItemSettings({
...loginItemOptions(),
openAtLogin,
openAsHidden: true,
path: process.execPath,
args: ['--hidden'],
})
}
+7 -6
View File
@@ -228,14 +228,12 @@ export async function startWebUiServer(port = DEFAULT_PORT): Promise<string> {
// setup is a #!/bin/sh wrapper, not a python interpreter, so detection
// resolves to /bin/sh and the bridge crashes (exit code 2) immediately.
const isWin = process.platform === 'win32'
const bundledPython = isWin
? join(pythonDir(), 'python.exe')
: join(pythonDir(), 'bin', 'python3')
const bundledPythonNoWindow = isWin
? join(pythonDir(), 'pythonw.exe')
: join(pythonDir(), 'bin', 'python3')
const bundledPython = isWin && existsSync(bundledPythonNoWindow)
? bundledPythonNoWindow
: isWin
? join(pythonDir(), 'python.exe')
: join(pythonDir(), 'bin', 'python3')
: bundledPython
const bridgePort = await getFreeTcpPort()
const workerPortBase = await getFreeTcpPortInRange(20000, 59000)
const loginShellPath = await getLoginShellPath()
@@ -255,6 +253,9 @@ export async function startWebUiServer(port = DEFAULT_PORT): Promise<string> {
NODE_ENV: 'production',
HERMES_DESKTOP: 'true',
HERMES_BIN: hermesBin(),
// The bridge and its per-profile workers need working stdout/stderr for
// ready handshakes. Use python.exe on Windows and hide windows at the
// process creation layer instead of switching the bridge to pythonw.exe.
HERMES_AGENT_BRIDGE_PYTHON: bundledPython,
HERMES_AGENT_CLI_PYTHON: existsSync(bundledPythonNoWindow) ? bundledPythonNoWindow : bundledPython,
HERMES_AGENT_ROOT: pythonDir(),
@@ -71,6 +71,40 @@ def _hidden_subprocess_kwargs() -> dict[str, int]:
return {"creationflags": getattr(subprocess, "CREATE_NO_WINDOW", 0)}
def _install_windows_hidden_subprocess_defaults() -> None:
"""Hide console windows for subprocesses launched inside desktop bridge runs.
The desktop bridge itself must keep stdout/stderr pipes for readiness and
worker handshakes, so it runs under python.exe. On Windows that means any
nested console executable, including git.exe from context expansion, can
flash a window unless the child process is created with CREATE_NO_WINDOW.
"""
if os.name != "nt":
return
if os.environ.get("HERMES_DESKTOP", "").strip().lower() != "true":
return
if getattr(subprocess, "_hermes_hidden_defaults_installed", False):
return
original_popen = subprocess.Popen
create_no_window = getattr(subprocess, "CREATE_NO_WINDOW", 0) or 0x08000000
class HiddenPopen(original_popen): # type: ignore[misc, valid-type]
def __init__(self, *args: Any, **kwargs: Any) -> None:
flags = kwargs.get("creationflags", 0) or 0
try:
kwargs["creationflags"] = int(flags) | create_no_window
except Exception:
kwargs["creationflags"] = create_no_window
super().__init__(*args, **kwargs)
subprocess.Popen = HiddenPopen # type: ignore[assignment]
subprocess._hermes_hidden_defaults_installed = True # type: ignore[attr-defined]
_install_windows_hidden_subprocess_defaults()
def _process_exists(pid: int) -> bool:
if pid <= 0:
return False