mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-28 06:51:16 +08:00
145 lines
5.1 KiB
Python
145 lines
5.1 KiB
Python
|
|
"""
|
||
|
|
Contextual first-touch onboarding hints.
|
||
|
|
|
||
|
|
Instead of blocking first-run questionnaires, show a one-time hint the *first*
|
||
|
|
time a user hits a behavior fork — message-while-running, first long-running
|
||
|
|
tool, etc. Each hint is shown once per install (tracked in ``config.yaml`` under
|
||
|
|
``onboarding.seen.<flag>``) and then never again.
|
||
|
|
|
||
|
|
Keep this module tiny and dependency-free so both the CLI and gateway can import
|
||
|
|
it without pulling in heavy modules.
|
||
|
|
"""
|
||
|
|
|
||
|
|
from __future__ import annotations
|
||
|
|
|
||
|
|
import logging
|
||
|
|
from pathlib import Path
|
||
|
|
from typing import Any, Mapping, Optional
|
||
|
|
|
||
|
|
logger = logging.getLogger(__name__)
|
||
|
|
|
||
|
|
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
# Flag names (stable — used as config.yaml keys under onboarding.seen)
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
|
||
|
|
BUSY_INPUT_FLAG = "busy_input_prompt"
|
||
|
|
TOOL_PROGRESS_FLAG = "tool_progress_prompt"
|
||
|
|
|
||
|
|
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
# Hint content
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
|
||
|
|
def busy_input_hint_gateway(mode: str) -> str:
|
||
|
|
"""Hint shown the first time a user messages while the agent is busy.
|
||
|
|
|
||
|
|
``mode`` is the effective busy_input_mode that was just applied, so the
|
||
|
|
message matches reality ("I just interrupted…" vs "I just queued…").
|
||
|
|
"""
|
||
|
|
if mode == "queue":
|
||
|
|
return (
|
||
|
|
"💡 First-time tip — I queued your message instead of interrupting. "
|
||
|
|
"Send `/busy interrupt` to make new messages stop the current task "
|
||
|
|
"immediately, or `/busy status` to check. This notice won't appear again."
|
||
|
|
)
|
||
|
|
return (
|
||
|
|
"💡 First-time tip — I just interrupted my current task to answer you. "
|
||
|
|
"Send `/busy queue` to queue follow-ups for after the current task instead, "
|
||
|
|
"or `/busy status` to check. This notice won't appear again."
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
def busy_input_hint_cli(mode: str) -> str:
|
||
|
|
"""CLI version of the busy-input hint (plain text, no markdown)."""
|
||
|
|
if mode == "queue":
|
||
|
|
return (
|
||
|
|
"(tip) Your message was queued for the next turn. "
|
||
|
|
"Use /busy interrupt to make Enter stop the current run instead. "
|
||
|
|
"This tip only shows once."
|
||
|
|
)
|
||
|
|
return (
|
||
|
|
"(tip) Your message interrupted the current run. "
|
||
|
|
"Use /busy queue to queue messages for the next turn instead. "
|
||
|
|
"This tip only shows once."
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
def tool_progress_hint_gateway() -> str:
|
||
|
|
return (
|
||
|
|
"💡 First-time tip — that tool took a while and I'm streaming every step. "
|
||
|
|
"If the progress messages feel noisy, send `/verbose` to cycle modes "
|
||
|
|
"(all → new → off). This notice won't appear again."
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
def tool_progress_hint_cli() -> str:
|
||
|
|
return (
|
||
|
|
"(tip) That tool ran for a while. Use /verbose to cycle tool-progress "
|
||
|
|
"display modes (all -> new -> off -> verbose). This tip only shows once."
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
# State read / write
|
||
|
|
# -------------------------------------------------------------------------
|
||
|
|
|
||
|
|
def _get_seen_dict(config: Mapping[str, Any]) -> Mapping[str, Any]:
|
||
|
|
onboarding = config.get("onboarding") if isinstance(config, Mapping) else None
|
||
|
|
if not isinstance(onboarding, Mapping):
|
||
|
|
return {}
|
||
|
|
seen = onboarding.get("seen")
|
||
|
|
return seen if isinstance(seen, Mapping) else {}
|
||
|
|
|
||
|
|
|
||
|
|
def is_seen(config: Mapping[str, Any], flag: str) -> bool:
|
||
|
|
"""Return True if the user has already been shown this first-touch hint."""
|
||
|
|
return bool(_get_seen_dict(config).get(flag))
|
||
|
|
|
||
|
|
|
||
|
|
def mark_seen(config_path: Path, flag: str) -> bool:
|
||
|
|
"""Persist ``onboarding.seen.<flag> = True`` to ``config_path``.
|
||
|
|
|
||
|
|
Uses the atomic YAML writer so a concurrent process can't observe a
|
||
|
|
partially-written file. Returns True on success, False on any error
|
||
|
|
(including the config file being absent — onboarding is best-effort).
|
||
|
|
"""
|
||
|
|
try:
|
||
|
|
import yaml
|
||
|
|
from utils import atomic_yaml_write
|
||
|
|
except Exception as e: # pragma: no cover — dependency issue
|
||
|
|
logger.debug("onboarding: failed to import yaml/utils: %s", e)
|
||
|
|
return False
|
||
|
|
|
||
|
|
try:
|
||
|
|
cfg: dict = {}
|
||
|
|
if config_path.exists():
|
||
|
|
with open(config_path, encoding="utf-8") as f:
|
||
|
|
cfg = yaml.safe_load(f) or {}
|
||
|
|
if not isinstance(cfg.get("onboarding"), dict):
|
||
|
|
cfg["onboarding"] = {}
|
||
|
|
seen = cfg["onboarding"].get("seen")
|
||
|
|
if not isinstance(seen, dict):
|
||
|
|
seen = {}
|
||
|
|
cfg["onboarding"]["seen"] = seen
|
||
|
|
if seen.get(flag) is True:
|
||
|
|
return True # already marked — nothing to do
|
||
|
|
seen[flag] = True
|
||
|
|
atomic_yaml_write(config_path, cfg)
|
||
|
|
return True
|
||
|
|
except Exception as e:
|
||
|
|
logger.debug("onboarding: failed to mark flag %s: %s", flag, e)
|
||
|
|
return False
|
||
|
|
|
||
|
|
|
||
|
|
__all__ = [
|
||
|
|
"BUSY_INPUT_FLAG",
|
||
|
|
"TOOL_PROGRESS_FLAG",
|
||
|
|
"busy_input_hint_gateway",
|
||
|
|
"busy_input_hint_cli",
|
||
|
|
"tool_progress_hint_gateway",
|
||
|
|
"tool_progress_hint_cli",
|
||
|
|
"is_seen",
|
||
|
|
"mark_seen",
|
||
|
|
]
|