快速开始

Canvas 插件重构

Canvas 插件重构

Canvas 使用率较低且处于实验阶段。应将它视为内置插件,而不是核心功能。核心可以保留通用 Gateway 网关、节点、HTTP、凭证、配置和原生客户端管道,但 Canvas 特定行为应位于 extensions/canvas 下。

目标

将 Canvas 所有权移至 extensions/canvas,同时保留当前的配对节点行为:

  • 面向智能体的 canvas 工具由 Canvas 插件注册
  • 只有在 Canvas 插件注册 Canvas 节点命令时,才允许这些命令
  • A2UI 主机/源文件位于 Canvas 插件下
  • Canvas 文档实体化位于 Canvas 插件下
  • CLI 命令实现位于 Canvas 插件下,或通过插件拥有的运行时 barrel 委托
  • 文档和插件清单将 Canvas 描述为实验性且由插件支持

非目标

  • 不要在此次重构中重新设计原生应用 Canvas UI。
  • 除非单独的产品决策说明应删除 Canvas,否则不要从 iOS、Android 或 macOS 中移除 Canvas 协议/客户端支持。
  • 不要仅为 Canvas 构建宽泛的插件服务框架,除非至少另一个内置插件需要相同的接缝。

当前分支状态

已完成:

  • extensions/canvas 中添加了内置插件包。
  • 添加了 extensions/canvas/openclaw.plugin.json
  • 将智能体 canvas 工具从 src/agents/tools/canvas-tool.ts 移至 extensions/canvas/src/tool.ts
  • src/agents/openclaw-tools.ts 中移除了核心对 createCanvasTool 的注册。
  • 将 Canvas 主机实现从 src/canvas-host 移至 extensions/canvas/src/host
  • 保留 extensions/canvas/runtime-api.ts,作为由插件拥有、用于测试、打包和外部公共 Canvas 辅助函数的兼容性 barrel。
  • 将 Canvas 文档实体化从 src/gateway/canvas-documents.ts 移至 extensions/canvas/src/documents.ts
  • 将 Canvas CLI 实现和 A2UI JSONL 辅助函数移入 extensions/canvas/src/cli.ts
  • 将 Canvas 主机 URL 和作用域能力辅助函数移入 extensions/canvas/src
  • 将 Canvas 节点命令默认值从硬编码核心列表移入插件 nodeInvokePolicies
  • plugins.entries.canvas.config.host 添加了插件拥有的 Canvas 主机配置。
  • 将 Canvas 和 A2UI HTTP 服务移到 Canvas 插件 HTTP 路由注册之后。
  • 为插件拥有的 HTTP 路由添加了通用插件 WebSocket 升级分发。
  • 用通用托管插件表面和节点能力辅助函数替换了 Canvas 特定的 Gateway 网关主机 URL 和节点能力凭证。
  • 添加了插件拥有的托管媒体解析器,使 Canvas 文档 URL 通过 Canvas 插件解析,而不是让核心导入 Canvas 文档内部实现。
  • 添加了 api.registerNodeCliFeature(...),让 Canvas 可以声明 openclaw nodes canvas 是插件拥有的节点功能,而无需手动拼写父命令路径。
  • 移除了生产 src/**extensions/canvas/runtime-api.js 的导入。
  • 将 A2UI bundle 源从 apps/shared/OpenClawKit/Tools/CanvasA2UI 移至 extensions/canvas/src/host/a2ui-app
  • 将 A2UI 构建/复制实现移到 extensions/canvas/scripts 下,并用通用内置插件资源钩子替换根构建接线。
  • 移除了运行时旧版顶层 canvasHost 配置别名。
  • 保留了 Canvas Doctor 迁移,使 openclaw doctor --fix 将旧 canvasHost 配置重写为 plugins.entries.canvas.config.host
  • 移除了 Gateway 网关协议 v4 后面的旧智能体 Canvas 协议兼容性。原生客户端和 Gateway 网关现在只使用 pluginSurfaceUrls.canvasnode.pluginSurface.refresh;已弃用的 canvasHostUrlcanvasCapabilitynode.canvas.capability.refresh 路径在此次实验性重构中故意不受支持。
  • 更新了生成的插件清单以包含 Canvas。
  • docs/plugins/reference/canvas.md 添加了插件参考文档。

已知仍由核心拥有的 Canvas 表面:

  • apps/ 下的原生应用 Canvas 处理程序仍有意消费 Canvas 插件表面
  • apps/ 下的原生应用 Canvas 协议/客户端处理程序
  • 已发布工件输出仍使用 dist/canvas-host/a2ui 以进行向后兼容的运行时查找,但复制步骤现在由插件拥有

目标形态

extensions/canvas 应拥有:

  • 插件清单和包元数据
  • 智能体工具注册
  • 节点调用命令策略
  • Canvas 主机和 A2UI 运行时
  • Canvas A2UI bundle 源和资源构建/复制脚本
  • Canvas 文档创建和资源解析
  • Canvas CLI 实现
  • Canvas 文档页面和插件清单条目

核心应只拥有通用接缝:

  • 插件发现和注册
  • 通用智能体工具注册表
  • 通用节点调用策略注册表
  • 通用 Gateway 网关 HTTP/凭证和 WebSocket 升级分发
  • 通用托管插件表面 URL 解析
  • 通用托管媒体解析器注册
  • 通用节点能力传输
  • 通用配置管道
  • 通用内置插件资源钩子发现

原生应用可以保留 Canvas 命令处理程序,作为协议客户端。它们不是插件运行时所有者。

迁移步骤

  1. plugins.entries.canvas.config.host 视为插件拥有的配置表面。
  2. 更新文档,使 Canvas 被描述为实验性内置插件。
  3. 运行聚焦 Canvas 的测试、插件清单检查、插件 SDK API 检查,以及受运行时边界影响的构建/类型门禁。

审计检查清单

在宣布重构完成之前:

  • rg "src/canvas-host|../canvas-host" 不返回任何实时源码导入。
  • rg "canvas-tool|createCanvasTool" src 找不到由核心拥有的 Canvas 工具实现。
  • rg "canvas.present|canvas.snapshot|canvas.a2ui" src/gateway 在通用插件策略测试之外找不到硬编码允许列表默认值。
  • rg "extensions/canvas/runtime-api" src --glob '!**/*.test.ts' 为空。
  • rg "canvas-documents" src 为空。
  • rg "registerNodesCanvasCommands|nodes-canvas" src 为空;Canvas 插件通过嵌套插件 CLI 元数据注册 openclaw nodes canvas
  • rg "createCanvasHostHandler|handleA2uiHttpRequest" src/gateway 不返回任何 Gateway 网关运行时所有权。
  • rg "apps/shared/OpenClawKit/Tools/CanvasA2UI|canvas-a2ui-copy|extensions/canvas/src/host/a2ui" scripts .github package.json 只找到兼容性包装器或插件拥有的路径。
  • pnpm plugins:inventory:check 通过。
  • pnpm plugin-sdk:api:check 通过,或生成的 API 基线被有意更新并经过评审。
  • 定向 Canvas 测试通过。
  • Canvas 主机/A2UI 路径的变更车道测试通过。
  • PR 正文明确说明 Canvas 是实验性的,并且由插件支持。

验证命令

迭代时使用定向本地检查:

pnpm test extensions/canvas/src/host/server.test.ts extensions/canvas/src/host/server.state-dir.test.ts extensions/canvas/src/host/file-resolver.test.ts
pnpm test src/gateway/server.plugin-node-capability-auth.test.ts src/gateway/server-import-boundary.test.ts
pnpm test extensions/canvas/src/config-migration.test.ts src/commands/doctor-legacy-config.migrations.test.ts
pnpm test test/scripts/changed-lanes.test.ts test/scripts/build-all.test.ts extensions/canvas/scripts/bundle-a2ui.test.ts test/scripts/bundled-plugin-assets.test.ts extensions/canvas/scripts/copy-a2ui.test.ts src/infra/run-node.test.ts
pnpm tsgo:extensions
pnpm plugins:inventory:check
pnpm plugin-sdk:api:check

如果运行时 barrel、惰性导入、打包或已发布插件表面发生变化,请在推送前运行 pnpm build