知识引擎/Hermes 知识引擎/构建记忆 Provider(提供者) 插件

原文链接:Memory Provider Plugins sidebar position: 8 title: "记忆 Provider 插件" description: "如何为 Hermes Agent 构建记忆 Provider 插件" 记忆 Provider 插件为 Hermes Agent 提供超越内置 ME

> 原文链接:Memory Provider Plugins


sidebar_position: 8 title: "记忆 Provider 插件" description: "如何为 Hermes Agent 构建记忆 Provider 插件"

构建记忆 Provider(提供者) 插件

记忆 Provider 插件为 Hermes Agent 提供超越内置 MEMORY.md 和 USER.md 的持久化、跨会话知识能力。本指南介绍如何构建一个记忆 Provider 插件。

:::tip 提示 记忆 Provider 是两种 Provider 插件(Provider Plugin) 类型之一。另一种是上下文引擎插件(Context Engine Plugin),用于替换内置的上下文压缩器。两者遵循相同的模式:单选、配置驱动、通过 hermes plugins 管理。 :::

目录结构

每个记忆 Provider 存放在 plugins/memory/<name>/ 中:

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

MemoryProvider 抽象基类(ABC)

你的插件需要实现 agent/memory_provider.py 中的 MemoryProvider 抽象基类(ABC, Abstract Base Class):

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 (property)始终
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)将非敏感配置写入原生位置(除非仅使用环境变量)

可选钩子(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)内置记忆写入时镜像到你的后端
shutdown()进程退出时清理连接

配置 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": "服务器区域",
            "default": "us-east",
            "choices": ["us-east", "eu-west", "ap-south"],
        },
        {
            "key": "project",
            "description": "项目标识符",
            "default": "hermes",
        },
    ]

带有 secret: Trueenv_var 的字段会写入 .env。非敏感字段会传递给 save_config()

:::tip 最小化 vs 完整 Schema get_config_schema() 中的每个字段都会在 hermes memory setup 期间被提示。具有许多选项的 Provider 应保持 Schema 最小化——仅包含用户必须配置的字段(API 密钥、必需的凭据)。将可选设置记录在配置文件参考中(例如 $HERMES_HOME/myprovider.json),而不是在设置时逐一提示。这样可以使设置向导保持快速,同时仍然支持高级配置。请参阅 Supermemory Provider 示例——它只提示输入 API 密钥;所有其他选项都存放在 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:
    """由记忆插件发现系统调用。"""
    ctx.register_memory_provider(MyMemoryProvider())

plugin.yaml

name: my-provider
version: 1.0.0
description: "对此 Provider 功能的简要描述。"
hooks:
  - on_session_end    # 列出你实现的钩子

线程契约

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 参数,而不是硬编码的 ~/.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 的完整端到端(E2E)测试模式,请参阅 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 命令

记忆 Provider 插件可以注册自己的 CLI 子命令树(例如 hermes my-provider statushermes my-provider config)。这使用基于约定的发现系统——无需修改核心文件。

工作原理

  1. 在你的插件目录中添加一个 cli.py 文件
  2. 定义一个 register_cli(subparser) 函数来构建 argparse 树
  3. 记忆插件系统在启动时通过 discover_plugin_cli_commands() 发现它
  4. 你的命令出现在 hermes <provider-name> <subcommand>

活跃 Provider 门控: 只有当你的 Provider 是配置中的活跃 memory.provider 时,你的 CLI 命令才会出现。如果用户没有配置你的 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)

参考实现

请参阅 plugins/memory/honcho/cli.py,其中包含 13 个子命令、跨 profile 管理(--target-profile)和配置读写的完整示例。

包含 CLI 的目录结构

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

单 Provider 规则

同一时间只能有一个外部记忆 Provider 处于活跃状态。如果用户尝试注册第二个,MemoryManager 会拒绝并发出警告。这可以防止工具 Schema 膨胀和后端冲突。

Continue Exploring

继续探索

这不是课程式的上一篇下一篇,而是从当前节点向外继续漫游。

开发者指南

构建上下文引擎插件(Context Engine Plugin)

原文链接:Context Engine Plugins sidebar position: 9 title: "上下文引擎插件" description: "如何构建替换内置 ContextCompressor 的上下文引擎插件" 上下文引擎插件用替代策略替换内置的 ContextCompressor 来管理对话上下文

开发者指南

架构

本页面是 Hermes Agent 内部结构的顶层地图. 使用它来了解代码库的整体结构,然后深入子系统文档获取实现细节。 如果你是首次接触此代码库: 1. 本页面 — 了解整体结构 2. Agent 循环内部机制 — AIAgent 如何工作 3. 提示词组装 — 系统提示词构建

开发者指南

贡献指南

感谢你对 Hermes Agent 的贡献!本指南涵盖开发环境设置、理解代码库以及如何让你的 PR 被合并。 我们按以下顺序重视贡献: 1. Bug 修复 — 崩溃、不正确行为、数据丢失 2. 跨平台兼容性 — macOS、不同 Linux 发行版、WSL2 3. 安全加固 — Shell 注入、Prompt 注入、路

开发者指南

Agent 循环内部机制

sidebar position: 3 title: "Agent 循环内部机制" description: "AIAgent 执行、API 模式、工具、回调和回退行为的详细解析" 核心编排引擎是 run agent.py 中的 AIAgent 类——大约 10,700 行代码,负责从 Prompt(提示词)组装到工具

开发者指南

提示词组装

sidebar position: 5 title: "提示词组装" description: "Hermes 如何构建系统提示词、保持缓存稳定性和注入临时层" Hermes 刻意将以下两者分离: - 缓存的系统提示词状态 - 临时的 API 调用时添加内容 这是项目中最关键的设计决策之一,因为它影响:

开发者指南

上下文压缩与缓存

Hermes Agent 使用双重压缩系统和 Anthropic 提示词缓存,在长对话中高效管理上下文窗口的使用。 源文件:agent/context engine.py(ABC)、agent/context compressor.py(默认引擎)、agent/prompt caching.py、gateway/run

Developer Guide

开发者指南

面向二次开发者,解释架构、运行时、上下文引擎、插件、工具与扩展机制。

20 篇文档20 个节点

当前节点

构建记忆 Provider(提供者) 插件

同主题继续探索

架构

本页面是 Hermes Agent 内部结构的顶层地图. 使用它来了解代码库的整体结构,然后深入子系统文档获取实现细节。 如果你是首次接触此代码库: 1. 本页面 — 了解整体结构 2. Agent 循环内部机制 — AIAgent 如何工作 3. 提示词组装 — 系统提示词构建

贡献指南

感谢你对 Hermes Agent 的贡献!本指南涵盖开发环境设置、理解代码库以及如何让你的 PR 被合并。 我们按以下顺序重视贡献: 1. Bug 修复 — 崩溃、不正确行为、数据丢失 2. 跨平台兼容性 — macOS、不同 Linux 发行版、WSL2 3. 安全加固 — Shell 注入、Prompt 注入、路

Agent 循环内部机制

sidebar position: 3 title: "Agent 循环内部机制" description: "AIAgent 执行、API 模式、工具、回调和回退行为的详细解析" 核心编排引擎是 run agent.py 中的 AIAgent 类——大约 10,700 行代码,负责从 Prompt(提示词)组装到工具

提示词组装

sidebar position: 5 title: "提示词组装" description: "Hermes 如何构建系统提示词、保持缓存稳定性和注入临时层" Hermes 刻意将以下两者分离: - 缓存的系统提示词状态 - 临时的 API 调用时添加内容 这是项目中最关键的设计决策之一,因为它影响:

上下文压缩与缓存

Hermes Agent 使用双重压缩系统和 Anthropic 提示词缓存,在长对话中高效管理上下文窗口的使用。 源文件:agent/context engine.py(ABC)、agent/context compressor.py(默认引擎)、agent/prompt caching.py、gateway/run

网关内部机制

sidebar position: 7 title: "网关内部机制" description: "消息网关如何启动、授权用户、路由会话和投递消息" 消息网关是一个长运行进程,通过统一架构将 Hermes 连接到 14+ 个外部消息平台。 当消息从任何平台到达时: 1. 平台适配器 接收原始事件,将其规范化为 Mess

相关节点

构建上下文引擎插件(Context Engine Plugin)

原文链接:Context Engine Plugins sidebar position: 9 title: "上下文引擎插件" description: "如何构建替换内置 ContextCompressor 的上下文引擎插件" 上下文引擎插件用替代策略替换内置的 ContextCompressor 来管理对话上下文

架构

本页面是 Hermes Agent 内部结构的顶层地图. 使用它来了解代码库的整体结构,然后深入子系统文档获取实现细节。 如果你是首次接触此代码库: 1. 本页面 — 了解整体结构 2. Agent 循环内部机制 — AIAgent 如何工作 3. 提示词组装 — 系统提示词构建

贡献指南

感谢你对 Hermes Agent 的贡献!本指南涵盖开发环境设置、理解代码库以及如何让你的 PR 被合并。 我们按以下顺序重视贡献: 1. Bug 修复 — 崩溃、不正确行为、数据丢失 2. 跨平台兼容性 — macOS、不同 Linux 发行版、WSL2 3. 安全加固 — Shell 注入、Prompt 注入、路

Agent 循环内部机制

sidebar position: 3 title: "Agent 循环内部机制" description: "AIAgent 执行、API 模式、工具、回调和回退行为的详细解析" 核心编排引擎是 run agent.py 中的 AIAgent 类——大约 10,700 行代码,负责从 Prompt(提示词)组装到工具

提示词组装

sidebar position: 5 title: "提示词组装" description: "Hermes 如何构建系统提示词、保持缓存稳定性和注入临时层" Hermes 刻意将以下两者分离: - 缓存的系统提示词状态 - 临时的 API 调用时添加内容 这是项目中最关键的设计决策之一,因为它影响:

上下文压缩与缓存

Hermes Agent 使用双重压缩系统和 Anthropic 提示词缓存,在长对话中高效管理上下文窗口的使用。 源文件:agent/context engine.py(ABC)、agent/context compressor.py(默认引擎)、agent/prompt caching.py、gateway/run