The dashboard's embedded Chat surface (/chat, /api/ws, /api/pty) was gated behind `hermes dashboard --tui` / HERMES_DASHBOARD_TUI=1. The desktop app and the dashboard's own Chat tab both drive the agent over the /api/ws + /api/pty WebSockets, so a dashboard started without the flag would pass the /api/status health check but slam the chat WebSocket shut with WS code 4403 — the app connects, reports "ready", and chat stays dead. This was the root cause behind multiple user reports of the desktop app failing to connect to a self-hosted gateway/dashboard, and it bit Docker and host installs alike. Make the embedded chat unconditional: - web_server.py: _DASHBOARD_EMBEDDED_CHAT_ENABLED defaults to True; drop the embedded_chat parameter and the runtime reassignment from start_server(). The WS gates still read the constant (now always true) so the seam — and its "rejects when disabled" contract test — stays meaningful. - main.py: remove the `--tui` argument from the dashboard subparser and the `embedded_chat = args.tui or HERMES_DASHBOARD_TUI==1` derivation. - web/: isDashboardEmbeddedChatEnabled() returns true unconditionally; drop the deprecated __HERMES_DASHBOARD_TUI__ alias and the dead LEGACY_TUI_RE scrape in the vite dev-token plugin. - apps/desktop/electron/main.cjs: drop `--tui` from the spawned dashboardArgs (it would now error with "unrecognized arguments: --tui") and the redundant HERMES_DASHBOARD_TUI env injection. - Docker: no s6 run-script change needed — the script never passed --tui; the HERMES_DASHBOARD_TUI env var is now simply a no-op, so the image works out of the box with no extra var. - Docs: remove every dashboard --tui / HERMES_DASHBOARD_TUI reference across the CLI reference, env-var reference, docker/desktop/web-dashboard guides, in-app tips, and the zh-Hans translations. The terminal `hermes --tui` / HERMES_TUI references are intentionally left untouched. Tests: 270 passing across web_server, dashboard lifecycle, host-header, auth-gate, and docker-override-scripts suites.
102 lines
3.3 KiB
TypeScript
102 lines
3.3 KiB
TypeScript
import { defineConfig, type Plugin } from "vite";
|
|
import react from "@vitejs/plugin-react";
|
|
import tailwindcss from "@tailwindcss/vite";
|
|
import path from "path";
|
|
|
|
const BACKEND = process.env.HERMES_DASHBOARD_URL ?? "http://127.0.0.1:9119";
|
|
|
|
/**
|
|
* In production the Python `hermes dashboard` server injects a one-shot
|
|
* session token into `index.html` (see `hermes_cli/web_server.py`). The
|
|
* Vite dev server serves its own `index.html`, so unless we forward that
|
|
* token, every protected `/api/*` call 401s.
|
|
*
|
|
* This plugin fetches the running dashboard's `index.html` on each dev page
|
|
* load, scrapes the `window.__HERMES_SESSION_TOKEN__` assignment, and
|
|
* re-injects it into the dev HTML. No-op in production builds.
|
|
*/
|
|
function hermesDevToken(): Plugin {
|
|
const TOKEN_RE = /window\.__HERMES_SESSION_TOKEN__\s*=\s*"([^"]+)"/;
|
|
const EMBEDDED_RE =
|
|
/window\.__HERMES_DASHBOARD_EMBEDDED_CHAT__\s*=\s*(true|false)/;
|
|
|
|
return {
|
|
name: "hermes:dev-session-token",
|
|
apply: "serve",
|
|
async transformIndexHtml() {
|
|
try {
|
|
const res = await fetch(BACKEND, { headers: { accept: "text/html" } });
|
|
const html = await res.text();
|
|
const match = html.match(TOKEN_RE);
|
|
if (!match) {
|
|
console.warn(
|
|
`[hermes] Could not find session token in ${BACKEND} — ` +
|
|
`is \`hermes dashboard\` running? /api calls will 401.`,
|
|
);
|
|
return;
|
|
}
|
|
const embeddedMatch = html.match(EMBEDDED_RE);
|
|
const embeddedJs = embeddedMatch ? embeddedMatch[1] : "true";
|
|
return [
|
|
{
|
|
tag: "script",
|
|
injectTo: "head",
|
|
children:
|
|
`window.__HERMES_SESSION_TOKEN__="${match[1]}";` +
|
|
`window.__HERMES_DASHBOARD_EMBEDDED_CHAT__=${embeddedJs};`,
|
|
},
|
|
];
|
|
} catch (err) {
|
|
console.warn(
|
|
`[hermes] Dashboard at ${BACKEND} unreachable — ` +
|
|
`start it with \`hermes dashboard\` or set HERMES_DASHBOARD_URL. ` +
|
|
`(${(err as Error).message})`,
|
|
);
|
|
}
|
|
},
|
|
};
|
|
}
|
|
|
|
export default defineConfig({
|
|
plugins: [react(), tailwindcss(), hermesDevToken()],
|
|
resolve: {
|
|
alias: {
|
|
"@": path.resolve(__dirname, "./src"),
|
|
},
|
|
// When @nous-research/ui is symlinked via `file:../../design-language`,
|
|
// Node's module resolution would pick up shared deps from
|
|
// design-language/node_modules/*, giving us two copies + breaking
|
|
// hooks (useRef-of-null), webgl contexts, etc. Force everything that
|
|
// exists in BOTH places to use the dashboard's copy.
|
|
//
|
|
// Don't list packages here that only exist in the DS (nanostores,
|
|
// @nanostores/react) — Vite dedupe errors out when it can't find
|
|
// them at the project root.
|
|
dedupe: [
|
|
"react",
|
|
"react-dom",
|
|
"@react-three/fiber",
|
|
"@observablehq/plot",
|
|
"three",
|
|
"leva",
|
|
"gsap",
|
|
],
|
|
},
|
|
build: {
|
|
outDir: "../hermes_cli/web_dist",
|
|
emptyOutDir: true,
|
|
},
|
|
server: {
|
|
proxy: {
|
|
"/api": {
|
|
target: BACKEND,
|
|
ws: true,
|
|
},
|
|
// Same host as `hermes dashboard` must serve these; Vite has no
|
|
// dashboard-plugins/* files, so without this, plugin scripts 404
|
|
// or receive index.html in dev.
|
|
"/dashboard-plugins": BACKEND,
|
|
},
|
|
},
|
|
});
|