* feat(github-copilot-sdk): add workspace skills support - Introduce ENABLE_WORKSPACE_SKILLS valve to enable/disable workspace custom tools discovery - Modify _build_session_config() to auto-load tools from .copilot-skills/ directory - Add workspace_skills_example.py template with 3 working example tools - Update README.md and README_CN.md with Workspace Skills guide and usage examples - Create v0.9.0.md and v0.9.0_CN.md release notes - Sync version to all docs files (index.md, index.zh.md, and main docs) - Bump version from 0.8.0 to 0.9.0 across all 7+ locations * docs: establish temp files handling policy (project-based, not /tmp) - Add TEMP_FILES_POLICY.md guideline for all skills and workflows - Update pr-submitter skill to use .temp/ directory instead of /tmp - Update release-prep skill documentation with temp file convention - Add .temp/ and .build/ entries to .gitignore - Create internal policy memo in /memories/repo/ This policy ensures: - All temporary files stay within project workspace (not system /tmp) - Alignment with OpenWebUI workspace isolation principles - Multi-user safety and cleanup traceability - Consistent handling across all skills and development workflows * fix(terminology): rename 'workspace skills' to 'workspace custom tools' for accuracy The term 'Skills' in Anthropic context refers to instruction-based frameworks (SKILL.md files with YAML frontmatter + markdown), not custom tool functions. Our implementation uses @define_tool decorator to define custom tools that the SDK auto-discovers from .copilot-skills/ directory. These are Tools, not Skills. Changes: - Rename ENABLE_WORKSPACE_SKILLS valve -> ENABLE_WORKSPACE_TOOLS - Update all documentation (README, README_CN, docs, release notes) - Fix section headings and descriptions throughout - Ensure consistent terminology across all files This is a terminology-only change; functionality remains identical. * feat(pipes): release v0.9.0 of GitHub Copilot SDK Pipe - Integrated OpenWebUI Skills Bridge and manage_skills tool - Reinforced status bar stability with session_finalized logic - Added persistent SDK config directory support * docs(pipes): add comprehensive guides and v0.9.0 notes for Copilot SDK - Added skill manager and best practices guides - Added publishing tool documentation - Included v0.9.0 release notes and deployment script - Updated usage guides
7.2 KiB
Skills 使用最佳实践
编写、组织和维护 Copilot SDK Skills 的简明指南。
理解 Skills 的工作机制
Skills 不是命令行工具,而是上下文注入的指令集:
- Copilot SDK 守护进程读取
skill_directories中每个SKILL.md文件 - 提取每个 skill 的 YAML
description字段 - 用户发消息时,SDK 将用户意图与所有 description 进行语义匹配
- 匹配成功后,SDK 触发
skill.invoked事件,将完整的 SKILL.md 正文注入对话上下文 - Agent 读取这些指令,使用
bash、create_file、view_file等工具来执行
关键理解:永远不要把 skill 的名字当作 bash 命令来运行(例如 finance-reporting)。Skill 本身就是指令,而不是可执行文件。
写好 description 字段
SKILL.md frontmatter 中的 description 是 主要触发机制,SDK 用它做语义路由。
应该这样做 ✅
- 以动词开头:"管理…"、"生成…"、"分析…"
- 明确写出 "Use when:" 场景——这是最可靠的触发信号
- 覆盖用户可能表达同一需求的多种说法
description: 根据大纲或主题生成 PowerPoint 演示文稿。
Use when: 创建幻灯片、制作演讲稿、导出 PPTX 文件、做 PPT。
不要这样做 ❌
- 模糊描述:"一个有用的多功能工具"
- 与其他 skill 描述重叠(会造成误触发)
- 省略 "Use when:" 示例(大幅降低触发准确率)
实用经验
如果两个人会用不同方式表达同一需求(如"做个幻灯片"vs"制作一个演讲 deck"),两种说法都应该出现在 description 中。
目录结构:什么放在哪里
skill-name/
├── SKILL.md ← 必须。Frontmatter + 核心指令
├── .owui_id ← 自动生成,禁止编辑或删除
├── references/ ← 可选。补充文档,按需加载
│ └── advanced.md
├── scripts/ ← 可选。辅助脚本(Shell/Python)
└── assets/ ← 可选。模板、样例文件、静态数据
何时用 references/
当内容属于以下情况时放入 references/:
- 仅边缘场景或高级用法才需要
- 内容太长,每次都注入会浪费上下文(> 约 100 行)
- 纯参考资料(API 规格、格式文档、示例)
使用"渐进式披露":Agent 先读 SKILL.md,仅在任务需要时才加载特定 reference 文件:
## 高级导出选项
详见 [references/export-options.md](references/export-options.md)。
何时内联在 SKILL.md
当内容属于以下情况时留在 SKILL.md:
- 每次运行 skill 都需要
- 足够短,不会拖慢上下文注入(总计 < 约 150 行)
- 是 skill 主流程的核心内容
命名约定
| 内容 | 规范 | 示例 |
|---|---|---|
| Skill 目录名 | kebab-case |
export-to-pptx |
Frontmatter name 字段 |
kebab-case,与目录名一致 |
export-to-pptx |
| 脚本文件名 | snake_case.py 或 .sh |
build_slide.py |
| Reference 文件名 | kebab-case.md |
advanced-options.md |
Skill 目录名避免空格和大写字母——SDK 使用目录名作为 skill 标识符。
编写高效的 SKILL.md 指令
以角色声明开头
告诉 Agent 在这个 skill 上下文中扮演什么角色:
# 导出为 PowerPoint
你是一个演示文稿构建器。你的任务是使用本 skill 目录中的脚本,将用户的内容转换为结构清晰的 PPTX 文件。
使用祈使句步骤
用编号步骤写指令,而不是大段散文:
1. 如果用户未提供大纲,先询问。
2. 运行 `python3 {scripts_dir}/build_slide.py --title "..." --output "{cwd}/output.pptx"`
3. 检查文件是否存在,确认成功。
4. 向用户提供文件的下载路径。
明确处理错误
告诉 Agent 出错时怎么做:
如果脚本以非零状态码退出,将 stderr 输出展示给用户并询问如何处理。
以收尾指令结束
任务完成后,总结创建的内容,并提醒用户文件的存放位置。
Skill 的适用范围
每个 skill 应该只做一件事。以下迹象说明 skill 太宽泛了:
- description 包含 4–5 个以上涵盖不同领域的 "Use when:" 条目
- SKILL.md 超过 300 行
- 已添加超过 3 个 reference 文件
当 skill 变得过大时,进行拆分:一个父 skill 负责路由,各子 skill 负责具体功能。
管理 shared/ 目录
shared/ 目录与 OpenWebUI 数据库双向同步:
- 通过 OpenWebUI UI 创建的 skill 会自动导入
shared/ - Agent 在
shared/中创建的 skill 在下次会话启动时导出到 OpenWebUI
安全操作方式
| 操作 | 方法 |
|---|---|
| 从 URL 安装 | python3 {scripts_dir}/install_skill.py --url <url> --dest {shared_dir} |
| 新建 skill | mkdir -p {shared_dir}/<name>/ && 创建 SKILL.md |
| 编辑 skill | 读取 → 修改 → 写回 SKILL.md |
| 删除 skill | rm -rf {shared_dir}/<name>/(不会删除 OpenWebUI UI 中的记录,需单独删除) |
| 列出 skills | python3 {scripts_dir}/list_skills.py --path {shared_dir} |
.owui_id 文件
每个与 OpenWebUI 同步的 skill 都有一个 .owui_id 文件,里面存储数据库 UUID。绝对不要编辑或删除此文件——它是文件系统与 OpenWebUI 数据库之间的关联纽带。一旦删除,下次同步时该 skill 会被视为新建项,可能产生重复。
会话生命周期意识
Skills 在会话开始时加载一次。在会话期间做的修改,下次会话才会生效。
| 时间点 | 发生的事 |
|---|---|
| 会话启动 | SDK 守护进程读取所有 SKILL.md;_sync_openwebui_skills 执行双向 DB↔文件同步 |
| 会话期间 | 新建/编辑/删除的 skill 文件已在磁盘上,但守护进程尚未加载 |
| 用户开启新会话 | 新 skill 生效;修改后的 description 开始触发 |
每次创建/编辑/删除后,必须告知用户:"此更改将在您开启新会话后生效。"
需要避免的反模式
| 反模式 | 问题所在 | 解决方式 |
|---|---|---|
把 <skill名> 当 bash 命令运行 |
Skill 不是可执行文件 | 阅读 SKILL.md 指令,用标准工具执行 |
编辑 .owui_id |
破坏数据库同步 | 永远不要碰这个文件 |
| 在 SKILL.md 中存储会话状态 | SKILL.md 是静态指令,不是状态文件 | 使用工作区中的独立文件保存会话状态 |
| description 过于宽泛 | 对每条消息都误触发 | 用 "Use when:" 缩窄到具体意图 |
| 把所有逻辑写进一个 500 行的 SKILL.md | 上下文注入慢,难以维护 | 拆分为 SKILL.md + references/*.md |
在 /tmp 创建 skill |
不持久,SDK 找不到 | 始终在 {shared_dir}/ 中创建 |
新建 Skill 快速检查清单
- 目录名为
kebab-case,与name字段一致 description以动词开头,包含 "Use when:" 示例- SKILL.md 以角色声明开头
- 指令使用祈使句编号步骤
- 过长的参考内容已移至
references/ - 脚本已放入
scripts/ - 确认:description 与其他已加载 skill 无重叠
- 已告知用户:"新 skill 在下次会话后生效"