Vai al contenuto principale

Documentation Index

Fetch the complete documentation index at: https://docs2.openclaw.ai/llms.txt

Use this file to discover all available pages before exploring further.

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 script 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 con api.on(...) dall’entry del plugin:
import { definePluginEntry } from "openclaw/plugin-sdk/plugin-entry";

export default definePluginEntry({
  id: "tool-preflight",
  name: "Tool Preflight",
  register(api) {
    api.on(
      "before_tool_call",
      async (event) => {
        if (event.toolName !== "web_search") {
          return;
        }

        return {
          requireApproval: {
            title: "Run web search",
            description: `Allow search query: ${String(event.params.query ?? "")}`,
            severity: "info",
            timeoutMs: 60_000,
            timeoutBehavior: "deny",
          },
        };
      },
      { priority: 50 },
    );
  },
});
Gli handler degli hook vengono eseguiti in sequenza in ordine decrescente di 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.
Gli operatori possono anche impostare budget degli hook senza modificare il codice del plugin:
{
  "plugins": {
    "entries": {
      "my-plugin": {
        "hooks": {
          "timeoutMs": 30000,
          "timeouts": {
            "before_prompt_build": 90000,
            "agent_end": 60000
          }
        }
      }
    }
  }
}
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’agente
  • before_model_resolve - esegue l’override del provider o del modello prima che i messaggi della sessione vengano caricati
  • agent_turn_prepare - consuma le iniezioni di turno del plugin in coda e aggiunge contesto nello stesso turno prima degli hook del prompt
  • before_prompt_build - aggiunge contesto dinamico o testo del system prompt prima della chiamata al modello
  • before_agent_start - fase combinata solo per compatibilità; preferisci i due hook sopra
  • before_agent_run - ispeziona il prompt finale e i messaggi della sessione prima dell’invio al modello e, facoltativamente, blocca l’esecuzione
  • before_agent_reply - interrompe anticipatamente il turno del modello con una risposta sintetica o silenzio
  • before_agent_finalize - ispeziona la risposta finale naturale e richiede un altro passaggio del modello
  • agent_end - osserva messaggi finali, stato di successo e durata dell’esecuzione
  • heartbeat_prompt_contribution - aggiunge contesto solo Heartbeat per monitor in background e plugin del ciclo di vita
Osservazione della conversazione
  • 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 risposta
  • llm_input - osserva l’input del provider (system prompt, prompt, cronologia)
  • llm_output - osserva l’output del provider
Strumenti
  • before_tool_call - riscrive i parametri dello strumento, blocca l’esecuzione o richiede approvazione
  • after_tool_call - osserva risultati dello strumento, errori e durata
  • tool_result_persist - riscrive il messaggio dell’assistente prodotto da un risultato dello strumento
  • before_message_write - ispeziona o blocca una scrittura di messaggio in corso (raro)
Messaggi e consegna
  • inbound_claim - rivendica un messaggio in ingresso prima dell’instradamento all’agente (risposte sintetiche)
  • message_received - osserva contenuto in ingresso, mittente, thread e metadati
  • message_sending - riscrive contenuto in uscita o annulla la consegna
  • message_sent - osserva il successo o il fallimento della consegna in uscita
  • before_dispatch - ispeziona o riscrive un dispatch in uscita prima del passaggio al canale
  • reply_dispatch - partecipa alla pipeline finale di dispatch della risposta
Sessioni e Compaction
  • session_start / session_end - traccia i confini del ciclo di vita della sessione. Il reason dell’evento è uno tra new, reset, idle, daily, compaction, deleted, shutdown, restart o unknown. I valori shutdown e restart vengono 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 Compaction
  • before_reset - osserva eventi di reset della sessione (/reset, reset programmatici)
Subagent
  • subagent_spawning / subagent_delivery_target / subagent_spawned / subagent_ended - coordina instradamento dei subagent e consegna del completamento
Ciclo di vita
  • gateway_start / gateway_stop - avvia o arresta servizi di proprietà del plugin con il Gateway
  • cron_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.toolName
  • event.params
  • event.derivedPaths opzionale, contenente suggerimenti best-effort sui percorsi target derivati dall’host per envelope di strumenti noti come apply_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.runId opzionale
  • event.toolCallId opzionale
  • campi di contesto come ctx.agentId, ctx.sessionKey, ctx.sessionId, ctx.runId, ctx.jobId (impostato su esecuzioni guidate da Cron) e ctx.trace diagnostico
Può restituire:
type BeforeToolCallResult = {
  params?: Record<string, unknown>;
  block?: boolean;
  blockReason?: string;
  requireApproval?: {
    title: string;
    description: string;
    severity?: "info" | "warning" | "critical";
    timeoutMs?: number;
    timeoutBehavior?: "allow" | "deny";
    pluginId?: string;
    onResolution?: (
      decision: "allow-once" | "allow-always" | "deny" | "timeout" | "cancelled",
    ) => Promise<void> | void;
  };
};
Regole:
  • block: true è terminale e salta gli handler con priorità inferiore.
  • block: false viene trattato come assenza di decisione.
  • params riscrive i parametri dello strumento per l’esecuzione.
  • requireApproval mette in pausa l’esecuzione dell’agente e chiede all’utente tramite approvazioni dei plugin. Il comando /approve può approvare sia approvazioni exec sia approvazioni plugin.
  • Un block: true con priorità inferiore può comunque bloccare dopo che un hook con priorità maggiore ha richiesto l’approvazione.
  • onResolution riceve la decisione di approvazione risolta - allow-once, allow-always, deny, timeout o cancelled.
I plugin inclusi che richiedono policy a livello host possono registrare policy di strumenti attendibili con 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 includere details 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.details prima del replay del provider e dell’input di Compaction, così i metadati non diventano contesto del modello.
  • Le voci di sessione persistite mantengono solo details limitati. I details troppo grandi vengono sostituiti con un riepilogo compatto e persistedDetailsTruncated: true.
  • tool_result_persist e before_message_write vengono eseguiti prima del limite finale di persistenza. Gli hook dovrebbero comunque mantenere piccoli i details restituiti ed evitare di inserire testo rilevante per il prompt solo in details; inserisci l’output dello strumento visibile al modello in content.

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. Restituisce providerOverride o modelOverride.
  • 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. Restituisce prependContext o appendContext.
  • before_prompt_build: riceve il prompt corrente e i messaggi della sessione. Restituisce prependContext, appendContext, systemPrompt, prependSystemContext o appendSystemContext.
  • heartbeat_prompt_contribution: viene eseguito solo per turni Heartbeat e restituisce prependContext o appendContext. È 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:
type BeforeAgentFinalizeRetry = {
  instruction: string;
  idempotencyKey?: string;
  maxAttempts?: number;
};
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": {
      "my-plugin": {
        "hooks": {
          "allowConversationAccess": true
        }
      }
    }
  }
}
Gli hook che modificano i prompt e le iniezioni persistenti per il turno successivo possono essere disabilitati per plugin con 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 tramite api.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: riscrivi content o restituisci { cancel: true }.
  • message_sent: osserva il successo o l’errore finale.
Per risposte TTS solo audio, 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_sending con cancel: true è terminale.
  • message_sending con cancel: false viene trattato come nessuna decisione.
  • Il content riscritto continua verso gli hook a priorità inferiore a meno che un hook successivo annulli la consegna.
  • message_sending può restituire cancelReason e metadata limitati con un annullamento. Le nuove API del ciclo di vita dei messaggi espongono questo come esito di consegna soppressa con motivo cancelled_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

Usa gateway_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_claim e message_received. Leggi BodyForAgent e i blocchi strutturati del contesto utente invece di analizzare testo envelope piatto. Vedi Envelope di canale in testo semplice → BodyForAgent.
  • before_agent_start resta per compatibilità. I nuovi plugins dovrebbero usare before_model_resolve e before_prompt_build invece della fase combinata.
  • onResolution in before_tool_call ora usa l’unione tipizzata PluginApprovalResolution (allow-once / allow-always / deny / timeout / cancelled) invece di una string libera.
Per l’elenco completo - registrazione della capability di memoria, profilo thinking del provider, provider di autenticazione esterni, tipi di discovery del provider, accessor del runtime dei task e rinomina command-authcommand-status - vedi Migrazione del Plugin SDK → Deprecazioni attive.

Correlati