Diffs
diffs est un outil de plugin optionnel avec de courtes consignes système intégrées et une skill compagnon qui transforme le contenu des modifications en artefact de diff en lecture seule pour les agents.
Il accepte soit :
- le texte
beforeetafter - un
patchunifié
- une URL de visionneuse Gateway pour une présentation dans le canvas
- un chemin de fichier rendu (PNG ou PDF) pour la distribution par message
- les deux sorties en un seul appel
Démarrage rapide
- Activez le plugin.
- Appelez
diffsavecmode: "view"pour les flux orientés canvas en priorité. - Appelez
diffsavecmode: "file"pour les flux de distribution de fichiers dans le chat. - Appelez
diffsavecmode: "both"lorsque vous avez besoin des deux artefacts.
Activer le plugin
Désactiver les consignes système intégrées
Si vous souhaitez conserver l’outildiffs activé tout en désactivant ses consignes intégrées dans le prompt système, définissez plugins.entries.diffs.hooks.allowPromptInjection sur false :
before_prompt_build du plugin diffs tout en gardant le plugin, l’outil et la skill compagnon disponibles.
Si vous souhaitez désactiver à la fois les consignes et l’outil, désactivez plutôt le plugin.
Workflow typique d’un agent
- L’agent appelle
diffs. - L’agent lit les champs
details. - L’agent :
- ouvre
details.viewerUrlaveccanvas present - envoie
details.filePathavecmessageen utilisantpathoufilePath - fait les deux
- ouvre
Exemples d’entrée
Before et after :Référence des entrées de l’outil
Tous les champs sont facultatifs sauf indication contraire :before(string) : texte d’origine. Obligatoire avecafterlorsquepatchest omis.after(string) : texte mis à jour. Obligatoire avecbeforelorsquepatchest omis.patch(string) : texte de diff unifié. Mutuellement exclusif avecbeforeetafter.path(string) : nom de fichier affiché pour le mode before/after.lang(string) : indice de remplacement de langue pour le mode before/after. Les valeurs inconnues reviennent au texte brut.title(string) : remplacement du titre de la visionneuse.mode("view" | "file" | "both") : mode de sortie. Par défaut : la valeur du plugindefaults.mode. Alias obsolète :"image"se comporte comme"file"et reste accepté pour compatibilité descendante.theme("light" | "dark") : thème de la visionneuse. Par défaut : la valeur du plugindefaults.theme.layout("unified" | "split") : disposition du diff. Par défaut : la valeur du plugindefaults.layout.expandUnchanged(boolean) : développe les sections inchangées lorsque le contexte complet est disponible. Option par appel uniquement (pas une clé de valeur par défaut du plugin).fileFormat("png" | "pdf") : format du fichier rendu. Par défaut : la valeur du plugindefaults.fileFormat.fileQuality("standard" | "hq" | "print") : préréglage de qualité pour le rendu PNG ou PDF.fileScale(number) : remplacement de l’échelle du périphérique (1-4).fileMaxWidth(number) : largeur maximale de rendu en pixels CSS (640-2400).ttlSeconds(number) : TTL de l’artefact en secondes pour la visionneuse et les sorties de fichier autonomes. Valeur par défaut : 1800, maximum : 21600.baseUrl(string) : remplacement de l’origine d’URL de la visionneuse. Remplace la valeur du pluginviewerBaseUrl. Doit être enhttpouhttps, sans query/hash.
format->fileFormatimageFormat->fileFormatimageQuality->fileQualityimageScale->fileScaleimageMaxWidth->fileMaxWidth
beforeetafter: 512 Kio maximum chacun.patch: 2 Mio maximum.path: 2048 octets maximum.lang: 128 octets maximum.title: 1024 octets maximum.- Limite de complexité du patch : 128 fichiers maximum et 120000 lignes au total.
patchavecbeforeouafterensemble est rejeté.- Limites de sécurité pour les fichiers rendus (s’appliquent au PNG et au PDF) :
fileQuality: "standard": 8 MP maximum (8 000 000 pixels rendus).fileQuality: "hq": 14 MP maximum (14 000 000 pixels rendus).fileQuality: "print": 24 MP maximum (24 000 000 pixels rendus).- Le PDF a également une limite maximale de 50 pages.
Contrat des détails de sortie
L’outil renvoie des métadonnées structurées sousdetails.
Champs partagés pour les modes qui créent une visionneuse :
artifactIdviewerUrlviewerPathtitleexpiresAtinputKindfileCountmodecontext(agentId,sessionId,messageChannel,agentAccountIdlorsqu’ils sont disponibles)
artifactIdexpiresAtfilePathpath(même valeur quefilePath, pour compatibilité avec l’outil de message)fileBytesfileFormatfileQualityfileScalefileMaxWidth
format(même valeur quefileFormat)imagePath(même valeur quefilePath)imageBytes(même valeur quefileBytes)imageQuality(même valeur quefileQuality)imageScale(même valeur quefileScale)imageMaxWidth(même valeur quefileMaxWidth)
mode: "view": champs de visionneuse uniquement.mode: "file": champs de fichier uniquement, aucun artefact de visionneuse.mode: "both": champs de visionneuse plus champs de fichier. Si le rendu du fichier échoue, la visionneuse est quand même renvoyée avecfileErroret l’alias de compatibilitéimageError.
Sections inchangées repliées
- La visionneuse peut afficher des lignes comme
N unmodified lines. - Les contrôles de développement sur ces lignes sont conditionnels et non garantis pour chaque type d’entrée.
- Les contrôles de développement apparaissent lorsque le diff rendu contient des données de contexte développables, ce qui est typique pour une entrée before/after.
- Pour de nombreuses entrées de patch unifié, les corps de contexte omis ne sont pas disponibles dans les hunks de patch analysés ; la ligne peut donc apparaître sans contrôles de développement. C’est un comportement attendu.
expandUnchangeds’applique uniquement lorsqu’un contexte développable existe.
Valeurs par défaut du plugin
Définissez des valeurs par défaut à l’échelle du plugin dans~/.openclaw/openclaw.json :
fontFamilyfontSizelineSpacinglayoutshowLineNumbersdiffIndicatorswordWrapbackgroundthemefileFormatfileQualityfileScalefileMaxWidthmode
viewerBaseUrl(string, facultatif)- Valeur de repli gérée par le plugin pour les liens de visionneuse renvoyés lorsqu’un appel d’outil ne transmet pas
baseUrl. - Doit être en
httpouhttps, sans query/hash.
- Valeur de repli gérée par le plugin pour les liens de visionneuse renvoyés lorsqu’un appel d’outil ne transmet pas
Configuration de sécurité
security.allowRemoteViewer(boolean, par défautfalse)false: les requêtes non loopback vers les routes de la visionneuse sont refusées.true: les visionneuses distantes sont autorisées si le chemin tokenisé est valide.
Cycle de vie et stockage des artefacts
- Les artefacts sont stockés dans le sous-dossier temporaire :
$TMPDIR/openclaw-diffs. - Les métadonnées de l’artefact de visionneuse contiennent :
- un identifiant d’artefact aléatoire (20 caractères hexadécimaux)
- un token aléatoire (48 caractères hexadécimaux)
createdAtetexpiresAt- le chemin
viewer.htmlstocké
- Le TTL par défaut de l’artefact est de 30 minutes lorsqu’il n’est pas spécifié.
- Le TTL maximal accepté pour la visionneuse est de 6 heures.
- Le nettoyage s’exécute de manière opportuniste après la création de l’artefact.
- Les artefacts expirés sont supprimés.
- Le nettoyage de repli supprime les dossiers obsolètes de plus de 24 heures lorsque les métadonnées sont absentes.
Comportement réseau et URL de la visionneuse
Route de la visionneuse :/plugins/diffs/view/{artifactId}/{token}
/plugins/diffs/assets/viewer.js/plugins/diffs/assets/viewer-runtime.js
baseUrl facultatif est donc également conservé pour ces requêtes de ressources.
Comportement de construction de l’URL :
- Si
baseUrlest fourni dans l’appel d’outil, il est utilisé après validation stricte. - Sinon, si la valeur du plugin
viewerBaseUrlest configurée, elle est utilisée. - Sans l’un ni l’autre remplacement, l’URL de la visionneuse utilise par défaut la loopback
127.0.0.1. - Si le mode de liaison de la gateway est
customet quegateway.customBindHostest défini, cet hôte est utilisé.
baseUrl :
- Doit commencer par
http://ouhttps://. - Query et hash sont rejetés.
- L’origine plus un chemin de base facultatif sont autorisés.
Modèle de sécurité
Renforcement de la visionneuse :- Loopback uniquement par défaut.
- Chemins de visionneuse tokenisés avec validation stricte de l’identifiant et du token.
- CSP de la réponse de la visionneuse :
default-src 'none'- scripts et ressources uniquement depuis self
- aucun
connect-srcsortant
- Limitation des échecs distants lorsque l’accès distant est activé :
- 40 échecs par 60 secondes
- verrouillage de 60 secondes (
429 Too Many Requests)
- Le routage des requêtes navigateur pour capture d’écran est refusé par défaut.
- Seules les ressources locales de la visionneuse depuis
http://127.0.0.1/plugins/diffs/assets/*sont autorisées. - Les requêtes réseau externes sont bloquées.
Exigences navigateur pour le mode fichier
mode: "file" et mode: "both" nécessitent un navigateur compatible Chromium.
Ordre de résolution :
browser.executablePathdans la configuration OpenClaw.- Variables d’environnement :
OPENCLAW_BROWSER_EXECUTABLE_PATHBROWSER_EXECUTABLE_PATHPLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH
- Repli vers la découverte de commande/chemin selon la plateforme.
Diff PNG/PDF rendering requires a Chromium-compatible browser...
Dépannage
Erreurs de validation d’entrée :Provide patch or both before and after text.- Incluez
beforeetafter, ou fournissezpatch.
- Incluez
Provide either patch or before/after input, not both.- Ne mélangez pas les modes d’entrée.
Invalid baseUrl: ...- Utilisez une origine
http(s)avec chemin facultatif, sans query/hash.
- Utilisez une origine
{field} exceeds maximum size (...)- Réduisez la taille de la charge utile.
- Rejet de patch volumineux
- Réduisez le nombre de fichiers du patch ou le nombre total de lignes.
- L’URL de la visionneuse pointe par défaut vers
127.0.0.1. - Pour les scénarios d’accès distant :
- définissez la valeur du plugin
viewerBaseUrl, ou - transmettez
baseUrlà chaque appel d’outil, ou - utilisez
gateway.bind=custometgateway.customBindHost
- définissez la valeur du plugin
- Si
gateway.trustedProxiesinclut loopback pour un proxy sur le même hôte (par exemple Tailscale Serve), les requêtes brutes de visionneuse loopback sans en-têtes transférés d’IP client échouent par défaut, conformément à la conception. - Pour cette topologie de proxy :
- préférez
mode: "file"oumode: "both"lorsque vous avez seulement besoin d’une pièce jointe, ou - activez intentionnellement
security.allowRemoteVieweret définissez la valeur du pluginviewerBaseUrlou transmettez unbaseUrlde proxy/public lorsque vous avez besoin d’une URL de visionneuse partageable
- préférez
- Activez
security.allowRemoteVieweruniquement si vous avez l’intention d’autoriser un accès externe à la visionneuse.
- Cela peut se produire avec une entrée patch lorsque le patch ne contient pas de contexte développable.
- C’est un comportement attendu et cela n’indique pas un échec de la visionneuse.
- L’artefact a expiré selon son TTL.
- Le token ou le chemin a été modifié.
- Le nettoyage a supprimé des données obsolètes.
Consignes opérationnelles
- Préférez
mode: "view"pour les revues interactives locales dans le canvas. - Préférez
mode: "file"pour les canaux de chat sortants qui nécessitent une pièce jointe. - Gardez
allowRemoteViewerdésactivé sauf si votre déploiement nécessite des URL de visionneuse distantes. - Définissez un
ttlSecondscourt et explicite pour les diffs sensibles. - Évitez d’envoyer des secrets dans l’entrée de diff lorsque ce n’est pas nécessaire.
- Si votre canal compresse fortement les images (par exemple Telegram ou WhatsApp), préférez la sortie PDF (
fileFormat: "pdf").
- Propulsé par Diffs.