快速开始
以下示例演示如何用 Hook 拦截危险命令——当 Agent 尝试执行rm -rf 时自动阻止。
第一步:创建脚本
~/.qoder/settings.json 中添加:
rm -rf 的命令,Hook 会阻止执行并提示 Agent。
配置
配置文件位置
Hook 配置从以下三个文件加载,三级配置会被合并执行:配置格式
| 字段 | 必填 | 说明 |
|---|---|---|
type | 是 | 固定为 "command" |
command | 是 | 要执行的 shell 命令 |
timeout | 否 | 超时时间(秒),默认 60 |
matcher | 否 | 匹配条件,不填则匹配所有 |
matcher 匹配规则
matcher 用于过滤 hook 的触发范围,不同事件匹配不同的字段(见各事件说明)。
| 写法 | 含义 | 示例 |
|---|---|---|
不填或 "*" | 匹配所有 | 所有工具都触发 |
| 精确值 | 精确匹配 | "Bash" 只匹配 Bash 工具 |
| 分隔 | 匹配多个值 | "Write|Edit" 匹配 Write 或 Edit |
| 正则表达式 | 正则匹配 | "mcp__.*" 匹配所有 MCP 工具 |
Hook 脚本编写
Hook 脚本通过 stdin 接收 JSON 输入,通过 exit code 和 stdout 输出来控制行为。本节说明所有事件通用的输入输出格式,各事件的额外字段见支持的事件。输入
Hook 脚本通过 stdin 接收 JSON 数据。所有事件都包含以下通用字段:| 字段 | 说明 |
|---|---|
session_id | 当前会话 ID |
cwd | 当前工作目录 |
hook_event_name | 触发的事件名称 |
jq 解析输入:
输出
Hook 通过 exit code 和 stdout 控制行为。 exit 0 表示成功。Qoder CLI 会解析 stdout,部分事件(如 UserPromptSubmit、SessionStart)支持纯文本上下文注入或 JSON 精细控制,具体见各事件说明。 exit 2 表示阻塞错误,仅对支持阻塞的事件生效。stdout 被忽略,stderr 作为错误信息反馈给 Agent。具体效果取决于事件:PreToolUse 阻止工具执行,UserPromptSubmit 拒绝 prompt,Stop 阻止 Agent 停止,等等。 其他 exit code 视为非阻塞错误,不影响执行流程,stderr 仅记录在日志中。环境变量
Hook 脚本执行时可以使用以下环境变量:| 变量 | 说明 |
|---|---|
QODER_PROJECT_DIR | 当前项目的工作目录 |
支持的事件
Qoder CLI 支持以下 Hook 事件,覆盖会话生命周期的各个阶段。SessionStart
会话开始时触发。 matcher 匹配: 会话来源| matcher 值 | 触发场景 |
|---|---|
startup | 新会话启动 |
resume | 恢复已有会话 |
compact | 上下文压缩完成后 |
SessionEnd
会话结束时触发。 matcher 匹配: 结束原因| matcher 值 | 触发场景 |
|---|---|
prompt_input_exit | 用户退出输入(Ctrl+D 等) |
other | 其他原因 |
UserPromptSubmit
用户提交 Prompt 后、Agent 处理前触发。可以为 Agent 注入额外上下文,也可以验证并阻止特定类型的 prompt。不支持 matcher,所有配置的 hook 都会执行。 通用字段之外,UserPromptSubmit 额外收到prompt 字段,包含用户提交的文本:
additionalContext 字段注入。纯文本更简单,但注意不要以 { 开头,否则会被当作 JSON 解析。
要阻止 prompt,可以用 exit 2(stderr 内容展示给用户),也可以 exit 0 输出包含 decision: "block" 的 JSON。exit 2 适合纯拦截场景;JSON 方式更灵活,适合在同一脚本中根据条件决定阻止还是放行。
| 字段 | 说明 |
|---|---|
decision | 设为 "block" 阻止 prompt 处理并从上下文中移除,不设置则放行 |
reason | 阻止时展示给用户的原因说明,不会添加到上下文 |
additionalContext | 注入给 Agent 的上下文字符串(通过 hookSpecificOutput 传递) |
JSON 格式不是必须的。简单场景下 exit 0 直接输出纯文本即可注入上下文,需要阻止 prompt 或精细控制时才需要 JSON。完整示例见提交 Prompt 时注入上下文和阻止包含敏感信息的 Prompt。
PreToolUse
工具执行前触发。可以阻止工具执行。 matcher 匹配: 工具名(如Bash、Write、Edit、Read、Glob、Grep,MCP 工具名如 mcp__server__tool)
额外输入字段:
PostToolUse
工具执行成功后触发。 matcher 匹配: 工具名 额外输入字段:PostToolUseFailure
工具执行失败后触发。 matcher 匹配: 工具名 额外输入字段:Stop
Agent 完成响应后触发(主 Agent,无待执行的工具调用时)。可以阻止 Agent 停止,让其继续工作。 阻止 Agent 停止: exit code 2,stderr 内容作为消息注入对话,Agent 继续工作。SubagentStart / SubagentStop
子 Agent 启动和完成时触发。SubagentStop 与 Stop 类似,可以阻止子 Agent 停止。 matcher 匹配: Agent 类型名 额外输入字段:PreCompact
上下文压缩前触发。 matcher 匹配: 触发方式| matcher 值 | 触发场景 |
|---|---|
manual | 用户手动执行 /compact |
auto | 上下文窗口满时自动触发 |
Notification
通知事件触发(权限请求、任务完成等)。 matcher 匹配: 通知类型| matcher 值 | 触发场景 |
|---|---|
permission | 权限请求通知 |
result | Agent 产出结果通知 |
PermissionRequest
工具执行需要用户授权时触发。 matcher 匹配: 工具名 额外输入字段:实用场景
桌面通知提醒
当 Agent 完成任务或需要授权时,弹出桌面通知。 脚本~/.qoder/hooks/notify.sh(macOS):
写文件后自动 Lint
每次 Agent 写入或编辑文件后,自动执行 lint 检查。 脚本${project}/.qoder/hooks/auto-lint.sh:
PostToolUse,matcher Write|Edit,command .qoder/hooks/auto-lint.sh。
让 Agent 继续工作
在 Agent 停止时检查是否还有未完成的任务,如果有则注入消息让 Agent 继续。 脚本~/.qoder/hooks/check-continue.sh:
Stop,command ~/.qoder/hooks/check-continue.sh。
提交 Prompt 时注入上下文
在每次用户提问前,自动注入当前 git 分支信息作为 Agent 上下文。 脚本~/.qoder/hooks/inject-branch.sh:
UserPromptSubmit,command ~/.qoder/hooks/inject-branch.sh。
阻止包含敏感信息的 Prompt
拦截包含密码、密钥等敏感关键词的 prompt,阻止其发送给 Agent。 脚本~/.qoder/hooks/block-sensitive.sh:
UserPromptSubmit,command ~/.qoder/hooks/block-sensitive.sh。