Tools
Differenze
diffs è uno strumento Plugin opzionale con una breve guida di sistema integrata e una Skill complementare che trasforma il contenuto delle modifiche in un artefatto diff in sola lettura per gli agenti.
Accetta uno tra:
- testo
beforeeafter - una
patchunificata
Può restituire:
- un URL del viewer del Gateway per la presentazione su canvas
- un percorso di file renderizzato (PNG o PDF) per la consegna nei messaggi
- entrambi gli output in una singola chiamata
Quando è abilitato, il Plugin antepone una guida d’uso concisa nello spazio del prompt di sistema ed espone anche una Skill dettagliata per i casi in cui l’agente ha bisogno di istruzioni più complete.
Avvio rapido
Install the plugin
openclaw plugins install diffs
Enable the plugin
{
plugins: {
entries: {
diffs: {
enabled: true,
},
},
},
}
Pick a mode
view
Flussi orientati al canvas: gli agenti chiamano diffs con mode: "view" e aprono details.viewerUrl con canvas present.
file
Consegna di file in chat: gli agenti chiamano diffs con mode: "file" e inviano details.filePath con message usando path o filePath.
both
Combinato: gli agenti chiamano diffs con mode: "both" per ottenere entrambi gli artefatti in una singola chiamata.
Disabilitare la guida di sistema integrata
Se vuoi mantenere abilitato lo strumento diffs ma disabilitare la sua guida integrata per il prompt di sistema, imposta plugins.entries.diffs.hooks.allowPromptInjection su false:
{
plugins: {
entries: {
diffs: {
enabled: true,
hooks: {
allowPromptInjection: false,
},
},
},
},
}
Questo blocca l’hook before_prompt_build del Plugin diffs mantenendo disponibili il Plugin, lo strumento e la Skill complementare.
Se vuoi disabilitare sia la guida sia lo strumento, disabilita invece il Plugin.
Flusso di lavoro tipico dell’agente
Call diffs
L’agente chiama lo strumento diffs con l’input.
Read details
L’agente legge i campi details dalla risposta.
Present
L’agente apre details.viewerUrl con canvas present, invia details.filePath con message usando path o filePath, oppure fa entrambe le cose.
Esempi di input
Before and after
{
"before": "# Hello\n\nOne",
"after": "# Hello\n\nTwo",
"path": "docs/example.md",
"mode": "view"
}
Patch
{
"patch": "diff --git a/src/example.ts b/src/example.ts\n--- a/src/example.ts\n+++ b/src/example.ts\n@@ -1 +1 @@\n-const x = 1;\n+const x = 2;\n",
"mode": "both"
}
Riferimento dell’input dello strumento
Tutti i campi sono opzionali salvo diversa indicazione.
beforestringTesto originale. Richiesto con after quando patch è omesso.
afterstringTesto aggiornato. Richiesto con before quando patch è omesso.
patchstringTesto diff unificato. Mutuamente esclusivo con before e after.
pathstringNome file da visualizzare per la modalità before and after.
langstringSuggerimento di override della lingua per la modalità before and after. I valori sconosciuti ripiegano su testo semplice.
titlestringOverride del titolo del viewer.
mode"view" | "file" | "both"Modalità di output. Il valore predefinito è il default del Plugin defaults.mode. Alias deprecato: "image" si comporta come "file" ed è ancora accettato per compatibilità all’indietro.
theme"light" | "dark"Tema del viewer. Il valore predefinito è il default del Plugin defaults.theme.
layout"unified" | "split"Layout del diff. Il valore predefinito è il default del Plugin defaults.layout.
expandUnchangedbooleanEspande le sezioni non modificate quando il contesto completo è disponibile. Opzione solo per singola chiamata (non una chiave predefinita del Plugin).
fileFormat"png" | "pdf"Formato del file renderizzato. Il valore predefinito è il default del Plugin defaults.fileFormat.
fileQuality"standard" | "hq" | "print"Preset di qualità per il rendering PNG o PDF.
fileScalenumberOverride della scala del dispositivo (1-4).
fileMaxWidthnumberLarghezza massima di rendering in pixel CSS (640-2400).
ttlSecondsnumberTTL dell’artefatto in secondi per gli output viewer e file autonomo. Massimo 21600.
baseUrlstringOverride dell’origine dell’URL del viewer. Sovrascrive viewerBaseUrl del Plugin. Deve essere http o https, senza query/hash.
Legacy input aliases
Ancora accettati per compatibilità all’indietro:
format->fileFormatimageFormat->fileFormatimageQuality->fileQualityimageScale->fileScaleimageMaxWidth->fileMaxWidth
Validation and limits
beforeeaftermassimo 512 KiB ciascuno.patchmassimo 2 MiB.pathmassimo 2048 byte.langmassimo 128 byte.titlemassimo 1024 byte.- Limite di complessità della patch: massimo 128 file e 120000 righe totali.
patchinsieme abeforeoafterviene rifiutato.- Limiti di sicurezza per il file renderizzato (si applicano a PNG e PDF):
fileQuality: "standard": massimo 8 MP (8.000.000 pixel renderizzati).fileQuality: "hq": massimo 14 MP (14.000.000 pixel renderizzati).fileQuality: "print": massimo 24 MP (24.000.000 pixel renderizzati).- Anche PDF ha un massimo di 50 pagine.
Contratto dei dettagli di output
Lo strumento restituisce metadati strutturati sotto details.
Viewer fields
Campi condivisi per le modalità che creano un viewer:
artifactIdviewerUrlviewerPathtitleexpiresAtinputKindfileCountmodecontext(agentId,sessionId,messageChannel,agentAccountIdquando disponibile)
File fields
Campi file quando viene renderizzato PNG o PDF:
artifactIdexpiresAtfilePathpath(stesso valore difilePath, per compatibilità con lo strumento messaggi)fileBytesfileFormatfileQualityfileScalefileMaxWidth
Compatibility aliases
Restituiti anche per i chiamanti esistenti:
format(stesso valore difileFormat)imagePath(stesso valore difilePath)imageBytes(stesso valore difileBytes)imageQuality(stesso valore difileQuality)imageScale(stesso valore difileScale)imageMaxWidth(stesso valore difileMaxWidth)
Riepilogo del comportamento delle modalità:
| Modalità | Cosa viene restituito |
|---|---|
"view" |
Solo i campi viewer. |
"file" |
Solo i campi file, nessun artefatto viewer. |
"both" |
Campi viewer più campi file. Se il rendering del file non riesce, il viewer viene comunque restituito con fileError e l’alias imageError. |
Sezioni non modificate compresse
- Il viewer può mostrare righe come
N unmodified lines. - I controlli di espansione su quelle righe sono condizionali e non garantiti per ogni tipo di input.
- I controlli di espansione appaiono quando il diff renderizzato contiene dati di contesto espandibili, cosa tipica per l’input before and after.
- Per molti input di patch unificate, i corpi del contesto omessi non sono disponibili negli hunk della patch analizzata, quindi la riga può apparire senza controlli di espansione. Questo è il comportamento previsto.
expandUnchangedsi applica solo quando esiste contesto espandibile.
Default del Plugin
Imposta i default a livello di Plugin in ~/.openclaw/openclaw.json:
{
plugins: {
entries: {
diffs: {
enabled: true,
config: {
defaults: {
fontFamily: "Fira Code",
fontSize: 15,
lineSpacing: 1.6,
layout: "unified",
showLineNumbers: true,
diffIndicators: "bars",
wordWrap: true,
background: true,
theme: "dark",
fileFormat: "png",
fileQuality: "standard",
fileScale: 2,
fileMaxWidth: 960,
mode: "both",
},
},
},
},
},
}
Default supportati:
fontFamilyfontSizelineSpacinglayoutshowLineNumbersdiffIndicatorswordWrapbackgroundthemefileFormatfileQualityfileScalefileMaxWidthmode
I parametri espliciti dello strumento sovrascrivono questi default.
Configurazione persistente dell’URL del viewer
viewerBaseUrlstringFallback di proprietà del Plugin per i link viewer restituiti quando una chiamata dello strumento non passa baseUrl. Deve essere http o https, senza query/hash.
{
plugins: {
entries: {
diffs: {
enabled: true,
config: {
viewerBaseUrl: "https://gateway.example.com/openclaw",
},
},
},
},
}
Configurazione di sicurezza
security.allowRemoteViewerbooleanfalse: le richieste non-loopback alle route del viewer vengono negate. true: i viewer remoti sono consentiti se il percorso tokenizzato è valido.
{
plugins: {
entries: {
diffs: {
enabled: true,
config: {
security: {
allowRemoteViewer: false,
},
},
},
},
},
}
Ciclo di vita e archiviazione degli artefatti
- Gli artefatti sono archiviati nella sottocartella temporanea:
$TMPDIR/openclaw-diffs. - I metadati dell’artefatto viewer contengono:
- ID artefatto casuale (20 caratteri esadecimali)
- token casuale (48 caratteri esadecimali)
createdAteexpiresAt- percorso
viewer.htmlarchiviato
- Il TTL predefinito dell’artefatto è 30 minuti quando non specificato.
- Il TTL massimo accettato per il viewer è 6 ore.
- La pulizia viene eseguita opportunisticamente dopo la creazione dell’artefatto.
- Gli artefatti scaduti vengono eliminati.
- La pulizia di fallback rimuove le cartelle obsolete più vecchie di 24 ore quando i metadati sono mancanti.
URL del viewer e comportamento di rete
Route del viewer:
/plugins/diffs/view/{artifactId}/{token}
Asset del viewer:
/plugins/diffs/assets/viewer.js/plugins/diffs/assets/viewer-runtime.js
Il documento viewer risolve questi asset in modo relativo all’URL del viewer, quindi un prefisso di percorso baseUrl opzionale viene preservato anche per entrambe le richieste degli asset.
Comportamento di costruzione dell’URL:
- Se viene fornito
baseUrlnella chiamata dello strumento, viene usato dopo una validazione rigorosa. - Altrimenti, se
viewerBaseUrldel Plugin è configurato, viene usato. - Senza nessuno dei due override, l’URL del viewer usa per default il loopback
127.0.0.1. - Se la modalità di bind del Gateway è
customegateway.customBindHostè impostato, viene usato quell’host.
Regole di baseUrl:
- Deve essere
http://ohttps://. - Query e hash vengono rifiutati.
- È consentita l’origine più un percorso base opzionale.
Modello di sicurezza
Rafforzamento del visualizzatore
- Solo loopback per impostazione predefinita.
- Percorsi del visualizzatore tokenizzati con convalida rigorosa di ID e token.
- CSP della risposta del visualizzatore:
default-src 'none'- script e asset solo dalla stessa origine
- nessun
connect-srcin uscita
- Limitazione dei miss remoti quando l'accesso remoto è abilitato:
- 40 errori ogni 60 secondi
- blocco di 60 secondi (
429 Too Many Requests)
Rafforzamento del rendering dei file
- Il routing delle richieste del browser per gli screenshot è negato per impostazione predefinita.
- Sono consentiti solo gli asset locali del visualizzatore da
http://127.0.0.1/plugins/diffs/assets/*. - Le richieste di rete esterne sono bloccate.
Requisiti del browser per la modalità file
mode: "file" e mode: "both" richiedono un browser compatibile con Chromium.
Ordine di risoluzione:
Configurazione
browser.executablePath nella configurazione di OpenClaw.
Variabili d'ambiente
OPENCLAW_BROWSER_EXECUTABLE_PATHBROWSER_EXECUTABLE_PATHPLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH
Fallback della piattaforma
Fallback di rilevamento dei comandi/percorsi della piattaforma.
Testo di errore comune:
Diff PNG/PDF rendering requires a Chromium-compatible browser...
Risolvi installando Chrome, Chromium, Edge o Brave, oppure impostando una delle opzioni del percorso dell'eseguibile indicate sopra.
Risoluzione dei problemi
Errori di convalida dell'input
Provide patch or both before and after text.— includi siabeforesiaafter, oppure forniscipatch.Provide either patch or before/after input, not both.— non mescolare le modalità di input.Invalid baseUrl: ...— usa un'originehttp(s)con percorso facoltativo, senza query/hash.{field} exceeds maximum size (...)— riduci la dimensione del payload.- Rifiuto di patch di grandi dimensioni — riduci il numero di file della patch o le righe totali.
Accessibilità del visualizzatore
- L'URL del visualizzatore viene risolto in
127.0.0.1per impostazione predefinita. - Per scenari di accesso remoto:
- imposta
viewerBaseUrldel plugin, oppure - passa
baseUrlper ogni chiamata dello strumento, oppure - usa
gateway.bind=customegateway.customBindHost
- imposta
- Se
gateway.trustedProxiesinclude loopback per un proxy sullo stesso host (ad esempio Tailscale Serve), le richieste raw al visualizzatore loopback senza intestazioni client-IP inoltrate falliscono in modo chiuso per progettazione. - Per quella topologia di proxy:
- preferisci
mode: "file"omode: "both"quando ti serve solo un allegato, oppure - abilita intenzionalmente
security.allowRemoteViewere impostaviewerBaseUrldel plugin o passa unbaseUrlproxy/pubblico quando ti serve un URL del visualizzatore condivisibile
- preferisci
- Abilita
security.allowRemoteViewersolo quando intendi consentire l'accesso esterno al visualizzatore.
La riga delle linee non modificate non ha pulsante di espansione
Questo può accadere per l'input patch quando la patch non contiene contesto espandibile. È previsto e non indica un errore del visualizzatore.
Artefatto non trovato
- Artefatto scaduto a causa del TTL.
- Token o percorso modificato.
- La pulizia ha rimosso dati obsoleti.
Indicazioni operative
- Preferisci
mode: "view"per revisioni interattive locali nel canvas. - Preferisci
mode: "file"per canali di chat in uscita che richiedono un allegato. - Mantieni
allowRemoteViewerdisabilitato a meno che la tua distribuzione non richieda URL remoti del visualizzatore. - Imposta
ttlSecondsbrevi ed espliciti per diff sensibili. - Evita di inviare segreti nell'input del diff quando non è necessario.
- Se il tuo canale comprime le immagini in modo aggressivo (ad esempio Telegram o WhatsApp), preferisci l'output PDF (
fileFormat: "pdf").