mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-24 10:52:21 +00:00
* feat(cli): /reasoning full to show complete thinking, not 10-line clamp The post-response Reasoning recap box hard-clamped long thinking to the first 10 lines, so there was no way to see the full reasoning trace after a turn (live streaming already shows it in full). Add display.reasoning_full (default off) plus /reasoning full|clamp to toggle it at runtime; the clamp truncation note now points at the command. Addresses repeated user requests to show all thinking tokens. * test(gateway): de-snapshot /reasoning help assertion The test froze the exact args-hint literal '/reasoning [level|show|hide]', which the new full/clamp args change to '[level|show|hide|full|clamp]'. Convert to an invariant: assert /reasoning is in help and carries its core args, not the exact hint string. * feat(tui): /reasoning full|clamp parity in tui_gateway The classic-CLI reasoning_full toggle had no TUI equivalent — typing /reasoning full in the TUI fell through to parse_reasoning_effort and errored. The TUI renders thinking as an expand/collapse section (no fixed 10-line recap), so map full -> sections.thinking=expanded (raw, uncapped via thinkingPreview mode='full') and clamp -> collapsed, persisting display.reasoning_full for cross-surface config consistency.
81 lines
2.6 KiB
Python
81 lines
2.6 KiB
Python
"""Tests for the CLI `/reasoning full` / `/reasoning clamp` recap toggle.
|
|
|
|
The post-response "Reasoning" recap box clamps long thinking to the first
|
|
10 lines. `/reasoning full` opts into uncapped display (Taelin's "show all
|
|
thinking tokens" ask); `/reasoning clamp` restores the 10-line collapse.
|
|
These assert the toggle sets the instance flag, persists to config.yaml,
|
|
and that the clamp gate honours the flag.
|
|
"""
|
|
|
|
import os
|
|
|
|
import yaml
|
|
|
|
from hermes_cli.cli_commands_mixin import CLICommandsMixin
|
|
from hermes_cli.config import DEFAULT_CONFIG
|
|
|
|
|
|
class _Stub(CLICommandsMixin):
|
|
"""Minimal carrier for the attributes `_handle_reasoning_command` reads."""
|
|
|
|
def __init__(self):
|
|
self.reasoning_config = None
|
|
self.show_reasoning = True
|
|
self.reasoning_full = False
|
|
self.agent = None
|
|
|
|
def _current_reasoning_callback(self):
|
|
return None
|
|
|
|
|
|
def test_default_config_clamps_reasoning():
|
|
# Behaviour contract: the recap defaults to clamped, not full.
|
|
assert DEFAULT_CONFIG["display"]["reasoning_full"] is False
|
|
|
|
|
|
def _seed_config(tmp_path, monkeypatch):
|
|
hh = tmp_path / ".hermes"
|
|
hh.mkdir()
|
|
(hh / "config.yaml").write_text("display:\n show_reasoning: true\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(hh))
|
|
# cli captures _hermes_home at import; force it to the temp home.
|
|
import cli
|
|
|
|
monkeypatch.setattr(cli, "_hermes_home", hh, raising=False)
|
|
return hh
|
|
|
|
|
|
def test_reasoning_full_sets_and_persists(tmp_path, monkeypatch):
|
|
hh = _seed_config(tmp_path, monkeypatch)
|
|
s = _Stub()
|
|
|
|
s._handle_reasoning_command("/reasoning full")
|
|
assert s.reasoning_full is True
|
|
saved = yaml.safe_load((hh / "config.yaml").read_text())
|
|
assert saved["display"]["reasoning_full"] is True
|
|
|
|
|
|
def test_reasoning_clamp_resets_and_persists(tmp_path, monkeypatch):
|
|
hh = _seed_config(tmp_path, monkeypatch)
|
|
s = _Stub()
|
|
s.reasoning_full = True
|
|
|
|
s._handle_reasoning_command("/reasoning clamp")
|
|
assert s.reasoning_full is False
|
|
saved = yaml.safe_load((hh / "config.yaml").read_text())
|
|
assert saved["display"]["reasoning_full"] is False
|
|
|
|
|
|
def test_reasoning_all_is_alias_for_full(tmp_path, monkeypatch):
|
|
_seed_config(tmp_path, monkeypatch)
|
|
s = _Stub()
|
|
s._handle_reasoning_command("/reasoning all")
|
|
assert s.reasoning_full is True
|
|
|
|
|
|
def test_clamp_gate_honours_flag():
|
|
# The display gate at cli.py: clamp only when long AND not reasoning_full.
|
|
reasoning = "\n".join(f"line{i}" for i in range(25))
|
|
lines = reasoning.strip().splitlines()
|
|
assert (len(lines) > 10 and not False) is True # full=False -> clamp
|
|
assert (len(lines) > 10 and not True) is False # full=True -> show all
|