Plugin SDK 概述
插件 SDK 是插件与核心之间的类型化契约。此页面是关于导入什么以及可以注册什么的参考。
始终从特定的子路径导入:
import { definePluginEntry } from "openclaw/plugin-sdk/plugin-entry";import { defineChannelPluginEntry } from "openclaw/plugin-sdk/channel-core";Each subpath is a small, self-contained module. This keeps startup fast and
prevents circular dependency issues. For 渠道-specific entry/build helpers,
prefer openclaw/plugin-sdk/channel-core; keep openclaw/plugin-sdk/core for
the broader umbrella surface and shared helpers such as
buildChannelConfigSchema.
For 渠道 config, publish the 渠道-owned JSON Schema through
openclaw.plugin.json#channelConfigs. The plugin-sdk/channel-config-schemaOpenClaw
subpath is for shared schema primitives and the generic builder. OpenClaw’s
bundled plugins use plugin-sdk/bundled-channel-config-schema for retained
bundled-渠道 schemas. Deprecated compatibility exports remain on
plugin-sdk/channel-config-schema-legacy; neither bundled schema subpath is a
pattern for new plugins.
插件 SDK 作为一组按区域分组的狭窄子路径公开(插件 入口、渠道、提供商、身份验证、运行时、功能、内存和保留的 捆绑插件辅助程序)。有关完整的目录——已分组并链接——请参阅 插件 SDK 子路径。
编译器入口清单位于
scripts/lib/plugin-sdk-entrypoints.json;包导出是从
公共子集生成的,该子集减去了 scripts/lib/plugin-sdk-private-local-only-subpaths.json 中列出的仓库本地测试/内部子路径。
运行 pnpm plugin-sdk:surface 以审核公共导出计数。足够陈旧且未被捆绑扩展生产代码使用的已弃用公共
子路径在 scripts/lib/plugin-sdk-deprecated-public-subpaths.json 中跟踪;广泛的
已弃用重新导出桶在 scripts/lib/plugin-sdk-deprecated-barrel-subpaths.json 中跟踪。
注册 API
Section titled “注册 API”register(api) 回调接收一个 OpenClawPluginApi 对象,其中包含以下
方法:
| 方法 | 注册内容 |
|---|---|
api.registerProvider(...) | 文本推理 (LLM) |
api.registerAgentHarness(...) | 实验性低级代理执行器 |
api.registerCliBackend(...) | 本地 CLI 推理后端 |
api.registerChannel(...) | 消息渠道 |
api.registerSpeechProvider(...) | 文本转语音 / STT 合成 |
api.registerRealtimeTranscriptionProvider(...) | 流式实时转录 |
api.registerRealtimeVoiceProvider(...) | 双工实时语音会话 |
api.registerMediaUnderstandingProvider(...) | 图像/音频/视频分析 |
api.registerImageGenerationProvider(...) | 图像生成 |
api.registerMusicGenerationProvider(...) | 音乐生成 |
api.registerVideoGenerationProvider(...) | 视频生成 |
api.registerWebFetchProvider(...) | Web 获取 / 抓取提供商 |
api.registerWebSearchProvider(...) | Web 搜索 |
对于具有固定工具名称的简单纯工具插件,请使用 defineToolPlugin。对于混合插件或完全动态的工具注册,请直接使用 api.registerTool(...)。
| 方法 | 注册内容 |
|---|---|
api.registerTool(tool, opts?) | Agent 工具(必需或 { optional: true }) |
api.registerCommand(def) | 自定义命令(绕过 LLM) |
当 Agent 需要简短的、由命令拥有的路由提示时,插件命令可以设置 agentPromptGuidance。保持该文本关于命令本身;不要将提供商或插件特定的策略添加到核心提示构建器中。
指导条目可能是旧式字符串(适用于每个提示表面),也可能是结构化条目:
agentPromptGuidance: ["Global command hint.", { text: "Only show this in the main PI prompt.", surfaces: ["pi_main"] }];结构化 surfaces 可能包括 pi_main、codex_app_server、cli_backend、acp_backend 或 subagent。对于有意的全表面指导,请省略 surfaces。不要传递空的 surfaces 数组;它会被拒绝,以免意外的范围丢失变成全局提示文本。
原生 Codex 应用服务器开发者的指令比其他提示表面更严格:只有显式限定范围为 codex_app_server 的指导才会被提升到那个更高优先级的通道。为了兼容性,旧式字符串指导和未限定范围的结构化指导仍然可用于非 Codex 提示表面。
| 方法 | 注册内容 |
|---|---|
api.registerHook(events, handler, opts?) | 事件钩子 |
api.registerHttpRoute(params) | Gateway(网关) HTTP 端点 |
api.registerGatewayMethod(name, handler) | Gateway(网关) RPC 方法 |
api.registerGatewayDiscoveryService(service) | 本地 Gateway(网关) 发现通告器 |
api.registerCli(registrar, opts?) | CLI 子命令 |
api.registerNodeCliFeature(registrar, opts?) | 下的节点特性 CLIopenclaw nodes |
api.registerService(service) | 后台服务 |
api.registerInteractiveHandler(registration) | 交互式处理程序 |
api.registerAgentToolResultMiddleware(...) | 运行时工具结果中间件 |
api.registerMemoryPromptSupplement(builder) | 附加的内存相邻提示部分 |
api.registerMemoryCorpusSupplement(adapter) | 附加的内存搜索/读取语料库 |
工作流插件的主机钩子
Section titled “工作流插件的主机钩子”主机钩子是插件的 SDK 接缝,适用于需要参与主机生命周期而不仅仅是添加提供商、渠道或工具的插件。它们是通用合约;计划模式可以使用它们,审批工作流、工作区策略网关、后台监视器、设置向导和 UI 伴生插件也可以使用它们。
| 方法 | 拥有的合约 |
|---|---|
api.session.state.registerSessionExtension(...) | 插件拥有的、通过 Gateway(网关) 会话投影的 JSON 兼容会话状态 |
api.session.workflow.enqueueNextTurnInjection(...) | 注入到单个会话的下一个 Agent 轮次中的持久且仅一次的上下文 |
api.registerTrustedToolPolicy(...) | 捆绑/受信任的插件前工具策略,可以阻止或重写工具参数 |
api.registerToolMetadata(...) | 工具目录显示元数据,不更改工具实现 |
api.registerCommand(...) | 作用域插件命令;命令结果可以设置 continueAgent: trueDiscord;Discord 原生命令支持 descriptionLocalizations |
api.session.controls.registerControlUiDescriptor(...) | 用于会话、工具、运行或设置表面的控制 UI 贡献描述符 |
api.lifecycle.registerRuntimeLifecycle(...) | 在重置/删除/重新加载路径上清理插件拥有的运行时资源的回调 |
api.agent.events.registerAgentEventSubscription(...) | 用于工作流状态和监视器的清理事件订阅 |
api.runContext.setRunContext(...) / getRunContext(...) / clearRunContext(...) | 每次运行的插件临时状态,在终端运行生命周期时清除 |
api.session.workflow.registerSessionSchedulerJob(...) | 插件拥有的调度器作业的清理元数据;不调度工作或创建任务记录 |
api.session.workflow.sendSessionAttachment(...) | 仅限捆绑的、主机介导的文件附件传递到活动的直接出站会话路由 |
api.session.workflow.scheduleSessionTurn(...) / unscheduleSessionTurnsByTag(...) | 仅限捆绑的基于 Cron 的计划会话轮次以及基于标签的清理 |
api.session.controls.registerSessionAction(...) | 客户端可以通过 Gateway(网关) 分发的类型化会话操作 |
对于新的插件代码,请使用分组的命名空间:
api.session.state.registerSessionExtension(...)api.session.workflow.enqueueNextTurnInjection(...)api.session.workflow.registerSessionSchedulerJob(...)api.session.workflow.sendSessionAttachment(...)api.session.workflow.scheduleSessionTurn(...)api.session.workflow.unscheduleSessionTurnsByTag(...)api.session.controls.registerSessionAction(...)api.session.controls.registerControlUiDescriptor(...)api.agent.events.registerAgentEventSubscription(...)api.agent.events.emitAgentEvent(...)api.runContext.setRunContext(...)/getRunContext(...)/clearRunContext(...)api.lifecycle.registerRuntimeLifecycle(...)
等效的扁平方法仍作为已弃用的兼容别名提供给现有插件。请不要添加调用
api.registerSessionExtension、api.enqueueNextTurnInjection、
api.registerControlUiDescriptor、api.registerRuntimeLifecycle、
api.registerAgentEventSubscription、api.emitAgentEvent、
api.setRunContext、api.getRunContext、api.clearRunContext、
api.registerSessionSchedulerJob、api.registerSessionAction、
api.sendSessionAttachment、api.scheduleSessionTurn 或
api.unscheduleSessionTurnsByTag 的新插件代码。
scheduleSessionTurn(...) 是基于 Gateway(网关) Cron 调度器的会话级便捷工具。Cron 负责计时并在轮次运行时创建后台任务记录;Plugin SDK 仅限制目标会话、插件拥有的命名和清理。当工作本身需要持久的多步骤任务流状态时,请在计划的轮次内使用 api.runtime.tasks.managedFlows。
这些契约有意划分了权限:
- 外部插件可以拥有会话扩展、UI 描述符、命令、工具元数据、下一次轮次注入和普通钩子。
- 受信任的工具策略在普通
before_tool_call钩子之前运行,并且仅限打包使用,因为它们参与主机安全策略。 - 保留的命令所有权仅限打包使用。外部插件应使用其自己的命令名称或别名。
allowPromptInjection=false禁用了提示变更钩子,包括agent_turn_prepare、before_prompt_build、heartbeat_prompt_contribution、 来自旧版before_agent_start的提示字段以及enqueueNextTurnInjection。
非 Plan 使用者的示例:
| 插件原型 | 使用的钩子 |
|---|---|
| 审批工作流 | 会话扩展、命令继续、下一轮注入、UI 描述符 |
| 预算/工作区策略网关 | 受信任工具策略、工具元数据、会话投影 |
| 后台生命周期监视器 | 运行时生命周期清理、代理事件订阅、会话调度器所有权/清理、心跳提示贡献、UI 描述符 |
| 设置或新手引导向导 | 会话扩展、作用域命令、控制 UI 描述符 |
何时使用工具结果中间件
捆绑插件可以在执行后、运行时将结果反馈给模型之前重写工具结果时使用 api.registerAgentToolResultMiddleware(...)。这是受信任的、与运行时无关的
用于异步输出缩减器(如 tokenjuice)的连接点。
捆绑插件必须为每个
目标运行时声明 contracts.agentToolResultMiddleware,例如 ["pi", "codex"]。外部插件
无法注册此中间件;对于不需要
模型前工具结果计时的工作,请保留正常的 OpenClaw 插件钩子。旧的仅限 Pi 的嵌入式
扩展工厂注册路径已被移除。
Gateway(网关) 发现注册
Section titled “Gateway(网关) 发现注册”api.registerGatewayDiscoveryService(...) 允许插件通过本地设备发现传输(如 mDNS/Gateway(网关))通告活动的 Bonjour。当启用本地设备发现时,OpenClaw 会在 Gateway(网关) 启动期间调用该服务,传入当前 Gateway(网关) 端口和非机密 TXT 提示数据,并在 Gateway(网关) 关闭期间调用返回的 stop 处理程序。
api.registerGatewayDiscoveryService({ id: "my-discovery", async advertise(ctx) { const handle = await startMyAdvertiser({ gatewayPort: ctx.gatewayPort, tls: ctx.gatewayTlsEnabled, displayName: ctx.machineDisplayName, }); return { stop: () => handle.stop() }; },});Gateway(网关) 设备发现插件不得将通告的 TXT 值视为机密信息或身份验证凭据。设备发现仅作为路由提示;信任关系仍由 Gateway(网关) 身份验证和 TLS 固定拥有。
CLI 注册元数据
Section titled “CLI 注册元数据”api.registerCli(registrar, opts?) 接受两种类型的命令元数据:
commands:由注册者拥有的显式命令名称descriptors:用于 CLI 帮助、路由和延迟加载插件 CLI 注册的解析时命令描述符parentPath:用于嵌套命令组的可选父命令路径,例如["nodes"]
对于配对节点功能,首选 api.registerNodeCliFeature(registrar, opts?)。它是 api.registerCli(..., { parentPath: ["nodes"] }) 的一个轻量级包装器,并使诸如 openclaw nodes canvas 之类的命令成为明确的插件拥有的节点功能。
如果您希望插件命令在常规根 CLI 路径中保持延迟加载,请提供覆盖该注册器公开的每个顶级命令根的 descriptors。
api.registerCli( async ({ program }) => { const { registerMatrixCli } = await import("./src/cli.js"); registerMatrixCli({ program }); }, { descriptors: [ { name: "matrix", description: "Manage Matrix accounts, verification, devices, and profile state", hasSubcommands: true, }, ], },);嵌套命令将解析后的父命令作为 program 接收:
api.registerCli( async ({ program }) => { const { registerNodesCanvasCommands } = await import("./src/cli.js"); registerNodesCanvasCommands(program); }, { parentPath: ["nodes"], descriptors: [ { name: "canvas", description: "Capture or render canvas content from a paired node", hasSubcommands: true, }, ], },);仅当您不需要延迟加载根 CLI 注册时,才单独使用 commands。该急切兼容性路径仍然受支持,但它不会安装用于解析时延迟加载的描述符支持的占位符。
CLI 后端注册
Section titled “CLI 后端注册”api.registerCliBackend(...) 允许插件拥有本地 CLI 后端(例如 claude-cli 或 my-cli)的默认配置。
- 后端
id成为模型引用(如my-cli/gpt-5)中的提供商前缀。 - 后端
config使用与agents.defaults.cliBackends.<id>相同的结构。 - 用户配置仍然优先。OpenClaw 会在运行 CLI 之前,将
agents.defaults.cliBackends.<id>与插件默认值合并。 - 当后端在合并后需要进行兼容性重写(例如标准化旧标志的结构)时,请使用
normalizeConfig。 - 对于属于 CLI 方言的请求范围 argv 重写(例如将 OpenClaw 的思考级别映射到原生 effort 标志),请使用
resolveExecutionArgs。
有关端到端创作指南,请参阅CLI 后端插件。
| 方法 | 注册内容 |
|---|---|
api.registerContextEngine(id, factory) | 上下文引擎(一次激活一个)。assemble() 回调接收 availableTools 和 citationsMode,以便引擎可以定制提示词补充内容。 |
api.registerMemoryCapability(capability) | 统一内存功能 |
api.registerMemoryPromptSection(builder) | 内存提示词部分构建器 |
api.registerMemoryFlushPlan(resolver) | 内存刷新计划解析器 |
api.registerMemoryRuntime(runtime) | 内存运行时适配器 |
内存嵌入适配器
Section titled “内存嵌入适配器”| 方法 | 注册内容 |
|---|---|
api.registerMemoryEmbeddingProvider(adapter) | 活动插件的内存嵌入适配器 |
registerMemoryCapability是首选的独占内存插件 API。registerMemoryCapability还可以公开publicArtifacts.listArtifacts(...),以便配套插件可以通过openclaw/plugin-sdk/memory-host-core使用导出的内存工件,而无需深入特定内存插件的私有布局。registerMemoryPromptSection、registerMemoryFlushPlan和registerMemoryRuntime是旧版兼容的专用内存插件 API。MemoryFlushPlan.model可以将刷新轮次固定到确切的provider/model引用(例如ollama/qwen3:8b),而无需继承活动的回退 链。registerMemoryEmbeddingProvider允许活动的内存插件注册一个 或多个嵌入适配器 ID(例如openai、gemini或自定义 插件定义的 ID)。- 用户配置(如
agents.defaults.memorySearch.provider和agents.defaults.memorySearch.fallback)将根据这些已注册的 适配器 ID 进行解析。
事件和生命周期
Section titled “事件和生命周期”| 方法 | 作用 |
|---|---|
api.on(hookName, handler, opts?) | 类型化生命周期钩子 |
api.onConversationBindingResolved(handler) | 会话绑定回调 |
有关示例、常用钩子名称和保护语义,请参阅 插件钩子。
钩子决策语义
Section titled “钩子决策语义”before_tool_call:返回{ block: true }是终局性的。一旦任何处理程序设置了它,将跳过低优先级的处理程序。before_tool_call:返回{ block: false }被视为无决策(与省略block相同),而不是覆盖。before_install:返回{ block: true }是终局性的。一旦任何处理程序设置了它,将跳过低优先级的处理程序。before_install:返回{ block: false }被视为无决策(与省略block相同),而不是覆盖。reply_dispatch:返回{ handled: true, ... }是终局性的。一旦任何处理程序声明了分派,将跳过低优先级的处理程序和默认模型分派路径。message_sending:返回{ cancel: true }是终局性的。一旦任何处理程序设置了它,将跳过低优先级的处理程序。message_sending:返回{ cancel: false }被视为无决策(与省略cancel相同),而不是覆盖。message_received:当需要入站线程/主题路由时,请使用类型化的threadId字段。将metadata用于渠道特定的额外信息。message_sending:在回退到渠道特定的metadata之前,使用类型化的replyToId/threadId路由字段。gateway_start:使用ctx.config、ctx.workspaceDir和ctx.getCron?.()处理网关拥有的启动状态,而不是依赖内部的gateway:startup钩子。cron_changed:观察网关拥有的 cron 生命周期变更。在同步外部唤醒调度器时使用event.job?.state?.nextRunAtMs和ctx.getCron?.(),并保持 OpenClaw 作为到期检查和执行的单一事实来源。
API 对象字段
Section titled “API 对象字段”| 字段 | 类型 | 描述 |
|---|---|---|
api.id | string | 插件 ID |
api.name | string | 显示名称 |
api.version | string? | 插件版本(可选) |
api.description | string? | 插件描述(可选) |
api.source | string | 插件源路径 |
api.rootDir | string? | 插件根目录(可选) |
api.config | OpenClawConfig | 当前配置快照(可用时为活动的内存运行时快照) |
api.pluginConfig | Record<string, unknown> | 来自 plugins.entries.<id>.config 的插件特定配置 |
api.runtime | PluginRuntime | 运行时辅助工具 |
api.logger | PluginLogger | 作用域日志记录器(debug、info、warn、error) |
api.registrationMode | PluginRegistrationMode | 当前加载模式;"setup-runtime" 是完整的条目启动/设置窗口之前的轻量级阶段 |
api.resolvePath(input) | (string) => string | 解析相对于插件根目录的路径 |
内部模块约定
Section titled “内部模块约定”在您的插件内部,使用本地桶文件(barrel files)进行内部导入:
my-plugin/ api.ts # Public exports for external consumers runtime-api.ts # Internal-only runtime exports index.ts # Plugin entry point setup-entry.ts # Lightweight setup-only entry (optional)Facade 加载的捆绑插件公开表面(api.ts、runtime-api.ts、
index.ts、setup-entry.ts 和类似的公共入口文件)在 OpenClaw
已在运行时优先使用活动运行时配置快照。如果尚不存在运行时快照,它们将回退到磁盘上已解析的配置文件。
打包的捆绑插件外观应通过 OpenClaw 的插件外观加载器加载;
从 dist/extensions/... 直接导入会绕过清单和运行时 sidecar 检查,而这些检查是打包安装用于插件自有代码的。
当辅助工具有意针对特定提供商(提供商)且尚不属于通用 SDK 子路径时,提供商插件可以公开一个狭窄的插件本地契约桶。捆绑示例:
- **Anthropic:公开的
api.ts/contract-api.ts接缝,用于 Claude beta-header 和service_tier流辅助工具。 - **
@openclaw/openai-provider:api.ts导出提供商构建器、 默认模型辅助工具和实时提供商构建器。 - **
@openclaw/openrouter-provider:api.ts导出提供商构建器 以及新手引导/配置辅助工具。
definePluginEntry 和 defineChannelPluginEntry 选项。
完整的 api.runtime 命名空间参考。
打包、清单和配置架构。
测试工具和 Lint 规则。
从已弃用的接口迁移。
深度架构和模型能力。