Skip to content

Latest commit

 

History

History
3001 lines (2332 loc) · 80.8 KB

File metadata and controls

3001 lines (2332 loc) · 80.8 KB

Hooks系统完整指南:自动化工作流的终极武器

课程信息

  • 作者:老金
  • GitHubhttps://github.com/KimYx0207
  • 公众号:老金带你玩AI
  • X(Twitter):老金带你玩AI
  • 个人博客https://aiking.dev
  • 预计学时:4-6小时
  • 难度等级:⭐⭐ 入门级(有Claude Code基础即可)
  • 更新日期:2026年4月
  • 适用版本:Claude Code v2.1.92(验证于2026-04-05)
  • 前置要求:已完成Claude Code安装和基础使用

本课学习目标

完成本课学习后,你将能够:

  1. 理解Hooks的核心价值:掌握Hooks与传统提示词的本质区别
  2. 配置第一个Hook:5分钟内完成最简单的Hook配置并看到效果
  3. 按事件族理解Hooks:理解工具、会话、任务、失败、文件系统、压缩和 Elicitation 等事件面
  4. 实现自动化工作流:Git提交检查、代码格式化、文件保护等实战场景
  5. 排查Hook故障:独立解决90%的常见配置和执行问题
  6. 安全使用Hooks:理解安全风险并正确配置权限

2026-04 差量更新(先读)

这章旧版最大的问题,是把 Hooks 讲成“固定 20 种类型”。现在这个说法已经不稳了。

  • 官方 Hook 事件面仍在扩展,不要再死记一个固定数字。
  • 当前主流处理器有 4 类commandhttppromptagent
  • 除了常见的 PreToolUse / PostToolUse / UserPromptSubmit,还应注意较新的事件,例如 PostToolUseFailurePermissionDeniedTaskCreatedTaskCompletedStopFailureCwdChangedFileChanged

因此,下面正文请用“事件族 + 处理器类型 + 典型场景”来理解,而不是背一个固定数量。

与 Hooks 相关的 v2.1.90 / v2.1.92 变更(GitHub Release 摘录)

下列句子均来自官方 What's changed英文原文便于逐字核对;教程不展开实现细节。

v2.1.90release):

  • Fixed Edit/Write failing with "File content has changed" when a PostToolUse format-on-save hook rewrites the file between consecutive edits
  • Fixed PreToolUse hooks that emit JSON to stdout and exit with code 2 not correctly blocking the tool call

v2.1.92release):

  • Fixed prompt-type Stop hooks incorrectly failing when the small fast model returns ok:false, and restored preventContinuation:true semantics for non-Stop prompt-type hooks

阅读提示:若 Hook 行为与上述条目相关但仍异常,以当前安装的 claude --version 及之后新 release 为准。


学习路径导航(先看这里!)

根据你的情况选择学习路径:这是一篇3000+行的长教程,不用全看!根据你的目标选择路径。

路径A:快速上手(30分钟)

适合人群:急着体验Hooks,想快速配一个看效果

只看这些章节(其他跳过):

✅ 术语表(5分钟) - 快速了解Hook核心概念
✅ 第一部分1.1-1.2:Hooks简介(5分钟) - 理解Hook是什么
✅ 第二部分:5分钟快速开始(15分钟) - 配置第一个Hook
✅ 第三部分3.1:PreToolUse基础(5分钟) - 最常用的Hook类型

30分钟后你能达到:成功配置第一个Hook,Claude Code能自动执行你的脚本


路径B:完整学习(4-6小时)

适合人群:想深入理解Hooks,掌握所有类型和高级用法

学习顺序:从头到尾所有章节

建议分段学习

  • 第1天(2小时):第1-3部分(理解+15种类型)
  • 第2天(2小时):第4-5部分(实战场景+故障排查)
  • 第3天(1小时):第6-7部分(FAQ+附录)

路径C:问题排查(10分钟)

适合人群:Hook配置出问题,需要快速解决

直接跳到这些章节

🔧 第五部分:故障排查 - 按错误类型查找解决方案
🔧 第六部分:FAQ - 20个常见问题解答

使用方法

  1. Ctrl + F 搜索你的错误信息关键词
  2. 找到对应的Q&A
  3. 按步骤解决

路径D:专项学习(30-60分钟/主题)

适合人群:已经会配置Hook,想学习特定功能

想学什么 看哪几节 预计时间
Git自动化 第四部分4.1节 45分钟
代码格式化 第四部分4.2节 30分钟
文件保护 第三部分3.1节 20分钟
提示词优化 第三部分3.3节 30分钟
安全最佳实践 第一部分1.4节 + 第五部分 40分钟

术语表(小白必读)

在开始之前,先了解这些关键术语。用生活类比帮助理解

术语 英文全称 通俗解释 生活类比
Hook - 在特定事件发生时自动执行的脚本 汽车传感器(检测到碰撞自动弹安全气囊)
PreToolUse Pre Tool Use 工具调用触发的Hook 机场安检门(登机前检查)
PostToolUse Post Tool Use 工具调用触发的Hook 快递签收后的自动通知
UserPromptSubmit User Prompt Submit 用户输入提交时触发的Hook 邮件发送前的拼写检查
Notification - 通知事件触发的Hook 手机APP推送通知
SessionStart Session Start 会话开始时触发的Hook 开机自动启动程序
SessionEnd Session End 会话结束时触发的Hook 关机前自动保存
Stop - AI停止响应时触发的Hook 紧急刹车后的状态保存
WorktreeCreate 🆕 Worktree Create 工作树创建时触发的Hook 新开一个平行工作台时的初始化
WorktreeRemove 🆕 Worktree Remove 工作树删除时触发的Hook 关闭平行工作台时的清理
Matcher - 匹配规则,决定Hook对哪些工具生效 筛选器(只检查特定行李)
Decision - PreToolUse Hook的返回决策 安检结果(放行/拦截/询问)
stdin Standard Input 标准输入,Hook接收数据的方式 传送带送入检查口
stdout Standard Output 标准输出,Hook返回结果的方式 检查结果显示屏
stderr Standard Error 标准错误输出,仅用于调试日志(不会显示在Claude Code界面) 后台监控日志(用户看不到)
timeout - 超时时间,Hook最长运行时间 限时检查(超时自动放行)
JSON JavaScript Object Notation 一种通用的数据格式,用花括号{}组织数据,settings.json配置文件就是JSON格式 标准化的表格模板
~(波浪号) Home Directory 用户的"家目录",macOS是/Users/用户名,Linux是/home/用户名,Windows对应C:\Users\用户名 你电脑上"我的文档"的上级目录

第一部分:Hooks简介(5分钟理解)

1.1 Hooks是什么

一句话理解:Hooks是Claude Code的"自动化传感器",在特定事件发生时自动执行你的脚本,实现100%可靠的自动化。

为什么需要Hooks?

没有Hooks之前(靠AI"记住")

问题:AI有时会"忘记"你的要求

你:每次写代码后帮我运行格式化
Claude:好的!(这次记住了)

...10分钟后...

Claude:代码写好了!
你:等等,你忘了格式化!
Claude:抱歉,我忘了...

有了Hooks之后(100%自动执行)

解决方案:不依赖AI记忆,配置Hook后自动执行

配置PostToolUse Hook → 监听Write工具 → 自动运行格式化脚本

Claude:代码写好了!
[Hook自动触发:运行 prettier --write xxx.js]
结果:代码已自动格式化,100%不会忘记

生活类比

  • 没有Hooks:靠人记住每次开车前检查轮胎(经常忘)
  • 有Hooks后:汽车传感器自动检测胎压,异常自动报警(100%可靠)

Hooks的核心价值

对比维度 提示词方式 Hooks方式
可靠性 不确定(AI可能忘记) 100%执行(确定性)
一致性 每次可能不同 每次完全相同
自动化 需要AI主动执行 事件触发自动执行
团队协作 每人都要提醒AI 配置一次,全员生效
适用场景 灵活建议 强制规则

1.2 Hooks能做什么(6个实际案例)

案例1:文件保护(PreToolUse)

场景:禁止Claude修改production目录下的文件

Hook触发:Claude尝试Write(file_path="production/config.js")
Hook检查:路径包含"production/"
Hook决策:deny(拒绝)
结果:Claude收到错误提示,文件未被修改

案例2:代码格式化(PostToolUse)

场景:每次保存代码后自动格式化

Hook触发:Claude成功执行Write(file_path="src/app.js")
Hook执行:运行 prettier --write src/app.js
结果:代码自动格式化,无需手动操作

案例3:提示词优化(UserPromptSubmit)

场景:自动在写作任务后追加写作规范

用户输入:"帮我写一篇关于AI的文章"
Hook检测:包含"写"和"文章"关键词
Hook追加:"\n\n## 写作规范\n1. 风格:接地气\n2. 字数:1500字"
Claude收到:原始输入 + 写作规范

案例4:Git提交检查(PreToolUse + Bash)

场景:提交前自动检查代码质量

Hook触发:Claude执行Bash(command="git commit -m xxx")
Hook执行:运行lint检查、测试、敏感信息扫描
Hook决策:全部通过 → allow;有问题 → deny
结果:低质量代码无法提交

案例5:会话初始化(SessionStart)

场景:启动Claude Code时自动加载项目配置

Hook触发:Claude Code启动
Hook执行:检查Python依赖是否安装
结果:缺少依赖时自动提示安装命令

案例6:桌面通知(Notification)

场景:Claude需要用户确认时发送桌面通知

Hook触发:Claude发送通知请求用户确认
Hook执行:调用系统通知API
结果:用户收到桌面弹窗,不会错过重要确认

1.3 Hooks执行流程

完整生命周期图

用户输入
    ↓
[UserPromptSubmit Hook] ← 可以修改/增强提示词
    ↓
Claude处理提示词
    ↓
决定调用工具(如Write)
    ↓
[PreToolUse Hook] ← 可以允许/拒绝/询问
    ↓
执行工具(如Write)
    ↓
[PostToolUse Hook] ← 可以执行后处理
    ↓
返回结果给用户

当前常见 Hook 事件族与触发时机(概念快照,不等于完整清单)

Hook类型 触发时机 典型用途 可否阻止后续操作
UserPromptSubmit 用户输入提交后 提示词优化、敏感词过滤 ✅ 是
PreToolUse 工具调用前 权限校验、参数验证 ✅ 是
PostToolUse / PostToolUseFailure 工具调用成功后 / 失败后 格式修复、自动测试、失败告警 ❌ 否
Notification 通知发送时 日志记录、桌面通知 ❌ 否
SessionStart 会话开始时 环境初始化 ❌ 否
SessionEnd 会话结束时 清理临时文件 ❌ 否
Stop / StopFailure AI 正常停止 / 异常停止时 保存状态、错误告警 ❌ 否
TaskCreated / TaskCompleted 子任务创建 / 完成时 子代理日志、任务收集 ❌ 否
PermissionDenied 权限被拒绝时 审计、自动补救提示 ❌ 否
PreCompact / PostCompact 上下文压缩前 / 后 保存关键上下文、记录 token 变化 ❌ 否
CwdChanged / FileChanged 工作目录切换 / 文件变化时 同步环境、触发检查 ❌ 否
Elicitation MCP 请求额外交互输入时 记录交互日志、输入校验 ❌ 否

💡 记忆方式:先记三大高频入口 UserPromptSubmitPreToolUsePostToolUse,再按“失败 / 任务 / 文件 / 压缩 / 交互”五个补充事件族扩展。

1.4 安全警告(重要!)

⚠️ 严重警告:Hooks可以执行任意Shell命令,这意味着配置不当可能导致:

  • 文件被删除或修改
  • 敏感信息泄露
  • 系统被恶意脚本攻击

安全最佳实践

风险 防护措施
恶意脚本 只运行你信任的脚本,不要从不明来源复制配置
权限过大 脚本只请求必要的权限,避免使用sudo
敏感信息 不要在脚本中硬编码密码/Token
无限循环 设置合理的timeout,避免脚本卡死
团队配置 代码审查.claude/settings.json变更

配置检查清单

□ 脚本来源可信吗?(自己写的/官方示例/信任的开源)
□ 脚本权限最小化了吗?(不需要sudo就不用)
□ 敏感信息用环境变量了吗?(不硬编码)
□ 设置了合理的timeout吗?(防止卡死)
□ 团队成员都知道这个Hook吗?(透明度)

第二部分:5分钟快速开始(立即见效)

本节目的:用最快速度配置第一个Hook,让你立即看到效果!

⏱️ 预计时间:5-10分钟

2.1 配置第一个Hook(最简单版本)

为什么选这个示例?

  • ✅ 最简单,只需要3个文件
  • ✅ 效果直观,立即看到输出
  • ✅ 无依赖,不需要安装任何东西

步骤1:创建Hook脚本目录

这一步要做什么:在项目根目录创建 .claude/hooks/ 目录

Windows系统(PowerShell):

# 进入你的项目目录
cd C:\你的项目路径

# 创建hooks目录
New-Item -ItemType Directory -Path ".claude\hooks" -Force

macOS/Linux系统:

# 进入你的项目目录
cd ~/你的项目路径

# 创建hooks目录
mkdir -p .claude/hooks

验证是否成功:

# 检查目录是否存在
ls .claude/hooks
# 应该显示空目录(暂时没有文件)

步骤2:创建最简单的Hook脚本

这一步要做什么:创建一个Python脚本,在每次Write工具执行后打印提示

创建文件 .claude/hooks/post-write-hello.py

💡 你有两种选择

选择A:在Claude Code对话框里说人话(推荐新手)

帮我创建.claude/hooks/post-write-hello.py文件,内容是一个PostToolUse Hook脚本,在Write工具执行后打印提示信息

(换行用 Shift + Enter,最后按 Enter 发送)

Claude Code会自动帮你创建正确格式的文件!

选择B:在终端里用命令行(熟悉命令行的用户) 见下方PowerShell/Bash代码

Windows(PowerShell):

@'
#!/usr/bin/env python3
"""
最简单的PostToolUse Hook示例
每次Write工具执行后记录日志
"""
import sys
import json
from pathlib import Path
from datetime import datetime

# 从stdin读取工具执行信息
try:
    input_data = json.loads(sys.stdin.read())
except:
    sys.exit(0)

# 获取工具名称和文件路径
tool_name = input_data.get('tool_name', '')
tool_input = input_data.get('tool_input', {})
file_path = tool_input.get('file_path', '')

# 只处理Write工具
if tool_name == 'Write':
    # PostToolUse Hook执行后处理任务
    # 注意:PostToolUse Hook无法向用户输出信息
    # Claude Code只会显示"Hook执行成功"
    # 如果需要调试,可以写入日志文件
    log_file = Path.home() / '.claude' / 'hooks' / 'post-write.log'
    log_file.parent.mkdir(parents=True, exist_ok=True)
    with open(log_file, 'a', encoding='utf-8') as f:
        timestamp = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
        f.write(f"[{timestamp}] ✅ 文件已保存: {file_path}\n")

sys.exit(0)
'@ | Out-File -FilePath ".claude\hooks\post-write-hello.py" -Encoding utf8

macOS/Linux:

cat > .claude/hooks/post-write-hello.py << 'EOF'
#!/usr/bin/env python3
"""
最简单的PostToolUse Hook示例
每次Write工具执行后记录日志
"""
import sys
import json
from pathlib import Path
from datetime import datetime

# 从stdin读取工具执行信息
try:
    input_data = json.loads(sys.stdin.read())
except:
    sys.exit(0)

# 获取工具名称和文件路径
tool_name = input_data.get('tool_name', '')
tool_input = input_data.get('tool_input', {})
file_path = tool_input.get('file_path', '')

# 只处理Write工具
if tool_name == 'Write':
    # PostToolUse Hook执行后处理任务
    # 注意:PostToolUse Hook无法向用户输出信息
    # Claude Code只会显示"Hook执行成功"
    # 如果需要调试,可以写入日志文件
    log_file = Path.home() / '.claude' / 'hooks' / 'post-write.log'
    log_file.parent.mkdir(parents=True, exist_ok=True)
    with open(log_file, 'a', encoding='utf-8') as f:
        timestamp = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
        f.write(f"[{timestamp}] ✅ 文件已保存: {file_path}\n")

sys.exit(0)

sys.exit(0)
EOF

# 添加执行权限(macOS/Linux必须)
chmod +x .claude/hooks/post-write-hello.py

验证脚本创建成功:

# 查看文件内容
cat .claude/hooks/post-write-hello.py
# 应该显示你刚才写入的Python代码

步骤3:配置settings.json

这一步要做什么:告诉Claude Code在PostToolUse时运行你的脚本

创建或编辑 .claude/settings.json

💡 你有两种选择

选择A:在Claude Code对话框里说人话(推荐新手)

帮我创建.claude/settings.json配置文件,配置PostToolUse Hook监听Write工具并运行post-write-hello.py脚本

(换行用 Shift + Enter,最后按 Enter 发送)

Claude Code会自动帮你创建正确格式的配置文件!

选择B:在终端里用命令行(熟悉命令行的用户) 见下方PowerShell/Bash代码

Windows(PowerShell):

@'
{
  "hooks": {
    "PostToolUse": [
      {
        "matcher": "Write",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/post-write-hello.py",
            "timeout": 10
          }
        ]
      }
    ]
  }
}
'@ | Out-File -FilePath ".claude\settings.json" -Encoding utf8

macOS/Linux:

cat > .claude/settings.json << 'EOF'
{
  "hooks": {
    "PostToolUse": [
      {
        "matcher": "Write",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/post-write-hello.py",
            "timeout": 10
          }
        ]
      }
    ]
  }
}
EOF

💡 配置说明

  • "PostToolUse":Hook类型,工具执行后触发
  • "matcher": "Write":只匹配Write工具
  • "command":要执行的脚本命令
  • "timeout": 10:超时10秒

步骤4:启动Claude Code并测试

这一步要做什么:重启Claude Code,让它读取新配置

# 在项目目录启动Claude Code
claude

测试命令

你:帮我创建一个test.txt文件,内容是"Hello Hooks!"

预期结果

Claude:我来帮你创建test.txt文件。

[Write工具执行]

✅ Hook执行成功

文件已创建成功!

💡 重要说明

  • PostToolUse Hook执行后,Claude Code只会显示"Hook执行成功"
  • Hook的日志已写入~/.claude/hooks/post-write.log文件
  • 你可以查看日志文件确认Hook确实执行了:
    cat ~/.claude/hooks/post-write.log
    # 应该看到类似:[2026-03-05 16:30:00] ✅ 文件已保存: /path/to/test.txt

关键确认:看到 Hook触发成功! 说明Hook配置正确并成功执行!

2.2 验证Hook工作正常

完整验证清单

  • .claude/hooks/ 目录存在
  • .claude/hooks/post-write-hello.py 文件存在且内容正确
  • .claude/settings.json 文件存在且JSON格式正确
  • macOS/Linux上脚本有执行权限(chmod +x
  • Claude Code启动时没有报错
  • 让Claude创建文件后看到Hook输出

如果没有看到Hook输出

  1. 检查JSON格式
# 验证JSON格式是否正确
python -c "import json; json.load(open('.claude/settings.json'))"
# 如果没报错说明格式正确
  1. 检查Python是否可用
python --version
# 应该显示Python 3.x
  1. 手动测试脚本
echo '{"tool_name": "Write", "tool_input": {"file_path": "test.txt"}}' | python .claude/hooks/post-write-hello.py
# 应该显示Hook触发成功的消息

2.3 恭喜完成第一个Hook!

你刚才完成了什么?

  1. ✅ 创建了Hook脚本目录
  2. ✅ 编写了第一个Hook脚本
  3. ✅ 配置了settings.json
  4. ✅ 验证了Hook正常工作

接下来可以

  • 继续学习15种Hook类型(第三部分)
  • 学习实战应用场景(第四部分)
  • 遇到问题查看故障排查(第五部分)

第三部分:15种Hook类型详解

本节目的:掌握所有Hook类型的用法

⏱️ 预计时间:1.5-2小时

3.1 PreToolUse(工具调用前)

一句话理解:PreToolUse就像"安检门",在工具执行前检查是否允许通过。

触发时机

在Claude准备调用工具(如Write、Edit、Bash)时,但尚未执行

输入参数(通过stdin的JSON)

{
  "tool_name": "Write",
  "tool_input": {
    "file_path": "C:/project/src/app.js",
    "content": "console.log('Hello World');"
  }
}
字段 类型 说明
tool_name string 工具名称(Write, Edit, Bash, Read等)
tool_input object 工具的输入参数

决策输出(通过stdout的JSON)

PreToolUse Hook可以返回决策指令控制工具是否执行。

新旧API并存:v2.1.49+ 推荐使用新版 hookSpecificOutput.permissionDecision 字段,旧版 decision 字段仍然支持但已废弃。

新版API(推荐) — 通过 hookSpecificOutput.permissionDecision 字段:

{
  "hookSpecificOutput": {
    "permissionDecision": "deny"
  },
  "message": "禁止修改production目录下的文件"
}

旧版API(已废弃但仍支持) — 通过 decision 字段:

{
  "decision": "deny",
  "message": "禁止修改production目录下的文件"
}

新版决策值hookSpecificOutput.permissionDecision):

决策值 说明 工具是否执行
"allow" 允许执行,绕过权限系统 ✅ 是
"deny" 拒绝执行,原因会反馈给Claude ❌ 否
"ask" 暂停,询问用户决定 🤔 等待用户决定
无输出 默认允许 ✅ 是

旧版决策值decision,已废弃):

旧版值 等同于新版 说明
"approve" "allow" 允许执行
"block" "deny" 拒绝执行

PostToolUse 和 UserPromptSubmit 的决策值:这两个Hook类型使用 "block" 来阻止/提供反馈,无输出则默认允许。

完整示例1:文件保护Hook

场景:禁止修改production/目录下的文件

脚本 .claude/hooks/pre-protect-production.py

#!/usr/bin/env python3
"""
PreToolUse Hook - 保护production目录
禁止Write/Edit工具修改production目录下的文件
"""
import sys
import json

# 读取stdin的JSON输入
try:
    input_data = json.loads(sys.stdin.read())
except json.JSONDecodeError:
    sys.exit(0)

tool_name = input_data.get('tool_name', '')
tool_input = input_data.get('tool_input', {})
file_path = tool_input.get('file_path', '')

# 只检查Write和Edit工具
if tool_name not in ['Write', 'Edit']:
    sys.exit(0)

# 规范化路径(处理Windows和Unix路径)
file_path_normalized = file_path.replace('\\', '/')

# 检查是否是保护目录
protected_dirs = ['production/', 'prod/', '.env']
for protected in protected_dirs:
    if protected in file_path_normalized:
        # 拒绝执行
        decision = {
            "decision": "deny",
            "message": f"❌ 禁止修改受保护的路径!\n路径: {file_path}\n原因: 包含受保护目录 '{protected}'\n\n请先切换到dev环境或手动操作。"
        }
        print(json.dumps(decision, ensure_ascii=False))
        sys.exit(0)

# 允许执行(无输出=默认allow)
sys.exit(0)

配置 .claude/settings.json

{
  "hooks": {
    "PreToolUse": [
      {
        "matcher": "Write|Edit",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/pre-protect-production.py",
            "timeout": 5
          }
        ]
      }
    ]
  }
}

运行效果

当Claude尝试修改production/config.json时:

Claude:我来修改production/config.json...

❌ 禁止修改受保护的路径!
路径: C:/project/production/config.json
原因: 包含受保护目录 'production/'

请先切换到dev环境或手动操作。

完整示例2:危险命令拦截Hook

场景:拦截危险的Bash命令(如rm -rf)

脚本 .claude/hooks/pre-block-dangerous-cmd.py

#!/usr/bin/env python3
"""
PreToolUse Hook - 拦截危险命令
阻止执行可能造成破坏的Shell命令
"""
import sys
import json
import re

# 危险命令模式
DANGEROUS_PATTERNS = [
    r'rm\s+-rf\s+/',           # rm -rf /
    r'rm\s+-rf\s+~',           # rm -rf ~
    r'rm\s+-rf\s+\*',          # rm -rf *
    r'rm\s+-rf\s+\.\.',        # rm -rf ..
    r':()\s*{\s*:\|:&\s*};:',  # Fork炸弹
    r'mkfs\.',                  # 格式化磁盘
    r'dd\s+if=.+of=/dev/',     # 覆盖磁盘
    r'>\s*/dev/sda',           # 覆盖磁盘
    r'chmod\s+-R\s+777\s+/',   # 危险权限
]

# 读取输入
try:
    input_data = json.loads(sys.stdin.read())
except json.JSONDecodeError:
    sys.exit(0)

tool_name = input_data.get('tool_name', '')
tool_input = input_data.get('tool_input', {})
command = tool_input.get('command', '')

# 只检查Bash工具
if tool_name != 'Bash':
    sys.exit(0)

# 检查危险模式
for pattern in DANGEROUS_PATTERNS:
    if re.search(pattern, command, re.IGNORECASE):
        decision = {
            "decision": "deny",
            "message": f"🚨 危险命令已拦截!\n\n命令: {command}\n\n匹配的危险模式: {pattern}\n\n如果确实需要执行,请在终端手动运行。"
        }
        print(json.dumps(decision, ensure_ascii=False))
        sys.exit(0)

sys.exit(0)

配置

{
  "hooks": {
    "PreToolUse": [
      {
        "matcher": "Bash",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/pre-block-dangerous-cmd.py",
            "timeout": 5
          }
        ]
      }
    ]
  }
}

3.2 PostToolUse(工具调用后)

一句话理解:PostToolUse就像"快递签收通知",在工具成功执行后自动触发后续操作。

触发时机

在工具成功执行后立即触发,可以处理工具的输出结果。

输入参数(通过stdin的JSON)

{
  "tool_name": "Write",
  "tool_input": {
    "file_path": "C:/project/src/app.js",
    "content": "console.log('Hello');"
  },
  "tool_output": {
    "success": true,
    "message": "File written successfully"
  }
}
字段 类型 说明
tool_name string 工具名称
tool_input object 工具的输入参数
tool_output object 工具的输出结果

输出格式

⚠️ 重要:PostToolUse Hook的输出机制

PostToolUse Hook无法向用户输出信息

  • 可以做:执行后处理任务(格式化、备份、测试、写日志文件)
  • 不能做:向用户显示信息(Claude Code只会显示"Hook执行成功/失败")
  • 常见误区print(..., file=sys.stderr)不会显示在界面,只在终端可见

如果需要向用户输出信息,请使用UserPromptSubmit HookadditionalContext机制。

PostToolUse Hook不返回决策(工具已经执行完了),只能:

  • 执行后处理任务(格式化、备份、测试)
  • 写入日志文件(用于调试)

完整示例1:自动代码格式化

场景:在Write工具保存.js/.ts文件后,自动运行Prettier格式化

脚本 .claude/hooks/post-auto-format.py

#!/usr/bin/env python3
"""
PostToolUse Hook - 自动代码格式化
保存代码文件后自动运行对应的格式化工具
"""
import sys
import json
import subprocess
from pathlib import Path

# 格式化工具配置
FORMATTERS = {
    '.js': 'npx prettier --write "{file}"',
    '.ts': 'npx prettier --write "{file}"',
    '.jsx': 'npx prettier --write "{file}"',
    '.tsx': 'npx prettier --write "{file}"',
    '.json': 'npx prettier --write "{file}"',
    '.css': 'npx prettier --write "{file}"',
    '.py': 'black "{file}"',
    '.go': 'gofmt -w "{file}"',
}

# 排除的目录
EXCLUDED_DIRS = {'node_modules', 'venv', '.venv', '__pycache__', 'dist', 'build', '.git'}

def should_format(file_path: str) -> bool:
    """检查是否应该格式化该文件"""
    path = Path(file_path)

    # 检查是否在排除目录中
    for part in path.parts:
        if part in EXCLUDED_DIRS:
            return False

    # 检查文件扩展名
    return path.suffix in FORMATTERS

def run_formatter(file_path: str) -> str:
    """运行格式化工具"""
    path = Path(file_path)
    suffix = path.suffix

    if suffix not in FORMATTERS:
        return None

    cmd = FORMATTERS[suffix].format(file=file_path)

    try:
        result = subprocess.run(
            cmd,
            shell=True,
            capture_output=True,
            text=True,
            timeout=30
        )
        if result.returncode == 0:
            return f"✅ 格式化成功"
        else:
            return f"⚠️ 格式化失败: {result.stderr[:100]}"
    except subprocess.TimeoutExpired:
        return "⚠️ 格式化超时"
    except FileNotFoundError:
        return "⚠️ 格式化工具未安装"
    except Exception as e:
        return f"⚠️ 格式化错误: {str(e)}"

def main():
    # 读取输入
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        return

    tool_name = input_data.get('tool_name', '')
    tool_input = input_data.get('tool_input', {})
    file_path = tool_input.get('file_path', '')

    # 只处理Write工具
    if tool_name != 'Write':
        return

    # 检查是否需要格式化
    if not file_path or not should_format(file_path):
        return

    # 运行格式化
    result = run_formatter(file_path)
    if result:
        print(f"\n[AutoFormat] {Path(file_path).name}: {result}", file=sys.stderr)

if __name__ == '__main__':
    main()

配置

{
  "hooks": {
    "PostToolUse": [
      {
        "matcher": "Write",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/post-auto-format.py",
            "timeout": 30
          }
        ]
      }
    ]
  }
}

运行效果

Claude:我来创建app.js文件...

[Write工具执行成功]

[AutoFormat] app.js: ✅ 格式化成功

完整示例2:自动备份Hook

场景:在Edit工具修改重要文件后,自动创建备份

脚本 .claude/hooks/post-auto-backup.py

#!/usr/bin/env python3
"""
PostToolUse Hook - 自动备份
编辑重要文件后自动创建.bak备份
"""
import sys
import json
import shutil
from pathlib import Path
from datetime import datetime

# 需要备份的目录
BACKUP_DIRS = ['config', 'src', 'docs', '.claude']

def main():
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        return

    tool_name = input_data.get('tool_name', '')
    tool_input = input_data.get('tool_input', {})
    file_path = tool_input.get('file_path', '')

    # 只处理Edit工具
    if tool_name != 'Edit':
        return

    # 检查是否在需要备份的目录中
    should_backup = any(dir_name in file_path for dir_name in BACKUP_DIRS)

    if should_backup and Path(file_path).exists():
        # 创建备份
        timestamp = datetime.now().strftime('%Y%m%d_%H%M%S')
        backup_path = f"{file_path}.{timestamp}.bak"

        try:
            shutil.copy2(file_path, backup_path)
            print(f"[Backup] ✅ 已创建备份: {Path(backup_path).name}", file=sys.stderr)
        except Exception as e:
            print(f"[Backup] ⚠️ 备份失败: {e}", file=sys.stderr)

if __name__ == '__main__':
    main()

3.3 UserPromptSubmit(用户提示词提交)

一句话理解:UserPromptSubmit就像"邮件发送前的自动补充",可以在用户输入发送给Claude之前自动增强或过滤。

触发时机

在用户提交提示词后,Claude处理之前

输入参数(通过stdin的JSON)

⚠️ 重要:UserPromptSubmit的输入是JSON格式,不是纯文本!

{
  "session_id": "abc123",
  "transcript_path": "/Users/.../.claude/projects/.../session.jsonl",
  "cwd": "/your/project/path",
  "permission_mode": "default",
  "hook_event_name": "UserPromptSubmit",
  "prompt": "请帮我写一篇关于AI的文章"
}
字段 类型 说明
session_id string 当前会话ID
prompt string 用户输入的原始提示词
cwd string Claude Code的工作目录
permission_mode string 权限模式(default/plan/bypassPermissions等)
hook_event_name string 事件类型标识

输出格式

Hook的stdout输出会被添加到AI上下文(作为系统消息注入)。

方式1:直接输出文本(会作为额外上下文添加)

## 写作要求
- 字数:1500字
- 风格:老金式接地气风格
- 包含实战案例

方式2:输出JSON格式(更多控制)

{
  "continue": true,
  "suppressOutput": false
}

完整示例:写作规范自动追加

场景:检测到写作任务时,自动追加写作规范

脚本 .claude/hooks/user-prompt-enhance.py

#!/usr/bin/env python3
"""
UserPromptSubmit Hook - 提示词自动增强
检测到特定任务时自动追加相关规范

【重要】:输入是JSON格式,必须先解析!
"""
import sys
import json

# 从stdin读取JSON输入
try:
    input_data = json.loads(sys.stdin.read())
except json.JSONDecodeError:
    # JSON解析失败,直接退出
    sys.exit(0)

# 获取用户输入的提示词
user_input = input_data.get('prompt', '').strip()

# 如果没有prompt字段,直接退出
if not user_input:
    sys.exit(0)

# 过滤简单回复(不需要增强)
simple_responses = ['好的', '是的', '继续', 'ok', 'yes', 'no', '确认', '取消']
if user_input.lower() in simple_responses or len(user_input) < 5:
    # 不需要增强,不输出任何内容
    sys.exit(0)

# 过滤斜杠命令
if user_input.startswith('/'):
    sys.exit(0)

# 检查是否是写作任务
writing_keywords = ['写', '文章', '生成', '创作', 'write', 'article', '内容']
is_writing_task = any(kw in user_input.lower() for kw in writing_keywords)

if is_writing_task:
    # 输出额外上下文(会添加到AI上下文中)
    enhancement = """
---
## 写作规范提醒(Hook自动注入)
1. **风格**:接地气、说人话,避免AI腔
2. **结构**:开头金句 -> 核心要点 -> 实战案例 -> 总结升华
3. **字数**:1500-2000字
4. **检查**:完成后运行 /pre-check 进行质量检查
---"""
    print(enhancement)
    print(f"[Hook] 已为写作任务注入规范", file=sys.stderr)

sys.exit(0)

💡 注意

  • 输入是JSON,必须用json.loads()解析
  • 用户原始输入在prompt字段中
  • 通过stdout输出JSON格式的additionalContext字段来注入上下文
  • stderr仅用于调试,不会显示在Claude Code界面

配置

{
  "hooks": {
    "UserPromptSubmit": [
      {
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/user-prompt-enhance.py",
            "timeout": 5
          }
        ]
      }
    ]
  }
}

运行效果

用户输入:

帮我写一篇关于Claude Code的介绍文章

Claude实际收到:

帮我写一篇关于Claude Code的介绍文章

---
## 写作规范提醒(自动追加)
1. **风格**:接地气、说人话,避免AI腔
2. **结构**:开头金句 -> 核心要点 -> 实战案例 -> 总结升华
3. **字数**:1500-2000字
4. **检查**:完成后运行 /pre-check 进行质量检查
---

3.4 Notification(通知)

一句话理解:Notification Hook就像"消息推送处理器",在Claude发送通知时触发。

触发时机

当Claude Code需要请求用户权限或提示输入空闲超过60秒时。

输入参数(通过stdin的JSON)

{
  "session_id": "abc123",
  "transcript_path": "/Users/.../.claude/projects/.../session.jsonl",
  "cwd": "/your/project/path",
  "hook_event_name": "Notification",
  "message": "Claude is waiting for your input"
}
字段 类型 说明
session_id string 当前会话ID
transcript_path string 会话记录文件路径
cwd string 工作目录
hook_event_name string 事件类型标识(固定为"Notification")
message string 通知消息内容(Notification特有字段

⚠️ 注意:根据官方实现,Notification Hook没有title字段,只有message字段。

完整示例:桌面通知Hook

场景:将Claude的通知转发到系统桌面通知

脚本 .claude/hooks/notification-desktop.py

#!/usr/bin/env python3
"""
Notification Hook - 桌面通知
将Claude的通知转发到系统桌面通知
"""
import sys
import json
import subprocess
import platform

def send_notification(title: str, message: str):
    """发送系统桌面通知"""
    system = platform.system()

    try:
        if system == 'Darwin':  # macOS
            subprocess.run([
                'osascript', '-e',
                f'display notification "{message}" with title "{title}"'
            ])
        elif system == 'Linux':
            subprocess.run(['notify-send', title, message])
        elif system == 'Windows':
            # Windows Toast通知(推荐方式)
            # 需要先安装:Install-Module -Name BurntToast -Scope CurrentUser
            try:
                # 优先使用BurntToast(更可靠)
                ps_cmd = f'New-BurntToastNotification -Text "{title}", "{message}"'
                result = subprocess.run(['powershell', '-Command', ps_cmd], capture_output=True)
                if result.returncode != 0:
                    raise Exception("BurntToast not available")
            except:
                # 回退方案:使用Windows原生Toast
                ps_cmd = f'''
                [Windows.UI.Notifications.ToastNotificationManager, Windows.UI.Notifications, ContentType = WindowsRuntime] | Out-Null
                $template = [Windows.UI.Notifications.ToastTemplateType]::ToastText02
                $xml = [Windows.UI.Notifications.ToastNotificationManager]::GetTemplateContent($template)
                $text = $xml.GetElementsByTagName("text")
                $text[0].AppendChild($xml.CreateTextNode("{title}")) | Out-Null
                $text[1].AppendChild($xml.CreateTextNode("{message}")) | Out-Null
                $toast = [Windows.UI.Notifications.ToastNotification]::new($xml)
                [Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("Claude Code").Show($toast)
                '''
                subprocess.run(['powershell', '-Command', ps_cmd], capture_output=True)
    except Exception as e:
        print(f"通知发送失败: {e}", file=sys.stderr)

def main():
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        return

    # 获取消息内容(注意:没有notification_type字段)
    message = input_data.get('message', '')
    session_id = input_data.get('session_id', '')

    if not message:
        return

    # 构建通知标题
    title = "Claude Code"

    # 发送桌面通知
    send_notification(title, message)
    print(f"[Notification] 已发送桌面通知: {message[:50]}...", file=sys.stderr)

if __name__ == '__main__':
    main()

3.5 SessionStart(会话开始)

一句话理解:SessionStart就像"开机自启动程序",在Claude Code启动时自动执行初始化任务。

触发时机

Claude Code启动时触发。

用途

  • 初始化环境
  • 检查依赖
  • 加载配置

完整示例:环境检查Hook

脚本 .claude/hooks/session-start-check.py

#!/usr/bin/env python3
"""
SessionStart Hook - 环境检查
启动时检查必需的工具和依赖是否已安装
"""
import sys
import shutil

# 检查必需的工具
required_tools = {
    'node': 'Node.js (npm install)',
    'python': 'Python 3.x',
    'git': 'Git版本控制',
}

# 可选但推荐的工具
optional_tools = {
    'prettier': 'Prettier代码格式化 (npm install -g prettier)',
    'black': 'Black Python格式化 (pip install black)',
}

missing_required = []
missing_optional = []

# 检查必需工具
for tool, desc in required_tools.items():
    if not shutil.which(tool):
        missing_required.append(f"  X {tool}: {desc}")

# 检查可选工具
for tool, desc in optional_tools.items():
    if not shutil.which(tool):
        missing_optional.append(f"  ! {tool}: {desc}")

# 输出检查结果
if missing_required or missing_optional:
    print("\n" + "="*50, file=sys.stderr)
    print("环境检查结果", file=sys.stderr)
    print("="*50, file=sys.stderr)

    if missing_required:
        print("\nX 缺少必需工具(请安装):", file=sys.stderr)
        for item in missing_required:
            print(item, file=sys.stderr)

    if missing_optional:
        print("\n! 缺少可选工具(建议安装):", file=sys.stderr)
        for item in missing_optional:
            print(item, file=sys.stderr)

    print("\n" + "="*50 + "\n", file=sys.stderr)
else:
    print("V 环境检查通过,所有工具已就绪", file=sys.stderr)

sys.exit(0)

配置

{
  "hooks": {
    "SessionStart": [
      {
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/session-start-check.py",
            "timeout": 10
          }
        ]
      }
    ]
  }
}

3.6 SessionEnd 和 Stop

SessionEnd(会话结束)

触发时机:Claude Code正常退出时

用途

  • 清理临时文件
  • 保存会话状态
  • 备份日志

示例脚本 .claude/hooks/session-end-cleanup.sh

#!/bin/bash
# SessionEnd Hook - 清理临时文件

temp_dir="$HOME/.claude/temp"
if [ -d "$temp_dir" ]; then
    rm -rf "$temp_dir"/*
    echo "V 临时文件已清理" >&2
fi

exit 0

Stop(AI停止响应)

触发时机:当Claude Code代理完成响应时

输入参数(通过stdin的JSON)

{
  "session_id": "abc123",
  "transcript_path": "/Users/.../.claude/projects/.../session.jsonl",
  "cwd": "/your/project/path",
  "hook_event_name": "Stop"
}

⚠️ 注意:Stop Hook没有reason字段,只有标准字段。

用途

  • 保存当前状态
  • 记录会话信息
  • 可以通过输出{"continue": false}强制停止

示例脚本 .claude/hooks/stop-save-state.py

#!/usr/bin/env python3
"""
Stop Hook - 保存状态
AI停止时保存当前会话状态
"""
import sys
import json
from datetime import datetime
from pathlib import Path

try:
    input_data = json.loads(sys.stdin.read())
except json.JSONDecodeError:
    input_data = {}

# 获取标准字段(注意:没有reason字段)
session_id = input_data.get('session_id', 'unknown')
cwd = input_data.get('cwd', str(Path.cwd()))

# 保存状态
state_dir = Path.home() / '.claude' / 'state'
state_dir.mkdir(parents=True, exist_ok=True)
state_file = state_dir / 'last-session-state.json'

state = {
    "stopped_at": datetime.now().isoformat(),
    "session_id": session_id,
    "project_dir": cwd
}

with open(state_file, 'w', encoding='utf-8') as f:
    json.dump(state, f, indent=2, ensure_ascii=False)

print(f"V 会话状态已保存到: {state_file}", file=sys.stderr)
sys.exit(0)

3.7 WorktreeCreate 和 WorktreeRemove(工作树管理)🆕

v2.1.49+ 新增:这两个Hook类型配合 Claude Code 内置的 Git Worktree 功能使用,用于 Git Worktree(工作树)的生命周期管理。

什么是 Git Worktree?

💡 生活类比:Git Worktree 就像在同一个项目里开了多个"平行工作台"。你可以在工作台A修bug,同时在工作台B开发新功能,互不干扰。Claude Code 使用 Worktree 来实现并行任务隔离。

WorktreeCreate(工作树创建时触发)

触发时机:Claude Code 创建新的 Git Worktree 时自动触发

典型用途

  • 初始化工作树特定的环境配置
  • 安装工作树所需的依赖
  • 设置工作树专属的环境变量
  • 记录工作树创建日志

配置示例

{
  "hooks": {
    "WorktreeCreate": [
      {
        "command": "echo \"新工作树已创建: $(date)\" >> ~/.claude/worktree.log",
        "timeout": 10000
      }
    ]
  }
}

输入数据(通过 stdin 接收 JSON):

{
  "worktree_path": "/path/to/worktree",
  "branch": "feature/new-feature"
}

WorktreeRemove(工作树删除时触发)

触发时机:Claude Code 删除 Git Worktree 时自动触发

典型用途

  • 清理工作树相关的临时文件
  • 释放工作树占用的资源
  • 记录工作树删除日志
  • 归档工作树的工作成果

配置示例

{
  "hooks": {
    "WorktreeRemove": [
      {
        "command": "echo \"工作树已删除: $(date)\" >> ~/.claude/worktree.log",
        "timeout": 10000
      }
    ]
  }
}

实战场景:工作树生命周期管理

{
  "hooks": {
    "WorktreeCreate": [
      {
        "command": ".claude/hooks/worktree-init.sh",
        "timeout": 30000
      }
    ],
    "WorktreeRemove": [
      {
        "command": ".claude/hooks/worktree-cleanup.sh",
        "timeout": 15000
      }
    ]
  }
}

⚠️ 注意:WorktreeCreate 和 WorktreeRemove 都是不可阻止的Hook,它们只用于执行附加操作,不能阻止工作树的创建或删除。

--worktree 启动参数的关系

🔥 重要:2026年2月(v2.1.49),Claude Code 正式内置了 Git Worktree 支持,这是一个核心功能级别的更新,不仅仅是 Hook。

--worktree-w)启动参数

# 在独立工作树中启动 Claude Code
claude --worktree
# 或简写
claude -w

工作原理

你的项目仓库(主工作目录)
├── .git/                    # 共享的 Git 历史
├── .claude/worktrees/       # 工作树存放目录(加到 .gitignore)
│   ├── worktree-abc123/     # Agent A 的独立工作目录
│   └── worktree-def456/     # Agent B 的独立工作目录
├── src/                     # 主工作目录的文件
└── ...

使用场景

场景 说明
并行开发 终端1: claude -w 开发新功能 / 终端2: claude -w 修复 bug
代码审查 主工作目录继续开发,工作树中运行审查 Agent
实验性修改 在工作树中试验方案,不影响主目录

配置步骤

# 1. 将工作树目录加到 .gitignore
echo ".claude/worktrees/" >> .gitignore

# 2. 启动第一个 Agent(在工作树中)
claude -w

# 3. 打开另一个终端,启动第二个 Agent(在另一个工作树中)
claude -w

Hook 的角色

  • Git 用户:直接使用 claude -w 即可,Hook 是可选的增强(如自动安装依赖)
  • 非 Git 用户(SVN/Perforce/Mercurial):通过 WorktreeCreate/WorktreeRemove Hook 自定义工作树的创建和清理逻辑,替代默认的 Git 行为

3.8 SubagentStart 和 SubagentStop(子代理生命周期)🆕

v2.1.49+ 新增:这两个Hook类型用于监控和管理子代理(Subagent/Task)的生命周期。

SubagentStart(子代理启动时触发)

触发时机:Claude Code 启动子代理(通过 Task 工具)时自动触发

典型用途

  • 记录子代理启动日志
  • 初始化子代理专属的环境配置
  • 监控并发子代理数量

配置示例

{
  "hooks": {
    "SubagentStart": [
      {
        "command": "echo \"子代理已启动: $(date)\" >> ~/.claude/subagent.log",
        "timeout": 5000
      }
    ]
  }
}

SubagentStop(子代理停止时触发)

触发时机:子代理完成任务或被终止时自动触发

典型用途

  • 收集子代理执行结果
  • 清理子代理使用的临时资源
  • 记录子代理执行耗时

3.9 PermissionRequest(权限请求)🆕

v2.1.49+ 新增:在权限请求时触发,可用于实现自动审批策略。

触发时机:Claude Code 请求执行需要用户授权的操作时

典型用途

  • 根据规则自动批准/拒绝权限请求
  • 记录权限请求审计日志
  • 实现基于项目的权限策略

配置示例

{
  "hooks": {
    "PermissionRequest": [
      {
        "command": "python .claude/hooks/permission-policy.py",
        "timeout": 5000
      }
    ]
  }
}

3.10 PreCompact(上下文压缩前)🆕

v2.1.49+ 新增:在 Claude Code 执行上下文压缩(Compact)前触发。

触发时机:当对话上下文即将被压缩时

典型用途

  • 在压缩前保存关键上下文信息
  • 记录压缩事件日志
  • 导出当前对话状态

3.11 ConfigChange(配置变更)🆕

v2.1.49+ 新增:在 Claude Code 配置发生变更时触发。

触发时机:settings.json 或其他配置文件被修改时

典型用途

  • 配置变更审计和日志记录
  • 自动同步配置到其他环境
  • 配置变更通知

3.12 TeammateIdle(队友空闲)🆕

v2.1.49+ 新增:在多代理协作场景中,当队友代理进入空闲状态时触发。

触发时机:协作中的队友代理完成当前任务、进入空闲状态时

典型用途

  • 自动分配待处理任务给空闲队友
  • 发送状态通知
  • 协调多代理工作流

3.13 新增Hook事件类型 🆕

v2.1+ 新增:以下Hook类型进一步扩展了Claude Code的事件覆盖范围,让你能捕捉到更多关键时刻。

🎯 生活类比:如果之前的Hook是"门窗报警器",这些新Hook就是"烟雾报警器、水浸传感器、燃气检测器"——覆盖了之前监控不到的异常场景。

StopFailure(API异常停止时触发)

触发时机:当API错误(429限流、401认证失败、500服务器错误等)导致会话异常停止时触发

与Stop的区别Stop 是正常结束(用户主动退出、任务完成),就像下班正常关灯锁门;StopFailure 是异常中断,就像突然停电——你需要知道发生了什么并采取措施。

典型用途

  • 发送告警通知(Slack/邮件/钉钉)
  • 记录错误日志用于后续分析
  • 触发自动重试或降级逻辑

配置示例

{
  "hooks": {
    "StopFailure": [
      {
        "command": "python .claude/hooks/alert-on-failure.py",
        "timeout": 10000
      }
    ]
  }
}

📝 输入数据:StopFailure的stdin JSON中包含 error 字段,携带具体的错误类型和消息,可用于区分限流、认证失败等不同场景。


PostCompact(上下文压缩后触发)

触发时机:执行 /compact 命令或上下文自动压缩完成后触发

🎯 生活类比:就像搬家后清点物品——压缩完成后,你想知道"丢掉了多少东西、还剩多少空间"。

典型用途

  • 记录压缩前后的token数量变化
  • 触发上下文恢复操作(如重新加载关键文件)
  • 日志记录用于监控上下文使用趋势

配置示例

{
  "hooks": {
    "PostCompact": [
      {
        "command": "echo \"[$(date)] 上下文已压缩\" >> ~/.claude/compact.log",
        "timeout": 5000
      }
    ]
  }
}

⚠️ 注意:PostCompact 是不可阻止的Hook,压缩已经完成,你只能执行后续操作。与 PreCompact(压缩前)配合使用效果更佳。


InstructionsLoaded(指令文件加载时触发)

触发时机:CLAUDE.md 等指令文件加载到上下文时触发

🎯 生活类比:就像新员工入职时检查培训手册是否齐全——确保AI读到了正确且完整的指令。

典型用途

  • 验证指令文件的完整性和正确性
  • 记录哪些指令文件被加载(便于调试)
  • 触发项目特定的初始化操作

配置示例

{
  "hooks": {
    "InstructionsLoaded": [
      {
        "command": "python .claude/hooks/verify-instructions.py",
        "timeout": 5000
      }
    ]
  }
}

📝 输入数据:stdin JSON中包含已加载指令文件的路径列表,可用于检查关键指令文件是否缺失。


Elicitation 和 ElicitationResult(MCP交互输入)🆕

v2.1+ 新增:这两个Hook配合MCP Elicitation功能使用,让你能监控和管理MCP服务器与用户之间的交互输入。

🎯 生活类比Elicitation 就像客服系统弹出问卷调查——MCP服务器需要向用户询问信息;ElicitationResult 就像用户填完问卷提交——你可以记录和验证填写的内容。

Elicitation — MCP请求输入时触发

触发时机:MCP服务器通过Elicitation API向用户发起交互输入请求时

典型用途

  • 记录MCP交互请求日志
  • 自动填充常用值(如默认配置)

ElicitationResult — 用户完成输入后触发

触发时机:用户完成MCP交互输入并提交后

典型用途

  • 验证用户输入数据的合法性
  • 记录交互结果用于审计
  • 触发后续自动化流程

配置示例

{
  "hooks": {
    "Elicitation": [
      {
        "command": "echo \"MCP请求输入: $(date)\" >> ~/.claude/elicitation.log",
        "timeout": 5000
      }
    ],
    "ElicitationResult": [
      {
        "command": "python .claude/hooks/validate-elicitation.py",
        "timeout": 5000
      }
    ]
  }
}

3.14 HTTP Hooks(远程Webhook集成)🆕

v2.1+ 新增:除了传统的Shell命令Hook,现在可以直接将Hook事件POST到远程URL,无需编写本地脚本。

🎯 生活类比:之前的Hook像是"自家装的门铃"——得自己接线、自己写响铃逻辑;HTTP Hook像是"接入物业监控中心"——事件发生时自动通知远程服务,你只管接收处理。

配置格式

{
  "hooks": {
    "PostToolUse": [
      {
        "type": "http",
        "url": "https://your-webhook.example.com/hook",
        "timeout": 5000
      }
    ]
  }
}

工作原理

  • 当Hook事件触发时,Claude Code 自动向配置的URL发送 POST 请求
  • 请求体为JSON格式,自动包含该Hook的标准输入数据(与Shell Hook的stdin内容相同)
  • 支持设置超时时间,防止远程服务无响应时阻塞工作流

Shell Hook vs HTTP Hook 对比

特性 Shell Hook HTTP Hook 🆕
执行方式 运行本地脚本/命令 POST JSON到远程URL
需要本地脚本 ✅ 是 ❌ 否
跨平台一致性 ❌ 需处理OS差异 ✅ 任何平台行为一致
适合场景 本地文件操作、代码检查 远程通知、日志收集、CI/CD
网络依赖 ❌ 无 ✅ 需要网络连接
配置复杂度 中(需写脚本) 低(只需URL)

适用场景

  • Slack/Discord通知:代码提交、构建完成时自动发送消息
  • CI/CD触发:Hook事件触发远程构建管道
  • 日志收集服务:将所有Hook事件发送到Datadog/ELK等平台
  • 团队协作:多人开发时自动同步状态到共享服务

⚠️ 安全提示:HTTP Hook会将Hook数据发送到外部服务,确保目标URL可信且使用HTTPS。避免在Hook数据中包含敏感信息(API密钥、密码等)。


第四部分:实战应用场景

本节目的:学习真实项目中的Hook应用

⏱️ 预计时间:1-1.5小时

4.1 Git自动化工作流

场景1:提交前自动检查

需求:在git commit前自动运行代码检查,包括:

  • 代码风格检查(lint)
  • 敏感信息检查(API Key等)
  • 分支保护(禁止直接提交main)

完整脚本 .claude/hooks/git-pre-commit-checker.py

#!/usr/bin/env python3
"""
Git提交前检查系统
在执行git commit前自动运行多项检查
"""
import sys
import json
import subprocess
import re
from pathlib import Path
from concurrent.futures import ThreadPoolExecutor, as_completed

# 配置
CONFIG = {
    'protected_branches': ['main', 'master', 'production'],
    'secret_patterns': [
        r'(?i)(api[_-]?key|apikey)\s*[=:]\s*["\']?[\w-]{20,}',
        r'(?i)(secret|password|passwd|pwd)\s*[=:]\s*["\']?[\w-]{8,}',
        r'(?i)(access[_-]?token|auth[_-]?token)\s*[=:]\s*["\']?[\w-]{20,}',
        r'sk-[a-zA-Z0-9]{20,}',  # OpenAI API Key
        r'ghp_[a-zA-Z0-9]{36,}',  # GitHub Token
    ],
}

def run_command(cmd, timeout=60):
    """运行命令并返回结果"""
    try:
        result = subprocess.run(cmd, shell=True, capture_output=True, text=True, timeout=timeout)
        return result.returncode, result.stdout, result.stderr
    except subprocess.TimeoutExpired:
        return -1, '', 'Command timed out'
    except Exception as e:
        return -1, '', str(e)

def check_branch():
    """检查分支规则"""
    code, stdout, _ = run_command('git rev-parse --abbrev-ref HEAD')
    if code != 0:
        return True, "无法获取当前分支"

    branch = stdout.strip()
    if branch in CONFIG['protected_branches']:
        return False, f"X 禁止直接提交到受保护分支: {branch}\n请使用Pull Request"

    return True, f"当前分支: {branch}"

def check_secrets():
    """检查敏感信息"""
    code, stdout, _ = run_command('git diff --cached')
    if code != 0:
        return True, "无法获取diff"

    findings = []
    for pattern in CONFIG['secret_patterns']:
        if re.search(pattern, stdout):
            findings.append(f"发现可疑模式: {pattern[:40]}...")

    if findings:
        return False, "X 发现可能的敏感信息:\n" + '\n'.join(findings)
    return True, "敏感信息检查通过"

def check_lint():
    """代码风格检查"""
    code, stdout, _ = run_command('git diff --cached --name-only --diff-filter=ACMR')
    if code != 0:
        return True, "无法获取变更文件列表"

    files = [f for f in stdout.strip().split('\n') if f]
    py_files = [f for f in files if f.endswith('.py')]
    js_files = [f for f in files if f.endswith(('.js', '.ts', '.jsx', '.tsx'))]

    errors = []

    # Python文件检查
    if py_files:
        code, stdout, stderr = run_command(f'ruff check {" ".join(py_files)}')
        if code != 0:
            errors.append(f"Python代码问题:\n{stdout or stderr}")

    # JavaScript/TypeScript文件检查
    if js_files:
        code, stdout, stderr = run_command(f'npx eslint {" ".join(js_files)} --quiet')
        if code != 0:
            errors.append(f"JS/TS代码问题:\n{stdout or stderr}")

    if errors:
        return False, '\n'.join(errors)
    return True, "代码风格检查通过"

def main():
    # 读取输入
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        return

    tool_name = input_data.get('tool_name', '')
    command = input_data.get('tool_input', {}).get('command', '')

    # 只处理git commit命令
    if tool_name != 'Bash' or 'git commit' not in command:
        return

    # 运行检查
    checks = [
        ('分支检查', check_branch),
        ('敏感信息', check_secrets),
        ('代码风格', check_lint),
    ]

    results = []
    all_passed = True

    # 并行执行检查
    with ThreadPoolExecutor(max_workers=3) as executor:
        future_to_check = {executor.submit(check[1]): check[0] for check in checks}
        for future in as_completed(future_to_check):
            check_name = future_to_check[future]
            try:
                passed, message = future.result()
                results.append((check_name, passed, message))
                if not passed:
                    all_passed = False
            except Exception as e:
                results.append((check_name, False, f"检查异常: {str(e)}"))
                all_passed = False

    # 输出报告
    print("\n" + "="*60, file=sys.stderr)
    print("Git提交前检查报告", file=sys.stderr)
    print("="*60, file=sys.stderr)

    for name, passed, message in results:
        status = "V PASS" if passed else "X FAIL"
        print(f"\n{status} {name}", file=sys.stderr)
        print(f"   {message}", file=sys.stderr)

    print("\n" + "="*60, file=sys.stderr)

    # 输出决策
    if not all_passed:
        decision = {
            "decision": "ask",
            "message": "检查未通过,是否仍要继续提交?"
        }
        print(json.dumps(decision, ensure_ascii=False))
    else:
        print("所有检查通过,允许提交", file=sys.stderr)

if __name__ == '__main__':
    main()

配置

{
  "hooks": {
    "PreToolUse": [
      {
        "matcher": "Bash",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/git-pre-commit-checker.py",
            "timeout": 120
          }
        ]
      }
    ]
  }
}

4.2 代码质量保障

场景:Markdown文章质量检查

需求:保存Markdown文章后自动检查:

  • 字数统计
  • 标题检查
  • 段落数量

脚本 .claude/hooks/post-article-quality.py

#!/usr/bin/env python3
"""
PostToolUse Hook - 文章质量检查
保存Markdown文件后自动进行质量检查
"""
import sys
import json
from pathlib import Path

def check_article_quality(file_path):
    """检查文章质量"""
    try:
        content = Path(file_path).read_text(encoding='utf-8')
    except Exception as e:
        return f"无法读取文件: {e}"

    # 统计指标
    char_count = len(content)
    word_count = len(content.split())
    has_title = content.strip().startswith('#')
    paragraphs = [p for p in content.split('\n\n') if p.strip()]
    paragraph_count = len(paragraphs)

    # 生成报告
    report = []
    report.append("\n" + "="*50)
    report.append("文章质量检查报告")
    report.append("="*50)
    report.append(f"\n字符数: {char_count} {'V' if char_count > 500 else '! 偏短'}")
    report.append(f"词数: {word_count}")
    report.append(f"标题: {'V 有' if has_title else 'X 缺少一级标题'}")
    report.append(f"段落数: {paragraph_count} {'V' if paragraph_count > 3 else '! 偏少'}")

    # 建议
    suggestions = []
    if char_count < 500:
        suggestions.append("- 建议增加内容,至少500字")
    if not has_title:
        suggestions.append("- 建议添加一级标题(# 标题)")
    if paragraph_count < 3:
        suggestions.append("- 建议增加段落,改善可读性")

    if suggestions:
        report.append("\n改进建议:")
        report.extend(suggestions)
    else:
        report.append("\nV 文章质量良好!")

    report.append("="*50 + "\n")

    return '\n'.join(report)

def main():
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        return

    tool_name = input_data.get('tool_name', '')
    tool_input = input_data.get('tool_input', {})
    file_path = tool_input.get('file_path', '')

    # 只处理Write工具和.md文件
    if tool_name != 'Write' or not file_path.endswith('.md'):
        return

    # 检查质量
    report = check_article_quality(file_path)
    print(report, file=sys.stderr)

if __name__ == '__main__':
    main()

4.3 完整配置示例

综合配置 .claude/settings.json

{
  "hooks": {
    "UserPromptSubmit": [
      {
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/user-prompt-enhance.py",
            "timeout": 5
          }
        ]
      }
    ],
    "PreToolUse": [
      {
        "matcher": "Write|Edit",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/pre-protect-production.py",
            "timeout": 5
          }
        ]
      },
      {
        "matcher": "Bash",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/pre-block-dangerous-cmd.py",
            "timeout": 5
          },
          {
            "type": "command",
            "command": "python .claude/hooks/git-pre-commit-checker.py",
            "timeout": 120
          }
        ]
      }
    ],
    "PostToolUse": [
      {
        "matcher": "Write",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/post-auto-format.py",
            "timeout": 30
          },
          {
            "type": "command",
            "command": "python .claude/hooks/post-article-quality.py",
            "timeout": 10
          }
        ]
      },
      {
        "matcher": "Edit",
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/post-auto-backup.py",
            "timeout": 10
          }
        ]
      }
    ],
    "SessionStart": [
      {
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/session-start-check.py",
            "timeout": 10
          }
        ]
      }
    ],
    "Notification": [
      {
        "hooks": [
          {
            "type": "command",
            "command": "python .claude/hooks/notification-desktop.py",
            "timeout": 5
          }
        ]
      }
    ]
  }
}

第五部分:故障排查

本节目的:快速解决Hook配置和执行问题

⏱️ 预计时间:按需查阅

5.1 Hook不执行

症状:配置了Hook但完全没有触发

排查步骤

  1. 检查配置文件路径
# 确认settings.json在正确位置
ls -la .claude/settings.json
# 应该存在且有内容
  1. 检查JSON格式
# 验证JSON格式
python -c "import json; json.load(open('.claude/settings.json'))"
# 没有报错说明格式正确
  1. 检查Matcher是否匹配
// 错误:matcher拼写错误
"matcher": "write"  // X 应该是大写W

// 正确
"matcher": "Write"  // V
  1. 检查脚本路径
# 确认脚本存在
ls -la .claude/hooks/your-hook.py

# macOS/Linux检查执行权限
chmod +x .claude/hooks/your-hook.py
  1. 重启Claude Code
# 退出当前会话
exit

# 重新启动
claude

5.2 Hook执行报错

症状:Hook触发了但报错退出

排查步骤

  1. 手动测试脚本
# 模拟输入测试
echo '{"tool_name": "Write", "tool_input": {"file_path": "test.txt"}}' | python .claude/hooks/your-hook.py

# 查看输出和错误
  1. 检查Python版本
python --version
# 应该是Python 3.x
  1. 检查依赖是否安装
# 如果脚本import了第三方库
pip install 缺少的库
  1. 查看stderr输出
# 在脚本中添加调试输出
import sys
print("DEBUG: 脚本开始执行", file=sys.stderr)
print(f"DEBUG: 收到输入: {input_data}", file=sys.stderr)

5.3 Hook超时

症状:Hook执行时间过长被强制终止

解决方案

  1. 增加timeout配置
{
  "type": "command",
  "command": "python .claude/hooks/slow-hook.py",
  "timeout": 120  // 增加到120秒
}
  1. 优化脚本性能
# 避免不必要的文件扫描
# 使用增量检查而不是全量检查
# 并行执行多个检查任务
  1. 异步处理
# 对于不需要阻塞的任务,可以后台执行
import subprocess
subprocess.Popen(['python', 'background-task.py'],
                 stdout=subprocess.DEVNULL,
                 stderr=subprocess.DEVNULL)

5.4 Windows特有问题

问题1:中文乱码

症状:Hook输出中文显示为乱码

解决方案

# 在脚本开头设置编码
import sys
import io
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
sys.stderr = io.TextIOWrapper(sys.stderr.buffer, encoding='utf-8')

问题2:路径分隔符

症状:Windows路径包含反斜杠导致问题

解决方案

# 统一处理路径
file_path = file_path.replace('\\', '/')

问题3:Batch脚本编码

症状:.bat文件中文乱码

解决方案

@echo off
chcp 65001 >nul
REM 脚本内容...

5.5 调试技巧

技巧1:日志文件

# 写入调试日志
from pathlib import Path
from datetime import datetime

log_file = Path.home() / '.claude' / 'hooks-debug.log'
log_file.parent.mkdir(parents=True, exist_ok=True)

with open(log_file, 'a', encoding='utf-8') as f:
    f.write(f"[{datetime.now()}] {message}\n")

技巧2:条件调试

import os

# 通过环境变量控制调试模式
DEBUG = os.getenv('CLAUDE_HOOK_DEBUG', '').lower() == 'true'

if DEBUG:
    print(f"DEBUG: {data}", file=sys.stderr)

技巧3:逐步排除法

// 先只保留一个Hook测试
{
  "hooks": {
    "PostToolUse": [
      {
        "matcher": "Write",
        "hooks": [
          {
            "type": "command",
            "command": "echo 'Hook triggered!' >&2"
          }
        ]
      }
    ]
  }
}

第六部分:FAQ(20个常见问题)

基础问题

Q1: Hook和CLAUDE.md有什么区别?

对比 Hook CLAUDE.md
执行方式 自动执行Shell命令 Claude解读后决定是否遵循
可靠性 100%执行 不确定(AI可能忘记)
用途 强制规则、自动化 提供上下文、建议

Q2: Hook可以用什么语言写?

任何可执行程序都可以:

  • Python(推荐,跨平台)
  • Bash/Shell(macOS/Linux)
  • Batch(Windows)
  • Node.js
  • Go、Rust等(需要编译)

Q3: Hook的timeout默认是多少?

默认60秒。建议根据任务复杂度设置:

  • 简单检查:5-10秒
  • 代码格式化:30秒
  • 完整测试:120秒

Q4: 一个事件可以配置多个Hook吗?

可以!多个Hook会按顺序执行:

{
  "matcher": "Write",
  "hooks": [
    {"type": "command", "command": "python hook1.py"},
    {"type": "command", "command": "python hook2.py"}
  ]
}

Q5: PreToolUse的decision有哪些值?

PreToolUse 有新旧两套API(新旧并存):

新版(推荐) — 通过 hookSpecificOutput.permissionDecision 字段:

效果
"allow" 允许执行,绕过权限系统
"deny" 拒绝执行,原因会反馈给Claude
"ask" 暂停,询问用户决定
无输出 默认允许

旧版(已废弃但仍支持) — 通过 decision 字段:

旧版值 等同于新版
"approve" "allow"
"block" "deny"

PostToolUse 和 UserPromptSubmit 使用 "block" 来阻止/提供反馈,无输出则默认允许。

配置问题

Q6: settings.json应该放在哪里?

项目根目录的 .claude/settings.json

Q7: Matcher支持正则表达式吗?

支持!例如:

  • "Write" - 精确匹配
  • "Write|Edit" - 匹配Write或Edit
  • ".*" - 匹配所有工具(慎用)

Q8: 如何让Hook只对特定目录的文件生效?

在脚本中检查文件路径:

if '/articles/' not in file_path:
    sys.exit(0)  # 不在目标目录,跳过

Q9: 环境变量怎么传递给Hook脚本?

Claude Code自动传递的环境变量:

  • CLAUDE_PROJECT_DIR - 项目根目录的绝对路径(Claude Code启动目录)

⚠️ 注意session_id不是环境变量,而是通过stdin的JSON输入传递!

使用方法:

import os
import json
import sys

# 从环境变量获取项目目录
project_dir = os.getenv('CLAUDE_PROJECT_DIR', os.getcwd())

# 从stdin的JSON获取session_id(不是环境变量!)
input_data = json.loads(sys.stdin.read())
session_id = input_data.get('session_id', '')

Q10: 如何临时禁用Hook?

方法1:重命名配置文件

mv .claude/settings.json .claude/settings.json.bak

方法2:注释掉Hook配置(JSON不支持注释,需要删除)

脚本问题

Q11: 如何读取Hook的输入?

import sys
import json

# stdin读取JSON
input_data = json.loads(sys.stdin.read())
tool_name = input_data.get('tool_name')

Q12: 如何在Hook中向用户输出信息?

重要:不同Hook类型的输出机制不同!

PostToolUse Hook:无法直接输出给用户

  • Claude Code只会显示"Hook执行成功/失败"
  • 如需调试,写入日志文件:
from pathlib import Path
log_file = Path.home() / '.claude' / 'hooks' / 'debug.log'
with open(log_file, 'a') as f:
    f.write(f"调试信息\n")

UserPromptSubmit Hook:通过stdout JSON的additionalContext字段

import json
output = {
    "hookSpecificOutput": {
        "hookEventName": "UserPromptSubmit",
        "additionalContext": "要显示给用户的内容"
    }
}
print(json.dumps(output))

PreToolUse Hook:通过stdout JSON返回决策

import json
decision = {
    "hookSpecificOutput": {
        "hookEventName": "PreToolUse",
        "permissionDecision": {
            "decision": "deny",
            "message": "拒绝原因(用户可见)"
        }
    }
}
print(json.dumps(decision))

stderr输出:仅用于调试,不会显示在Claude Code界面

print("调试信息", file=sys.stderr)  # 只在终端可见,用户看不到

Q13: 如何返回决策(PreToolUse)?

使用stdout输出JSON:

print(json.dumps({"decision": "deny", "message": "原因"}))

Q14: 脚本报错会影响Claude Code吗?

不会!Hook脚本出错不会阻止Claude Code运行,只是该Hook功能失效。

Q15: 如何处理Windows/macOS/Linux兼容性?

import platform
system = platform.system()

if system == 'Windows':
    # Windows特定代码
elif system == 'Darwin':  # macOS
    # macOS特定代码
else:  # Linux
    # Linux特定代码

高级问题

Q16: Hook可以修改Claude的输出吗?

不能直接修改。但可以:

  • PostToolUse后修改文件内容
  • UserPromptSubmit修改用户输入

Q17: 多个Hook的执行顺序是什么?

按配置文件中的顺序依次执行。

Q18: Hook可以调用Claude API吗?

可以,但要注意:

  • 会消耗额外Token
  • 可能导致无限循环
  • 建议设置调用限制

Q19: 如何在团队中共享Hook配置?

.claude/ 目录加入Git版本控制:

git add .claude/settings.json
git add .claude/hooks/
git commit -m "Add Claude Code hooks"

Q20: Hook有性能影响吗?

有一定影响:

  • 每次工具调用都会触发Hook
  • 复杂脚本会增加延迟
  • 建议优化脚本性能,设置合理timeout

附录A:配置速查表

Hook类型速查

Hook类型 触发时机 输入格式 输出格式 可阻止 重要
UserPromptSubmit 用户输入后 JSON 文本(注入上下文) V
PreToolUse 工具调用前 JSON JSON决策 V
PostToolUse 工具调用后 JSON X
Notification 通知发送时 JSON X
SessionStart 会话开始 X
SessionEnd 会话结束 X
Stop AI停止 JSON X
SubagentStart 🆕 子代理启动 JSON X
SubagentStop 🆕 子代理停止 JSON X
PermissionRequest 🆕 权限请求 JSON JSON决策 V
PreCompact 🆕 上下文压缩前 JSON X
ConfigChange 🆕 配置变更 JSON X
TeammateIdle 🆕 队友空闲 JSON X
WorktreeCreate 工作树创建 JSON X
WorktreeRemove 工作树删除 JSON X
StopFailure 🆕 API异常停止 JSON X
PostCompact 🆕 上下文压缩后 JSON X
InstructionsLoaded 🆕 指令文件加载 JSON X
Elicitation 🆕 MCP请求输入 JSON X
ElicitationResult 🆕 MCP输入完成 JSON X

常用工具名速查

工具名 功能 常用Hook 重要
Write 写入文件 PostToolUse格式化
Edit 编辑文件 PostToolUse备份
Read 读取文件 PreToolUse权限控制
Bash 执行命令 PreToolUse危险命令拦截
Glob 文件搜索 -
Grep 内容搜索 -
WebSearch 网络搜索 -

Decision值速查

PreToolUse 新版APIhookSpecificOutput.permissionDecision,推荐):

含义 工具执行 重要
"allow" 允许,绕过权限系统 V
"deny" 拒绝,原因反馈给Claude X
"ask" 暂停,询问用户 ?
无输出 默认允许 V

PreToolUse 旧版APIdecision,已废弃但仍支持):

旧版值 等同于新版
"approve" "allow"
"block" "deny"

PostToolUse / UserPromptSubmit

含义
"block" 阻止/提供反馈
无输出 默认允许

附录B:完整脚本模板

Python脚本模板

#!/usr/bin/env python3
"""
Hook名称 - 功能描述
"""
import sys
import json

def main():
    # 读取输入
    try:
        input_data = json.loads(sys.stdin.read())
    except json.JSONDecodeError:
        sys.exit(0)

    tool_name = input_data.get('tool_name', '')
    tool_input = input_data.get('tool_input', {})

    # 你的逻辑
    # ...

    # 输出到Claude Code界面
    print("信息", file=sys.stderr)

    # 如果是PreToolUse,输出决策
    # print(json.dumps({"decision": "allow"}))

if __name__ == '__main__':
    main()

Bash脚本模板

#!/bin/bash
# Hook名称 - 功能描述

# 读取stdin
input_json=$(cat)

# 使用Python解析JSON
tool_name=$(echo "$input_json" | python3 -c "import sys,json; print(json.load(sys.stdin).get('tool_name',''))")

# 你的逻辑
# ...

# 输出日志
echo "信息" >&2

exit 0

附录C:参考资源

官方文档

社区资源

相关课程

  • 第3部分:Commands系统 - Slash命令开发
  • 第4部分:MCP集成 - 外部工具连接
  • 第6部分:Skills定制 - 技能包开发

学习总结

通过本课学习,你已经掌握:

  1. Hooks核心概念:理解Hook是什么、为什么需要、能做什么
  2. 15种Hook类型:PreToolUse、PostToolUse、UserPromptSubmit、Notification、Stop、SessionStart、SessionEnd、SubagentStart、SubagentStop、PermissionRequest、PreCompact、ConfigChange、TeammateIdle、WorktreeCreate、WorktreeRemove全部类型
  3. 配置方法:settings.json配置格式、Matcher语法、timeout设置
  4. 实战场景:Git自动化、代码格式化、文件保护、质量检查
  5. 故障排查:常见问题诊断和解决方法
  6. 安全意识:Hook安全风险和最佳实践

下一步建议

  1. 从第二部分的简单示例开始实践
  2. 根据你的项目需求,选择合适的Hook类型
  3. 参考第四部分的实战案例,逐步构建自己的自动化工作流
  4. 遇到问题查阅第五部分和第六部分

记住:Hooks是自动化的终极武器,合理使用可以让你的开发效率翻倍!


文档版本:v1.3(2026-04:v2.1.90 / v2.1.92 release 摘录) 最后更新:2026年4月5日 作者:老金

v1.2(2026-04-05)

  • 差量更新:增加 v2.1.90(PostToolUse 与 PreToolUse 相关修复)、v2.1.92(prompt-type Stop hooks)官方 release What's changed 英文原文摘录(附链接)

v1.1 修正内容(2025-12-23)

修正项 问题 修正后
UserPromptSubmit输入格式 错误描述为"纯文本" 修正为JSON格式
UserPromptSubmit示例脚本 直接读文本导致脚本无法工作 使用json.loads解析
速查表UserPromptSubmit行 输入格式列写"纯文本"与正文矛盾 修正为JSON,输出列修正为"文本(注入上下文)"
Notification输入格式 字段不完整 添加完整字段说明
Stop Hook输入格式 错误声称有reason字段 删除不存在的字段
环境变量说明 错误声称有CLAUDE_SESSION_ID 明确session_id在JSON中
Windows通知脚本 NotifyIcon不稳定 改用BurntToast/原生Toast