- Bloquear operaciones peligrosas antes de que se ejecuten, como comandos de shell destructivos o acceso a archivos no autorizado
- Registrar y auditar cada llamada de herramienta para cumplimiento, depuración o análisis
- Transformar entradas y salidas para desinfectar datos, inyectar credenciales o redirigir rutas de archivos
- Requerir aprobación humana para acciones sensibles como escrituras en bases de datos o llamadas a API
- Rastrear el ciclo de vida de la sesión para gestionar estado, limpiar recursos o enviar notificaciones
Cómo funcionan los hooks
Se dispara un evento
Algo sucede durante la ejecución del agente y el SDK dispara un evento: una herramienta está a punto de ser llamada (
PreToolUse), una herramienta devolvió un resultado (PostToolUse), un subagente se inició o se detuvo, el agente está inactivo, o la ejecución finalizó. Vea la lista completa de eventos.El SDK recopila hooks registrados
El SDK verifica si hay hooks registrados para ese tipo de evento. Esto incluye hooks de devolución de llamada que pasa en
options.hooks y hooks de comandos de shell de archivos de configuración cuando la entrada settingSources o setting_sources correspondiente está habilitada, lo cual lo está para las opciones predeterminadas de query().Los matchers filtran qué hooks se ejecutan
Si un hook tiene un patrón
matcher (como "Write|Edit"), el SDK lo prueba contra el objetivo del evento (por ejemplo, el nombre de la herramienta). Los hooks sin un matcher se ejecutan para cada evento de ese tipo.Se ejecutan las funciones de devolución de llamada
Cada hook coincidente recibe su función de devolución de llamada con información sobre lo que está sucediendo: el nombre de la herramienta, sus argumentos, el ID de sesión y otros detalles específicos del evento.
Su devolución de llamada devuelve una decisión
Después de realizar cualquier operación (registro, llamadas a API, validación), su devolución de llamada devuelve un objeto de salida que le dice al agente qué hacer: permitir la operación, bloquearla, modificar la entrada o inyectar contexto en la conversación.
PreToolUse (paso 1) con un matcher "Write|Edit" (paso 3) para que la devolución de llamada solo se dispare para herramientas de escritura de archivos. Cuando se activa, la devolución de llamada recibe la entrada de la herramienta (paso 4), verifica si la ruta del archivo apunta a un archivo .env, y devuelve permissionDecision: "deny" para bloquear la operación (paso 5):
Hooks disponibles
El SDK proporciona hooks para diferentes etapas de la ejecución del agente. Algunos hooks están disponibles en ambos SDK, mientras que otros son solo para TypeScript.| Evento de Hook | SDK de Python | SDK de TypeScript | Qué lo dispara | Caso de uso de ejemplo |
|---|---|---|---|---|
PreToolUse | Sí | Sí | Solicitud de llamada de herramienta (puede bloquear o modificar) | Bloquear comandos de shell peligrosos |
PostToolUse | Sí | Sí | Resultado de ejecución de herramienta | Registrar todos los cambios de archivo en pista de auditoría |
PostToolUseFailure | Sí | Sí | Fallo de ejecución de herramienta | Manejar o registrar errores de herramienta |
PostToolBatch | No | Sí | Un lote completo de llamadas de herramienta se resuelve, una vez por lote antes de la siguiente llamada del modelo | Inyectar convenciones una vez para todo el lote |
UserPromptSubmit | Sí | Sí | Envío de solicitud del usuario | Inyectar contexto adicional en solicitudes |
MessageDisplay | No | Sí | Un mensaje del asistente con texto se completa, una vez por mensaje con el texto completo del mensaje | Redactar o reformatear el texto mostrado sin cambiar la transcripción |
Stop | Sí | Sí | Detención de ejecución del agente | Guardar estado de sesión antes de salir |
SubagentStart | Sí | Sí | Inicialización de subagente | Rastrear generación de tareas paralelas |
SubagentStop | Sí | Sí | Finalización de subagente | Agregar resultados de tareas paralelas |
PreCompact | Sí | Sí | Solicitud de compactación de conversación | Archivar transcripción completa antes de resumir |
PermissionRequest | Sí | Sí | Se mostraría diálogo de permiso | Manejo de permisos personalizado |
SessionStart | No | Sí | Inicialización de sesión | Inicializar registro y telemetría |
SessionEnd | No | Sí | Terminación de sesión | Limpiar recursos temporales |
Notification | Sí | Sí | Mensajes de estado del agente | Enviar actualizaciones de estado del agente a Slack o PagerDuty |
Setup | No | Sí | Configuración/mantenimiento de sesión | Ejecutar tareas de inicialización |
TeammateIdle | No | Sí | El compañero se vuelve inactivo | Reasignar trabajo o notificar |
TaskCompleted | No | Sí | Tarea de fondo se completa | Agregar resultados de tareas paralelas |
ConfigChange | No | Sí | Archivo de configuración cambia | Recargar configuración dinámicamente |
WorktreeCreate | No | Sí | Git worktree creado | Rastrear espacios de trabajo aislados |
WorktreeRemove | No | Sí | Git worktree eliminado | Limpiar recursos de espacio de trabajo |
Configurar hooks
Para configurar un hook, páselo en el campohooks de sus opciones de agente (ClaudeAgentOptions en Python, el objeto options en TypeScript):
hooks es un diccionario (Python) u objeto (TypeScript) donde:
- Las claves son nombres de eventos de hook (por ejemplo,
'PreToolUse','PostToolUse','Stop') - Los valores son matrices de matchers, cada una conteniendo un patrón de filtro opcional y sus funciones de devolución de llamada
Matchers
Use matchers para filtrar cuándo se disparan sus devoluciones de llamada. El campomatcher coincide con un valor diferente dependiendo del tipo de evento de hook. Por ejemplo, los hooks basados en herramientas coinciden con el nombre de la herramienta, mientras que los hooks Notification coinciden con el tipo de notificación. Vea la referencia de hooks de Claude Code para la lista completa de valores de matcher para cada tipo de evento.
Los matchers del SDK siguen las mismas reglas que matchers en archivos de configuración: un matcher que contiene solo letras, dígitos, _ y | se compara como una cadena exacta, con | separando alternativas, por lo que Write|Edit coincide exactamente con esas dos herramientas. Un matcher de *, una cadena vacía, u omitir el matcher completamente coincide con cada ocurrencia del evento; un matcher que contiene cualquier otro carácter se evalúa como una expresión regular, por lo que ^mcp__ coincide con cada herramienta MCP. Un matcher como mcp__memory contiene solo letras y guiones bajos, por lo que se compara como una cadena exacta y no coincide con ninguna herramienta; use mcp__memory__.* para coincidir con cada herramienta de ese servidor.
| Opción | Tipo | Predeterminado | Descripción |
|---|---|---|---|
matcher | string | undefined | Patrón coincidido contra el campo de filtro del evento, siguiendo las reglas de comparación anteriores. Para hooks de herramientas, este es el nombre de la herramienta. Las herramientas integradas incluyen Bash, Read, Write, Edit, Glob, Grep, WebFetch, Agent y otros (vea Tipos de entrada de herramienta para la lista completa). Las herramientas MCP usan el patrón mcp__<server>__<action>. |
hooks | HookCallback[] | - | Requerido. Matriz de funciones de devolución de llamada a ejecutar cuando el patrón coincide |
timeout | number | 60 | Tiempo de espera en segundos |
matcher para dirigirse a herramientas específicas siempre que sea posible. Un matcher con 'Bash' solo se ejecuta para comandos Bash, mientras que omitir el patrón ejecuta sus devoluciones de llamada para cada ocurrencia del evento. Tenga en cuenta que para hooks basados en herramientas, los matchers solo filtran por nombre de herramienta, no por rutas de archivo u otros argumentos. Para filtrar por ruta de archivo, verifique tool_input.file_path dentro de su devolución de llamada.
Funciones de devolución de llamada
Entradas
Cada devolución de llamada de hook recibe tres argumentos:- Datos de entrada: un objeto tipado que contiene detalles del evento. Cada tipo de hook tiene su propia forma de entrada (por ejemplo,
PreToolUseHookInputincluyetool_nameytool_input, mientras queNotificationHookInputincluyemessage). Vea las definiciones de tipo completas en las referencias del SDK de TypeScript y Python.- Todas las entradas de hook comparten
session_id,cwdyhook_event_name. agent_idyagent_typese rellenan cuando el hook se dispara dentro de un subagente. En TypeScript, estos están en la entrada de hook base y disponibles para todos los tipos de hook. En Python, están solo enPreToolUse,PostToolUseyPostToolUseFailure.
- Todas las entradas de hook comparten
- ID de uso de herramienta (
str | None/string | undefined): correlaciona eventosPreToolUseyPostToolUsepara la misma llamada de herramienta. - Contexto: en TypeScript, contiene una propiedad
signal(AbortSignal) para cancelación. En Python, este argumento está reservado para uso futuro.
Salidas
Su devolución de llamada devuelve un objeto con dos categorías de campos:- Campos de nivel superior funcionan igual en cada evento:
systemMessagemuestra un mensaje al usuario, ycontinue(continue_en Python) determina si el agente sigue ejecutándose después de este hook. hookSpecificOutputcontrola la operación actual. Los campos dentro dependen del tipo de evento de hook. Para hooksPreToolUse, aquí es donde establecepermissionDecision("allow","deny","ask"o"defer"),permissionDecisionReasoneupdatedInput. Devolver"defer"finaliza la consulta para que pueda reanudarla más tarde. Para hooksPostToolUse, puede estableceradditionalContextpara agregar información al resultado de la herramienta. Para reemplazar la salida de la herramienta antes de que Claude la vea, establezcaupdatedToolOutput, que funciona para cualquier herramienta en ambos SDK. El campo anteriorupdatedMCPToolOutputreemplaza solo la salida de herramientas MCP y está deprecado.
{} para permitir la operación sin cambios. Los hooks de devolución de llamada del SDK usan el mismo formato de salida JSON que hooks de comandos de shell de Claude Code, que documenta cada campo y opción específica del evento. Para las definiciones de tipo del SDK, vea las referencias del SDK de TypeScript y Python.
Cuando se aplican múltiples hooks o reglas de permiso, deny tiene prioridad sobre defer, que tiene prioridad sobre ask, que tiene prioridad sobre allow. Si algún hook devuelve
deny, la operación se bloquea independientemente de otros hooks.Salida asincrónica
De forma predeterminada, el agente espera a que su hook devuelva antes de continuar. Si su hook realiza un efecto secundario (registro, envío de webhook) y no necesita influir en el comportamiento del agente, puede devolver una salida asincrónica en su lugar. Esto le dice al agente que continúe inmediatamente sin esperar a que el hook termine:| Campo | Tipo | Descripción |
|---|---|---|
async | true | Señala modo asincrónico. El agente continúa sin esperar. En Python, use async_ para evitar la palabra clave reservada. |
asyncTimeout | number | Tiempo de espera opcional en milisegundos para la operación de fondo |
Las salidas asincrónicas no pueden bloquear, modificar o inyectar contexto en la operación ya que el agente ya ha avanzado. Úselas solo para efectos secundarios como registro, métricas o notificaciones.
Ejemplos
Modificar entrada de herramienta
Este ejemplo intercepta llamadas de herramienta Write y reescribe el argumentofile_path para anteponer /sandbox, redirigiendo todas las escrituras de archivo a un directorio aislado. La devolución de llamada devuelve updatedInput con la ruta modificada y permissionDecision: 'allow' para aprobar automáticamente la operación reescrita:
Cuando use
updatedInput, también debe incluir permissionDecision: 'allow' para aprobar automáticamente la entrada modificada o permissionDecision: 'ask' para mostrársela al usuario. Con 'defer', updatedInput se ignora. Siempre devuelva un nuevo objeto en lugar de mutar el tool_input original.Agregar contexto y bloquear una herramienta
Este ejemplo bloquea escrituras en el directorio/etc y explica por qué tanto al modelo como al usuario:
permissionDecision: 'deny'detiene la llamada de herramienta.permissionDecisionReasonle dice al modelo por qué, para que evite reintentar.systemMessagemuestra al usuario qué sucedió.
Aprobar automáticamente herramientas específicas
De forma predeterminada, el agente puede solicitar permiso antes de usar ciertas herramientas. Este ejemplo aprueba automáticamente herramientas del sistema de archivos de solo lectura (Read, Glob, Grep) devolviendopermissionDecision: 'allow', permitiéndoles ejecutarse sin confirmación del usuario mientras deja todas las otras herramientas sujetas a verificaciones de permiso normales:
Registrar múltiples hooks
Cuando se dispara un evento, todos los hooks coincidentes se ejecutan en paralelo. Para decisiones de permiso, el resultado más restrictivo gana: un únicodeny bloquea la llamada de herramienta independientemente de lo que devuelvan los otros hooks. Debido a que el orden de finalización es no determinista, escriba cada hook para actuar de forma independiente en lugar de depender de que otro hook se haya ejecutado primero.
El ejemplo a continuación registra tres verificaciones independientes para cada llamada de herramienta:
Filtrar con matchers de múltiples herramientas
Use matchers de múltiples herramientas para compartir una devolución de llamada entre herramientas relacionadas. Este ejemplo registra tres matchers con diferentes alcances:- Una lista exacta separada por tuberías (
Write|Edit|Delete) disparafile_security_hooksolo para herramientas de modificación de archivos. - Una expresión regular (
^mcp__) disparamcp_audit_hookpara cualquier herramienta MCP cuyo nombre comience conmcp__. - Un matcher omitido dispara
global_loggerpara cada llamada de herramienta independientemente del nombre.
Rastrear actividad de subagente
Use hooksSubagentStop para monitorear cuándo los subagentes terminan su trabajo. Vea el tipo de entrada completo en las referencias del SDK de TypeScript y Python. Este ejemplo registra un resumen cada vez que un subagente se completa:
Realizar solicitudes HTTP desde hooks
Los hooks pueden realizar operaciones asincrónicas como solicitudes HTTP. Capture errores dentro de su hook en lugar de dejarlos propagarse, ya que una excepción no manejada puede interrumpir el agente. Este ejemplo envía un webhook después de que cada herramienta se completa, registrando qué herramienta se ejecutó y cuándo. El hook captura errores para que un webhook fallido no interrumpa el agente:Reenviar notificaciones a Slack
Use hooksNotification para recibir notificaciones del sistema del agente y reenviarlas a servicios externos. Las notificaciones se disparan para tipos de evento tales como:
permission_promptcuando Claude necesita permisoidle_promptcuando Claude está esperando entradaauth_successcuando la autenticación se completaelicitation_dialog,elicitation_complete, yelicitation_responsepara flujos de elicitación de solicitud del usuario
message con una descripción legible por humanos y opcionalmente un title.
Este ejemplo reenvía cada notificación a un canal de Slack. Requiere una URL de webhook entrante de Slack, que crea agregando una aplicación a su espacio de trabajo de Slack y habilitando webhooks entrantes:
Solucionar problemas comunes
Hook no se dispara
- Verifique que el nombre del evento de hook sea correcto y sensible a mayúsculas (
PreToolUse, nopreToolUse) - Verifique que su patrón de matcher coincida exactamente con el nombre de la herramienta
- Asegúrese de que el hook esté bajo el tipo de evento correcto en
options.hooks - Para hooks que no son de herramientas como
StopySubagentStop, los matchers coinciden contra campos diferentes (vea patrones de matcher) - Los hooks pueden no dispararse cuando el agente alcanza el límite
max_turnsporque la sesión termina antes de que los hooks puedan ejecutarse
Matcher no filtra como se esperaba
Los matchers solo coinciden con nombres de herramientas, no con rutas de archivo u otros argumentos. Para filtrar por ruta de archivo, verifiquetool_input.file_path dentro de su hook:
Tiempo de espera del hook
- Aumente el valor
timeouten la configuraciónHookMatcher - Use el
AbortSignaldel tercer argumento de devolución de llamada para manejar la cancelación correctamente en TypeScript
Herramienta bloqueada inesperadamente
- Verifique todos los hooks
PreToolUsepara devoluciones depermissionDecision: 'deny' - Agregue registro a sus hooks para ver qué
permissionDecisionReasonestán devolviendo - Verifique que los patrones de matcher no sean demasiado amplios (un matcher vacío coincide con todas las herramientas)
Entrada modificada no aplicada
-
Asegúrese de que
updatedInputesté dentro dehookSpecificOutput, no en el nivel superior: -
Devuelva
permissionDecision: 'allow'para aprobar automáticamente la entrada modificada, u'ask'para mostrarla al usuario para su aprobación -
Incluya
hookEventNameenhookSpecificOutputpara identificar para qué tipo de hook es la salida
Hooks de sesión no disponibles en Python
SessionStart y SessionEnd pueden registrarse como hooks de devolución de llamada del SDK en TypeScript, pero no están disponibles en el SDK de Python (HookEvent los omite). En Python, solo están disponibles como hooks de comandos de shell definidos en archivos de configuración (por ejemplo, .claude/settings.json). Para cargar hooks de comandos de shell desde su aplicación SDK, incluya la fuente de configuración apropiada con setting_sources o settingSources:
client.receive_response() como su disparador.
Solicitudes de permiso de subagente multiplicándose
Al generar múltiples subagentes, cada uno puede solicitar permisos por separado. Los subagentes no heredan automáticamente los permisos del agente padre. Para evitar solicitudes repetidas, use hooksPreToolUse para aprobar automáticamente herramientas específicas, o configure reglas de permiso que se apliquen a sesiones de subagente.
Bucles recursivos de hook con subagentes
Un hookUserPromptSubmit que genera subagentes puede crear bucles infinitos si esos subagentes disparan el mismo hook. Para prevenir esto:
- Verifique un indicador de subagente en la entrada del hook antes de generar
- Use una variable compartida o estado de sesión para rastrear si ya está dentro de un subagente
- Alcance los hooks para ejecutarse solo para la sesión del agente de nivel superior
systemMessage no aparece en la salida
El camposystemMessage muestra un mensaje al usuario, no al modelo. Por defecto, el SDK no expone la salida de hooks en el flujo de mensajes, por lo que el mensaje puede no aparecer a menos que establezca includeHookEvents (include_hook_events en Python). Para pasar contexto al modelo en su lugar, devuelva additionalContext.
Si necesita exponer decisiones de hook a su aplicación de manera confiable, regístrelas por separado o use un canal de salida dedicado.
Recursos relacionados
- Referencia de hooks de Claude Code: esquemas completos de entrada/salida JSON, documentación de eventos y patrones de matcher
- Guía de hooks de Claude Code: ejemplos de hooks de comandos de shell y tutoriales
- Referencia del SDK de TypeScript: tipos de hook, definiciones de entrada/salida y opciones de configuración
- Referencia del SDK de Python: tipos de hook, definiciones de entrada/salida y opciones de configuración
- Permisos: controlar qué puede hacer su agente
- Herramientas personalizadas: crear herramientas para extender las capacidades del agente