mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-13 09:01:54 +00:00
* feat(dashboard): unify multi-profile management — one machine dashboard, global profile switcher The dashboard becomes a machine-level management surface with one write-target selector, replacing per-profile dashboard fragmentation. Backend: - profile param (query or body) on /api/config (get/put/raw), /api/env (get/put/delete/reveal), /api/mcp/servers (list/add/remove/test/enabled), /api/mcp/catalog (list/install), /api/model/info, /api/model/set — all scoped through the existing _profile_scope() context manager - model/set restructured: expensive-model warning (await) runs before the scope; the config write runs sync inside the scope in a worker thread - MCP catalog installs + git-bootstrap entries spawn 'hermes -p <profile>' - chat PTY: ?profile= on /api/pty points the child's HERMES_HOME at the profile dir (its own gateway subprocess, config/skills/memory/state.db all profile-bound); in-process gateway attach skipped when scoped CLI launch unification: - '<profile> dashboard' routes to the machine dashboard: attach (open browser at ?profile=) when one is listening, else re-exec pinned to the default profile with --open-profile preselecting the launcher - --isolated preserves the old dedicated per-profile server behavior - start_server(initial_profile=...) appends ?profile= to the auto-open URL Frontend: - ProfileProvider + sidebar ProfileSwitcher: ONE global selector, URL- persisted (?profile=), mirrored into fetchJSON which auto-appends the param to the scoped endpoint families (explicit params win) - app-wide amber banner names the managed profile - SkillsPage's page-local selector (from the skills-scoping PR) folded into the global context — single source of truth - ChatPage threads the scope into the PTY WS URL; switching profiles remounts the terminal into a fresh scoped session Omitted profile keeps legacy behavior everywhere. * docs(dashboard): document machine-level multi-profile management - web-dashboard.md: 'Managing multiple profiles' section (switcher, URL deep-links, unified launch, --isolated, scoped Chat, what stays per-profile) + --isolated in the options table - profiles.md: 'From the dashboard' subsection + set-as-active vs switcher clarification - cli-commands.md: --isolated flag + profile-alias launch example * fix(dashboard): address profile-unification review findings Review findings (dev review on PR #44007): 1. HIGH — stale page state on profile switch: pages load data on mount and didn't consume the profile scope, so a page opened under profile A kept showing A's state while writes silently targeted the newly selected B. Fixed structurally: ProfileKeyedRoutes wraps the routed page tree and keys it by the selected profile, remounting every page (fresh state + refetch) on switch. ChatPage keeps its own remount (channel keyed on scopedProfile). 2. HIGH — /api/model/auxiliary read was unscoped while /api/model/set wrote scoped (Models page could show default's aux pins while editing worker's). Endpoint now takes profile + _profile_scope, added to PROFILE_SCOPED_PREFIXES, HTTPException re-raise so ghost profiles 404 instead of 500. Regression test asserts read/write symmetry with differing worker/default aux config. 3. MEDIUM — tools post-setup spawned unscoped from the profile-aware drawer. Now spawns 'hermes -p <profile> tools post-setup <key>' (same mechanism as hub installs); drawer threads its profile prop. Most hooks install machine-level artifacts where the scope is inert, but hooks reading config/env now see the drawer's HERMES_HOME. 4. LOW — ty warnings: env Optional asserts before subscript/membership, fastapi import replaced with web_server.HTTPException re-use. 298 tests green across the four affected suites; tsc -b + vite build green; aux scoping E2E-verified with real imports. * fix(dashboard): address second profile-unification review (gille) 1. BLOCKER — profile scope dropped on sidebar navigation: ProfileProvider derived the selection from the current URL, and nav links are bare paths, so clicking Config from /skills?profile=worker silently reset the write target. State is now the source of truth; an effect re-asserts ?profile= onto the new location after every navigation (URL stays a synchronized projection for deep links/refresh), and an incoming URL param (e.g. 'Manage skills & tools' links) still wins. 2. BLOCKER — /api/model/options unscoped while model/set wrote scoped: the picker context (current model/provider, custom providers, per-profile .env auth state) now loads inside _profile_scope; added to PROFILE_SCOPED_PREFIXES. Test: a worker-only current-model pin appears in the scoped payload and not the unscoped one. 3. BLOCKER — MCP test-server probe escaped the scope after the config read: the probe now re-enters _profile_scope inside the worker thread so env-placeholder expansion resolves against the selected profile's .env. Known limit (documented): the probe's dedicated MCP event-loop thread doesn't inherit the contextvar (OAuth token paths). Test asserts get_hermes_home() inside the probe == the worker profile dir. 4. BLOCKER — broad excepts swallowed unknown-profile 404s: /api/model/info degraded to 200-with-empty-model-info and /api/mcp/catalog to a silently-empty catalog. Both re-raise HTTPException; 404 regression tests added for info/options/catalog. Polish: scope banner clears the fixed mobile header (mt-14 lg:mt-0); --open-profile hidden via argparse.SUPPRESS (internal re-exec flag); attach-path test now asserts the opened ?profile= URL. (Stale-page-state + /api/model/auxiliary findings from this review were already fixed in92bcd1568— the review ran againste600f6951.) 35 tests in the two new suites + 274 in the adjacent ones, all green; tsc -b + vite build green; scoping E2E-verified with real imports. * docs(dashboard)+fix: self-review pass — Profiles page section, REST profile-param tip, body-beats-query precedence Docs: - web-dashboard.md: add the missing 'Profiles' subsection to Pages (cards, create/builder, manage-skills jump, set-as-active vs switcher distinction, editors); REST API section gets a profile-scoped-endpoints tip documenting ?profile= / body profile / 404 semantics / /api/pty - (profiles.md + cli-commands.md were already updated ine600f6951) Precedence fix: scoped endpoints taking BOTH a query param and a body field now resolve body.profile first. The SPA's fetchJSON injects the query param from the GLOBAL switcher; an explicit body.profile (e.g. Profile Builder flows writing into a specific new profile) is the more specific intent and must not be overridden by whatever the sidebar happens to be set to. Matches the documented 'explicit beats global' contract in api.ts. Verified: 304 tests green across the four suites; tsc -b + vite build green; docusaurus build green (only pre-existing broken-link warnings, none from this PR's pages).
385 lines
16 KiB
Python
385 lines
16 KiB
Python
"""Regression tests for the machine-dashboard multi-profile unification.
|
|
|
|
The dashboard is ONE machine-level management surface: config, env, MCP,
|
|
model, and chat-PTY endpoints accept an optional ``profile`` so the global
|
|
profile switcher can target any profile's HERMES_HOME. These tests pin:
|
|
reads/writes land in the REQUESTED profile, the dashboard's own profile
|
|
stays untouched, and the chat PTY env is scoped via HERMES_HOME.
|
|
"""
|
|
import pytest
|
|
import yaml
|
|
|
|
|
|
@pytest.fixture
|
|
def isolated_profiles(tmp_path, monkeypatch, _isolate_hermes_home):
|
|
"""Isolated default home + one named profile, each with config + .env."""
|
|
from hermes_constants import get_hermes_home
|
|
from hermes_cli import profiles
|
|
|
|
default_home = get_hermes_home()
|
|
profiles_root = default_home / "profiles"
|
|
worker_home = profiles_root / "worker_beta"
|
|
for home in (default_home, worker_home):
|
|
home.mkdir(parents=True, exist_ok=True)
|
|
(home / "config.yaml").write_text("{}\n", encoding="utf-8")
|
|
(worker_home / ".env").write_text("", encoding="utf-8")
|
|
|
|
monkeypatch.setattr(profiles, "_get_default_hermes_home", lambda: default_home)
|
|
monkeypatch.setattr(profiles, "_get_profiles_root", lambda: profiles_root)
|
|
return {"default": default_home, "worker_beta": worker_home}
|
|
|
|
|
|
@pytest.fixture
|
|
def client(monkeypatch, isolated_profiles):
|
|
try:
|
|
from starlette.testclient import TestClient
|
|
except ImportError:
|
|
pytest.skip("fastapi/starlette not installed")
|
|
|
|
import hermes_state
|
|
from hermes_constants import get_hermes_home
|
|
from hermes_cli.web_server import app, _SESSION_HEADER_NAME, _SESSION_TOKEN
|
|
|
|
monkeypatch.setattr(hermes_state, "DEFAULT_DB_PATH", get_hermes_home() / "state.db")
|
|
c = TestClient(app)
|
|
c.headers[_SESSION_HEADER_NAME] = _SESSION_TOKEN
|
|
return c
|
|
|
|
|
|
def _cfg(home):
|
|
return yaml.safe_load((home / "config.yaml").read_text()) or {}
|
|
|
|
|
|
class TestProfileScopedConfig:
|
|
def test_config_put_lands_in_target_profile_only(self, client, isolated_profiles):
|
|
resp = client.put(
|
|
"/api/config",
|
|
json={"config": {"timezone": "Mars/Olympus"}, "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
assert _cfg(isolated_profiles["worker_beta"]).get("timezone") == "Mars/Olympus"
|
|
assert _cfg(isolated_profiles["default"]).get("timezone") != "Mars/Olympus"
|
|
|
|
def test_config_get_reads_target_profile(self, client, isolated_profiles):
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"timezone: Venus/Cloud\n", encoding="utf-8"
|
|
)
|
|
resp = client.get("/api/config", params={"profile": "worker_beta"})
|
|
assert resp.status_code == 200
|
|
assert resp.json().get("timezone") == "Venus/Cloud"
|
|
# Unscoped read sees the dashboard's own config.
|
|
resp = client.get("/api/config")
|
|
assert resp.json().get("timezone") != "Venus/Cloud"
|
|
|
|
def test_config_query_param_equivalent_to_body(self, client, isolated_profiles):
|
|
"""The SPA's fetchJSON injects ?profile= — must scope like body.profile."""
|
|
resp = client.put(
|
|
"/api/config?profile=worker_beta",
|
|
json={"config": {"timezone": "Pluto/Far"}},
|
|
)
|
|
assert resp.status_code == 200
|
|
assert _cfg(isolated_profiles["worker_beta"]).get("timezone") == "Pluto/Far"
|
|
assert _cfg(isolated_profiles["default"]).get("timezone") != "Pluto/Far"
|
|
|
|
def test_config_raw_round_trip_scoped(self, client, isolated_profiles):
|
|
resp = client.put(
|
|
"/api/config/raw",
|
|
json={"yaml_text": "timezone: Io/Volcano\n", "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
resp = client.get("/api/config/raw", params={"profile": "worker_beta"})
|
|
assert "Io/Volcano" in resp.json()["yaml"]
|
|
resp = client.get("/api/config/raw")
|
|
assert "Io/Volcano" not in resp.json()["yaml"]
|
|
|
|
def test_unknown_profile_404(self, client, isolated_profiles):
|
|
resp = client.get("/api/config", params={"profile": "ghost"})
|
|
assert resp.status_code == 404
|
|
|
|
|
|
class TestProfileScopedEnv:
|
|
def test_env_set_lands_in_target_profile_only(self, client, isolated_profiles):
|
|
resp = client.put(
|
|
"/api/env",
|
|
json={"key": "FAL_KEY", "value": "test-fal-123", "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
worker_env = (isolated_profiles["worker_beta"] / ".env").read_text()
|
|
assert "test-fal-123" in worker_env
|
|
default_env_path = isolated_profiles["default"] / ".env"
|
|
if default_env_path.exists():
|
|
assert "test-fal-123" not in default_env_path.read_text()
|
|
|
|
def test_env_list_reads_target_profile(self, client, isolated_profiles):
|
|
(isolated_profiles["worker_beta"] / ".env").write_text(
|
|
"FAL_KEY=worker-only-value\n", encoding="utf-8"
|
|
)
|
|
resp = client.get("/api/env", params={"profile": "worker_beta"})
|
|
assert resp.status_code == 200
|
|
assert resp.json()["FAL_KEY"]["is_set"] is True
|
|
resp = client.get("/api/env")
|
|
assert resp.json()["FAL_KEY"]["is_set"] is False
|
|
|
|
def test_env_delete_scoped(self, client, isolated_profiles):
|
|
(isolated_profiles["worker_beta"] / ".env").write_text(
|
|
"FAL_KEY=doomed\n", encoding="utf-8"
|
|
)
|
|
resp = client.request(
|
|
"DELETE",
|
|
"/api/env",
|
|
json={"key": "FAL_KEY", "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
assert "doomed" not in (isolated_profiles["worker_beta"] / ".env").read_text()
|
|
|
|
|
|
class TestProfileScopedMcp:
|
|
def test_mcp_add_and_list_scoped(self, client, isolated_profiles):
|
|
resp = client.post(
|
|
"/api/mcp/servers",
|
|
json={"name": "scoped-srv", "url": "http://localhost:1234/sse",
|
|
"profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
|
|
worker_cfg = _cfg(isolated_profiles["worker_beta"])
|
|
assert "scoped-srv" in worker_cfg.get("mcp_servers", {})
|
|
assert "scoped-srv" not in _cfg(isolated_profiles["default"]).get("mcp_servers", {})
|
|
|
|
listing = client.get("/api/mcp/servers", params={"profile": "worker_beta"}).json()
|
|
assert any(s["name"] == "scoped-srv" for s in listing["servers"])
|
|
listing = client.get("/api/mcp/servers").json()
|
|
assert not any(s["name"] == "scoped-srv" for s in listing["servers"])
|
|
|
|
def test_mcp_enabled_toggle_scoped(self, client, isolated_profiles):
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"mcp_servers:\n srv1:\n url: http://x/sse\n", encoding="utf-8"
|
|
)
|
|
resp = client.put(
|
|
"/api/mcp/servers/srv1/enabled",
|
|
json={"enabled": False, "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
worker_cfg = _cfg(isolated_profiles["worker_beta"])
|
|
assert worker_cfg["mcp_servers"]["srv1"]["enabled"] is False
|
|
|
|
def test_mcp_probe_runs_inside_profile_scope(
|
|
self, client, isolated_profiles, monkeypatch
|
|
):
|
|
"""The test-server probe must execute with the selected profile's
|
|
scope active so env-placeholder expansion reads the profile's .env,
|
|
matching the config the server was saved into."""
|
|
import hermes_cli.mcp_config as mcp_config
|
|
from hermes_constants import get_hermes_home
|
|
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"mcp_servers:\n probe-srv:\n url: http://x/sse\n",
|
|
encoding="utf-8",
|
|
)
|
|
seen = {}
|
|
|
|
def fake_probe(name, config, connect_timeout=30):
|
|
seen["home"] = str(get_hermes_home())
|
|
return [("tool-a", "desc")]
|
|
|
|
monkeypatch.setattr(mcp_config, "_probe_single_server", fake_probe)
|
|
resp = client.post(
|
|
"/api/mcp/servers/probe-srv/test", params={"profile": "worker_beta"}
|
|
)
|
|
assert resp.status_code == 200
|
|
assert resp.json()["ok"] is True
|
|
assert seen["home"] == str(isolated_profiles["worker_beta"])
|
|
|
|
def test_mcp_remove_scoped(self, client, isolated_profiles):
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"mcp_servers:\n srv2:\n url: http://x/sse\n", encoding="utf-8"
|
|
)
|
|
# Removing from the DASHBOARD's profile must 404 (srv2 lives in worker).
|
|
resp = client.delete("/api/mcp/servers/srv2")
|
|
assert resp.status_code == 404
|
|
resp = client.delete("/api/mcp/servers/srv2", params={"profile": "worker_beta"})
|
|
assert resp.status_code == 200
|
|
assert "srv2" not in _cfg(isolated_profiles["worker_beta"]).get("mcp_servers", {})
|
|
|
|
|
|
class TestProfileScopedModel:
|
|
def test_model_set_main_scoped(self, client, isolated_profiles):
|
|
resp = client.post(
|
|
"/api/model/set",
|
|
json={
|
|
"scope": "main",
|
|
"provider": "openrouter",
|
|
"model": "test/model-1",
|
|
"confirm_expensive_model": True,
|
|
"profile": "worker_beta",
|
|
},
|
|
)
|
|
assert resp.status_code == 200
|
|
worker_cfg = _cfg(isolated_profiles["worker_beta"])
|
|
model_cfg = worker_cfg.get("model", {})
|
|
assert isinstance(model_cfg, dict)
|
|
assert model_cfg.get("provider") == "openrouter"
|
|
default_model = _cfg(isolated_profiles["default"]).get("model", {})
|
|
if isinstance(default_model, dict):
|
|
assert default_model.get("default") != "test/model-1"
|
|
|
|
def test_auxiliary_read_scoped_matches_write_target(
|
|
self, client, isolated_profiles
|
|
):
|
|
"""Reads and writes must scope symmetrically: an aux pin written to
|
|
the worker profile must show up ONLY in the worker-scoped read.
|
|
(Regression: /api/model/auxiliary used to read unscoped while
|
|
/api/model/set wrote scoped — the Models page displayed the
|
|
dashboard profile's pins while editing the selected profile's.)"""
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"auxiliary:\n vision:\n provider: openrouter\n"
|
|
" model: worker/vision-pin\n",
|
|
encoding="utf-8",
|
|
)
|
|
resp = client.get("/api/model/auxiliary", params={"profile": "worker_beta"})
|
|
assert resp.status_code == 200
|
|
vision = next(t for t in resp.json()["tasks"] if t["task"] == "vision")
|
|
assert vision["model"] == "worker/vision-pin"
|
|
|
|
# Unscoped read = the dashboard's own profile, which has no pin.
|
|
resp = client.get("/api/model/auxiliary")
|
|
assert resp.status_code == 200
|
|
vision = next(t for t in resp.json()["tasks"] if t["task"] == "vision")
|
|
assert vision["model"] != "worker/vision-pin"
|
|
|
|
def test_auxiliary_unknown_profile_404(self, client, isolated_profiles):
|
|
resp = client.get("/api/model/auxiliary", params={"profile": "ghost"})
|
|
assert resp.status_code == 404
|
|
|
|
def test_model_options_scoped_to_profile(self, client, isolated_profiles):
|
|
"""The Models picker must read the SAME profile model/set writes —
|
|
current model/provider in the payload come from the scoped config."""
|
|
(isolated_profiles["worker_beta"] / "config.yaml").write_text(
|
|
"model:\n provider: openrouter\n default: worker/current-pin\n",
|
|
encoding="utf-8",
|
|
)
|
|
resp = client.get("/api/model/options", params={"profile": "worker_beta"})
|
|
assert resp.status_code == 200
|
|
body = resp.json()
|
|
# The payload carries the current selection somewhere stable; assert
|
|
# the worker pin appears in the scoped response and not the unscoped.
|
|
assert "worker/current-pin" in resp.text
|
|
resp = client.get("/api/model/options")
|
|
assert resp.status_code == 200
|
|
assert "worker/current-pin" not in resp.text
|
|
assert isinstance(body, dict)
|
|
|
|
def test_model_options_unknown_profile_404(self, client, isolated_profiles):
|
|
resp = client.get("/api/model/options", params={"profile": "ghost"})
|
|
assert resp.status_code == 404
|
|
|
|
def test_model_info_unknown_profile_404(self, client, isolated_profiles):
|
|
"""Regression: the broad except used to convert the 404 into a 200
|
|
with empty model info ("no model set" — silently wrong)."""
|
|
resp = client.get("/api/model/info", params={"profile": "ghost"})
|
|
assert resp.status_code == 404
|
|
|
|
def test_mcp_catalog_unknown_profile_404(self, client, isolated_profiles):
|
|
resp = client.get("/api/mcp/catalog", params={"profile": "ghost"})
|
|
assert resp.status_code == 404
|
|
|
|
|
|
class TestProfileScopedPostSetup:
|
|
def test_post_setup_spawns_with_profile_flag(
|
|
self, client, isolated_profiles, monkeypatch
|
|
):
|
|
"""Post-setup runs in a -p scoped subprocess so hooks that read
|
|
config / write per-profile state see the same HERMES_HOME the rest
|
|
of the drawer's writes targeted."""
|
|
import hermes_cli.web_server as web_server
|
|
|
|
calls = []
|
|
|
|
class _FakeProc:
|
|
pid = 777
|
|
|
|
monkeypatch.setattr(
|
|
web_server,
|
|
"_spawn_hermes_action",
|
|
lambda subcommand, name: calls.append(list(subcommand)) or _FakeProc(),
|
|
)
|
|
monkeypatch.setattr(
|
|
"hermes_cli.tools_config.valid_post_setup_keys",
|
|
lambda: {"agent_browser"},
|
|
)
|
|
resp = client.post(
|
|
"/api/tools/toolsets/browser/post-setup",
|
|
json={"key": "agent_browser", "profile": "worker_beta"},
|
|
)
|
|
assert resp.status_code == 200
|
|
assert calls == [
|
|
["-p", "worker_beta", "tools", "post-setup", "agent_browser"]
|
|
]
|
|
|
|
def test_post_setup_without_profile_keeps_legacy_argv(
|
|
self, client, isolated_profiles, monkeypatch
|
|
):
|
|
import hermes_cli.web_server as web_server
|
|
|
|
calls = []
|
|
|
|
class _FakeProc:
|
|
pid = 777
|
|
|
|
monkeypatch.setattr(
|
|
web_server,
|
|
"_spawn_hermes_action",
|
|
lambda subcommand, name: calls.append(list(subcommand)) or _FakeProc(),
|
|
)
|
|
monkeypatch.setattr(
|
|
"hermes_cli.tools_config.valid_post_setup_keys",
|
|
lambda: {"agent_browser"},
|
|
)
|
|
resp = client.post(
|
|
"/api/tools/toolsets/browser/post-setup",
|
|
json={"key": "agent_browser"},
|
|
)
|
|
assert resp.status_code == 200
|
|
assert calls == [["tools", "post-setup", "agent_browser"]]
|
|
|
|
|
|
class TestProfileScopedChatPty:
|
|
def test_chat_argv_scopes_hermes_home(self, isolated_profiles, monkeypatch):
|
|
import hermes_cli.web_server as web_server
|
|
|
|
monkeypatch.setattr(
|
|
"hermes_cli.main._make_tui_argv",
|
|
lambda root, tui_dev=False: (["cat"], None),
|
|
raising=False,
|
|
)
|
|
argv, cwd, env = web_server._resolve_chat_argv(profile="worker_beta")
|
|
assert env is not None
|
|
assert env["HERMES_HOME"] == str(isolated_profiles["worker_beta"])
|
|
# Scoped chat must NOT attach to the dashboard's in-memory gateway.
|
|
assert "HERMES_TUI_GATEWAY_URL" not in env
|
|
|
|
def test_chat_argv_unscoped_keeps_legacy_env(self, isolated_profiles, monkeypatch):
|
|
import hermes_cli.web_server as web_server
|
|
|
|
monkeypatch.setattr(
|
|
"hermes_cli.main._make_tui_argv",
|
|
lambda root, tui_dev=False: (["cat"], None),
|
|
raising=False,
|
|
)
|
|
argv, cwd, env = web_server._resolve_chat_argv()
|
|
assert env is not None
|
|
assert env.get("HERMES_HOME") != str(isolated_profiles["worker_beta"])
|
|
|
|
def test_chat_argv_unknown_profile_raises(self, isolated_profiles, monkeypatch):
|
|
import hermes_cli.web_server as web_server
|
|
|
|
monkeypatch.setattr(
|
|
"hermes_cli.main._make_tui_argv",
|
|
lambda root, tui_dev=False: (["cat"], None),
|
|
raising=False,
|
|
)
|
|
# Reuse the HTTPException class web_server itself raises — avoids a
|
|
# direct fastapi import (unresolvable in the ty lint environment).
|
|
with pytest.raises(web_server.HTTPException) as exc:
|
|
web_server._resolve_chat_argv(profile="ghost")
|
|
assert exc.value.status_code == 404
|