跳到主要内容

构建 Memory Provider 插件

Memory provider 插件为 Hermes Agent 提供跨 session 的持久化知识,超越内置的 MEMORY.md 和 USER.md。本指南介绍如何构建一个 memory provider 插件。

目录结构

每个 memory provider 位于 plugins/memory/<name>/ 目录下:

plugins/memory/my-provider/
├── __init__.py # MemoryProvider 实现 + register() 入口
├── plugin.yaml # 元数据(名称、描述、hooks)
└── README.md # 安装说明、配置参考、工具列表

MemoryProvider 抽象基类

你的插件需要实现 agent/memory_provider.py 中的 MemoryProvider 抽象基类:

from agent.memory_provider import MemoryProvider

class MyMemoryProvider(MemoryProvider):
@property
def name(self) -> str:
return "my-provider"

def is_available(self) -> bool:
"""检查该 provider 是否可以激活。不得发起网络请求。"""
return bool(os.environ.get("MY_API_KEY"))

def initialize(self, session_id: str, **kwargs) -> None:
"""在 agent 启动时调用一次。

kwargs 始终包含:
hermes_home (str): 当前活跃的 HERMES_HOME 路径,用于存储数据。
"""
self._api_key = os.environ.get("MY_API_KEY", "")
self._session_id = session_id

# ... 实现其余方法

必须实现的方法

核心生命周期

方法调用时机是否必须实现?
name(属性)始终
is_available()agent 初始化时,激活前 — 不得发起网络请求
initialize(session_id, **kwargs)agent 启动时
get_tool_schemas()初始化后,用于注入工具
handle_tool_call(name, args)agent 调用你的工具时(如果有工具的话)

配置

方法用途是否必须实现?
get_config_schema()声明 hermes memory setup 的配置字段
save_config(values, hermes_home)将非密钥配置写入原生位置(纯环境变量 provider 除外)

可选 Hooks

方法调用时机使用场景
system_prompt_block()组装系统提示词时静态 provider 信息
prefetch(query)每次 API 调用前返回召回的上下文
queue_prefetch(query)每轮对话后为下一轮预热
sync_turn(user, assistant)每轮对话完成后持久化对话内容
on_session_end(messages)对话结束时最终提取/刷新
on_pre_compress(messages)上下文压缩前在丢弃前保存洞察
on_memory_write(action, target, content)内置 memory 写入时同步到你的后端
shutdown()进程退出时清理连接

Config Schema

get_config_schema() 返回字段描述符列表,供 hermes memory setup 使用:

def get_config_schema(self):
return [
{
"key": "api_key",
"description": "My Provider API key",
"secret": True, # → 写入 .env
"required": True,
"env_var": "MY_API_KEY", # 显式环境变量名
"url": "https://my-provider.com/keys", # 获取密钥的地址
},
{
"key": "region",
"description": "Server region",
"default": "us-east",
"choices": ["us-east", "eu-west", "ap-south"],
},
{
"key": "project",
"description": "Project identifier",
"default": "hermes",
},
]

设置了 secret: Trueenv_var 的字段会写入 .env,非密钥字段则传递给 save_config()

提示

最简 Schema vs 完整 Schema get_config_schema() 中的每个字段都会在 hermes memory setup 期间被提示输入。选项较多的 provider 应保持 schema 最简——只包含用户必须配置的字段(API key、必要凭证)。将可选配置记录在配置文件参考中(例如 $HERMES_HOME/myprovider.json),而不是在设置向导中全部提示。这样既能保持设置向导快速高效,又支持高级配置。可参考 Supermemory provider 的示例——它只提示输入 API key,所有其他选项均位于 supermemory.json 中。

保存配置

def save_config(self, values: dict, hermes_home: str) -> None:
"""将非密钥配置写入原生位置。"""
import json
from pathlib import Path
config_path = Path(hermes_home) / "my-provider.json"
config_path.write_text(json.dumps(values, indent=2))

对于纯环境变量的 provider,保留默认的空操作即可。

插件入口

def register(ctx) -> None:
"""由 memory 插件发现系统调用。"""
ctx.register_memory_provider(MyMemoryProvider())

plugin.yaml

name: my-provider
version: 1.0.0
description: "该 provider 功能的简短描述。"
hooks:
- on_session_end # 列出你实现的 hooks

线程契约

sync_turn() 必须是非阻塞的。 如果你的后端存在延迟(API 调用、LLM 处理),请在守护线程中执行:

def sync_turn(self, user_content, assistant_content):
def _sync():
try:
self._api.ingest(user_content, assistant_content)
except Exception as e:
logger.warning("同步失败:%s", e)

if self._sync_thread and self._sync_thread.is_alive():
self._sync_thread.join(timeout=5.0)
self._sync_thread = threading.Thread(target=_sync, daemon=True)
self._sync_thread.start()

Profile 隔离

所有存储路径必须使用 initialize() 中的 hermes_home kwarg,而不是硬编码 ~/.hermes

# 正确 — 限定在 profile 范围内
from hermes_constants import get_hermes_home
data_dir = get_hermes_home() / "my-provider"

# 错误 — 在所有 profile 间共享
data_dir = Path("~/.hermes/my-provider").expanduser()

测试

完整的端到端测试示例(使用真实 SQLite provider)请参见 tests/agent/test_memory_plugin_e2e.py

from agent.memory_manager import MemoryManager

mgr = MemoryManager()
mgr.add_provider(my_provider)
mgr.initialize_all(session_id="test-1", platform="cli")

# 测试工具路由
result = mgr.handle_tool_call("my_tool", {"action": "add", "content": "test"})

# 测试生命周期
mgr.sync_all("user msg", "assistant msg")
mgr.on_session_end([])
mgr.shutdown_all()

添加 CLI 命令

Memory provider 插件可以注册自己的 CLI 子命令树(例如 hermes my-provider statushermes my-provider config)。该功能通过基于约定的发现系统实现——无需修改核心文件。

工作原理

  1. 在插件目录中添加 cli.py 文件
  2. 定义 register_cli(subparser) 函数,用于构建 argparse 命令树
  3. memory 插件系统在启动时通过 discover_plugin_cli_commands() 自动发现
  4. 你的命令将出现在 hermes <provider-name>

激活 provider 限制: 你的 CLI 命令仅在你的 provider 是配置中激活的 memory.provider 时才会出现。如果用户未配置你的 provider,你的命令不会在 hermes --help 中显示。

示例

# plugins/memory/my-provider/cli.py

def my_command(args):
"""由 argparse 调度的处理函数。"""
sub = getattr(args, "my_command", None)
if sub == "status":
print("Provider 处于活跃状态且已连接。")
elif sub == "config":
print("正在显示配置...")
else:
print("用法:hermes my-provider <status|config>")

def register_cli(subparser) -> None:
"""构建 hermes my-provider 的 argparse 命令树。

在 argparse 设置阶段由 discover_plugin_cli_commands() 调用。
"""
subs = subparser.add_subparsers(dest="my_command")
subs.add_parser("status", help="显示 provider 状态")
subs.add_parser("config", help="显示 provider 配置")
subparser.set_defaults(func=my_command)

参考实现

完整示例(包含 13 个子命令、跨 profile 管理(--target-profile)及配置读写)请参见 plugins/memory/honcho/cli.py

包含 CLI 的目录结构

plugins/memory/my-provider/
├── __init__.py # MemoryProvider 实现 + register()
├── plugin.yaml # 元数据
├── cli.py # register_cli(subparser) — CLI 命令
└── README.md # 安装说明

单一 Provider 规则

同一时间只能激活一个外部 memory provider。如果用户尝试注册第二个,MemoryManager 将以警告拒绝。这可防止 tool schema 膨胀和后端冲突。