14 KiB
14 KiB
OpenSpec 说明
适用于使用 OpenSpec 进行规范驱动开发的 AI 编码助手的说明。
快速检查清单
- 搜索现有工作:
openspec spec list --long、openspec list(仅在全文搜索时使用rg) - 确定范围:新功能 vs 修改现有功能
- 选择唯一的
change-id:kebab-case,动词开头(add-、update-、remove-、refactor-) - 搭建:
proposal.md、tasks.md、design.md(仅在需要时),以及每个受影响功能的增量规范 - 编写增量:使用
## ADDED|MODIFIED|REMOVED|RENAMED Requirements;每个需求至少包含一个#### Scenario: - 验证:
openspec validate [change-id] --strict并修复问题 - 请求批准:在提案获得批准之前不要开始实施
三阶段工作流
阶段 1:创建变更
在以下情况下创建提案:
- 添加功能或功能特性
- 进行破坏性更改(API、架构)
- 更改架构或模式
- 优化性能(改变行为)
- 更新安全模式
触发词(示例):
- "帮我创建一个变更提案"
- "帮我规划一个变更"
- "我想创建一个提案"
- "我想创建一个规范提案"
- "我想创建一个规范"
宽松匹配指导:
- 包含以下之一:
proposal、change、spec - 与以下之一组合:
create、plan、make、start、help
跳过提案的情况:
- Bug 修复(恢复预期行为)
- 拼写错误、格式、注释
- 依赖更新(非破坏性)
- 配置更改
- 现有行为的测试
工作流
- 查看
openspec/project.md、openspec list和openspec list --specs以了解当前上下文。 - 选择一个唯一的动词开头的
change-id,并在openspec/changes/<id>/下搭建proposal.md、tasks.md、可选的design.md和规范增量。 - 使用
## ADDED|MODIFIED|REMOVED Requirements起草规范增量,每个需求至少包含一个#### Scenario:。 - 运行
openspec validate <id> --strict并在分享提案之前解决所有问题。
阶段 2:实施变更
将这些步骤作为待办事项跟踪,逐一完成。
- 阅读 proposal.md - 了解要构建的内容
- 阅读 design.md(如果存在) - 查看技术决策
- 阅读 tasks.md - 获取实施清单
- 按顺序实施任务 - 按顺序完成
- 确认完成 - 在更新状态之前,确保
tasks.md中的每个项目都已完成 - 更新清单 - 所有工作完成后,将每个任务设置为
- [x],使列表反映实际情况 - 批准门控 - 在提案经过审查和批准之前不要开始实施
阶段 3:归档变更
部署后,创建单独的 PR:
- 将
changes/[name]/→changes/archive/YYYY-MM-DD-[name]/ - 如果功能已更改,更新
specs/ - 对于仅工具更改,使用
openspec archive <change-id> --skip-specs --yes(始终显式传递变更 ID) - 运行
openspec validate --strict以确认归档的变更通过检查
任何任务之前
上下文检查清单:
- 在
specs/[capability]/spec.md中阅读相关规范 - 检查
changes/中的待处理变更是否存在冲突 - 阅读
openspec/project.md了解约定 - 运行
openspec list查看活动变更 - 运行
openspec list --specs查看现有功能
创建规范之前:
- 始终检查功能是否已存在
- 优先修改现有规范而不是创建重复项
- 使用
openspec show [spec]查看当前状态 - 如果请求不明确,在搭建之前提出 1-2 个澄清问题
搜索指导
- 枚举规范:
openspec spec list --long(或脚本使用--json) - 枚举变更:
openspec list(或openspec change list --json- 已弃用但可用) - 显示详细信息:
- 规范:
openspec show <spec-id> --type spec(使用--json进行过滤) - 变更:
openspec show <change-id> --json --deltas-only
- 规范:
- 全文搜索(使用 ripgrep):
rg -n "Requirement:|Scenario:" openspec/specs
快速开始
CLI 命令
# 基本命令
openspec list # 列出活动变更
openspec list --specs # 列出规范
openspec show [item] # 显示变更或规范
openspec validate [item] # 验证变更或规范
openspec archive <change-id> [--yes|-y] # 部署后归档(非交互式运行添加 --yes)
# 项目管理
openspec init [path] # 初始化 OpenSpec
openspec update [path] # 更新说明文件
# 交互模式
openspec show # 提示选择
openspec validate # 批量验证模式
# 调试
openspec show [change] --json --deltas-only
openspec validate [change] --strict
命令标志
--json- 机器可读输出--type change|spec- 消除项目歧义--strict- 全面验证--no-interactive- 禁用提示--skip-specs- 归档时不更新规范--yes/-y- 跳过确认提示(非交互式归档)
目录结构
openspec/
├── project.md # 项目约定
├── specs/ # 当前真相 - 已构建的内容
│ └── [capability]/ # 单一专注的功能
│ ├── spec.md # 需求和场景
│ └── design.md # 技术模式
├── changes/ # 提案 - 应该更改的内容
│ ├── [change-name]/
│ │ ├── proposal.md # 原因、内容、影响
│ │ ├── tasks.md # 实施清单
│ │ ├── design.md # 技术决策(可选;参见标准)
│ │ └── specs/ # 增量变更
│ │ └── [capability]/
│ │ └── spec.md # ADDED/MODIFIED/REMOVED
│ └── archive/ # 已完成的变更
创建变更提案
决策树
新请求?
├─ 修复规范行为的 Bug? → 直接修复
├─ 拼写错误/格式/注释? → 直接修复
├─ 新功能/能力? → 创建提案
├─ 破坏性更改? → 创建提案
├─ 架构更改? → 创建提案
└─ 不明确? → 创建提案(更安全)
提案结构
-
创建目录:
changes/[change-id]/(kebab-case,动词开头,唯一) -
编写 proposal.md:
# 变更:[变更的简要描述]
## 原因
[关于问题/机会的 1-2 句话]
## 变更内容
- [变更的要点列表]
- [用 **BREAKING** 标记破坏性更改]
## 影响
- 受影响的规范:[列出功能]
- 受影响的代码:[关键文件/系统]
- 创建规范增量:
specs/[capability]/spec.md
## ADDED Requirements
### Requirement: 新功能
系统应提供...
#### Scenario: 成功情况
- **WHEN** 用户执行操作
- **THEN** 预期结果
## MODIFIED Requirements
### Requirement: 现有功能
[完整的修改后需求]
## REMOVED Requirements
### Requirement: 旧功能
**原因**:[为什么移除]
**迁移**:[如何处理]
如果多个功能受到影响,在 changes/[change-id]/specs/<capability>/spec.md 下创建多个增量文件——每个功能一个。
- 创建 tasks.md:
## 1. 实施
- [ ] 1.1 创建数据库架构
- [ ] 1.2 实现 API 端点
- [ ] 1.3 添加前端组件
- [ ] 1.4 编写测试
- 在需要时创建 design.md:
如果以下任何情况适用,则创建
design.md;否则省略:
- 跨领域更改(多个服务/模块)或新的架构模式
- 新的外部依赖或重大的数据模型更改
- 安全、性能或迁移复杂性
- 在编码之前从技术决策中受益的模糊性
最小 design.md 骨架:
## 上下文
[背景、约束、利益相关者]
## 目标 / 非目标
- 目标:[...]
- 非目标:[...]
## 决策
- 决策:[内容和原因]
- 考虑的替代方案:[选项 + 理由]
## 风险 / 权衡
- [风险] → 缓解措施
## 迁移计划
[步骤、回滚]
## 开放问题
- [...]
规范文件格式
关键:场景格式
正确(使用 #### 标题):
#### Scenario: 用户登录成功
- **WHEN** 提供有效凭据
- **THEN** 返回 JWT 令牌
错误(不要使用项目符号或粗体):
- **Scenario: 用户登录** ❌
**Scenario**: 用户登录 ❌
### Scenario: 用户登录 ❌
每个需求必须至少有一个场景。
需求措辞
- 对规范性需求使用 SHALL/MUST(除非有意非规范性,否则避免 should/may)
增量操作
## ADDED Requirements- 新功能## MODIFIED Requirements- 更改的行为## REMOVED Requirements- 已弃用的功能## RENAMED Requirements- 名称更改
标题与 trim(header) 匹配 - 忽略空白。
何时使用 ADDED vs MODIFIED
- ADDED:引入可以独立作为需求的新功能或子功能。当更改是正交的(例如,添加"斜杠命令配置")而不是改变现有需求的语义时,优先使用 ADDED。
- MODIFIED:更改现有行为、范围或验收标准。始终粘贴完整的、更新的需求内容(标题 + 所有场景)。归档器将用您在此处提供的内容替换整个需求;部分增量将丢失先前的详细信息。
- RENAMED:仅在名称更改时使用。如果您还更改行为,请使用 RENAMED(名称)加上 MODIFIED(内容),引用新名称。
常见陷阱:使用 MODIFIED 添加新关注点而不包含先前的文本。这会在归档时导致详细信息丢失。如果您没有明确更改现有需求,请在 ADDED 下添加新需求。
正确编写 MODIFIED 需求:
- 在
openspec/specs/<capability>/spec.md中找到现有需求。 - 复制整个需求块(从
### Requirement: ...到其场景)。 - 将其粘贴到
## MODIFIED Requirements下并编辑以反映新行为。 - 确保标题文本完全匹配(忽略空白)并至少保留一个
#### Scenario:。
RENAMED 示例:
## RENAMED Requirements
- FROM: `### Requirement: Login`
- TO: `### Requirement: User Authentication`
故障排除
常见错误
"变更必须至少有一个增量"
- 检查
changes/[name]/specs/是否存在 .md 文件 - 验证文件具有操作前缀(## ADDED Requirements)
"需求必须至少有一个场景"
- 检查场景是否使用
#### Scenario:格式(4 个井号) - 不要对场景标题使用项目符号或粗体
静默场景解析失败
- 需要精确格式:
#### Scenario: Name - 使用以下命令调试:
openspec show [change] --json --deltas-only
验证提示
# 始终使用严格模式进行全面检查
openspec validate [change] --strict
# 调试增量解析
openspec show [change] --json | jq '.deltas'
# 检查特定需求
openspec show [spec] --json -r 1
成功路径脚本
# 1) 探索当前状态
openspec spec list --long
openspec list
# 可选全文搜索:
# rg -n "Requirement:|Scenario:" openspec/specs
# rg -n "^#|Requirement:" openspec/changes
# 2) 选择变更 id 并搭建
CHANGE=add-two-factor-auth
mkdir -p openspec/changes/$CHANGE/{specs/auth}
printf "## 原因\n...\n\n## 变更内容\n- ...\n\n## 影响\n- ...\n" > openspec/changes/$CHANGE/proposal.md
printf "## 1. 实施\n- [ ] 1.1 ...\n" > openspec/changes/$CHANGE/tasks.md
# 3) 添加增量(示例)
cat > openspec/changes/$CHANGE/specs/auth/spec.md << 'EOF'
## ADDED Requirements
### Requirement: 双因素认证
用户必须在登录时提供第二个因素。
#### Scenario: 需要 OTP
- **WHEN** 提供有效凭据
- **THEN** 需要 OTP 挑战
EOF
# 4) 验证
openspec validate $CHANGE --strict
多功能示例
openspec/changes/add-2fa-notify/
├── proposal.md
├── tasks.md
└── specs/
├── auth/
│ └── spec.md # ADDED: 双因素认证
└── notifications/
└── spec.md # ADDED: OTP 电子邮件通知
auth/spec.md
## ADDED Requirements
### Requirement: 双因素认证
...
notifications/spec.md
## ADDED Requirements
### Requirement: OTP 电子邮件通知
...
最佳实践
简单优先
- 默认 <100 行新代码
- 单文件实现,直到证明不足
- 没有明确理由时避免框架
- 选择无聊、经过验证的模式
复杂性触发器
仅在以下情况下添加复杂性:
- 性能数据表明当前解决方案太慢
- 具体的规模要求(>1000 用户,>100MB 数据)
- 需要抽象的多个经过验证的用例
清晰的引用
- 使用
file.ts:42格式表示代码位置 - 将规范引用为
specs/auth/spec.md - 链接相关变更和 PR
功能命名
- 使用动词-名词:
user-auth、payment-capture - 每个功能单一目的
- 10 分钟可理解性规则
- 如果描述需要"AND",则拆分
变更 ID 命名
- 使用 kebab-case,简短且描述性:
add-two-factor-auth - 优先使用动词开头的前缀:
add-、update-、remove-、refactor- - 确保唯一性;如果已使用,追加
-2、-3等
工具选择指南
| 任务 | 工具 | 原因 |
|---|---|---|
| 按模式查找文件 | Glob | 快速模式匹配 |
| 搜索代码内容 | Grep | 优化的正则表达式搜索 |
| 读取特定文件 | Read | 直接文件访问 |
| 探索未知范围 | Task | 多步骤调查 |
错误恢复
变更冲突
- 运行
openspec list查看活动变更 - 检查重叠的规范
- 与变更所有者协调
- 考虑合并提案
验证失败
- 使用
--strict标志运行 - 检查 JSON 输出以获取详细信息
- 验证规范文件格式
- 确保场景格式正确
缺少上下文
- 首先阅读 project.md
- 检查相关规范
- 查看最近的归档
- 请求澄清
快速参考
阶段指示器
changes/- 已提议,尚未构建specs/- 已构建并部署archive/- 已完成的变更
文件用途
proposal.md- 原因和内容tasks.md- 实施步骤design.md- 技术决策spec.md- 需求和行为
CLI 要点
openspec list # 正在进行什么?
openspec show [item] # 查看详细信息
openspec validate --strict # 是否正确?
openspec archive <change-id> [--yes|-y] # 标记完成(自动化添加 --yes)
记住:规范是真相。变更是提案。保持它们同步。