Hooks Reference
Maigo 註冊三個 hook,定義在 hooks/hooks.json。
只要 plugin 載入就自動生效,使用者不用設定。
SessionStart — hooks/repo_detect.py
session 開啟時觸發。偵測目前 repo 是否命中已知 project,若命中就 emit systemMessage 要求 agent 載入對應的 project-aware skill;未命中則 silent approve。讓 contributor 進到熟悉的 codebase 時,自動拿到該 repo 的慣例知識 (命名、測試模式、PR 規範等),不用使用者手動引用。
偵測流程
- 讀 stdin JSON 取
cwd(缺欄位 → fallbackos.getcwd()) - 對
REPO_RULES內每個 rule,依序跑其detectors - 任一 detector 命中即視為 rule 命中(OR 邏輯)
- 命中 → emit
approve+ systemMessage(要求載入skills/<skill>/SKILL.md) - 全部未命中 → emit
approve+ 空 systemMessage(silent)
目前 registry
| Project | Skill | Detector 條件 |
|---|---|---|
apache-airflow |
airflow-aware |
git remote 含 apache/airflow,或 airflow/__init__.py 存在且 airflow/models/dag.py / airflow/dag.py 至少有一個 |
commitizen-tools-commitizen |
commitizen-aware |
git remote 含 commitizen-tools/commitizen,或 commitizen/__init__.py 存在且 commitizen/cli.py / commitizen/commands/__init__.py / commitizen/bump.py 至少有一個 |
Detector 類型
type |
參數 | 命中條件 |
|---|---|---|
git_remote |
pattern: <substring> |
git config --get remote.origin.url 輸出含 pattern |
file_structure |
all_of: [paths] / any_of: [paths] |
all_of 全部存在 且(若有 any_of)至少一個存在 |
Fail-open 情況
- stdin JSON 解析失敗 → 視為空 dict,仍 fallback
os.getcwd()繼續跑 - 個別 detector 拋例外(
subprocess.TimeoutExpired/FileNotFoundError/OSError)→ skip 該 detector,繼續下一個 - 個別 rule
match_rule拋例外 → skip 該 rule,繼續下一個 - 頂層 unhandled exception → stderr 印一行,emit 空 approve
Timeout
5 秒上限(hooks/hooks.json 設定),單個 git subprocess 內部另設 3 秒上限
(GIT_TIMEOUT_SEC)。偵測都是本機檔案或 git config 讀取,毫秒級完成;5 秒
為保護性上限。
Add New Project Entry
擴充偵測範圍走兩步:
- 加 registry entry — 編輯
hooks/repo_detect.py的REPO_RULESlist, append 一個 dict:{ "name": "<project-name>", "skill": "<skill-name>", # 對應 skills/<skill-name>/SKILL.md "detectors": [ {"type": "git_remote", "pattern": "<org>/<repo>"}, # 可選:加 file_structure detector 當備援 {"type": "file_structure", "all_of": ["<sentinel-file>"], "any_of": ["<alt-file-1>", "<alt-file-2>"]}, ], } - 建對應 skill — 依 skills.md 加新 skill 的 checklist 建
skills/<skill-name>/SKILL.md。 skill 內容定位為「knowledge layer」(contributor 慣例),參考skills/airflow-aware/SKILL.md結構: 開頭寫**Loaded by**: repo-detect hook (SessionStart) when <project> is detected, 後接 When to apply / 命名 / 環境 / 測試 / PR 等子段。
完成後跑 python3 scripts/validate_plugin.py 確認 skill cross-ref 通過(check #8 會抓 repo_detect.py 引用的 skills/<name>/ 是否存在)。
TeammateIdle — hooks/teammate_quality_check.py
agent 跑完輸出送回 orchestrator 時觸發。 檢查輸出符合該角色的最低規格,不符合就 block 並要求補完。
各角色擋的條件
| Agent | 必須包含 | 違反時的 block message |
|---|---|---|
| Raana | ## Loaded memory entries 段(即使無相關 entry 也要明寫「(無相關 entry)」) |
「缺 memory 載入回報」 |
| Tomori | ## Loaded memory entries 段 |
「缺 memory 載入回報」 |
| Tomori | 提到 /tmp/maigo/<repo>/plan.md 或 /tmp/maigo/<repo>/review-rubric.md 路徑 |
「沒提到計畫檔路徑」 |
| Tomori | 結構段落:## Goal / ## Steps / ## Rubric / ## Acceptance / ## 目標 / ## 步驟 之一 |
「缺計畫結構」 |
| Soyo | ## Loaded memory entries 段 |
「缺 memory 載入回報」 |
| Soyo | verdict 字串:APPROVED / NEEDS_CHANGES / BLOCKED |
「沒下 verdict」 |
| Soyo | checklist 項目:[x] / [X] / [ ] |
「沒 checklist」 |
| Soyo | 非 APPROVED 時:must-fix / 改法 / evidence / 待補 之一 |
「擋下卻沒列 must-fix」 |
| Taki | exit <number> 模式 |
「沒貼 exit code」 |
| Taki | PASS 或 FAIL 之一 |
「沒給最終 verdict」 |
| Taki | 不能包含 should work / looks good / 應該可以 / 看起來沒問題 等 hedge 語 |
「verifier 只能拿 exit code 講話」 |
| Anon | 至少一個 file path reference(regex 抓 *.py / *.md / *.yml / *.yaml / *.json / *.toml / *.txt / *.sh / *.cfg) |
「沒看到檔案路徑 reference」 |
不檢查的角色
- 不在已知名單的角色 → 預設通過
Fail-open 情況
- malformed input(沒
teammate_role或teammate_output)→ approve - input 不是有效 JSON → approve
Timeout
30 秒上限。hook 只做 regex match,理論上毫秒級完成;30 秒為保護性上限,不應在正常情況被觸發。
加新規格
編輯 hooks/teammate_quality_check.py,在 ROLE_HANDLERS 字典加一個新 mapping,
並寫對應的 check_<role> 函式。每個 handler 都要呼叫 emit("block", ...) 或 emit("approve", ...)。
Stop — hooks/verify_completion.py
任務宣告完成前觸發。即使 orchestrator 想跳過 Taki 也擋下。
偵測順序
| 偵測到 | 跑什麼指令 |
|---|---|
uv.lock |
uv run pytest |
pyproject.toml 或 setup.py + tests/ 或 test/ 目錄 |
pytest |
package.json 內 scripts.test 存在 |
npm test --silent |
Cargo.toml |
cargo test --quiet |
go.mod |
go test ./... |
| 都沒有 | 跳過(no-op approve) |
設定檔(放在 user 專案的 .claude/ 下)
| 檔案 | 行為 |
|---|---|
skip-test-verification |
第一行非空非註解視為原因,整個檢查跳過 |
test-command |
完全覆寫 test 指令(用 shlex.split 解析,支援引號) |
known-test-failures |
已知失敗名單(一行一個),不擋這些;只擋「新的」失敗 |
Fatal markers
偵測到 \bImportError: / \bModuleNotFoundError: / \bSyntaxError:(必須有冒號,
避免誤判 test 名稱裡的字串)→ 視為 collection 錯,比 test 失敗更優先擋下,
message 強調「這不是 test fail,是 import 錯」。
抓 failure 名稱的 regex
| 框架 | 模式 |
|---|---|
| pytest | FAILED <name> 或 <file>::<test> FAILED |
| jest | FAIL <file>.test.[jt]sx? |
| cargo test | test <name> ... FAILED |
| go test | --- FAIL: <name> |
抓不出名稱但 exit 非 0 → 仍 block,附最後 500 chars 原始 output。
Timeout
預設 90 秒(hook 自身 timeout 120 秒,留 30 秒 buffer)。
編輯 TEST_TIMEOUT_SEC 常數可調。
Fail-open 情況
- 偵測不到任何專案類型(沒有
uv.lock/pyproject.toml/package.json/Cargo.toml/go.mod)→ approve(no-op) .claude/skip-test-verification存在 → approve 並記錄原因- 本次 session 無未提交的檔案修改(read-only session)→ approve,跳過 test 驗證。偵測方式:
git status --porcelain回傳 exit 0 且 stdout 為空;returncode != 0(非 git repo 等)→ fail-open,照常跑 test - stdin JSON 解析失敗或無
cwd欄位 → fallback 到os.getcwd(),照常嘗試偵測;如果偵測不到還是會 no-op approve
觀察 hook 行為
要看 hook 真的有跑、回傳什麼:
# 模擬 TeammateIdle 觸發
python3 -c "import json,sys; print(json.dumps({'teammate_role':'Soyo','teammate_output':'## Verdict\nBLOCKED\n## Checklist\n- [ ] foo'}))" \
| python3 hooks/teammate_quality_check.py
# 模擬 Stop 觸發
python3 -c "import json,sys; print(json.dumps({'cwd':'/path/to/project'}))" \
| python3 hooks/verify_completion.py