Diferencias
diffs es una herramienta opcional de plugin con una guía breve integrada en el sistema y una skill complementaria que convierte contenido de cambios en un artefacto de diferencias de solo lectura para agentes.
Acepta una de estas opciones:
- texto
beforeyafter - un
patchunificado
- una URL de visor del gateway para presentación en canvas
- una ruta de archivo renderizado (PNG o PDF) para entrega por mensaje
- ambas salidas en una sola llamada
Inicio rápido
- Habilita el plugin.
- Llama a
diffsconmode: "view"para flujos centrados primero en canvas. - Llama a
diffsconmode: "file"para flujos de entrega de archivos en chat. - Llama a
diffsconmode: "both"cuando necesites ambos artefactos.
Habilitar el plugin
Deshabilitar la guía integrada del sistema
Si quieres mantener habilitada la herramientadiffs pero deshabilitar su guía integrada del system prompt, establece plugins.entries.diffs.hooks.allowPromptInjection en false:
before_prompt_build del plugin diffs mientras mantiene disponibles el plugin, la herramienta y la skill complementaria.
Si quieres deshabilitar tanto la guía como la herramienta, deshabilita el plugin en su lugar.
Flujo de trabajo típico del agente
- El agente llama a
diffs. - El agente lee los campos
details. - El agente:
- abre
details.viewerUrlconcanvas present - envía
details.filePathconmessageusandopathofilePath - hace ambas cosas
- abre
Ejemplos de entrada
Antes y después:Referencia de entrada de la herramienta
Todos los campos son opcionales salvo que se indique lo contrario:before(string): texto original. Obligatorio junto conaftercuando se omitepatch.after(string): texto actualizado. Obligatorio junto conbeforecuando se omitepatch.patch(string): texto de diff unificado. Mutuamente excluyente conbeforeyafter.path(string): nombre de archivo para mostrar en el modo antes y después.lang(string): sugerencia de anulación del idioma para el modo antes y después. Los valores desconocidos vuelven a texto sin formato.title(string): anulación del título del visor.mode("view" | "file" | "both"): modo de salida. Usa de forma predeterminada el valor del plugindefaults.mode. Alias obsoleto:"image"se comporta como"file"y sigue aceptándose por compatibilidad con versiones anteriores.theme("light" | "dark"): tema del visor. Usa de forma predeterminada el valor del plugindefaults.theme.layout("unified" | "split"): disposición del diff. Usa de forma predeterminada el valor del plugindefaults.layout.expandUnchanged(boolean): expande las secciones sin cambios cuando hay contexto completo disponible. Opción solo por llamada (no es una clave predeterminada del plugin).fileFormat("png" | "pdf"): formato del archivo renderizado. Usa de forma predeterminada el valor del plugindefaults.fileFormat.fileQuality("standard" | "hq" | "print"): ajuste de calidad para renderizado PNG o PDF.fileScale(number): anulación de escala del dispositivo (1-4).fileMaxWidth(number): ancho máximo de renderizado en píxeles CSS (640-2400).ttlSeconds(number): TTL del artefacto en segundos para el visor y las salidas de archivo independientes. Predeterminado 1800, máximo 21600.baseUrl(string): anulación del origen de la URL del visor. Anula elviewerBaseUrldel plugin. Debe serhttpohttps, sin query/hash.
format->fileFormatimageFormat->fileFormatimageQuality->fileQualityimageScale->fileScaleimageMaxWidth->fileMaxWidth
beforeyaftertienen un máximo de 512 KiB cada uno.patchtiene un máximo de 2 MiB.pathtiene un máximo de 2048 bytes.langtiene un máximo de 128 bytes.titletiene un máximo de 1024 bytes.- Límite de complejidad del patch: máximo 128 archivos y 120000 líneas totales.
- Se rechaza usar
patchjunto conbeforeoafter. - Límites de seguridad del archivo renderizado (se aplican a PNG y PDF):
fileQuality: "standard": máximo 8 MP (8,000,000 píxeles renderizados).fileQuality: "hq": máximo 14 MP (14,000,000 píxeles renderizados).fileQuality: "print": máximo 24 MP (24,000,000 píxeles renderizados).- PDF también tiene un máximo de 50 páginas.
Contrato de detalles de salida
La herramienta devuelve metadatos estructurados endetails.
Campos compartidos para los modos que crean un visor:
artifactIdviewerUrlviewerPathtitleexpiresAtinputKindfileCountmodecontext(agentId,sessionId,messageChannel,agentAccountIdcuando están disponibles)
artifactIdexpiresAtfilePathpath(mismo valor quefilePath, para compatibilidad con la herramienta de mensajes)fileBytesfileFormatfileQualityfileScalefileMaxWidth
format(mismo valor quefileFormat)imagePath(mismo valor quefilePath)imageBytes(mismo valor quefileBytes)imageQuality(mismo valor quefileQuality)imageScale(mismo valor quefileScale)imageMaxWidth(mismo valor quefileMaxWidth)
mode: "view": solo campos del visor.mode: "file": solo campos de archivo, sin artefacto de visor.mode: "both": campos del visor más campos de archivo. Si falla el renderizado del archivo, el visor igualmente se devuelve confileErrory el alias de compatibilidadimageError.
Secciones sin cambios colapsadas
- El visor puede mostrar filas como
N unmodified lines. - Los controles de expansión en esas filas son condicionales y no están garantizados para cada tipo de entrada.
- Los controles de expansión aparecen cuando el diff renderizado tiene datos de contexto expandibles, algo típico en entradas antes y después.
- Para muchas entradas de patch unificado, los cuerpos de contexto omitidos no están disponibles en los bloques parseados del patch, por lo que la fila puede aparecer sin controles de expansión. Este es el comportamiento esperado.
expandUnchangedse aplica solo cuando existe contexto expandible.
Valores predeterminados del plugin
Establece valores predeterminados globales del plugin en~/.openclaw/openclaw.json:
fontFamilyfontSizelineSpacinglayoutshowLineNumbersdiffIndicatorswordWrapbackgroundthemefileFormatfileQualityfileScalefileMaxWidthmode
viewerBaseUrl(string, opcional)- Valor de respaldo propiedad del plugin para los enlaces del visor devueltos cuando una llamada a la herramienta no pasa
baseUrl. - Debe ser
httpohttps, sin query/hash.
- Valor de respaldo propiedad del plugin para los enlaces del visor devueltos cuando una llamada a la herramienta no pasa
Configuración de seguridad
security.allowRemoteViewer(boolean, predeterminadofalse)false: se deniegan las solicitudes que no son loopback a las rutas del visor.true: se permiten visores remotos si la ruta tokenizada es válida.
Ciclo de vida y almacenamiento de artefactos
- Los artefactos se almacenan en la subcarpeta temporal:
$TMPDIR/openclaw-diffs. - Los metadatos del artefacto del visor contienen:
- ID de artefacto aleatorio (20 caracteres hexadecimales)
- token aleatorio (48 caracteres hexadecimales)
createdAtyexpiresAt- ruta
viewer.htmlalmacenada
- El TTL predeterminado del artefacto es de 30 minutos cuando no se especifica.
- El TTL máximo aceptado del visor es de 6 horas.
- La limpieza se ejecuta de forma oportunista después de crear el artefacto.
- Los artefactos vencidos se eliminan.
- La limpieza de respaldo elimina carpetas obsoletas de más de 24 horas cuando faltan los metadatos.
URL del visor y comportamiento de red
Ruta del visor:/plugins/diffs/view/{artifactId}/{token}
/plugins/diffs/assets/viewer.js/plugins/diffs/assets/viewer-runtime.js
baseUrl también se conserva para ambas solicitudes de recursos.
Comportamiento de construcción de URL:
- Si se proporciona
baseUrlen la llamada a la herramienta, se usa después de una validación estricta. - En caso contrario, si el plugin tiene configurado
viewerBaseUrl, se usa ese valor. - Sin ninguna de esas anulaciones, la URL del visor usa de forma predeterminada loopback
127.0.0.1. - Si el modo de enlace del gateway es
customygateway.customBindHostestá configurado, se usa ese host.
baseUrl:
- Debe ser
http://ohttps://. - Se rechazan query y hash.
- Se permite el origen más una ruta base opcional.
Modelo de seguridad
Fortalecimiento del visor:- Solo loopback de forma predeterminada.
- Rutas de visor tokenizadas con validación estricta de ID y token.
- CSP de respuesta del visor:
default-src 'none'- scripts y recursos solo desde self
- sin
connect-srcsaliente
- Limitación de fallos remotos cuando el acceso remoto está habilitado:
- 40 fallos por 60 segundos
- bloqueo de 60 segundos (
429 Too Many Requests)
- El enrutamiento de solicitudes del navegador de capturas está denegado por defecto.
- Solo se permiten recursos locales del visor desde
http://127.0.0.1/plugins/diffs/assets/*. - Se bloquean las solicitudes de red externas.
Requisitos del navegador para el modo de archivo
mode: "file" y mode: "both" necesitan un navegador compatible con Chromium.
Orden de resolución:
browser.executablePathen la configuración de OpenClaw.- Variables de entorno:
OPENCLAW_BROWSER_EXECUTABLE_PATHBROWSER_EXECUTABLE_PATHPLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH
- Respaldo de detección de comandos/rutas de la plataforma.
Diff PNG/PDF rendering requires a Chromium-compatible browser...
Solución de problemas
Errores de validación de entrada:Provide patch or both before and after text.- Incluye
beforeyafter, o proporcionapatch.
- Incluye
Provide either patch or before/after input, not both.- No mezcles modos de entrada.
Invalid baseUrl: ...- Usa un origen
http(s)con ruta opcional, sin query/hash.
- Usa un origen
{field} exceeds maximum size (...)- Reduce el tamaño de la carga útil.
- Rechazo de patch grande
- Reduce el número de archivos del patch o el total de líneas.
- La URL del visor se resuelve a
127.0.0.1de forma predeterminada. - Para escenarios de acceso remoto:
- configura
viewerBaseUrlen el plugin, o - pasa
baseUrlpor llamada a la herramienta, o - usa
gateway.bind=customygateway.customBindHost
- configura
- Si
gateway.trustedProxiesincluye loopback para un proxy del mismo host (por ejemplo Tailscale Serve), las solicitudes sin procesar del visor en loopback sin encabezados de IP de cliente reenviada fallan de forma cerrada por diseño. - Para esa topología de proxy:
- prefiere
mode: "file"omode: "both"cuando solo necesites un adjunto, o - habilita intencionalmente
security.allowRemoteViewery configuraviewerBaseUrlen el plugin o pasa unbaseUrlde proxy/público cuando necesites una URL de visor compartible
- prefiere
- Habilita
security.allowRemoteViewersolo cuando realmente quieras acceso externo al visor.
- Esto puede ocurrir con entrada de patch cuando el patch no contiene contexto expandible.
- Esto es esperable y no indica un fallo del visor.
- El artefacto venció por TTL.
- El token o la ruta cambiaron.
- La limpieza eliminó datos obsoletos.
Guía operativa
- Prefiere
mode: "view"para revisiones interactivas locales en canvas. - Prefiere
mode: "file"para canales de chat salientes que necesitan un adjunto. - Mantén
allowRemoteViewerdeshabilitado a menos que tu implementación necesite URL remotas del visor. - Establece un
ttlSecondscorto y explícito para diferencias sensibles. - Evita enviar secretos en la entrada del diff cuando no sea necesario.
- Si tu canal comprime imágenes de forma agresiva (por ejemplo Telegram o WhatsApp), prefiere salida PDF (
fileFormat: "pdf").
- Impulsado por Diffs.