Skip to Content
User GuideFeaturesHooks

Qwen Code Hooks Documentation

Overview

Qwen Code hooks provide a powerful mechanism for extending and customizing the behavior of the Qwen Code application. Hooks allow users to execute custom scripts or programs at specific points in the application lifecycle, such as before tool execution, after tool execution, at session start/end, and during other key events.

Hooks are enabled by default. You can temporarily disable all hooks by setting disableAllHooks to true in your settings file (at the top level, alongside hooks):

{ "disableAllHooks": true, "hooks": { "PreToolUse": [...] } }

This disables all hooks without deleting their configurations.

What are Hooks?

Hooks are user-defined scripts or programs that are automatically executed by Qwen Code at predefined points in the application flow. They allow users to:

  • Monitor and audit tool usage
  • Enforce security policies
  • Inject additional context into conversations
  • Customize application behavior based on events
  • Integrate with external systems and services
  • Modify tool inputs or responses programmatically

Hook Architecture

The Qwen Code hook system consists of several key components:

  1. Hook Registry: Stores and manages all configured hooks
  2. Hook Planner: Determines which hooks should run for each event
  3. Hook Runner: Executes individual hooks with proper context
  4. Hook Aggregator: Combines results from multiple hooks
  5. Hook Event Handler: Coordinates the firing of hooks for events

Hook Events

Hooks fire at specific points during a Qwen Code session. When an event fires and a matcher matches, Qwen Code passes JSON context about the event to your hook handler. For command hooks, input arrives on stdin. Your handler can inspect the input, take action, and optionally return a decision. Some events fire once per session, while others fire repeatedly inside the agentic loop.

Hook Lifecycle Diagram

The following table lists all available hook events in Qwen Code:

Event NameDescriptionUse Case
PreToolUseFired before tool executionPermission checking, input validation, logging
PostToolUseFired after successful tool executionLogging, output processing, monitoring
PostToolUseFailureFired when tool execution failsError handling, alerting, remediation
NotificationFired when notifications are sentNotification customization, logging
UserPromptSubmitFired when user submits a promptInput processing, validation, context injection
SessionStartFired when a new session startsInitialization, context setup
StopFired before Qwen concludes its responseFinalization, cleanup
SubagentStartFired when a subagent startsSubagent initialization
SubagentStopFired when a subagent stopsSubagent finalization
PreCompactFired before conversation compactionPre-compaction processing
SessionEndFired when a session endsCleanup, reporting
PermissionRequestFired when permission dialogs are displayedPermission automation, policy enforcement

Input/Output Rules

Hook Input Structure

All hooks receive standardized input in JSON format through stdin. Common fields included in every hook event:

{ "session_id": "string", "transcript_path": "string", "cwd": "string", "hook_event_name": "string", "timestamp": "string" }

Event-specific fields are added based on the hook type. Below are the event-specific fields for each hook event:

Individual Hook Event Details

PreToolUse

Purpose: Executed before a tool is used to allow for permission checks, input validation, or context injection.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "tool_name": "name of the tool being executed", "tool_input": "object containing the tool's input parameters", "tool_use_id": "unique identifier for this tool use instance" }

Output Options:

  • hookSpecificOutput.permissionDecision: “allow”, “deny”, or “ask” (REQUIRED)
  • hookSpecificOutput.permissionDecisionReason: explanation for the decision (REQUIRED)
  • hookSpecificOutput.updatedInput: modified tool input parameters to use instead of original
  • hookSpecificOutput.additionalContext: additional context information

Note: While standard hook output fields like decision and reason are technically supported by the underlying class, the official interface expects the hookSpecificOutput with permissionDecision and permissionDecisionReason.

Example Output:

{ "hookSpecificOutput": { "hookEventName": "PreToolUse", "permissionDecision": "allow", "permissionDecisionReason": "My reason here", "updatedInput": { "field_to_modify": "new value" }, "additionalContext": "Current environment: production. Proceed with caution." } }

PostToolUse

Purpose: Executed after a tool completes successfully to process results, log outcomes, or inject additional context.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "tool_name": "name of the tool that was executed", "tool_input": "object containing the tool's input parameters", "tool_response": "object containing the tool's response", "tool_use_id": "unique identifier for this tool use instance" }

Output Options:

  • decision: “allow”, “deny”, “block” (defaults to “allow” if not specified)
  • reason: reason for the decision
  • hookSpecificOutput.additionalContext: additional information to be included

Example Output:

{ "decision": "allow", "reason": "Tool executed successfully", "hookSpecificOutput": { "additionalContext": "File modification recorded in audit log" } }

PostToolUseFailure

Purpose: Executed when a tool execution fails to handle errors, send alerts, or record failures.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "tool_use_id": "unique identifier for the tool use", "tool_name": "name of the tool that failed", "tool_input": "object containing the tool's input parameters", "error": "error message describing the failure", "is_interrupt": "boolean indicating if failure was due to user interruption (optional)" }

Output Options:

  • hookSpecificOutput.additionalContext: error handling information
  • Standard hook output fields

Example Output:

{ "hookSpecificOutput": { "additionalContext": "Error: File not found. Failure logged in monitoring system." } }

UserPromptSubmit

Purpose: Executed when the user submits a prompt to modify, validate, or enrich the input.

Event-specific fields:

{ "prompt": "the user's submitted prompt text" }

Output Options:

  • decision: “allow”, “deny”, “block”, or “ask”
  • reason: human-readable explanation for the decision
  • hookSpecificOutput.additionalContext: additional context to append to the prompt (optional)

Note: Since UserPromptSubmitOutput extends HookOutput, all standard fields are available but only additionalContext in hookSpecificOutput is specifically defined for this event.

Example Output:

{ "decision": "allow", "reason": "Prompt reviewed and approved", "hookSpecificOutput": { "additionalContext": "Remember to follow company coding standards." } }

SessionStart

Purpose: Executed when a new session starts to perform initialization tasks.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "source": "startup | resume | clear | compact", "model": "the model being used", "agent_type": "the type of agent if applicable (optional)" }

Output Options:

  • hookSpecificOutput.additionalContext: context to be available in the session
  • Standard hook output fields

Example Output:

{ "hookSpecificOutput": { "additionalContext": "Session started with security policies enabled." } }

SessionEnd

Purpose: Executed when a session ends to perform cleanup tasks.

Event-specific fields:

{ "reason": "clear | logout | prompt_input_exit | bypass_permissions_disabled | other" }

Output Options:

  • Standard hook output fields (typically not used for blocking)

Stop

Purpose: Executed before Qwen concludes its response to provide final feedback or summaries.

Event-specific fields:

{ "stop_hook_active": "boolean indicating if stop hook is active", "last_assistant_message": "the last message from the assistant" }

Output Options:

  • decision: “allow”, “deny”, “block”, or “ask”
  • reason: human-readable explanation for the decision
  • stopReason: feedback to include in the stop response
  • continue: set to false to stop execution
  • hookSpecificOutput.additionalContext: additional context information

Note: Since StopOutput extends HookOutput, all standard fields are available but the stopReason field is particularly relevant for this event.

Example Output:

{ "decision": "block", "reason": "Must be provided when Qwen Code is blocked from stopping" }

SubagentStart

Purpose: Executed when a subagent (like the Task tool) is started to set up context or permissions.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "agent_id": "identifier for the subagent", "agent_type": "type of agent (Bash, Explorer, Plan, Custom, etc.)" }

Output Options:

  • hookSpecificOutput.additionalContext: initial context for the subagent
  • Standard hook output fields

Example Output:

{ "hookSpecificOutput": { "additionalContext": "Subagent initialized with restricted permissions." } }

SubagentStop

Purpose: Executed when a subagent finishes to perform finalization tasks.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "stop_hook_active": "boolean indicating if stop hook is active", "agent_id": "identifier for the subagent", "agent_type": "type of agent", "agent_transcript_path": "path to the subagent's transcript", "last_assistant_message": "the last message from the subagent" }

Output Options:

  • decision: “allow”, “deny”, “block”, or “ask”
  • reason: human-readable explanation for the decision

Example Output:

{ "decision": "block", "reason": "Must be provided when Qwen Code is blocked from stopping" }

PreCompact

Purpose: Executed before conversation compaction to prepare or log the compaction.

Event-specific fields:

{ "trigger": "manual | auto", "custom_instructions": "custom instructions currently set" }

Output Options:

  • hookSpecificOutput.additionalContext: context to include before compaction
  • Standard hook output fields

Example Output:

{ "hookSpecificOutput": { "additionalContext": "Compacting conversation to maintain optimal context window." } }

Notification

Purpose: Executed when notifications are sent to customize or intercept them.

Event-specific fields:

{ "message": "notification message content", "title": "notification title (optional)", "notification_type": "permission_prompt | idle_prompt | auth_success" }

Note: elicitation_dialog type is defined but not currently implemented.

Output Options:

  • hookSpecificOutput.additionalContext: additional information to include
  • Standard hook output fields

Example Output:

{ "hookSpecificOutput": { "additionalContext": "Notification processed by monitoring system." } }

PermissionRequest

Purpose: Executed when permission dialogs are displayed to automate decisions or update permissions.

Event-specific fields:

{ "permission_mode": "default | plan | auto_edit | yolo", "tool_name": "name of the tool requesting permission", "tool_input": "object containing the tool's input parameters", "permission_suggestions": "array of suggested permissions (optional)" }

Output Options:

  • hookSpecificOutput.decision: structured object with permission decision details:
    • behavior: “allow” or “deny”
    • updatedInput: modified tool input (optional)
    • updatedPermissions: modified permissions (optional)
    • message: message to show to user (optional)
    • interrupt: whether to interrupt the workflow (optional)

Example Output:

{ "hookSpecificOutput": { "decision": { "behavior": "allow", "message": "Permission granted based on security policy", "interrupt": false } } }

Hook Configuration

Hooks are configured in Qwen Code settings, typically in .qwen/settings.json or user configuration files:

{ "hooks": { "PreToolUse": [ { "matcher": "^bash$", // Regex to match tool names "sequential": false, // Whether to run hooks sequentially "hooks": [ { "type": "command", "command": "/path/to/script.sh", "name": "security-check", "description": "Run security checks before tool execution", "timeout": 30000 } ] } ], "SessionStart": [ { "hooks": [ { "type": "command", "command": "echo 'Session started'", "name": "session-init" } ] } ] } }

Matcher Patterns

Matchers allow filtering hooks based on context. Not all hook events support matchers:

Event TypeEventsMatcher SupportMatcher Target (Values)
Tool EventsPreToolUse, PostToolUse, PostToolUseFailure, PermissionRequest✅ Yes (regex)Tool name: bash, read_file, write_file, edit, glob, grep_search, etc.
Subagent EventsSubagentStart, SubagentStop✅ Yes (regex)Agent type: Bash, Explorer, etc.
Session EventsSessionStart✅ Yes (regex)Source: startup, resume, clear, compact
Session EventsSessionEnd✅ Yes (regex)Reason: clear, logout, prompt_input_exit, bypass_permissions_disabled, other
Notification EventsNotification✅ Yes (exact)Type: permission_prompt, idle_prompt, auth_success
Compact EventsPreCompact✅ Yes (exact)Trigger: manual, auto
Prompt EventsUserPromptSubmit❌ NoN/A
Stop EventsStop❌ NoN/A

Matcher Syntax:

  • Regex pattern matched against the target field
  • Empty string "" or "*" matches all events of that type
  • Standard regex syntax supported (e.g., ^bash$, read.*, (bash|run_shell_command))

Examples:

{ "hooks": { "PreToolUse": [ { "matcher": "^bash$", // Only match bash tool "hooks": [...] }, { "matcher": "read.*", // Match read_file, read_multiple_files, etc. "hooks": [...] }, { "matcher": "", // Match all tools (same as "*" or omitting matcher) "hooks": [...] } ], "SubagentStart": [ { "matcher": "^(Bash|Explorer)$", // Only match Bash and Explorer agents "hooks": [...] } ], "SessionStart": [ { "matcher": "^(startup|resume)$", // Only match startup and resume sources "hooks": [...] } ] } }

Hook Execution

Parallel vs Sequential Execution

  • By default, hooks execute in parallel for better performance
  • Use sequential: true in hook definition to enforce order-dependent execution
  • Sequential hooks can modify input for subsequent hooks in the chain

Security Model

  • Hooks run in the user’s environment with user privileges
  • Project-level hooks require trusted folder status
  • Timeouts prevent hanging hooks (default: 60 seconds)

Exit Codes

Hook scripts communicate their result through exit codes:

Exit CodeMeaningBehavior
0Successstdout/stderr not shown
2Blocking errorShow stderr to model and block tool call
OtherNon-blocking errorShow stderr to user only but continue tool call

Examples:

#!/bin/bash # Success (exit 0 is default, can be omitted) echo '{"decision": "allow"}' exit 0 # Blocking error - prevents operation echo "Dangerous operation blocked by security policy" >&2 exit 2

Note: If no exit code is specified, the script defaults to 0 (success).

Best Practices

Example 1: Security Validation Hook

A PreToolUse hook that logs and potentially blocks dangerous commands:

security_check.sh

#!/bin/bash # Read input from stdin INPUT=$(cat) # Parse the input to extract tool info TOOL_NAME=$(echo "$INPUT" | jq -r '.tool_name') TOOL_INPUT=$(echo "$INPUT" | jq -r '.tool_input') # Check for potentially dangerous operations if echo "$TOOL_INPUT" | grep -qiE "(rm.*-rf|mv.*\/|chmod.*777)"; then echo '{ "decision": "deny", "reason": "Potentially dangerous operation detected", "hookSpecificOutput": { "hookEventName": "PreToolUse", "permissionDecision": "deny", "permissionDecisionReason": "Dangerous command blocked by security policy" } }' exit 2 # Blocking error fi # Allow the operation with a log echo "INFO: Tool $TOOL_NAME executed safely at $(date)" >> /var/log/qwen-security.log # Allow with additional context echo '{ "decision": "allow", "reason": "Operation approved by security checker", "hookSpecificOutput": { "hookEventName": "PreToolUse", "permissionDecision": "allow", "permissionDecisionReason": "Security check passed", "additionalContext": "Command approved by security policy" } }' exit 0

Configure in .qwen/settings.json:

{ "hooks": { "PreToolUse": [ { "hooks": [ { "type": "command", "command": "${SECURITY_CHECK_SCRIPT}", "name": "security-checker", "description": "Security validation for bash commands", "timeout": 10000 } ] } ] } }

Example 2: User Prompt Validation Hook

A UserPromptSubmit hook that validates user prompts for sensitive information and provides context for long prompts:

prompt_validator.py

import json import sys import re # Load input from stdin try: input_data = json.load(sys.stdin) except json.JSONDecodeError as e: print(f"Error: Invalid JSON input: {e}", file=sys.stderr) exit(1) user_prompt = input_data.get("prompt", "") # Sensitive words list sensitive_words = ["password", "secret", "token", "api_key"] # Check for sensitive information for word in sensitive_words: if re.search(rf"\b{word}\b", user_prompt.lower()): # Block prompts containing sensitive information output = { "decision": "block", "reason": f"Prompt contains sensitive information '{word}'. Please remove sensitive content and resubmit.", "hookSpecificOutput": { "hookEventName": "UserPromptSubmit" } } print(json.dumps(output)) exit(0) # Check prompt length and add warning context if too long if len(user_prompt) > 1000: output = { "hookSpecificOutput": { "hookEventName": "UserPromptSubmit", "additionalContext": "Note: User submitted a long prompt. Please read carefully and ensure all requirements are understood." } } print(json.dumps(output)) exit(0) # No processing needed for normal cases exit(0)

Troubleshooting

  • Check application logs for hook execution details
  • Verify hook script permissions and executability
  • Ensure proper JSON formatting in hook outputs
  • Use specific matcher patterns to avoid unintended hook execution
Last updated on