mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-29 06:31:32 +00:00
Translates the full English docs corpus (335 files) into Simplified Chinese under website/i18n/zh-Hans/. Combined with PR #31895 (cross- locale link fix), the 简体中文 locale toggle now serves a complete Chinese site with working cross-page navigation. Pipeline: - Claude Sonnet 4.6 via OpenRouter, 8-way concurrent - Preserves frontmatter keys, code blocks, MDX/JSX, link URLs, brand names, and technical jargon (prompt/token/hook/MCP/ACP/etc.) - Translates only frontmatter title/description and prose - Two largest files (configuration.md 93KB, research-paper-writing.md 107KB) retried with 64K max_tokens after initial fence-drift - 3 manual post-fixes for MDX edge cases the model didn't escape: < in optional-skills-catalog table, double-quotes in an alt= tag, and a bare URL adjacent to a full-width period Cost: ~$30 total (Sonnet 4.6 input $3/M + output $15/M). Verified `npm run build` succeeds for both en and zh-Hans locales, no double-prefixed /docs/zh-Hans/docs/ URLs in rendered output, all in-page navigation resolves correctly. Translations are machine-generated and may need human review on specific pages — but they're an enormous improvement over the previous state (3 zh-Hans pages out of 335).
6.8 KiB
6.8 KiB
| sidebar_position | title | description |
|---|---|---|
| 8 | 扩展 CLI | 构建包装 CLI,通过自定义 widget、快捷键和布局变更来扩展 Hermes TUI |
扩展 CLI
Hermes 在 HermesCLI 上暴露了受保护的扩展 hook(钩子),使包装 CLI 可以添加 widget、快捷键和布局自定义,而无需覆盖超过 1000 行的 run() 方法。这样可以让你的扩展与内部变更解耦。
扩展点
共有五个扩展接缝可用:
| Hook | 用途 | 何时覆盖 |
|---|---|---|
_get_extra_tui_widgets() |
向布局注入 widget | 需要持久 UI 元素(面板、状态栏、迷你播放器)时 |
_register_extra_tui_keybindings(kb, *, input_area) |
添加键盘快捷键 | 需要热键(切换面板、传输控制、模态快捷键)时 |
_build_tui_layout_children(**widgets) |
完全控制 widget 排序 | 需要重新排序或包装现有 widget 时(少见) |
process_command() |
添加自定义斜杠命令 | 需要处理 /mycommand 时(已有 hook) |
_build_tui_style_dict() |
自定义 prompt_toolkit 样式 | 需要自定义颜色或样式时(已有 hook) |
前三个是新增的受保护 hook,后两个已存在。
快速开始:包装 CLI
#!/usr/bin/env python3
"""my_cli.py — Example wrapper CLI that extends Hermes."""
from cli import HermesCLI
from prompt_toolkit.layout import FormattedTextControl, Window
from prompt_toolkit.filters import Condition
class MyCLI(HermesCLI):
def __init__(self, **kwargs):
super().__init__(**kwargs)
self._panel_visible = False
def _get_extra_tui_widgets(self):
"""Add a toggleable info panel above the status bar."""
cli_ref = self
return [
Window(
FormattedTextControl(lambda: "📊 My custom panel content"),
height=1,
filter=Condition(lambda: cli_ref._panel_visible),
),
]
def _register_extra_tui_keybindings(self, kb, *, input_area):
"""F2 toggles the custom panel."""
cli_ref = self
@kb.add("f2")
def _toggle_panel(event):
cli_ref._panel_visible = not cli_ref._panel_visible
def process_command(self, cmd: str) -> bool:
"""Add a /panel slash command."""
if cmd.strip().lower() == "/panel":
self._panel_visible = not self._panel_visible
state = "visible" if self._panel_visible else "hidden"
print(f"Panel is now {state}")
return True
return super().process_command(cmd)
if __name__ == "__main__":
cli = MyCLI()
cli.run()
运行:
cd ~/.hermes/hermes-agent
source .venv/bin/activate
python my_cli.py
Hook 参考
_get_extra_tui_widgets()
返回要插入 TUI 布局的 prompt_toolkit widget 列表。Widget 出现在间隔区与状态栏之间——位于输入区上方、主输出区下方。
def _get_extra_tui_widgets(self) -> list:
return [] # default: no extra widgets
每个 widget 应为 prompt_toolkit 容器(如 Window、ConditionalContainer、HSplit)。使用 ConditionalContainer 或 filter=Condition(...) 可使 widget 支持切换显示。
from prompt_toolkit.layout import ConditionalContainer, Window, FormattedTextControl
from prompt_toolkit.filters import Condition
def _get_extra_tui_widgets(self):
return [
ConditionalContainer(
Window(FormattedTextControl("Status: connected"), height=1),
filter=Condition(lambda: self._show_status),
),
]
_register_extra_tui_keybindings(kb, *, input_area)
在 Hermes 注册自身快捷键之后、布局构建之前调用。将你的快捷键添加到 kb。
def _register_extra_tui_keybindings(self, kb, *, input_area):
pass # default: no extra keybindings
参数:
kb— prompt_toolkit 应用的KeyBindings实例input_area— 主TextAreawidget,用于读取或操作用户输入
def _register_extra_tui_keybindings(self, kb, *, input_area):
cli_ref = self
@kb.add("f3")
def _clear_input(event):
input_area.text = ""
@kb.add("f4")
def _insert_template(event):
input_area.text = "/search "
避免与内置快捷键冲突:Enter(提交)、Escape Enter(换行)、Ctrl-C(中断)、Ctrl-D(退出)、Tab(接受自动建议)。F2 及以上的功能键和 Ctrl 组合键通常是安全的。
_build_tui_layout_children(**widgets)
仅在需要完全控制 widget 排序时才覆盖此方法。大多数扩展应使用 _get_extra_tui_widgets() 代替。
def _build_tui_layout_children(self, *, sudo_widget, secret_widget,
approval_widget, clarify_widget, model_picker_widget=None,
spinner_widget=None, spacer, status_bar, input_rule_top,
image_bar, input_area, input_rule_bot, voice_status_bar,
completions_menu) -> list:
默认实现返回(值为 None 的 widget 会被过滤掉):
[
Window(height=0), # anchor
sudo_widget, # sudo password prompt (conditional)
secret_widget, # secret input prompt (conditional)
approval_widget, # dangerous command approval (conditional)
clarify_widget, # clarify question UI (conditional)
model_picker_widget, # model picker overlay (conditional)
spinner_widget, # thinking spinner (conditional)
spacer, # fills remaining vertical space
*self._get_extra_tui_widgets(), # YOUR WIDGETS GO HERE
status_bar, # model/token/context status line
input_rule_top, # ─── border above input
image_bar, # attached images indicator
input_area, # user text input
input_rule_bot, # ─── border below input
voice_status_bar, # voice mode status (conditional)
completions_menu, # autocomplete dropdown
]
布局示意图
默认布局从上到下:
- 输出区 — 滚动的对话历史
- 间隔区
- 额外 widget — 来自
_get_extra_tui_widgets() - 状态栏 — 模型、上下文占比、已用时间
- 图片栏 — 已附加图片数量
- 输入区 — 用户 prompt(提示词)
- 语音状态 — 录音指示器
- 补全菜单 — 自动补全建议
使用技巧
- 状态变更后刷新显示:调用
self._invalidate()触发 prompt_toolkit 重绘。 - 访问 agent 状态:
self.agent、self.model、self.conversation_history均可直接使用。 - 自定义样式:覆盖
_build_tui_style_dict()并为自定义样式类添加条目。 - 斜杠命令:覆盖
process_command(),处理自己的命令,其余一律调用super().process_command(cmd)。 - 不要覆盖
run(),除非绝对必要——扩展 hook 的存在正是为了避免这种耦合。