mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-05 02:31:47 +00:00
The old CWD heuristic was fooled by: 1. TERMINAL_CWD persisted to .env by `hermes config set terminal.cwd` 2. Inherited TERMINAL_CWD from parent hermes processes 3. Only resolved when config had a placeholder value (not explicit paths) Fix: - load_cli_config() unconditionally uses os.getcwd() for local backend - TERMINAL_CWD always force-exported in CLI mode (overrides stale values) - Gateway sets _HERMES_GATEWAY=1 marker so lazy cli.py imports don't clobber - Remove terminal.cwd from config-set .env sync map (prevents re-poisoning) - Clarify setup wizard label as 'Gateway working directory' Closes #19214
101 lines
3.4 KiB
Python
101 lines
3.4 KiB
Python
"""Tests for CLI/TUI CWD resolution in load_cli_config().
|
|
|
|
Rules:
|
|
- Local backend CLI/TUI: always os.getcwd(), ignoring config and inherited env.
|
|
- Non-local with placeholder: pop cwd for backend default.
|
|
- Non-local with explicit path: keep as-is.
|
|
"""
|
|
|
|
import os
|
|
import pytest
|
|
|
|
_CWD_PLACEHOLDERS = (".", "auto", "cwd")
|
|
|
|
|
|
def _resolve_cwd(terminal_config: dict, defaults: dict, env: dict):
|
|
"""Mirror the CWD resolution logic from cli.py load_cli_config()."""
|
|
effective_backend = terminal_config.get("env_type", "local")
|
|
|
|
if effective_backend == "local":
|
|
terminal_config["cwd"] = "/fake/getcwd"
|
|
defaults["terminal"]["cwd"] = terminal_config["cwd"]
|
|
elif terminal_config.get("cwd") in _CWD_PLACEHOLDERS:
|
|
terminal_config.pop("cwd", None)
|
|
|
|
# Bridge: TERMINAL_CWD always exported in CLI, skipped in gateway
|
|
_is_gateway = env.get("_HERMES_GATEWAY") == "1"
|
|
if "cwd" in terminal_config:
|
|
if _is_gateway:
|
|
pass # don't touch env
|
|
else:
|
|
env["TERMINAL_CWD"] = str(terminal_config["cwd"])
|
|
|
|
return env.get("TERMINAL_CWD", "")
|
|
|
|
|
|
class TestLocalBackendCli:
|
|
"""Local backend always uses os.getcwd()."""
|
|
|
|
def test_explicit_config_ignored(self):
|
|
env = {}
|
|
tc = {"cwd": "/explicit/path", "env_type": "local"}
|
|
d = {"terminal": {"cwd": "/explicit/path"}}
|
|
assert _resolve_cwd(tc, d, env) == "/fake/getcwd"
|
|
|
|
def test_inherited_env_overwritten(self):
|
|
env = {"TERMINAL_CWD": "/parent/hermes"}
|
|
tc = {"cwd": "/home/user", "env_type": "local"}
|
|
d = {"terminal": {"cwd": "/home/user"}}
|
|
assert _resolve_cwd(tc, d, env) == "/fake/getcwd"
|
|
|
|
def test_placeholder_resolved(self):
|
|
env = {}
|
|
tc = {"cwd": "."}
|
|
d = {"terminal": {"cwd": "."}}
|
|
assert _resolve_cwd(tc, d, env) == "/fake/getcwd"
|
|
|
|
def test_env_and_no_config_file(self):
|
|
env = {"TERMINAL_CWD": "/stale/value"}
|
|
tc = {"cwd": ".", "env_type": "local"}
|
|
d = {"terminal": {"cwd": "."}}
|
|
assert _resolve_cwd(tc, d, env) == "/fake/getcwd"
|
|
|
|
|
|
class TestNonLocalBackends:
|
|
"""Non-local backends use config or per-backend defaults."""
|
|
|
|
def test_placeholder_popped(self):
|
|
env = {}
|
|
tc = {"cwd": ".", "env_type": "docker"}
|
|
d = {"terminal": {"cwd": "."}}
|
|
assert _resolve_cwd(tc, d, env) == ""
|
|
|
|
def test_explicit_path_kept(self):
|
|
env = {}
|
|
tc = {"cwd": "/srv/app", "env_type": "ssh"}
|
|
d = {"terminal": {"cwd": "/srv/app"}}
|
|
assert _resolve_cwd(tc, d, env) == "/srv/app"
|
|
|
|
def test_auto_placeholder_popped(self):
|
|
env = {}
|
|
tc = {"cwd": "auto", "env_type": "modal"}
|
|
d = {"terminal": {"cwd": "auto"}}
|
|
assert _resolve_cwd(tc, d, env) == ""
|
|
|
|
|
|
class TestGatewayLazyImport:
|
|
"""Gateway lazy import of cli.py must not clobber TERMINAL_CWD."""
|
|
|
|
def test_gateway_cwd_preserved(self):
|
|
env = {"_HERMES_GATEWAY": "1", "TERMINAL_CWD": "/home/user/project"}
|
|
tc = {"cwd": "/home/user", "env_type": "local"}
|
|
d = {"terminal": {"cwd": "/home/user"}}
|
|
result = _resolve_cwd(tc, d, env)
|
|
assert result == "/home/user/project"
|
|
|
|
def test_cli_overwrites_stale_env(self):
|
|
env = {"TERMINAL_CWD": "/stale/from/dotenv"}
|
|
tc = {"cwd": "/home/user", "env_type": "local"}
|
|
d = {"terminal": {"cwd": "/home/user"}}
|
|
result = _resolve_cwd(tc, d, env)
|
|
assert result == "/fake/getcwd"
|