Configurare il vostro primo hook
Per creare un hook, aggiungete un bloccohooks a un file di impostazioni. Questa procedura crea un hook di notifica desktop, in modo da ricevere un avviso ogni volta che Claude sta aspettando il vostro input invece di guardare il terminale.
Aggiungere l'hook alle vostre impostazioni
Aprite Se il vostro file di impostazioni ha già una chiave
~/.claude/settings.json e aggiungete un hook Notification. L’esempio sottostante utilizza osascript per macOS; consultate Ricevere una notifica quando Claude ha bisogno di input per i comandi Linux e Windows.hooks, unite la voce Notification ad essa piuttosto che sostituire l’intero oggetto. Potete anche chiedere a Claude di scrivere l’hook per voi descrivendo quello che volete nella CLI.Verificare la configurazione
Digitate
/hooks per aprire il browser degli hooks. Vedrete un elenco di tutti gli eventi hook disponibili, con un conteggio accanto a ogni evento che ha hook configurati. Selezionate Notification per confermare che il vostro nuovo hook appare nell’elenco. Selezionando l’hook vengono mostrati i suoi dettagli: l’evento, il matcher, il tipo, il file di origine e il comando.Cosa potete automatizzare
Gli hooks vi permettono di eseguire codice in punti chiave del ciclo di vita di Claude Code: formattare file dopo le modifiche, bloccare comandi prima che si eseguano, inviare notifiche quando Claude ha bisogno di input, iniettare contesto all’inizio della sessione, e altro ancora. Per l’elenco completo degli eventi hook, consultate il riferimento Hooks. Ogni esempio include un blocco di configurazione pronto all’uso che aggiungete a un file di impostazioni. I modelli più comuni:- Ricevere una notifica quando Claude ha bisogno di input
- Formattare automaticamente il codice dopo le modifiche
- Bloccare le modifiche ai file protetti
- Reiniettare il contesto dopo la compattazione
- Controllare le modifiche di configurazione
- Ricaricare l’ambiente quando la directory o i file cambiano
- Approvare automaticamente specifici prompt di autorizzazione
Ricevere una notifica quando Claude ha bisogno di input
Ricevete una notifica desktop ogni volta che Claude finisce di lavorare e ha bisogno del vostro input, in modo da poter passare ad altri compiti senza controllare il terminale. Questo hook utilizza l’eventoNotification, che si attiva quando Claude sta aspettando input o autorizzazione. Ogni scheda sottostante utilizza il comando di notifica nativo della piattaforma. Aggiungete questo a ~/.claude/settings.json:
- macOS
- Linux
- Windows (PowerShell)
Formattare automaticamente il codice dopo le modifiche
Eseguite automaticamente Prettier su ogni file che Claude modifica, in modo che la formattazione rimanga coerente senza intervento manuale. Questo hook utilizza l’eventoPostToolUse con un matcher Edit|Write, quindi si esegue solo dopo gli strumenti di modifica dei file. Il comando estrae il percorso del file modificato con jq e lo passa a Prettier. Aggiungete questo a .claude/settings.json nella radice del vostro progetto:
Gli esempi Bash in questa pagina utilizzano
jq per l’analisi JSON. Installatelo con brew install jq (macOS), apt-get install jq (Debian/Ubuntu), o consultate i download di jq.Bloccare le modifiche ai file protetti
Impedite a Claude di modificare file sensibili come.env, package-lock.json, o qualsiasi cosa in .git/. Claude riceve un feedback che spiega perché la modifica è stata bloccata, in modo da poter adattare il suo approccio.
Questo esempio utilizza un file di script separato che l’hook chiama. Lo script controlla il percorso del file di destinazione rispetto a un elenco di modelli protetti ed esce con il codice 2 per bloccare la modifica.
Rendere lo script eseguibile (macOS/Linux)
Gli script degli hook devono essere eseguibili affinché Claude Code li esegua:
Reiniettare il contesto dopo la compattazione
Quando la finestra di contesto di Claude si riempie, la compattazione riassume la conversazione per liberare spazio. Questo può perdere dettagli importanti. Utilizzate un hookSessionStart con un matcher compact per reiniettare il contesto critico dopo ogni compattazione.
Qualsiasi testo che il vostro comando scrive su stdout viene aggiunto al contesto di Claude. Questo esempio ricorda a Claude le convenzioni del progetto e il lavoro recente. Aggiungete questo a .claude/settings.json nella radice del vostro progetto:
echo con qualsiasi comando che produce output dinamico, come git log --oneline -5 per mostrare i commit recenti. Per iniettare contesto all’inizio di ogni sessione, considerate di utilizzare CLAUDE.md invece. Per le variabili di ambiente, consultate CLAUDE_ENV_FILE nel riferimento.
Controllare le modifiche di configurazione
Tracciate quando i file di impostazioni o skills cambiano durante una sessione. L’eventoConfigChange si attiva quando un processo esterno o un editor modifica un file di configurazione, in modo da poter registrare le modifiche per la conformità o bloccare le modifiche non autorizzate.
Questo esempio aggiunge ogni modifica a un registro di controllo. Aggiungete questo a ~/.claude/settings.json:
user_settings, project_settings, local_settings, policy_settings, o skills. Per bloccare una modifica dall’avere effetto, uscite con il codice 2 o restituite {"decision": "block"}. Consultate il riferimento ConfigChange per lo schema di input completo.
Ricaricare l’ambiente quando la directory o i file cambiano
Alcuni progetti impostano variabili di ambiente diverse a seconda di quale directory siete. Strumenti come direnv lo fanno automaticamente nella vostra shell, ma lo strumento Bash di Claude non raccoglie quei cambiamenti da solo. Un hookCwdChanged risolve questo: si esegue ogni volta che Claude cambia directory, in modo da poter ricaricare le variabili corrette per la nuova posizione. L’hook scrive i valori aggiornati su CLAUDE_ENV_FILE, che Claude Code applica prima di ogni comando Bash. Aggiungete questo a ~/.claude/settings.json:
FileChanged con un matcher che elenca i nomi dei file da guardare (separati da pipe). Il matcher sia configura quali file guardare che filtra quali hook si eseguono. Questo esempio guarda .envrc e .env per i cambiamenti nella directory corrente:
watchPaths, e i dettagli di CLAUDE_ENV_FILE.
Approvare automaticamente specifici prompt di autorizzazione
Saltate la finestra di dialogo di approvazione per le chiamate di strumenti che consentite sempre. Questo esempio approva automaticamenteExitPlanMode, lo strumento che Claude chiama quando finisce di presentare un piano e chiede di procedere, in modo da non essere richiesto ogni volta che un piano è pronto.
A differenza degli esempi di codice di uscita sopra, l’approvazione automatica richiede che il vostro hook scriva una decisione JSON su stdout. Un hook PermissionRequest si attiva quando Claude Code sta per mostrare una finestra di dialogo di autorizzazione, e restituire "behavior": "allow" la risponde per vostro conto.
Il matcher limita l’hook a ExitPlanMode solo, in modo che nessun altro prompt sia interessato. Aggiungete questo a ~/.claude/settings.json:
updatedPermissions con una voce setMode. Il valore mode è qualsiasi modalità di autorizzazione come default, acceptEdits, o bypassPermissions, e destination: "session" la applica solo per la sessione corrente.
Per passare la sessione a acceptEdits, il vostro hook scrive questo JSON su stdout:
.* o lasciare il matcher vuoto approverebbe automaticamente ogni prompt di autorizzazione, incluse le scritture di file e i comandi shell. Consultate il riferimento PermissionRequest per l’insieme completo di campi di decisione.
Come funzionano gli hooks
Gli eventi hook si attivano in punti specifici del ciclo di vita di Claude Code. Quando un evento si attiva, tutti gli hooks corrispondenti si eseguono in parallelo, e i comandi hook identici vengono automaticamente deduplicati. La tabella sottostante mostra ogni evento e quando si attiva:| Event | When it fires |
|---|---|
SessionStart | When a session begins or resumes |
UserPromptSubmit | When you submit a prompt, before Claude processes it |
PreToolUse | Before a tool call executes. Can block it |
PermissionRequest | When a permission dialog appears |
PermissionDenied | When a tool call is denied by the auto mode classifier. Return {retry: true} to tell the model it may retry the denied tool call |
PostToolUse | After a tool call succeeds |
PostToolUseFailure | After a tool call fails |
Notification | When Claude Code sends a notification |
SubagentStart | When a subagent is spawned |
SubagentStop | When a subagent finishes |
TaskCreated | When a task is being created via TaskCreate |
TaskCompleted | When a task is being marked as completed |
Stop | When Claude finishes responding |
StopFailure | When the turn ends due to an API error. Output and exit code are ignored |
TeammateIdle | When an agent team teammate is about to go idle |
InstructionsLoaded | When a CLAUDE.md or .claude/rules/*.md file is loaded into context. Fires at session start and when files are lazily loaded during a session |
ConfigChange | When a configuration file changes during a session |
CwdChanged | When the working directory changes, for example when Claude executes a cd command. Useful for reactive environment management with tools like direnv |
FileChanged | When a watched file changes on disk. The matcher field specifies which filenames to watch |
WorktreeCreate | When a worktree is being created via --worktree or isolation: "worktree". Replaces default git behavior |
WorktreeRemove | When a worktree is being removed, either at session exit or when a subagent finishes |
PreCompact | Before context compaction |
PostCompact | After context compaction completes |
Elicitation | When an MCP server requests user input during a tool call |
ElicitationResult | After a user responds to an MCP elicitation, before the response is sent back to the server |
SessionEnd | When a session terminates |
type che determina come si esegue. La maggior parte degli hooks utilizza "type": "command", che esegue un comando shell. Sono disponibili altri tre tipi:
"type": "http": POST dei dati dell’evento a un URL. Consultate HTTP hooks."type": "prompt": valutazione LLM a turno singolo. Consultate Hooks basati su prompt."type": "agent": verifica multi-turno con accesso agli strumenti. Consultate Hooks basati su agenti.
Leggere l’input e restituire l’output
Gli hooks comunicano con Claude Code attraverso stdin, stdout, stderr e codici di uscita. Quando un evento si attiva, Claude Code passa i dati specifici dell’evento come JSON allo stdin del vostro script. Il vostro script legge quei dati, fa il suo lavoro, e dice a Claude Code cosa fare dopo tramite il codice di uscita.Input dell’hook
Ogni evento include campi comuni comesession_id e cwd, ma ogni tipo di evento aggiunge dati diversi. Ad esempio, quando Claude esegue un comando Bash, un hook PreToolUse riceve qualcosa di simile su stdin:
UserPromptSubmit ricevono il testo prompt invece, gli hook SessionStart ricevono la source (startup, resume, clear, compact), e così via. Consultate Campi di input comuni nel riferimento per i campi condivisi, e la sezione di ogni evento per gli schemi specifici dell’evento.
Output dell’hook
Il vostro script dice a Claude Code cosa fare dopo scrivendo su stdout o stderr e uscendo con un codice specifico. Ad esempio, un hookPreToolUse che vuole bloccare un comando:
- Exit 0: l’azione procede. Per gli hook
UserPromptSubmiteSessionStart, qualsiasi cosa scriviate su stdout viene aggiunta al contesto di Claude. - Exit 2: l’azione è bloccata. Scrivete un motivo su stderr, e Claude lo riceve come feedback in modo da poter adattarsi.
- Qualsiasi altro codice di uscita: l’azione procede. Stderr viene registrato ma non mostrato a Claude. Attivate la modalità verbose con
Ctrl+Oper vedere questi messaggi nella trascrizione.
Output JSON strutturato
I codici di uscita vi danno due opzioni: consentire o bloccare. Per un controllo maggiore, uscite con 0 e stampate un oggetto JSON su stdout invece.Utilizzate exit 2 per bloccare con un messaggio stderr, o exit 0 con JSON per un controllo strutturato. Non mescolateli: Claude Code ignora JSON quando uscite con 2.
PreToolUse può negare una chiamata di strumento e dire a Claude perché, o escalarlo all’utente per l’approvazione:
permissionDecision e annulla la chiamata dello strumento, quindi alimenta permissionDecisionReason di nuovo a Claude come feedback. Queste tre opzioni sono specifiche per PreToolUse:
"allow": procedi senza mostrare un prompt di autorizzazione interattivo. Le regole di negazione e richiesta, incluse le liste di negazione gestite dall’azienda, si applicano ancora"deny": annulla la chiamata dello strumento e invia il motivo a Claude"ask": mostra il prompt di autorizzazione all’utente come al solito
"allow" salta il prompt interattivo ma non sostituisce le regole di autorizzazione. Se una regola di negazione corrisponde alla chiamata dello strumento, la chiamata viene bloccata anche quando il vostro hook restituisce "allow". Se una regola di richiesta corrisponde, l’utente viene comunque richiesto. Questo significa che le regole di negazione da qualsiasi ambito di impostazioni, incluse le impostazioni gestite, hanno sempre la precedenza sulle approvazioni degli hook.
Altri eventi utilizzano modelli di decisione diversi. Ad esempio, gli hook PostToolUse e Stop utilizzano un campo decision: "block" di livello superiore, mentre PermissionRequest utilizza hookSpecificOutput.decision.behavior. Consultate la tabella di riepilogo nel riferimento per una suddivisione completa per evento.
Per gli hook UserPromptSubmit, utilizzate additionalContext invece per iniettare testo nel contesto di Claude. Gli hooks basati su prompt (type: "prompt") gestiscono l’output diversamente: consultate Hooks basati su prompt.
Filtrare gli hooks con i matcher
Senza un matcher, un hook si attiva su ogni occorrenza del suo evento. I matcher vi permettono di restringerlo. Ad esempio, se volete eseguire un formattatore solo dopo le modifiche ai file (non dopo ogni chiamata di strumento), aggiungete un matcher al vostro hookPostToolUse:
"Edit|Write" è un modello regex che corrisponde al nome dello strumento. L’hook si attiva solo quando Claude utilizza lo strumento Edit o Write, non quando utilizza Bash, Read, o qualsiasi altro strumento.
Ogni tipo di evento corrisponde a un campo specifico. I matcher supportano stringhe esatte e modelli regex:
| Evento | Cosa filtra il matcher | Valori matcher di esempio |
|---|---|---|
PreToolUse, PostToolUse, PostToolUseFailure, PermissionRequest | nome dello strumento | Bash, Edit|Write, mcp__.* |
SessionStart | come è iniziata la sessione | startup, resume, clear, compact |
SessionEnd | perché è terminata la sessione | clear, resume, logout, prompt_input_exit, bypass_permissions_disabled, other |
Notification | tipo di notifica | permission_prompt, idle_prompt, auth_success, elicitation_dialog |
SubagentStart | tipo di agente | Bash, Explore, Plan, o nomi di agenti personalizzati |
PreCompact, PostCompact | cosa ha attivato la compattazione | manual, auto |
SubagentStop | tipo di agente | stessi valori di SubagentStart |
ConfigChange | fonte di configurazione | user_settings, project_settings, local_settings, policy_settings, skills |
StopFailure | tipo di errore | rate_limit, authentication_failed, billing_error, invalid_request, server_error, max_output_tokens, unknown |
InstructionsLoaded | motivo del caricamento | session_start, nested_traversal, path_glob_match, include, compact |
Elicitation | nome del server MCP | i vostri nomi di server MCP configurati |
ElicitationResult | nome del server MCP | stessi valori di Elicitation |
FileChanged | nome del file (basename del file modificato) | .envrc, .env, qualsiasi nome di file che volete guardare |
UserPromptSubmit, Stop, TeammateIdle, TaskCompleted, WorktreeCreate, WorktreeRemove, CwdChanged | nessun supporto matcher | si attiva sempre su ogni occorrenza |
- Registrare ogni comando Bash
- Corrispondere agli strumenti MCP
- Pulire alla fine della sessione
Corrispondere solo alle chiamate dello strumento
Bash e registrare ogni comando in un file. L’evento PostToolUse si attiva dopo che il comando è completato, quindi tool_input.command contiene quello che è stato eseguito. L’hook riceve i dati dell’evento come JSON su stdin, e jq -r '.tool_input.command' estrae solo la stringa del comando, che >> aggiunge al file di log:Configurare la posizione dell’hook
Dove aggiungete un hook determina il suo ambito:| Posizione | Ambito | Condivisibile |
|---|---|---|
~/.claude/settings.json | Tutti i vostri progetti | No, locale alla vostra macchina |
.claude/settings.json | Singolo progetto | Sì, può essere committato nel repo |
.claude/settings.local.json | Singolo progetto | No, gitignored |
| Impostazioni di policy gestite | Organizzazione intera | Sì, controllato dall’amministratore |
Plugin hooks/hooks.json | Quando il plugin è abilitato | Sì, raggruppato con il plugin |
| Skill o agente frontmatter | Mentre la skill o l’agente è attivo | Sì, definito nel file del componente |
/hooks in Claude Code per sfogliare tutti gli hooks configurati raggruppati per evento. Per disabilitare tutti gli hooks contemporaneamente, impostate "disableAllHooks": true nel vostro file di impostazioni.
Se modificate i file di impostazioni direttamente mentre Claude Code è in esecuzione, il file watcher normalmente raccoglie i cambiamenti degli hook automaticamente.
Hooks basati su prompt
Per decisioni che richiedono giudizio piuttosto che regole deterministiche, utilizzate gli hooktype: "prompt". Invece di eseguire un comando shell, Claude Code invia il vostro prompt e i dati di input dell’hook a un modello Claude (Haiku per impostazione predefinita) per prendere la decisione. Potete specificare un modello diverso con il campo model se avete bisogno di più capacità.
L’unico lavoro del modello è restituire una decisione sì/no come JSON:
"ok": true: l’azione procede"ok": false: l’azione è bloccata. Il"reason"del modello viene alimentato di nuovo a Claude in modo da poter adattarsi.
Stop per chiedere al modello se tutti i compiti richiesti sono completi. Se il modello restituisce "ok": false, Claude continua a lavorare e utilizza il reason come sua prossima istruzione:
Hooks basati su agenti
Quando la verifica richiede l’ispezione di file o l’esecuzione di comandi, utilizzate gli hooktype: "agent". A differenza degli hook di prompt che effettuano una singola chiamata LLM, gli hook di agenti generano un subagent che può leggere file, cercare codice e utilizzare altri strumenti per verificare le condizioni prima di restituire una decisione.
Gli hook di agenti utilizzano lo stesso formato di risposta "ok" / "reason" degli hook di prompt, ma con un timeout predefinito più lungo di 60 secondi e fino a 50 turni di utilizzo dello strumento.
Questo esempio verifica che i test passino prima di consentire a Claude di fermarsi:
HTTP hooks
Utilizzate gli hooktype: "http" per POST dei dati dell’evento a un endpoint HTTP invece di eseguire un comando shell. L’endpoint riceve lo stesso JSON che un hook di comando riceverebbe su stdin, e restituisce i risultati attraverso il corpo della risposta HTTP utilizzando lo stesso formato JSON.
Gli HTTP hooks sono utili quando volete che un server web, una funzione cloud o un servizio esterno gestisca la logica dell’hook: ad esempio, un servizio di controllo condiviso che registra gli eventi di utilizzo dello strumento in un team.
Questo esempio pubblica ogni utilizzo dello strumento a un servizio di registrazione locale:
hookSpecificOutput appropriati. I codici di stato HTTP da soli non possono bloccare le azioni.
I valori dell’intestazione supportano l’interpolazione delle variabili di ambiente utilizzando la sintassi $VAR_NAME o ${VAR_NAME}. Solo le variabili elencate nell’array allowedEnvVars vengono risolte; tutti gli altri riferimenti $VAR rimangono vuoti.
Per le opzioni di configurazione complete e la gestione delle risposte, consultate HTTP hooks nel riferimento.
Limitazioni e risoluzione dei problemi
Limitazioni
- Gli hook di comando comunicano solo attraverso stdout, stderr e codici di uscita. Non possono attivare comandi o chiamate di strumenti direttamente. Gli HTTP hooks comunicano attraverso il corpo della risposta invece.
- Il timeout dell’hook è di 10 minuti per impostazione predefinita, configurabile per hook con il campo
timeout(in secondi). - Gli hook
PostToolUsenon possono annullare le azioni poiché lo strumento è già stato eseguito. - Gli hook
PermissionRequestnon si attivano in modalità non interattiva (-p). Utilizzate gli hookPreToolUseper le decisioni di autorizzazione automatizzate. - Gli hook
Stopsi attivano ogni volta che Claude finisce di rispondere, non solo al completamento dell’attività. Non si attivano su interruzioni dell’utente. Gli errori API attivano StopFailure invece.
Hook non si attiva
L’hook è configurato ma non si esegue mai.- Eseguite
/hookse confermate che l’hook appare sotto l’evento corretto - Controllate che il modello del matcher corrisponda esattamente al nome dello strumento (i matcher sono sensibili alle maiuscole)
- Verificate che state attivando il tipo di evento corretto (ad esempio,
PreToolUsesi attiva prima dell’esecuzione dello strumento,PostToolUsesi attiva dopo) - Se utilizzate gli hook
PermissionRequestin modalità non interattiva (-p), passate aPreToolUseinvece
Errore dell’hook nell’output
Vedete un messaggio come “PreToolUse hook error: …” nella trascrizione.- Il vostro script è uscito con un codice diverso da zero inaspettatamente. Testatelo manualmente inviando JSON di esempio:
- Se vedete “command not found”, utilizzate percorsi assoluti o
$CLAUDE_PROJECT_DIRper fare riferimento agli script - Se vedete “jq: command not found”, installate
jqo utilizzate Python/Node.js per l’analisi JSON - Se lo script non si esegue affatto, rendetelo eseguibile:
chmod +x ./my-hook.sh
/hooks non mostra hook configurati
Avete modificato un file di impostazioni ma gli hooks non appaiono nel menu.
- Le modifiche ai file vengono normalmente raccolte automaticamente. Se non sono apparse dopo alcuni secondi, il file watcher potrebbe aver perso il cambiamento: riavviate la vostra sessione per forzare un ricaricamento.
- Verificate che il vostro JSON sia valido (le virgole finali e i commenti non sono consentiti)
- Confermate che il file di impostazioni è nella posizione corretta:
.claude/settings.jsonper gli hook del progetto,~/.claude/settings.jsonper gli hook globali
L’hook Stop si esegue per sempre
Claude continua a lavorare in un ciclo infinito invece di fermarsi. Il vostro script di hook Stop deve controllare se ha già attivato una continuazione. Analizzate il campostop_hook_active dall’input JSON e uscite presto se è true:
Convalida JSON non riuscita
Claude Code mostra un errore di analisi JSON anche se il vostro script di hook produce JSON valido. Quando Claude Code esegue un hook, genera una shell che fornisce il vostro profilo (~/.zshrc o ~/.bashrc). Se il vostro profilo contiene istruzioni echo incondizionate, quell’output viene anteposto al vostro JSON dell’hook:
$- contiene i flag della shell, e i significa interattiva. Gli hooks si eseguono in shell non interattive, quindi l’echo viene saltato.
Tecniche di debug
Attivate la modalità verbose conCtrl+O per vedere l’output dell’hook nella trascrizione, o eseguite claude --debug per i dettagli di esecuzione completi incluso quali hook hanno corrisposto e i loro codici di uscita.
Ulteriori informazioni
- Riferimento Hooks: schemi di eventi completi, formato di output JSON, hook asincroni e hook di strumenti MCP
- Considerazioni sulla sicurezza: esaminate prima di distribuire gli hooks in ambienti condivisi o di produzione
- Esempio di validatore di comandi Bash: implementazione di riferimento completa