Gli hook dei Plugin sono punti di estensione in-process per i plugin OpenClaw. Usali quando un plugin deve ispezionare o modificare esecuzioni dell’agente, chiamate agli strumenti, flusso dei messaggi, ciclo di vita della sessione, instradamento dei subagent, installazioni o avvio del Gateway. Usa invece gli hook interni quando vuoi un piccolo scriptDocumentation Index
Fetch the complete documentation index at: https://docs2.openclaw.ai/llms.txt
Use this file to discover all available pages before exploring further.
HOOK.md installato dall’operatore per eventi di comando e Gateway come
/new, /reset, /stop, agent:bootstrap o gateway:startup.
Avvio rapido
Registra hook di plugin tipizzati conapi.on(...) dall’entry del plugin:
priority. Gli hook con la stessa priorità
mantengono l’ordine di registrazione.
api.on(name, handler, opts?) accetta:
priority- ordinamento degli handler (i valori più alti vengono eseguiti per primi).timeoutMs- budget opzionale per singolo hook. Quando impostato, il runner degli hook interrompe quell’handler dopo che il budget è scaduto e continua con il successivo, invece di lasciare che configurazioni lente o lavoro di recupero consumino il timeout del modello configurato dal chiamante. Omettilo per usare il timeout predefinito di osservazione/decisione che il runner degli hook applica genericamente.
hooks.timeouts.<hookName> sovrascrive hooks.timeoutMs, che sovrascrive il valore
api.on(..., { timeoutMs }) definito dall’autore del plugin. Ogni valore configurato deve
essere un intero positivo non superiore a 600000 millisecondi. Preferisci override per singolo hook
per hook notoriamente lenti, così un plugin non ottiene un budget più lungo
ovunque.
Ogni hook riceve event.context.pluginConfig, la configurazione risolta per il
plugin che ha registrato quell’handler. Usala per decisioni degli hook che richiedono
le opzioni correnti del plugin; OpenClaw la inietta per handler senza mutare l’oggetto
evento condiviso visto dagli altri plugin.
Catalogo degli hook
Gli hook sono raggruppati in base alla superficie che estendono. I nomi in grassetto accettano un risultato decisionale (blocco, annullamento, override o richiesta di approvazione); tutti gli altri sono solo di osservazione. Turno dell’agentebefore_model_resolve- esegue l’override del provider o del modello prima che i messaggi della sessione vengano caricatiagent_turn_prepare- consuma le iniezioni di turno del plugin in coda e aggiunge contesto nello stesso turno prima degli hook del promptbefore_prompt_build- aggiunge contesto dinamico o testo del system prompt prima della chiamata al modellobefore_agent_start- fase combinata solo per compatibilità; preferisci i due hook soprabefore_agent_run- ispeziona il prompt finale e i messaggi della sessione prima dell’invio al modello e, facoltativamente, blocca l’esecuzionebefore_agent_reply- interrompe anticipatamente il turno del modello con una risposta sintetica o silenziobefore_agent_finalize- ispeziona la risposta finale naturale e richiede un altro passaggio del modelloagent_end- osserva messaggi finali, stato di successo e durata dell’esecuzioneheartbeat_prompt_contribution- aggiunge contesto solo Heartbeat per monitor in background e plugin del ciclo di vita
model_call_started/model_call_ended- osserva metadati sanificati di chiamata provider/modello, tempistiche, esito e hash limitati degli ID richiesta senza contenuto del prompt o della rispostallm_input- osserva l’input del provider (system prompt, prompt, cronologia)llm_output- osserva l’output del provider
before_tool_call- riscrive i parametri dello strumento, blocca l’esecuzione o richiede approvazioneafter_tool_call- osserva risultati dello strumento, errori e duratatool_result_persist- riscrive il messaggio dell’assistente prodotto da un risultato dello strumentobefore_message_write- ispeziona o blocca una scrittura di messaggio in corso (raro)
inbound_claim- rivendica un messaggio in ingresso prima dell’instradamento all’agente (risposte sintetiche)message_received- osserva contenuto in ingresso, mittente, thread e metadatimessage_sending- riscrive contenuto in uscita o annulla la consegnamessage_sent- osserva il successo o il fallimento della consegna in uscitabefore_dispatch- ispeziona o riscrive un dispatch in uscita prima del passaggio al canalereply_dispatch- partecipa alla pipeline finale di dispatch della risposta
session_start/session_end- traccia i confini del ciclo di vita della sessione. Ilreasondell’evento è uno tranew,reset,idle,daily,compaction,deleted,shutdown,restartounknown. I valorishutdownerestartvengono attivati dal finalizer di shutdown del Gateway quando il processo viene arrestato o riavviato mentre le sessioni sono ancora attive, così i plugin downstream (come store di memoria o trascrizioni) possono finalizzare righe fantasma che altrimenti resterebbero in stato aperto tra i riavvii. Il finalizer è limitato, così un plugin lento non può bloccare SIGTERM/SIGINT.before_compaction/after_compaction- osserva o annota cicli di Compactionbefore_reset- osserva eventi di reset della sessione (/reset, reset programmatici)
subagent_spawning/subagent_delivery_target/subagent_spawned/subagent_ended- coordina instradamento dei subagent e consegna del completamento
gateway_start/gateway_stop- avvia o arresta servizi di proprietà del plugin con il Gatewaycron_changed- osserva modifiche del ciclo di vita Cron di proprietà del gateway (aggiunto, aggiornato, rimosso, avviato, terminato, pianificato)before_install- ispeziona scansioni di installazione di skill o plugin e, facoltativamente, blocca
Criteri per le chiamate agli strumenti
before_tool_call riceve:
event.toolNameevent.paramsevent.derivedPathsopzionale, contenente suggerimenti best-effort sui percorsi target derivati dall’host per envelope di strumenti noti comeapply_patch; quando presenti, questi percorsi possono essere incompleti o possono sovrastimare ciò che lo strumento toccherà effettivamente (per esempio, con input malformati o parziali)event.runIdopzionaleevent.toolCallIdopzionale- campi di contesto come
ctx.agentId,ctx.sessionKey,ctx.sessionId,ctx.runId,ctx.jobId(impostato su esecuzioni guidate da Cron) ectx.tracediagnostico
block: trueè terminale e salta gli handler con priorità inferiore.block: falseviene trattato come assenza di decisione.paramsriscrive i parametri dello strumento per l’esecuzione.requireApprovalmette in pausa l’esecuzione dell’agente e chiede all’utente tramite approvazioni dei plugin. Il comando/approvepuò approvare sia approvazioni exec sia approvazioni plugin.- Un
block: truecon priorità inferiore può comunque bloccare dopo che un hook con priorità maggiore ha richiesto l’approvazione. onResolutionriceve la decisione di approvazione risolta -allow-once,allow-always,deny,timeoutocancelled.
api.registerTrustedToolPolicy(...). Queste vengono eseguite prima dei normali
hook before_tool_call e prima delle decisioni dei plugin esterni. Usale solo
per gate considerati attendibili dall’host come policy dello spazio di lavoro, applicazione del budget o
sicurezza di workflow riservati. I plugin esterni dovrebbero usare i normali hook before_tool_call.
Persistenza dei risultati degli strumenti
I risultati degli strumenti possono includeredetails strutturati per rendering UI, diagnostica,
instradamento di media o metadati di proprietà del plugin. Tratta details come metadati runtime,
non come contenuto del prompt:
- OpenClaw rimuove
toolResult.detailsprima del replay del provider e dell’input di Compaction, così i metadati non diventano contesto del modello. - Le voci di sessione persistite mantengono solo
detailslimitati. I details troppo grandi vengono sostituiti con un riepilogo compatto epersistedDetailsTruncated: true. tool_result_persistebefore_message_writevengono eseguiti prima del limite finale di persistenza. Gli hook dovrebbero comunque mantenere piccoli idetailsrestituiti ed evitare di inserire testo rilevante per il prompt solo indetails; inserisci l’output dello strumento visibile al modello incontent.
Hook di prompt e modello
Usa gli hook specifici di fase per i nuovi plugin:before_model_resolve: riceve solo il prompt corrente e i metadati degli allegati. RestituisceproviderOverrideomodelOverride.agent_turn_prepare: riceve il prompt corrente, i messaggi di sessione preparati e le eventuali iniezioni in coda esattamente una volta svuotate per questa sessione. RestituisceprependContextoappendContext.before_prompt_build: riceve il prompt corrente e i messaggi della sessione. RestituisceprependContext,appendContext,systemPrompt,prependSystemContextoappendSystemContext.heartbeat_prompt_contribution: viene eseguito solo per turni Heartbeat e restituisceprependContextoappendContext. È pensato per monitor in background che devono riassumere lo stato corrente senza modificare i turni avviati dall’utente.
before_agent_start resta disponibile per compatibilità. Preferisci gli hook espliciti sopra
così il tuo plugin non dipende da una fase combinata legacy.
before_agent_run viene eseguito dopo la costruzione del prompt e prima di qualsiasi input al modello,
inclusi il caricamento di immagini locali al prompt e l’osservazione llm_input. Riceve
l’input utente corrente come prompt, più la cronologia della sessione caricata in messages
e il system prompt attivo. Restituisce { outcome: "block", reason, message? }
per interrompere l’esecuzione prima che il modello possa leggere il prompt. reason è interno;
message è la sostituzione visibile all’utente. Gli unici esiti supportati sono
pass e block; forme decisionali non supportate falliscono in modo chiuso.
Quando un’esecuzione viene bloccata, OpenClaw archivia solo il testo sostitutivo in
message.content più metadati di blocco non sensibili come l’ID del plugin bloccante
e il timestamp. Il testo originale dell’utente non viene conservato nella trascrizione o nel contesto futuro.
Le ragioni interne di blocco sono trattate come sensibili ed escluse da
trascrizione, cronologia, broadcast, log e payload diagnostici. L’osservabilità
dovrebbe usare campi sanificati come ID del bloccatore, esito, timestamp o una categoria
sicura.
before_agent_start e agent_end includono event.runId quando OpenClaw può
identificare l’esecuzione attiva. Lo stesso valore è disponibile anche su ctx.runId.
Le esecuzioni guidate da Cron espongono anche ctx.jobId (l’ID del job Cron di origine) così
gli hook dei plugin possono limitare metriche, effetti collaterali o stato a uno specifico job
pianificato.
Per esecuzioni originate da canale, ctx.messageProvider è la superficie del provider come
discord o telegram, mentre ctx.channelId è l’identificatore del target della conversazione
quando OpenClaw può derivarne uno dalla chiave di sessione o dai metadati di consegna.
agent_end è un hook di osservazione e viene eseguito fire-and-forget dopo il turno. Il
runner degli hook applica un timeout di 30 secondi, così un plugin bloccato o un endpoint
di embedding non può lasciare la promessa dell’hook pendente per sempre. Un timeout viene registrato e
OpenClaw continua; non annulla il lavoro di rete di proprietà del plugin a meno che anche il
plugin non usi un proprio segnale di interruzione.
Usa model_call_started e model_call_ended per la telemetria delle chiamate al provider
che non deve ricevere prompt raw, cronologia, risposte, header, corpi delle richieste
o ID delle richieste del provider. Questi hook includono metadati stabili come
runId, callId, provider, model, api/transport facoltativi, durationMs/outcome
terminali e upstreamRequestIdHash quando OpenClaw può derivare un hash limitato
dell’ID richiesta del provider.
before_agent_finalize viene eseguito solo quando un harness sta per accettare una risposta
finale naturale dell’assistente. Non è il percorso di annullamento /stop e non viene
eseguito quando l’utente interrompe un turno. Restituisci { action: "revise", reason } per chiedere
all’harness un ulteriore passaggio del modello prima della finalizzazione, { action: "finalize", reason? } per forzare la finalizzazione, oppure ometti un risultato per continuare.
Gli hook Stop nativi di Codex vengono inoltrati in questo hook come decisioni OpenClaw
before_agent_finalize.
Quando restituiscono action: "revise", i plugins possono includere metadati retry per rendere
il passaggio extra del modello limitato e sicuro da riprodurre:
instruction viene aggiunto al motivo di revisione inviato all’harness.
idempotencyKey consente all’host di contare i tentativi per la stessa richiesta del plugin in
decisioni di finalizzazione equivalenti, e maxAttempts limita quanti passaggi extra
l’host consentirà prima di continuare con la risposta finale naturale.
I plugins non bundled che necessitano di hook di conversazione raw (before_model_resolve,
before_agent_reply, llm_input, llm_output, before_agent_finalize,
agent_end o before_agent_run) devono impostare:
plugins.entries.<id>.hooks.allowPromptInjection=false.
Estensioni di sessione e iniezioni per il turno successivo
I plugins di workflow possono conservare un piccolo stato di sessione compatibile con JSON tramiteapi.registerSessionExtension(...) e aggiornarlo tramite il metodo Gateway
sessions.pluginPatch. Le righe di sessione proiettano lo stato delle estensioni registrate
tramite pluginExtensions, consentendo alla UI di controllo e ad altri client di visualizzare
lo stato di proprietà del plugin senza conoscere gli internals del plugin.
Usa api.enqueueNextTurnInjection(...) quando un plugin ha bisogno che un contesto persistente
raggiunga il prossimo turno del modello esattamente una volta. OpenClaw svuota le iniezioni in coda prima
degli hook dei prompt, elimina le iniezioni scadute e deduplica per idempotencyKey
per plugin. Questa è la seam corretta per riprese dopo approvazione, riepiloghi di policy,
delta dei monitor in background e continuazioni di comandi che dovrebbero essere visibili
al modello nel turno successivo ma non dovrebbero diventare testo permanente del prompt di sistema.
Le semantiche di cleanup fanno parte del contratto. Il cleanup delle estensioni di sessione e
le callback di cleanup del ciclo di vita runtime ricevono reset, delete, disable o
restart. L’host rimuove lo stato persistente dell’estensione di sessione del plugin proprietario
e le iniezioni pendenti per il turno successivo per reset/delete/disable; restart conserva
lo stato di sessione persistente mentre le callback di cleanup consentono ai plugins di rilasciare job dello scheduler,
contesto di esecuzione e altre risorse out-of-band della vecchia generazione runtime.
Hook dei messaggi
Usa gli hook dei messaggi per routing a livello di canale e policy di consegna:message_received: osserva contenuto in entrata, mittente,threadId,messageId,senderId, correlazione facoltativa di run/sessione e metadati.message_sending: riscrivicontento restituisci{ cancel: true }.message_sent: osserva il successo o l’errore finale.
content può contenere la trascrizione parlata nascosta
anche quando il payload del canale non ha testo/caption visibile. Riscrivere quel
content aggiorna solo la trascrizione visibile all’hook; non viene renderizzata come
caption media.
I contesti degli hook dei messaggi espongono campi di correlazione stabili quando disponibili:
ctx.sessionKey, ctx.runId, ctx.messageId, ctx.senderId, ctx.trace,
ctx.traceId, ctx.spanId, ctx.parentSpanId e ctx.callDepth. Preferisci
questi campi di prima classe prima di leggere i metadati legacy.
Preferisci i campi tipizzati threadId e replyToId prima di usare metadati specifici del canale.
Regole decisionali:
message_sendingconcancel: trueè terminale.message_sendingconcancel: falseviene trattato come nessuna decisione.- Il
contentriscritto continua verso gli hook a priorità inferiore a meno che un hook successivo annulli la consegna. message_sendingpuò restituirecancelReasonemetadatalimitati con un annullamento. Le nuove API del ciclo di vita dei messaggi espongono questo come esito di consegna soppressa con motivocancelled_by_message_sending_hook; la consegna diretta legacy continua a restituire un array di risultati vuoto per compatibilità.message_sentè solo osservazione. Gli errori degli handler vengono registrati e non modificano il risultato della consegna.
Hook di installazione
before_install viene eseguito dopo la scansione integrata per installazioni di skill e plugin.
Restituisci risultati aggiuntivi oppure { block: true, blockReason } per interrompere
l’installazione.
block: true è terminale. block: false viene trattato come nessuna decisione.
Ciclo di vita del Gateway
Usagateway_start per servizi plugin che necessitano di stato di proprietà del Gateway. Il
contesto espone ctx.config, ctx.workspaceDir e ctx.getCron?.() per
ispezione e aggiornamenti di cron. Usa gateway_stop per liberare risorse
a lunga esecuzione.
Non fare affidamento sull’hook interno gateway:startup per servizi runtime
di proprietà del plugin.
cron_changed scatta per eventi del ciclo di vita cron di proprietà del gateway con un payload
evento tipizzato che copre i motivi added, updated, removed, started, finished
e scheduled. L’evento trasporta uno snapshot PluginHookGatewayCronJob
(inclusi state.nextRunAtMs, state.lastRunStatus e
state.lastError quando presenti) più un PluginHookGatewayCronDeliveryStatus
di not-requested | delivered | not-delivered | unknown. Gli eventi rimossi
trasportano comunque lo snapshot del job eliminato così gli scheduler esterni possono
riconciliare lo stato. Usa ctx.getCron?.() e ctx.config dal contesto runtime
quando sincronizzi scheduler di risveglio esterni e mantieni OpenClaw come
fonte di verità per i controlli delle scadenze e l’esecuzione.
Deprecazioni imminenti
Alcune superfici adiacenti agli hook sono deprecate ma ancora supportate. Migra prima della prossima major release:- Envelope di canale in testo semplice negli handler
inbound_claimemessage_received. LeggiBodyForAgente i blocchi strutturati del contesto utente invece di analizzare testo envelope piatto. Vedi Envelope di canale in testo semplice → BodyForAgent. before_agent_startresta per compatibilità. I nuovi plugins dovrebbero usarebefore_model_resolveebefore_prompt_buildinvece della fase combinata.onResolutioninbefore_tool_callora usa l’unione tipizzataPluginApprovalResolution(allow-once/allow-always/deny/timeout/cancelled) invece di unastringlibera.
command-auth → command-status - vedi
Migrazione del Plugin SDK → Deprecazioni attive.
Correlati
- Migrazione del Plugin SDK - deprecazioni attive e timeline di rimozione
- Creazione di plugins
- Panoramica del Plugin SDK
- Entry point dei plugin
- Hook interni
- Internals dell’architettura dei plugin