Merge PR #39: docs: §13/§18/§23 更新 synchronize fallback + merge 通知
This commit was merged in pull request #39.
This commit is contained in:
@@ -838,7 +838,7 @@ Agent spawn 走生产 openclaw 的决策理由:
|
|||||||
| CI 标准门控 → 代码审查 | **CI 通过后 daemon Webhook 转发 Mail 给审查者** | Mail 通知司马懿 Review | Gitea Webhook `pull_request` → daemon Webhook 模块 → Mail API |
|
| CI 标准门控 → 代码审查 | **CI 通过后 daemon Webhook 转发 Mail 给审查者** | Mail 通知司马懿 Review | Gitea Webhook `pull_request` → daemon Webhook 模块 → Mail API |
|
||||||
| 代码审查 → 修改(不通过) | 审查者提交 Review 意见 | daemon Webhook 转发 Mail 通知改动者(附 Review 摘要) | Gitea Webhook `pull_request_review` → daemon Webhook 模块 → Mail API |
|
| 代码审查 → 修改(不通过) | 审查者提交 Review 意见 | daemon Webhook 转发 Mail 通知改动者(附 Review 摘要) | Gitea Webhook `pull_request_review` → daemon Webhook 模块 → Mail API |
|
||||||
| 代码审查 → Merge(通过) | **审查者点 Approve** | daemon Webhook 转发 Mail 通知改动者 merge | Gitea Webhook `pull_request_review` → daemon Webhook 模块 → Mail API |
|
| 代码审查 → Merge(通过) | **审查者点 Approve** | daemon Webhook 转发 Mail 通知改动者 merge | Gitea Webhook `pull_request_review` → daemon Webhook 模块 → Mail API |
|
||||||
| Merge → 部署 | **merge 到 main 自动触发** | 无需通知(自动化) | Gitea Actions `on: push: branches: [main]` |
|
| Merge → 部署 | **merge 到 main 自动触发** | Mail 通知 PR 作者合并完成(PR #38) | Gitea Actions `on: push: branches: [main]` |
|
||||||
| 部署 → E2E | **部署 job 成功后触发 E2E job** | E2E 结果评论到 merge commit | Gitea Actions `needs: [deploy]` |
|
| 部署 → E2E | **部署 job 成功后触发 E2E job** | E2E 结果评论到 merge commit | Gitea Actions `needs: [deploy]` |
|
||||||
| E2E/部署 → Issue关闭 | 庞统或改动者手动确认后关闭 | Issue 关闭通知关注者 | Gitea API `PATCH /repos/{owner}/{repo}/issues/{id}` state=closed |
|
| E2E/部署 → Issue关闭 | 庞统或改动者手动确认后关闭 | Issue 关闭通知关注者 | Gitea API `PATCH /repos/{owner}/{repo}/issues/{id}` state=closed |
|
||||||
| CI失败 → Issue评论 | **CI 失败自动评论** → daemon Webhook 转发 Mail 通知改动者 | 评论到关联 Issue + Mail 推送 Agent | Gitea Actions `if: failure()` 写 PR评论 → daemon Webhook 监听 `issue_comment` → Mail |
|
| CI失败 → Issue评论 | **CI 失败自动评论** → daemon Webhook 转发 Mail 通知改动者 | 评论到关联 Issue + Mail 推送 Agent | Gitea Actions `if: failure()` 写 PR评论 → daemon Webhook 监听 `issue_comment` → Mail |
|
||||||
@@ -1303,7 +1303,7 @@ Layer 3: Mail 执行层(Agent 接口)
|
|||||||
| `push` | 代码推送 | commit hash, 分支, 作者 | 不需要转发(Actions 自动处理) |
|
| `push` | 代码推送 | commit hash, 分支, 作者 | 不需要转发(Actions 自动处理) |
|
||||||
| `pull_request` (opened) | PR 创建 | PR ID, 标题, 分支, 作者 | → Mail 通知司马懿 Review |
|
| `pull_request` (opened) | PR 创建 | PR ID, 标题, 分支, 作者 | → Mail 通知司马懿 Review |
|
||||||
| `pull_request_review` (submitted) | Review 提交 | PR ID, 审查者, 结论(APPROVE/REQUEST_CHANGES), 评论 | → Mail 通知张飞 Review 结果 |
|
| `pull_request_review` (submitted) | Review 提交 | PR ID, 审查者, 结论(APPROVE/REQUEST_CHANGES), 评论 | → Mail 通知张飞 Review 结果 |
|
||||||
| `pull_request` (closed/merged) | PR 合并 | PR ID, 合并 commit | 不需要转发(Actions 自动触发 deploy) |
|
| `pull_request` (closed/merged) | PR 合并 | PR ID, 合并 commit | Mail 通知 PR 作者合并完成(PR #38 恢复) |
|
||||||
| `issue_comment` | PR/Issue 评论 | 评论者, 内容 | CI workflow 写的失败评论 → 转发 Mail |
|
| `issue_comment` | PR/Issue 评论 | 评论者, 内容 | CI workflow 写的失败评论 → 转发 Mail |
|
||||||
| `issues` (opened+assigned) | Issue 创建/指派 | Issue ID, 标题, 被指派人 | → Mail 通知开发者 |
|
| `issues` (opened+assigned) | Issue 创建/指派 | Issue ID, 标题, 被指派人 | → Mail 通知开发者 |
|
||||||
| `release` | Release 创建 | tag, 名称 | 触发完整 CI+部署 |
|
| `release` | Release 创建 | tag, 名称 | 触发完整 CI+部署 |
|
||||||
@@ -1336,8 +1336,8 @@ async def handle_gitea_webhook(event: dict, x_gitea_event: str = Header(...), x_
|
|||||||
pr_author = to_agent_id(event["pull_request"]["user"]["login"])
|
pr_author = to_agent_id(event["pull_request"]["user"]["login"])
|
||||||
await send_mail(to="simayi-challenger", title=f"Review 请求: PR #{event['number']}", ...)
|
await send_mail(to="simayi-challenger", title=f"Review 请求: PR #{event['number']}", ...)
|
||||||
elif action == "closed" and event["pull_request"]["merged"]:
|
elif action == "closed" and event["pull_request"]["merged"]:
|
||||||
# merge 不需要通知,Actions 自动处理
|
# PR #38: 通知 PR 作者合并完成
|
||||||
pass
|
await _handle_pr_closed(event)
|
||||||
|
|
||||||
elif x_gitea_event == "pull_request_review":
|
elif x_gitea_event == "pull_request_review":
|
||||||
state = event["review"]["state"]
|
state = event["review"]["state"]
|
||||||
|
|||||||
@@ -119,3 +119,28 @@
|
|||||||
- 姜维第一次分析给出了错误根因(Gitea 双 notifier),第二次深入调查后自我纠正
|
- 姜维第一次分析给出了错误根因(Gitea 双 notifier),第二次深入调查后自我纠正
|
||||||
- 庞统把姜维的第一次结论当事实汇报给主公,没有标注"这是姜维的调查结论,尚未独立验证"
|
- 庞统把姜维的第一次结论当事实汇报给主公,没有标注"这是姜维的调查结论,尚未独立验证"
|
||||||
- **改进**:SOUL.md 新增规则——推测 vs 事实显式标注、引用他人结论时标注来源、结论被推翻时及时更正
|
- **改进**:SOUL.md 新增规则——推测 vs 事实显式标注、引用他人结论时标注来源、结论被推翻时及时更正
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## PR #38 新增场景(synchronize fallback + merge 通知)
|
||||||
|
|
||||||
|
> 2026-06-12 新增,对应 PR #38 的设计变更
|
||||||
|
|
||||||
|
### 步骤 9:synchronize fallback ✅
|
||||||
|
- 操作:创建 PR(无 review 历史)→ push 新 commit 到 PR 分支
|
||||||
|
- 触发事件:`pull_request` (synchronize)
|
||||||
|
- 预期:`simayi-challenger`(默认 reviewer)收到"请重新 review" Mail
|
||||||
|
- 验证点:
|
||||||
|
- PR 无 review 历史时,`_fetch_latest_reviewer()` 返回 None → fallback 到 `simayi-challenger`
|
||||||
|
- Mail to 正确(默认 reviewer 而非跳过通知)
|
||||||
|
- 模板使用 `review_updated.md`
|
||||||
|
|
||||||
|
### 步骤 10:merge 通知 ✅
|
||||||
|
- 操作:PR 通过 Review 后 merge
|
||||||
|
- 触发事件:`pull_request` (closed) + `merged=true`
|
||||||
|
- 预期:PR 作者收到"PR 已合并" Mail
|
||||||
|
- 验证点:
|
||||||
|
- Mail to 正确(PR 作者)
|
||||||
|
- `merged_by` 字段正确提取(payload `merged_by` → fallback `sender`)
|
||||||
|
- 模板使用 `review_merged.md`
|
||||||
|
- 纯 closed(非 merged)不触发通知
|
||||||
|
|||||||
@@ -28,6 +28,7 @@
|
|||||||
|---|---|---|---|---|
|
|---|---|---|---|---|
|
||||||
| E1 | PR 更新(push 新 commit)→ 通知 reviewer | `pull_request.synchronize` | **高** | review 驳回→修改→重 review 的关键闭环 |
|
| E1 | PR 更新(push 新 commit)→ 通知 reviewer | `pull_request.synchronize` | **高** | review 驳回→修改→重 review 的关键闭环 |
|
||||||
| ~~E2~~ | ~~PR 合并通知~~ | ~~已删除~~ | ~~—~~ | ~~和 §22 CD 成功通知重叠,已删~~ |
|
| ~~E2~~ | ~~PR 合并通知~~ | ~~已删除~~ | ~~—~~ | ~~和 §22 CD 成功通知重叠,已删~~ |
|
||||||
|
| E2 | PR 合并 → 通知 PR 作者 | `pull_request` (closed+merged) | **高** | PR #38 恢复:CD 通知语义不同(部署状态 vs 合并信息),文档 PR 无 CD 流程仍需通知 |
|
||||||
| E3 | Review 评论(COMMENTED)→ 通知 PR 作者 | `pull_request_review` (COMMENTED) | 中 | reviewer 讨论提问,作者应知道 |
|
| E3 | Review 评论(COMMENTED)→ 通知 PR 作者 | `pull_request_review` (COMMENTED) | 中 | reviewer 讨论提问,作者应知道 |
|
||||||
| E4 | PR 上普通评论 → 通知相关人 | `issue_comment` (on PR) | 低 | 非关键路径 |
|
| E4 | PR 上普通评论 → 通知相关人 | `issue_comment` (on PR) | 低 | 非关键路径 |
|
||||||
|
|
||||||
@@ -64,17 +65,42 @@ async def _handle_pull_request(payload: Dict[str, Any]) -> None:
|
|||||||
新增 `_handle_pr_synchronize`:
|
新增 `_handle_pr_synchronize`:
|
||||||
1. 从 payload 取 PR 信息(number、title、author、head sha)
|
1. 从 payload 取 PR 信息(number、title、author、head sha)
|
||||||
2. 查询最近一次 review(Gitea API `GET /repos/{owner}/{repo}/pulls/{number}/reviews`)取 reviewer
|
2. 查询最近一次 review(Gitea API `GET /repos/{owner}/{repo}/pulls/{number}/reviews`)取 reviewer
|
||||||
3. 如果没有 review 记录(首次 push 后 reviewer 还没 review),跳过(opened 事件已经通知过了)
|
3. 如果没有 review 记录(`_fetch_latest_reviewer()` 返回 None),fallback 到默认 reviewer `simayi-challenger`,而非跳过通知(PR #38 改动:确保无 review 历史时也能通知默认审查者)
|
||||||
4. 渲染 `review_updated.md` 模板,发送 Mail 给 reviewer
|
4. 渲染 `review_updated.md` 模板,发送 Mail 给 reviewer
|
||||||
|
|
||||||
**关键设计决策**:
|
**关键设计决策**:
|
||||||
- 不用 `requested_reviewers`(可能为空),用最近 review 的提交者
|
- 不用 `requested_reviewers`(可能为空),用最近 review 的提交者
|
||||||
- 只在有 review 历史时才通知(避免 opened + synchronize 重复通知)
|
- 无 review 历史时 fallback 到默认 reviewer `simayi-challenger`(PR #38:避免 opened + synchronize 间隔较短时 reviewer 未收到任何通知)
|
||||||
- Mail from 用 `system`
|
- Mail from 用 `system`
|
||||||
|
|
||||||
### ~~Handler 2:PR 合并通知~~ — 已删除
|
### Handler 2:`_handle_pr_closed`(PR 合并通知)— PR #38 恢复
|
||||||
|
|
||||||
> 司马懿 review 指出与 §22 CD 成功通知重叠。CD 成功通知已隐含合并信息,无需单独发 merged 通知。
|
**触发**:`pull_request` 事件 + `action=closed` + `merged=true`
|
||||||
|
|
||||||
|
**通知对象**:PR 作者
|
||||||
|
|
||||||
|
**实现**:
|
||||||
|
|
||||||
|
修改 `_handle_pull_request` 的 action 分发,新增 `closed` 分支:
|
||||||
|
|
||||||
|
```python
|
||||||
|
async def _handle_pull_request(payload: Dict[str, Any]) -> None:
|
||||||
|
action = payload.get("action", "")
|
||||||
|
if action == "opened":
|
||||||
|
await _handle_pr_opened(payload)
|
||||||
|
elif action == "synchronize":
|
||||||
|
await _handle_pr_synchronize(payload)
|
||||||
|
elif action == "closed" and payload.get("pull_request", {}).get("merged"):
|
||||||
|
await _handle_pr_closed(payload)
|
||||||
|
# 其他 action 忽略
|
||||||
|
```
|
||||||
|
|
||||||
|
新增 `_handle_pr_closed`:
|
||||||
|
1. 从 payload 取 PR 信息(number、title、merged_by)
|
||||||
|
2. `merged_by` 优先从 `payload["pull_request"]["merged_by"]` 取,若为空则 fallback 到 `payload["sender"]`(PR #38:兼容不同 Gitea 版本和 merge 方式)
|
||||||
|
3. 渲染 `review_merged.md` 模板,发送 Mail 给 PR 作者
|
||||||
|
|
||||||
|
**恢复说明**:此前因与 §22 CD 成功通知重叠而删除。但实际场景中 CD 通知发的是部署状态,PR 作者更关心的是"谁帮我 merge 了"这个信息,两者语义不同。且 CD 流程不一定每次都触发(如文档 PR),merge 通知仍需独立存在。(PR #38 恢复)
|
||||||
|
|
||||||
### 新增 Handler 3:review COMMENTED 处理
|
### 新增 Handler 3:review COMMENTED 处理
|
||||||
|
|
||||||
@@ -101,7 +127,7 @@ async def _handle_pull_request(payload: Dict[str, Any]) -> None:
|
|||||||
| 模板文件 | 变量 | 说明 |
|
| 模板文件 | 变量 | 说明 |
|
||||||
|---|---|---|
|
|---|---|---|
|
||||||
| `review_updated.md` | repo, pr_number, pr_title, pr_author, branch, new_sha, reviewer | PR 有新 commit,请重新 review |
|
| `review_updated.md` | repo, pr_number, pr_title, pr_author, branch, new_sha, reviewer | PR 有新 commit,请重新 review |
|
||||||
| ~~`pr_merged.md`~~ | ~~已删除~~ | ~~—~~ |
|
| `review_merged.md` | repo, pr_number, pr_title, pr_author, merged_by | PR 已合并,通知作者(PR #38 恢复) |
|
||||||
| `review_comment.md` | repo, pr_number, pr_title, reviewer, comment_body | reviewer 提交了评论 |
|
| `review_comment.md` | repo, pr_number, pr_title, reviewer, comment_body | reviewer 提交了评论 |
|
||||||
|
|
||||||
### `_EVENT_HANDLERS` 无需改动
|
### `_EVENT_HANDLERS` 无需改动
|
||||||
@@ -122,9 +148,9 @@ async def _handle_pull_request(payload: Dict[str, Any]) -> None:
|
|||||||
|
|
||||||
| 文件 | 改动 |
|
| 文件 | 改动 |
|
||||||
|---|---|
|
|---|---|
|
||||||
| `src/api/toolchain_routes.py` | 修改 `_handle_pull_request`(扩展 action 分发)+ 新增 `_handle_pr_synchronize` + 修改 `_handle_pull_request_review`(支持 COMMENTED) |
|
| `src/api/toolchain_routes.py` | 修改 `_handle_pull_request`(扩展 action 分发 + closed 分支)+ 新增 `_handle_pr_synchronize` + `_handle_pr_closed` + 修改 `_handle_pull_request_review`(支持 COMMENTED) |
|
||||||
| `templates/toolchain/review_updated.md` | 新增 |
|
| `templates/toolchain/review_updated.md` | 新增 |
|
||||||
| ~~`templates/toolchain/pr_merged.md`~~ | ~~已删除~~ |
|
| `templates/toolchain/review_merged.md` | 新增(PR #38 恢复) |
|
||||||
| `templates/toolchain/review_comment.md` | 新增 |
|
| `templates/toolchain/review_comment.md` | 新增 |
|
||||||
| `src/daemon/toolchain_templates.py` | `_TEMPLATE_MAP` 新增 3 个映射 |
|
| `src/daemon/toolchain_templates.py` | `_TEMPLATE_MAP` 新增 3 个映射 |
|
||||||
| `docs/design/23-toolchain-pr-lifecycle.md` | 本文档 |
|
| `docs/design/23-toolchain-pr-lifecycle.md` | 本文档 |
|
||||||
@@ -134,8 +160,9 @@ async def _handle_pull_request(payload: Dict[str, Any]) -> None:
|
|||||||
在 `sanguo/moziplus-v2` 测试仓库上 E2E 验证:
|
在 `sanguo/moziplus-v2` 测试仓库上 E2E 验证:
|
||||||
|
|
||||||
1. **synchronize**:创建 PR → review 驳回 → push 新 commit → 验证 reviewer 收到"请重新 review" Mail
|
1. **synchronize**:创建 PR → review 驳回 → push 新 commit → 验证 reviewer 收到"请重新 review" Mail
|
||||||
~~2. merged~~:已删除
|
2. **synchronize fallback**(PR #38):创建 PR → push commit(无 review 历史)→ 验证默认 reviewer (`simayi-challenger`) 收到通知
|
||||||
3. **COMMENTED**:review 提交纯评论 → 验证 PR 作者收到通知
|
3. **merge 通知**(PR #38 恢复):PR merge → 验证 PR 作者收到合并通知 Mail
|
||||||
|
4. **COMMENTED**:review 提交纯评论 → 验证 PR 作者收到通知
|
||||||
|
|
||||||
## 风险评估
|
## 风险评估
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user