# HOOKS-README contains all the details, scripts, and instructions for the hooks ## Hook Events Overview - [Official 27 Hooks](https://code.claude.com/docs/en/hooks) Claude Code provides several hook events that run at different points in the workflow: | # | Hook | Description | Options | |:-:|------|-------------|---------| | 1 | `PreToolUse` | Runs before tool calls (can block them) | `async`, `timeout: 5000`, `tool_use_id` | | 2 | `PermissionRequest` | Runs when Claude Code requests permission from the user | `async`, `timeout: 5000`, `permission_suggestions` | | 3 | `PostToolUse` | Runs after tool calls complete successfully | `async`, `timeout: 5000`, `tool_response`, `tool_use_id` | | 4 | `PostToolUseFailure` | Runs after tool calls fail | `async`, `timeout: 5000`, `error`, `is_interrupt`, `tool_use_id` | | 5 | `UserPromptSubmit` | Runs when the user submits a prompt, before Claude processes it | `async`, `timeout: 5000`, `prompt` | | 6 | `Notification` | Runs when Claude Code sends notifications | `async`, `timeout: 5000`, `notification_type`, `message`, `title` | | 7 | `Stop` | Runs when Claude Code finishes responding | `async`, `timeout: 5000`, `last_assistant_message`, `stop_hook_active` | | 8 | `SubagentStart` | Runs when subagent tasks start | `async`, `timeout: 5000`, `agent_id`, `agent_type` | | 9 | `SubagentStop` | Runs when subagent tasks complete | `async`, `timeout: 5000`, `agent_id`, `agent_type`, `last_assistant_message`, `agent_transcript_path`, `stop_hook_active` | | 10 | `PreCompact` | Runs before Claude Code is about to run a compact operation | `async`, `timeout: 5000`, `once`, `compact_trigger` | | 11 | `PostCompact` | Runs after Claude Code completes a compact operation | `async`, `timeout: 5000`, `compact_trigger` | | 12 | `SessionStart` | Runs when Claude Code starts a new session or resumes an existing session | `async`, `timeout: 5000`, `once`, `agent_type`, `model`, `source` | | 13 | `SessionEnd` | Runs when Claude Code session ends | `async`, `timeout: 5000`, `once`, `reason` | | 14 | `Setup` | Runs when Claude Code runs the /setup command for project initialization | `async`, `timeout: 30000` | | 15 | `TeammateIdle` | Runs when a teammate agent becomes idle (experimental agent teams) | `async`, `timeout: 5000`, `teammate_name`, `team_name` | | 16 | `TaskCreated` | Runs when a task is being created via the TaskCreate tool (experimental agent teams) | `async`, `timeout: 5000`, `task_id`, `task_subject`, `task_description`, `teammate_name`, `team_name` | | 17 | `TaskCompleted` | Runs when a background task completes (experimental agent teams) | `async`, `timeout: 5000`, `task_id`, `task_subject`, `task_description`, `teammate_name`, `team_name` | | 18 | `ConfigChange` | Runs when a configuration file changes during a session | `async`, `timeout: 5000`, `file_path`, `source` | | 19 | `WorktreeCreate` | Runs when agent worktree isolation creates worktrees for custom VCS setup | `async`, `timeout: 5000`, `worktree_path`, `worktree_name`, `base_branch` | | 20 | `WorktreeRemove` | Runs when agent worktree isolation removes worktrees for custom VCS teardown | `async`, `timeout: 5000`, `worktree_path`, `worktree_name` | | 21 | `InstructionsLoaded` | Runs when CLAUDE.md or `.claude/rules/*.md` files are loaded into context | `async`, `timeout: 5000`, `file_path`, `memory_type`, `load_reason`, `globs`, `trigger_file_path`, `parent_file_path` | | 22 | `Elicitation` | Runs when an MCP server requests user input during a tool call | `async`, `timeout: 5000`, `mcp_server_name`, `tool_name`, `form_fields` | | 23 | `ElicitationResult` | Runs after a user responds to an MCP elicitation, before the response is sent back to the server | `async`, `timeout: 5000`, `mcp_server_name`, `tool_name`, `user_response`, `form_fields` | | 24 | `StopFailure` | Runs when the turn ends due to an API error (rate limit, auth failure, etc.) | `async`, `timeout: 5000`, `error`, `error_details`, `last_assistant_message` | | 25 | `CwdChanged` | Runs when the working directory changes during a session (reactive env management, e.g. direnv) | `async`, `timeout: 5000`, `old_cwd`, `new_cwd` | | 26 | `FileChanged` | Runs when watched files change during a session (reactive env management, e.g. direnv). **Requires `matcher` with pipe-separated basenames** (e.g. `.envrc\|.env`) to specify which files to watch | `async`, `timeout: 5000`, `file_path`, `change_type` | | 27 | `PermissionDenied` | Runs after auto mode classifier denies a tool call. Return `{retry: true}` to tell the model it can retry | `async`, `timeout: 5000`, `tool_name`, `tool_input`, `tool_use_id`, `reason` | > **Note:** Hooks 15-17 (`TeammateIdle`, `TaskCreated`, and `TaskCompleted`) require the experimental agent teams feature. Set `CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS=1` when launching Claude Code to enable them. ### Not in Official Docs The following items exist in the [Claude Code Changelog](https://github.com/anthropics/claude-code/blob/main/CHANGELOG.md) but are **not listed** in the [Official Hooks Reference](https://code.claude.com/docs/en/hooks): | Item | Added In | Changelog Reference | Notes | |------|----------|-------------------|-------| | `Setup` hook | [v2.1.10](https://github.com/anthropics/claude-code/blob/main/CHANGELOG.md#2110) | "Added new Setup hook event that can be triggered via `--init`, `--init-only`, or `--maintenance` CLI flags for repository setup and maintenance operations" | Not listed in official hooks reference page (26 hooks listed, Setup excluded) | | Agent frontmatter hooks | [v2.1.0](https://github.com/anthropics/claude-code/blob/main/CHANGELOG.md#210) | "Added hooks support to agent frontmatter, allowing agents to define PreToolUse, PostToolUse, and Stop hooks scoped to the agent's lifecycle" | Changelog only mentions 3 hooks, but testing confirms **6 hooks** actually fire in agent sessions: PreToolUse, PostToolUse, PermissionRequest, PostToolUseFailure, Stop, SubagentStop. Not all 27 hooks are supported. | ## Prerequisites Before using hooks, ensure you have **Python 3** installed on your system. ### Required Software #### All Platforms (Windows, macOS, Linux) - **Python 3**: Required for running the hook scripts - Verify installation: `python3 --version` **Installation Instructions:** - **Windows**: Download from [python.org](https://www.python.org/downloads/) or install via `winget install Python.Python.3` - **macOS**: Install via `brew install python3` (requires [Homebrew](https://brew.sh/)) - **Linux**: Install via `sudo apt install python3` (Ubuntu/Debian) or `sudo yum install python3` (RHEL/CentOS) ### Audio Players (Optional - Automatically Detected) The hook scripts automatically detect and use the appropriate audio player for your platform: - **macOS**: Uses `afplay` (built-in, no installation needed) - **Linux**: Uses `paplay` from `pulseaudio-utils` - install via `sudo apt install pulseaudio-utils` - **Windows**: Uses built-in `winsound` module (included with Python) ### How Hooks Are Executed The hooks are configured in `.claude/settings.json` to run directly with Python 3: ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py" } ``` ## Configuring Hooks (Enable/Disable) Hooks can be easily enabled or disabled at both the global and individual levels. ### Disable All Hooks at Once Edit `.claude/settings.local.json` and set: ```json { "disableAllHooks": true } ``` **Note:** The `.claude/settings.local.json` file is git-ignored, so each user can configure their own hook preferences without affecting the team's shared settings in `.claude/settings.json`. > **Managed Settings:** If an administrator has configured hooks through managed policy settings, `disableAllHooks` set in user, project, or local settings cannot disable those managed hooks (fixed in v2.1.49). ### Disable Individual Hooks For granular control, you can disable specific hooks by editing the hooks configuration files. #### Configuration Files There are two configuration files for managing individual hooks: 1. **`.claude/hooks/config/hooks-config.json`** - The shared/default configuration that is committed to git 2. **`.claude/hooks/config/hooks-config.local.json`** - Your personal overrides (git-ignored) The local config file (`.local.json`) takes precedence over the shared config, allowing each developer to customize their hook behavior without affecting the team. #### Shared Configuration Edit `.claude/hooks/config/hooks-config.json` for team-wide defaults: ```json { "disableLogging": false, "disablePreToolUseHook": false, "disablePermissionRequestHook": false, "disablePostToolUseHook": false, "disablePostToolUseFailureHook": false, "disableUserPromptSubmitHook": false, "disableNotificationHook": false, "disableStopHook": false, "disableSubagentStartHook": false, "disableSubagentStopHook": false, "disablePreCompactHook": false, "disablePostCompactHook": false, "disableElicitationHook": false, "disableElicitationResultHook": false, "disableStopFailureHook": false, "disableSessionStartHook": false, "disableSessionEndHook": false, "disableSetupHook": false, "disableTeammateIdleHook": false, "disableTaskCompletedHook": false, "disableConfigChangeHook": false, "disableWorktreeCreateHook": false, "disableWorktreeRemoveHook": false, "disableInstructionsLoadedHook": false, "disableCwdChangedHook": false, "disableFileChangedHook": false, "disablePermissionDeniedHook": false } ``` **Configuration Options:** - `disableLogging`: Set to `true` to disable logging hook events to `.claude/hooks/logs/hooks-log.jsonl` (useful to prevent log file growth) #### Local Configuration (Personal Overrides) Create or edit `.claude/hooks/config/hooks-config.local.json` for personal preferences: ```json { "disableLogging": true, "disablePostToolUseHook": true, "disableSessionStartHook": true } ``` In this example, logging is disabled, and the PostToolUse and SessionStart hooks are overridden locally. All other hooks will use the shared configuration values. **Note:** Individual hook toggles are checked by the hook script (`.claude/hooks/scripts/hooks.py`). Local settings override shared settings, and if a hook is disabled, the script exits silently without playing any sounds or executing hook logic. ### Text to Speech (TTS) website used to generate sounds: https://elevenlabs.io/ voice used: Samara X ## Agent Frontmatter Hooks Claude Code 2.1.0 introduced support for agent-specific hooks defined in agent frontmatter files. These hooks only run within the agent's lifecycle and support a subset of hook events. ### Supported Agent Hooks Agent frontmatter hooks support **6 hooks** (not all 27). The changelog originally mentioned only 3, but testing confirms 6 hooks actually fire in agent sessions: - `PreToolUse`: Runs before the agent uses a tool - `PostToolUse`: Runs after the agent completes a tool use - `PermissionRequest`: Runs when a tool requires user permission - `PostToolUseFailure`: Runs after a tool call fails - `Stop`: Runs when the agent finishes - `SubagentStop`: Runs when a subagent completes > **Note:** The [v2.1.0 changelog](https://github.com/anthropics/claude-code/blob/main/CHANGELOG.md#210) only mentions 3 hooks: *"Added hooks support to agent frontmatter, allowing agents to define PreToolUse, PostToolUse, and Stop hooks scoped to the agent's lifecycle"*. However, testing with the `claude-code-hook-agent` confirms that 6 hooks actually fire in agent sessions. The remaining 21 hooks (e.g., Notification, SessionStart, SessionEnd, etc.) do not fire in agent contexts. > > **Update (Feb 2026):** The [official hooks reference](https://code.claude.com/docs/en/hooks) now states *"All hook events are supported"* for skill/agent frontmatter hooks. This may mean support has expanded beyond the 6 hooks originally tested. Re-testing recommended to verify if additional hooks now fire in agent sessions. ### Agent Sound Folders Agent-specific sounds are stored in separate folders: - `.claude/hooks/sounds/agent_pretooluse/` - `.claude/hooks/sounds/agent_posttooluse/` - `.claude/hooks/sounds/agent_permissionrequest/` - `.claude/hooks/sounds/agent_posttoolusefailure/` - `.claude/hooks/sounds/agent_stop/` - `.claude/hooks/sounds/agent_subagentstop/` ### Creating an Agent with Hooks 1. Create an agent definition file in `.claude/agents/`: ```markdown --- name: my-agent description: Description of what this agent does hooks: PreToolUse: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: PreToolUse PostToolUse: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: PostToolUse PermissionRequest: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: PermissionRequest PostToolUseFailure: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: PostToolUseFailure Stop: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: Stop SubagentStop: - type: command command: python3 ${CLAUDE_PROJECT_DIR}/.claude/hooks/scripts/hooks.py --agent=my-agent timeout: 5000 async: true statusMessage: SubagentStop --- Your agent instructions here... ``` 2. Add sound files to the agent sound folders: - `agent_pretooluse/agent_pretooluse.wav` - `agent_posttooluse/agent_posttooluse.wav` - `agent_permissionrequest/agent_permissionrequest.wav` - `agent_posttoolusefailure/agent_posttoolusefailure.wav` - `agent_stop/agent_stop.wav` - `agent_subagentstop/agent_subagentstop.wav` ### Example: Weather Fetcher Agent See `.claude/agents/claude-code-hook-agent.md` for a complete example of an agent with hooks configured. ### Hook Option: `once: true` The `once: true` option ensures a hook only runs once per session: ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py", "timeout": 5000, "once": true } ``` This is useful for hooks like `SessionStart`, `SessionEnd`, and `PreCompact` that should only trigger once. > **Note:** The `once` option is for **skills only, not agents**. It works in settings-based hooks and skill frontmatter, but is not supported in agent frontmatter hooks. ### Hook Option: `async: true` Hooks can run in the background without blocking Claude Code's execution by adding `"async": true`: ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py", "timeout": 5000, "async": true } ``` **When to use async hooks:** - Logging and analytics - Notifications and sound effects - Any side-effect that shouldn't slow down Claude Code This project uses `async: true` for all hooks since sound notifications are side-effects that don't need to block execution. The `timeout` specifies how long the async hook can run before being terminated. ### Hook Option: `asyncRewake` (since v2.1.72, undocumented) The `asyncRewake` option combines async execution with the ability to wake the model on failure: ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py", "asyncRewake": true } ``` When `asyncRewake` is `true`, the hook runs in the background (implies `async`) but if it exits with code 2 (blocking error), it wakes the model to handle the error. This is useful for hooks that are normally non-blocking but need to surface critical failures. Discovered in the settings schema `propertyNames` — not yet in official documentation. ### Hook Option: `statusMessage` The `statusMessage` field sets a custom spinner message displayed to the user while the hook is running: ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py", "timeout": 5000, "async": true, "statusMessage": "PreToolUse" } ``` This project sets `statusMessage` to the hook event name on all hooks, so the spinner briefly shows which hook is firing (e.g., "PreToolUse", "SessionStart", "Stop"). This is most visible for synchronous hooks; for async hooks the message flashes briefly before the hook runs in the background. ### Hook Option: `if` (since v2.1.85) The `if` field adds conditional execution to hooks using permission rule syntax. When set, the hook process is only spawned if the condition matches — reducing unnecessary process spawning: ```json { "hooks": { "PreToolUse": [{ "matcher": "Bash", "hooks": [{ "type": "command", "command": "./validate-git.sh", "if": "Bash(git *)" }] }] } } ``` **Key details:** - Uses permission rule syntax: `Bash(git *)`, `Edit(*.ts)`, `mcp__.*` - Only applies to tool event hooks: `PreToolUse`, `PostToolUse`, `PostToolUseFailure`, `PermissionRequest` - Set at the handler level (per-handler granularity), not the matcher level - Without `if`, the hook process spawns on every matcher match — with `if`, it only spawns when the condition also matches - This project does not use `if` since all hooks fire for sound playback regardless of tool arguments ## Hook Types Claude Code supports four hook handler types. This project uses `command` hooks for all sound playback. ### `type: "command"` (used by this project) Runs a shell command. Receives JSON input via stdin, communicates results through exit codes and stdout. ```json { "type": "command", "command": "python3 .claude/hooks/scripts/hooks.py", "timeout": 5000, "async": true } ``` ### `type: "prompt"` Sends a prompt to a Claude model for single-turn evaluation. The model returns a yes/no decision as JSON (`{"ok": true/false, "reason": "..."}`). Useful for decisions that require judgment rather than deterministic rules. ```json { "type": "prompt", "prompt": "Check if all tasks are complete. $ARGUMENTS", "timeout": 30 } ``` **Supported events:** PreToolUse, PostToolUse, PostToolUseFailure, PermissionRequest, UserPromptSubmit, Stop, SubagentStop, TaskCreated, TaskCompleted. **Command-only events (not supported for prompt/agent types):** ConfigChange, CwdChanged, Elicitation, ElicitationResult, FileChanged, InstructionsLoaded, Notification, PermissionDenied, PostCompact, PreCompact, SessionEnd, SessionStart, Setup, StopFailure, SubagentStart, TeammateIdle, WorktreeCreate, WorktreeRemove. ### `type: "agent"` Spawns a subagent with multi-turn tool access (Read, Grep, Glob) to verify conditions before returning a decision. Same response format as prompt hooks. Useful when verification requires inspecting actual files or test output. ```json { "type": "agent", "prompt": "Verify that all unit tests pass. $ARGUMENTS", "timeout": 120 } ``` ### `type: "http"` (since v2.1.63) POSTs JSON to a URL and receives a JSON response, instead of running a shell command. Useful for integrating with external services or webhooks. HTTP hooks are routed through the sandbox network proxy when sandboxing is enabled. ```json { "type": "http", "url": "http://localhost:8080/hooks/pre-tool-use", "timeout": 30, "headers": { "Authorization": "Bearer $MY_TOKEN" }, "allowedEnvVars": ["MY_TOKEN"] } ``` **Not supported for:** ConfigChange, CwdChanged, Elicitation, ElicitationResult, FileChanged, InstructionsLoaded, Notification, PermissionDenied, PostCompact, PreCompact, SessionEnd, SessionStart, Setup, StopFailure, SubagentStart, TeammateIdle, WorktreeCreate, WorktreeRemove (command-only events). Headers support environment variable interpolation with `$VAR_NAME`, but only for variables explicitly listed in `allowedEnvVars`. ## Environment Variables Claude Code provides these environment variables to hook scripts: | Variable | Availability | Description | |----------|-------------|-------------| | `$CLAUDE_PROJECT_DIR` | All hooks | Project root directory. Wrap in quotes for paths with spaces | | `$CLAUDE_ENV_FILE` | SessionStart, CwdChanged, FileChanged | File path for persisting environment variables for subsequent Bash commands. Use append (`>>`) to preserve variables from other hooks | | `${CLAUDE_PLUGIN_ROOT}` | Plugin hooks | Plugin's root directory, for scripts bundled with a plugin | | `$CLAUDE_CODE_REMOTE` | All hooks | Set to `"true"` in remote web environments, not set in local CLI | | `${CLAUDE_SKILL_DIR}` | Skill hooks | Skill's own directory, for scripts bundled with a skill (since v2.1.69) | | `${CLAUDE_PLUGIN_DATA}` | Plugin hooks | Plugin's persistent data directory that survives plugin updates (since v2.1.78) | | `CLAUDE_CODE_SESSIONEND_HOOKS_TIMEOUT_MS` | SessionEnd hooks | Override SessionEnd hook timeout in milliseconds. Prior to v2.1.74, SessionEnd hooks were killed after 1.5s regardless of configured `timeout`. Now respects the hook's `timeout` value, or this env var if set (since v2.1.74) | | `session_id` (via stdin JSON) | All hooks | Current session ID, received as part of the JSON input on stdin (not an environment variable) | ### Common Input Fields (stdin JSON) Every hook receives a JSON object on stdin containing these common fields, in addition to any hook-specific fields listed in the Options column above: | Field | Type | Description | |-------|------|-------------| | `hook_event_name` | string | Name of the hook event that fired (e.g., `"PreToolUse"`, `"Stop"`) | | `session_id` | string | Current session identifier | | `transcript_path` | string | Path to the conversation transcript JSON file | | `cwd` | string | Current working directory | | `permission_mode` | string | Current permission mode: `default`, `plan`, `acceptEdits`, `dontAsk`, or `bypassPermissions` | | `agent_id` | string | Unique subagent identifier. Present when the hook fires inside a subagent context (since v2.1.69) | | `agent_type` | string | Agent type name (e.g. `Bash`, `Explore`, `Plan`, or custom). Present when using `--agent ` flag or inside a subagent (since v2.1.69) | > **Note:** Hook-specific fields (e.g., `tool_name` for PreToolUse, `last_assistant_message` for Stop) are listed in the Options column of the [Hook Events Overview](#hook-events-overview---official-27-hooks) table above. ## Hooks Management Commands Claude Code provides built-in commands for managing hooks: - **`/hooks`** — Interactive hook management UI. View, add, and delete hooks without editing JSON files. Hooks are labeled by source: `[User]`, `[Project]`, `[Local]`, `[Plugin]`. You can also toggle `disableAllHooks` from this menu. - **`claude hooks reload`** — Reload hooks configuration without restarting the session. Useful after editing settings files (since v2.0.47). ## MCP Tool Matchers For `PreToolUse`, `PostToolUse`, and `PermissionRequest` hooks, you can match MCP (Model Context Protocol) tools using the pattern `mcp____`: ```json { "hooks": { "PreToolUse": [{ "matcher": "mcp__memory__.*", "hooks": [{ "type": "command", "command": "echo 'MCP memory tool used'" }] }] } } ``` Full regex is supported: `mcp__memory__.*` (all tools from memory server), `mcp__.*__write.*` (any write tool from any server). ### Per-Hook Matcher Reference Matchers filter which events trigger a hook. Not all hooks support matchers — hooks without matcher support always fire. | Hook | Matcher Field | Possible Values | Example | |------|--------------|-----------------|---------| | `PreToolUse` | `tool_name` | Any tool name: `Bash`, `Read`, `Edit`, `Write`, `Glob`, `Grep`, `Agent`, `WebFetch`, `WebSearch`, `AskUserQuestion`, `ExitPlanMode`, `mcp__*` | `"matcher": "Bash"` | | `PermissionRequest` | `tool_name` | Same as PreToolUse | `"matcher": "mcp__memory__.*"` | | `PostToolUse` | `tool_name` | Same as PreToolUse | `"matcher": "Write"` | | `PostToolUseFailure` | `tool_name` | Same as PreToolUse | `"matcher": "Bash"` | | `Notification` | `notification_type` | `permission_prompt`, `idle_prompt`, `auth_success`, `elicitation_dialog` | `"matcher": "permission_prompt"` | | `SubagentStart` | `agent_type` | `Bash`, `Explore`, `Plan`, or custom agent name | `"matcher": "Bash"` | | `SubagentStop` | `agent_type` | `Bash`, `Explore`, `Plan`, or custom agent name | `"matcher": "Bash"` | | `SessionStart` | `source` | `startup`, `resume`, `clear`, `compact` | `"matcher": "startup"` | | `SessionEnd` | `reason` | `clear`, `resume`, `logout`, `prompt_input_exit`, `bypass_permissions_disabled`, `other` | `"matcher": "logout"` | | `PreCompact` | `compact_trigger` | `manual`, `auto` | `"matcher": "auto"` | | `PostCompact` | `compact_trigger` | `manual`, `auto` | `"matcher": "manual"` | | `Elicitation` | `mcp_server_name` | MCP server name | `"matcher": "my-mcp-server"` | | `ElicitationResult` | `mcp_server_name` | MCP server name | `"matcher": "my-mcp-server"` | | `ConfigChange` | `source` | `user_settings`, `project_settings`, `local_settings`, `policy_settings`, `skills` | `"matcher": "project_settings"` | | `UserPromptSubmit` | — | No matcher support | Always fires | | `Stop` | — | No matcher support | Always fires | | `TeammateIdle` | — | No matcher support | Always fires | | `TaskCreated` | — | No matcher support | Always fires | | `TaskCompleted` | — | No matcher support | Always fires | | `WorktreeCreate` | — | No matcher support | Always fires | | `WorktreeRemove` | — | No matcher support | Always fires | | `InstructionsLoaded` | `load_reason` | `session_start`, `nested_traversal`, `path_glob_match`, `include`, `compact` | `"matcher": "session_start"` | | `StopFailure` | `error` | `rate_limit`, `authentication_failed`, `billing_error`, `invalid_request`, `server_error`, `max_output_tokens`, `unknown` | `"matcher": "rate_limit"` | | `CwdChanged` | — | No matcher support | Always fires | | `FileChanged` | `filename` (basename) | Pipe-separated basenames: `.envrc`, `.env`, `.env.local` | `"matcher": ".envrc\|.env"` | | `Setup` | — | No matcher support | Always fires | | `PermissionDenied` | `tool_name` | Tool names (same as PreToolUse) | `"matcher": "Bash"` | ## Known Issues & Workarounds ### Agent Stop Hook Bug (SubagentStop vs Stop) **Bug Report:** [GitHub Issue #19220](https://github.com/anthropics/claude-code/issues/19220) **Issue:** When defining a `Stop` hook in an agent's frontmatter, the `hook_event_name` passed to the hook script is `"SubagentStop"` instead of `"Stop"`. This contradicts the official documentation and breaks consistency with other agent hooks (`PreToolUse` and `PostToolUse`), which correctly pass their configured names. | Hook | Defined As | Received As | Status | |------|------------|-------------|--------| | PreToolUse | `PreToolUse:` | `"PreToolUse"` | ✅ Correct | | PostToolUse | `PostToolUse:` | `"PostToolUse"` | ✅ Correct | | Stop | `Stop:` | `"SubagentStop"` | ❌ Inconsistent | **Status:** The [official hooks reference](https://code.claude.com/docs/en/hooks#hooks-in-skills-and-agents) now documents this as expected behavior: *"For subagents, Stop hooks are automatically converted to SubagentStop since that is the event that fires when a subagent completes."* This project handles it via the `AGENT_HOOK_SOUND_MAP` in `hooks.py`, which has a separate `SubagentStop` entry that maps to the `agent_subagentstop` sound folder. ### PreToolUse `updatedInput` for AskUserQuestion (since v2.1.85) When a `PreToolUse` hook matches `AskUserQuestion`, it can return `updatedInput` to auto-respond to the question — enabling headless integrations to programmatically answer user questions without manual input: ```json { "hookSpecificOutput": { "updatedInput": { "question": "Do you want to proceed?", "answer": "yes" } } } ``` This is useful for CI/CD pipelines, automated testing, or any context where Claude Code runs without a human at the terminal. Not yet in official docs pages — sourced from GitHub changelog v2.1.85. ### PreToolUse Decision Control Deprecation The `PreToolUse` hook previously used top-level `decision` and `reason` fields for blocking tool calls. These are now **deprecated**. Use `hookSpecificOutput.permissionDecision` and `hookSpecificOutput.permissionDecisionReason` instead: | Deprecated | Current | |-----------|---------| | `"decision": "approve"` | `"hookSpecificOutput": { "permissionDecision": "allow" }` | | `"decision": "block"` | `"hookSpecificOutput": { "permissionDecision": "deny" }` | This does not affect this project since `hooks.py` uses async sound playback and does not use decision control. ## Decision Control Patterns Different hooks use different output schemas for blocking or controlling execution. This project does not use decision control (all hooks are async sound playback), but for reference: | Hook(s) | Control Method | Values | |---------|---------------|--------| | PreToolUse | `hookSpecificOutput.permissionDecision` | `allow`, `deny`, `ask`, `defer` (headless `-p` mode only, v2.1.89+) | | PreToolUse | `hookSpecificOutput.autoAllow` | `true` — auto-approve future uses of this tool (since v2.0.76) | | PermissionRequest | `hookSpecificOutput.decision.behavior` | `allow`, `deny` | | Stop, SubagentStop, ConfigChange | Top-level `decision` | `block` | | TeammateIdle, TaskCreated, TaskCompleted | `continue` + exit code 2 | `{"continue": false, "stopReason": "..."}` — JSON decision control added in v2.1.70. TaskCreated also uses exit code 2 to block task creation (stderr fed back to model) | | UserPromptSubmit | Can modify `prompt` field | Returns modified prompt via stdout | | WorktreeCreate | Non-zero exit + stdout path | Non-zero exit fails creation; stdout provides worktree path | | Elicitation | `hookSpecificOutput.action` + `hookSpecificOutput.content` | `accept`, `decline`, `cancel` — control MCP elicitation response | | ElicitationResult | `hookSpecificOutput.action` + `hookSpecificOutput.content` | `accept`, `decline`, `cancel` — override user response before sending to server | | PermissionDenied | `hookSpecificOutput.retry` | `true` — signal that model may retry the denied tool call (v2.1.89+) | ### Universal JSON Output Fields All hooks can return these fields via stdout JSON: | Field | Type | Description | |-------|------|-------------| | `continue` | bool | If `false`, stops Claude entirely | | `stopReason` | string | Message shown when `continue` is false | | `suppressOutput` | bool | Hides stdout from verbose mode | | `systemMessage` | string | Warning message shown to user | | `additionalContext` | string | Context added to Claude's conversation | ## Hook Deduplication & External Changes - **Hook deduplication:** Identical hook handlers defined in multiple settings locations run only once in parallel, preventing duplicate execution. - **External change detection:** Claude Code warns when hooks are modified externally (e.g., by another process editing settings files) during an active session.