mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-18 04:41:56 +00:00
Adds a pure-local recap of recent session activity — turn counts, tools used, files touched, last user ask, last assistant reply — appended to the existing /status output. Useful when juggling multiple sessions and you want a one-glance reminder of where this one left off. Inspired by Claude Code 2.1.114's /recap, but folded into /status so we don't add a 6th info command. Pure local computation: no LLM call, no auxiliary model, no prompt-cache invalidation, instant and free. Salvage of #18587 — kept the shared hermes_cli.session_recap.build_recap helper and its 13 unit tests, dropped the /recap slash command + ACTIVE_SESSION_BYPASS_COMMANDS entry + Level-2 bypass since /status already covers both surfaces. Tailored to hermes-agent's tool vocabulary: file-editing tools (patch, write_file, read_file, skill_manage, skill_view) surface touched paths; tool-call counts highlight which classes of work drove the session. Source: https://code.claude.com/docs/en/whats-new/2026-w17
316 lines
11 KiB
Python
316 lines
11 KiB
Python
"""Session recap — summarize what's happened in the current session.
|
||
|
||
Inspired by Claude Code's `/recap` command (v2.1.114, April 2026), which
|
||
shows a one-line summary of what happened while a terminal was unfocused
|
||
so users juggling multiple sessions can re-orient quickly.
|
||
|
||
Source: https://code.claude.com/docs/en/whats-new/2026-w17
|
||
|
||
Differences from Claude Code:
|
||
- Pure local computation from the in-memory conversation history. No
|
||
LLM call, no auxiliary model, no prompt-cache invalidation. A
|
||
recap should be instant and free.
|
||
- Works unchanged on CLI and every gateway platform (Telegram,
|
||
Discord, Slack, …) because both call into the same ``build_recap``
|
||
helper. Claude Code only shows this on the CLI.
|
||
- Tailored to hermes-agent's tool vocabulary (``terminal``, ``patch``,
|
||
``write_file``, ``delegate_task``, ``browser_*``, ``web_*``) — the
|
||
recap surfaces which classes of work were most active.
|
||
"""
|
||
from __future__ import annotations
|
||
|
||
import os
|
||
from collections import Counter
|
||
from typing import Any, Iterable, List, Mapping, Optional, Sequence, Tuple
|
||
|
||
# How many recent user/assistant turns we consider "recent activity".
|
||
_RECENT_TURN_WINDOW = 20
|
||
|
||
# How many characters of the latest user prompt to show.
|
||
_PROMPT_PREVIEW_CHARS = 140
|
||
|
||
# How many characters of the latest assistant text to show.
|
||
_ASSISTANT_PREVIEW_CHARS = 200
|
||
|
||
# How many recently-touched files to list.
|
||
_MAX_FILES_LISTED = 5
|
||
|
||
# Tool names that identify a file-editing action and the argument key that
|
||
# holds the path.
|
||
_FILE_EDIT_TOOLS: Mapping[str, str] = {
|
||
"write_file": "path",
|
||
"patch": "path",
|
||
"read_file": "path",
|
||
"skill_manage": "file_path",
|
||
"skill_view": "file_path",
|
||
}
|
||
|
||
|
||
def _coerce_text(value: Any) -> str:
|
||
"""Flatten assistant/user ``content`` into a plain string.
|
||
|
||
Content can be a string or a list of content blocks (for multimodal
|
||
or reasoning models). We concatenate every text-like block and
|
||
ignore the rest.
|
||
"""
|
||
if value is None:
|
||
return ""
|
||
if isinstance(value, str):
|
||
return value
|
||
if isinstance(value, list):
|
||
parts: List[str] = []
|
||
for block in value:
|
||
if isinstance(block, str):
|
||
parts.append(block)
|
||
continue
|
||
if isinstance(block, Mapping):
|
||
text = block.get("text")
|
||
if isinstance(text, str) and text:
|
||
parts.append(text)
|
||
return "\n".join(parts)
|
||
return str(value)
|
||
|
||
|
||
def _tool_call_name_and_args(tool_call: Any) -> Tuple[str, Mapping[str, Any]]:
|
||
"""Extract ``(name, arguments_dict)`` from a tool_call entry.
|
||
|
||
``arguments`` may be a JSON string or a dict depending on provider.
|
||
Return an empty dict if it cannot be parsed.
|
||
"""
|
||
if not isinstance(tool_call, Mapping):
|
||
return "", {}
|
||
fn = tool_call.get("function") or {}
|
||
if not isinstance(fn, Mapping):
|
||
return "", {}
|
||
name = str(fn.get("name") or "") or ""
|
||
raw_args = fn.get("arguments")
|
||
if isinstance(raw_args, Mapping):
|
||
return name, raw_args
|
||
if isinstance(raw_args, str) and raw_args:
|
||
try:
|
||
import json
|
||
|
||
parsed = json.loads(raw_args)
|
||
if isinstance(parsed, Mapping):
|
||
return name, parsed
|
||
except Exception:
|
||
return name, {}
|
||
return name, {}
|
||
|
||
|
||
def _iter_assistant_tool_calls(
|
||
messages: Sequence[Mapping[str, Any]],
|
||
) -> Iterable[Tuple[str, Mapping[str, Any]]]:
|
||
for msg in messages:
|
||
if not isinstance(msg, Mapping):
|
||
continue
|
||
if msg.get("role") != "assistant":
|
||
continue
|
||
tool_calls = msg.get("tool_calls") or []
|
||
if not isinstance(tool_calls, list):
|
||
continue
|
||
for tc in tool_calls:
|
||
name, args = _tool_call_name_and_args(tc)
|
||
if name:
|
||
yield name, args
|
||
|
||
|
||
def _count_visible_turns(
|
||
messages: Sequence[Mapping[str, Any]],
|
||
) -> Tuple[int, int, int]:
|
||
"""Return ``(user_turn_count, assistant_turn_count, tool_message_count)``."""
|
||
users = assistants = tools = 0
|
||
for msg in messages:
|
||
if not isinstance(msg, Mapping):
|
||
continue
|
||
role = msg.get("role")
|
||
if role == "user":
|
||
users += 1
|
||
elif role == "assistant":
|
||
assistants += 1
|
||
elif role == "tool":
|
||
tools += 1
|
||
return users, assistants, tools
|
||
|
||
|
||
def _latest_user_prompt(
|
||
messages: Sequence[Mapping[str, Any]],
|
||
) -> Optional[str]:
|
||
for msg in reversed(messages):
|
||
if isinstance(msg, Mapping) and msg.get("role") == "user":
|
||
text = _coerce_text(msg.get("content")).strip()
|
||
if text:
|
||
return text
|
||
return None
|
||
|
||
|
||
def _latest_assistant_text(
|
||
messages: Sequence[Mapping[str, Any]],
|
||
) -> Optional[str]:
|
||
for msg in reversed(messages):
|
||
if not isinstance(msg, Mapping):
|
||
continue
|
||
if msg.get("role") != "assistant":
|
||
continue
|
||
text = _coerce_text(msg.get("content")).strip()
|
||
if text:
|
||
return text
|
||
return None
|
||
|
||
|
||
def _recent_window(
|
||
messages: Sequence[Mapping[str, Any]], window: int = _RECENT_TURN_WINDOW
|
||
) -> List[Mapping[str, Any]]:
|
||
"""Return the tail slice of ``messages`` covering at most ``window``
|
||
user+assistant turns (tool messages ride along inside the window).
|
||
|
||
Iterating from the end, we count user and assistant messages and
|
||
keep everything from the first message that falls within the window.
|
||
"""
|
||
count = 0
|
||
cut = 0
|
||
for i in range(len(messages) - 1, -1, -1):
|
||
msg = messages[i]
|
||
if isinstance(msg, Mapping) and msg.get("role") in ("user", "assistant"):
|
||
count += 1
|
||
if count >= window:
|
||
cut = i
|
||
break
|
||
else:
|
||
return list(messages)
|
||
return list(messages[cut:])
|
||
|
||
|
||
def _shortened_path(path: str) -> str:
|
||
"""Show a path relative to cwd when possible, otherwise with ~ expansion."""
|
||
if not path:
|
||
return path
|
||
try:
|
||
abs_path = os.path.abspath(os.path.expanduser(path))
|
||
cwd = os.getcwd()
|
||
if abs_path == cwd:
|
||
return "."
|
||
if abs_path.startswith(cwd + os.sep):
|
||
return abs_path[len(cwd) + 1 :]
|
||
home = os.path.expanduser("~")
|
||
if abs_path.startswith(home + os.sep):
|
||
return "~/" + abs_path[len(home) + 1 :]
|
||
return abs_path
|
||
except Exception:
|
||
return path
|
||
|
||
|
||
def _summarise_tool_activity(
|
||
tool_calls: Sequence[Tuple[str, Mapping[str, Any]]],
|
||
) -> Tuple[List[Tuple[str, int]], List[str]]:
|
||
"""Return ``(tool_counts_sorted, recently_edited_files)``.
|
||
|
||
``tool_counts_sorted`` is descending by count, keeping the full list
|
||
so callers can truncate for display. ``recently_edited_files`` lists
|
||
distinct paths (most recent first) from file-editing tools.
|
||
"""
|
||
counter: Counter[str] = Counter()
|
||
files_seen: List[str] = []
|
||
files_set: set[str] = set()
|
||
# Walk in reverse so "most recent first" drops out of order-preserved iteration.
|
||
for name, args in reversed(list(tool_calls)):
|
||
counter[name] += 1
|
||
arg_key = _FILE_EDIT_TOOLS.get(name)
|
||
if arg_key:
|
||
path = args.get(arg_key)
|
||
if isinstance(path, str) and path and path not in files_set:
|
||
files_set.add(path)
|
||
files_seen.append(_shortened_path(path))
|
||
# Restore "reverse of reverse" for correct counts; Counter ignores order
|
||
# so only files_seen needed the reversal. Fix ordering: currently
|
||
# files_seen is newest→oldest which is what we want for display.
|
||
tool_counts = sorted(counter.items(), key=lambda kv: (-kv[1], kv[0]))
|
||
return tool_counts, files_seen
|
||
|
||
|
||
def _truncate(text: str, limit: int) -> str:
|
||
text = " ".join(text.split()) # collapse newlines for a compact one-liner
|
||
if len(text) <= limit:
|
||
return text
|
||
return text[: limit - 1].rstrip() + "…"
|
||
|
||
|
||
def build_recap(
|
||
messages: Sequence[Mapping[str, Any]],
|
||
*,
|
||
session_title: Optional[str] = None,
|
||
session_id: Optional[str] = None,
|
||
platform: Optional[str] = None,
|
||
) -> str:
|
||
"""Build a multi-line recap of recent activity.
|
||
|
||
Inputs:
|
||
messages: the full conversation history as a list of
|
||
chat-completion-style dicts (``role``, ``content``,
|
||
``tool_calls``, …).
|
||
session_title: optional human title (from SessionDB).
|
||
session_id: optional session id.
|
||
platform: optional hint (``"cli"``, ``"telegram"``, …). Does not
|
||
change behavior today but is accepted for forward compat.
|
||
|
||
The output is plain text designed to render well in both a terminal
|
||
(with 80-col wrapping) and a gateway message bubble.
|
||
"""
|
||
_ = platform # reserved for future use
|
||
lines: List[str] = []
|
||
|
||
header_bits: List[str] = ["Session recap"]
|
||
if session_title:
|
||
header_bits.append(f"— {session_title}")
|
||
elif session_id:
|
||
header_bits.append(f"— {session_id[:8]}")
|
||
lines.append(" ".join(header_bits))
|
||
|
||
if not messages:
|
||
lines.append(" (nothing to recap — no messages yet)")
|
||
return "\n".join(lines)
|
||
|
||
users, assistants, tool_msgs = _count_visible_turns(messages)
|
||
window = _recent_window(messages)
|
||
win_users, win_assistants, _ = _count_visible_turns(window)
|
||
|
||
scope = (
|
||
f"{win_users} user turn{'s' if win_users != 1 else ''} / "
|
||
f"{win_assistants} assistant repl{'ies' if win_assistants != 1 else 'y'}"
|
||
)
|
||
if (users, assistants) != (win_users, win_assistants):
|
||
scope += f" (of {users}/{assistants} total)"
|
||
lines.append(f" Recent: {scope}, {tool_msgs} tool result{'s' if tool_msgs != 1 else ''}")
|
||
|
||
tool_calls = list(_iter_assistant_tool_calls(window))
|
||
tool_counts, files = _summarise_tool_activity(tool_calls)
|
||
if tool_counts:
|
||
top = ", ".join(f"{name}×{count}" for name, count in tool_counts[:5])
|
||
extra = len(tool_counts) - 5
|
||
if extra > 0:
|
||
top += f" (+{extra} more)"
|
||
lines.append(f" Tools used: {top}")
|
||
if files:
|
||
shown = files[:_MAX_FILES_LISTED]
|
||
extra = len(files) - len(shown)
|
||
entry = ", ".join(shown)
|
||
if extra > 0:
|
||
entry += f" (+{extra} more)"
|
||
lines.append(f" Files touched: {entry}")
|
||
|
||
latest_user = _latest_user_prompt(window)
|
||
if latest_user:
|
||
lines.append(f" Last ask: {_truncate(latest_user, _PROMPT_PREVIEW_CHARS)}")
|
||
|
||
latest_reply = _latest_assistant_text(window)
|
||
if latest_reply:
|
||
lines.append(f" Last reply: {_truncate(latest_reply, _ASSISTANT_PREVIEW_CHARS)}")
|
||
|
||
if len(lines) == 2:
|
||
# Only the header + scope line — nothing substantive to show.
|
||
lines.append(" (no assistant activity yet in this window)")
|
||
|
||
return "\n".join(lines)
|
||
|
||
|
||
__all__ = ["build_recap"]
|