@saulwade/swl-ses 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +238 -0
- package/README.md +560 -0
- package/_userland/agentes/.gitkeep +0 -0
- package/_userland/habilidades/.gitkeep +0 -0
- package/agentes/.evolved.json +9 -0
- package/agentes/accesibilidad-wcag-swl.md +692 -0
- package/agentes/arquitecto-swl.md +238 -0
- package/agentes/auto-evolucion-swl.md +854 -0
- package/agentes/backend-api-swl.md +470 -0
- package/agentes/backend-csharp-swl.md +418 -0
- package/agentes/backend-go-swl.md +388 -0
- package/agentes/backend-java-swl.md +279 -0
- package/agentes/backend-node-swl.md +477 -0
- package/agentes/backend-python-swl.md +608 -0
- package/agentes/backend-rust-swl.md +362 -0
- package/agentes/backend-workers-swl.md +480 -0
- package/agentes/cloud-infra-swl.md +485 -0
- package/agentes/consolidador-swl.md +539 -0
- package/agentes/datos-swl.md +584 -0
- package/agentes/depurador-swl.md +349 -0
- package/agentes/devops-ci-swl.md +374 -0
- package/agentes/disenador-ui-swl.md +558 -0
- package/agentes/documentador-swl.md +343 -0
- package/agentes/evals/arquitecto-swl.evals.json +56 -0
- package/agentes/evals/auto-evolucion-swl.evals.json +68 -0
- package/agentes/evals/implementador-swl.evals.json +56 -0
- package/agentes/evals/orquestador-swl.evals.json +60 -0
- package/agentes/evals/perfilador-usuario-swl.evals.json +60 -0
- package/agentes/evals/red-team-swl.evals.json +59 -0
- package/agentes/evals/revisor-codigo-swl.evals.json +59 -0
- package/agentes/frontend-angular-swl.md +627 -0
- package/agentes/frontend-css-swl.md +720 -0
- package/agentes/frontend-react-swl.md +696 -0
- package/agentes/frontend-swl.md +500 -0
- package/agentes/frontend-tailwind-swl.md +830 -0
- package/agentes/implementador-swl.md +328 -0
- package/agentes/investigador-swl.md +430 -0
- package/agentes/investigador-ux-swl.md +500 -0
- package/agentes/llm-apps-swl.md +276 -0
- package/agentes/migrador-swl.md +417 -0
- package/agentes/mobile-android-swl.md +509 -0
- package/agentes/mobile-cross-swl.md +539 -0
- package/agentes/mobile-ios-swl.md +500 -0
- package/agentes/mobile-testing-swl.md +300 -0
- package/agentes/notificador-swl.md +916 -0
- package/agentes/observabilidad-swl.md +436 -0
- package/agentes/orquestador-swl.md +884 -0
- package/agentes/pagos-swl.md +283 -0
- package/agentes/perfilador-usuario-swl.md +306 -0
- package/agentes/planificador-swl.md +402 -0
- package/agentes/producto-prd-swl.md +587 -0
- package/agentes/red-team-swl.md +216 -0
- package/agentes/release-manager-swl.md +568 -0
- package/agentes/rendimiento-swl.md +714 -0
- package/agentes/resolutor-build-swl.md +243 -0
- package/agentes/revisor-angular-swl.md +276 -0
- package/agentes/revisor-codigo-swl.md +348 -0
- package/agentes/revisor-csharp-swl.md +262 -0
- package/agentes/revisor-go-swl.md +257 -0
- package/agentes/revisor-java-swl.md +255 -0
- package/agentes/revisor-kotlin-swl.md +271 -0
- package/agentes/revisor-nextjs-swl.md +279 -0
- package/agentes/revisor-php-swl.md +269 -0
- package/agentes/revisor-react-swl.md +276 -0
- package/agentes/revisor-rust-swl.md +344 -0
- package/agentes/revisor-seguridad-swl.md +390 -0
- package/agentes/revisor-swift-swl.md +266 -0
- package/agentes/revisor-typescript-swl.md +344 -0
- package/agentes/sre-swl.md +265 -0
- package/agentes/tdd-qa-swl.md +354 -0
- package/agentes/ux-disenador-swl.md +501 -0
- package/bin/lib/bot-comandos.js +1030 -0
- package/bin/lib/bot-discovery.js +182 -0
- package/bin/lib/bot-git.js +142 -0
- package/bin/swl-ses.js +325 -0
- package/bin/swl-telegram-bot.js +442 -0
- package/bin/swl-telegram-bot.plist +21 -0
- package/bin/swl-telegram-bot.service +14 -0
- package/comandos/swl/.evolved.json +23 -0
- package/comandos/swl/actualizar.md +174 -0
- package/comandos/swl/adoptar-proyecto.md +207 -0
- package/comandos/swl/aprender.md +701 -0
- package/comandos/swl/auditar-deps.md +134 -0
- package/comandos/swl/autoresearch.md +170 -0
- package/comandos/swl/ayuda.md +224 -0
- package/comandos/swl/brainstorm.md +50 -0
- package/comandos/swl/checkpoint.md +330 -0
- package/comandos/swl/compactar.md +283 -0
- package/comandos/swl/configurar-ci.md +227 -0
- package/comandos/swl/contexto.md +112 -0
- package/comandos/swl/contribuir.md +233 -0
- package/comandos/swl/crear-skill.md +292 -0
- package/comandos/swl/cron.md +196 -0
- package/comandos/swl/dashboard.md +146 -0
- package/comandos/swl/discutir-fase.md +230 -0
- package/comandos/swl/ejecutar-fase.md +135 -0
- package/comandos/swl/evaluar-skill.md +487 -0
- package/comandos/swl/evolucion-estado.md +142 -0
- package/comandos/swl/evolucionar.md +259 -0
- package/comandos/swl/exportar-vault.md +189 -0
- package/comandos/swl/gateway.md +158 -0
- package/comandos/swl/inbox.md +116 -0
- package/comandos/swl/instalar.md +220 -0
- package/comandos/swl/instintos.md +86 -0
- package/comandos/swl/mapear-codebase.md +312 -0
- package/comandos/swl/mcp-status.md +175 -0
- package/comandos/swl/metricas.md +270 -0
- package/comandos/swl/modelo.md +102 -0
- package/comandos/swl/notificaciones.md +396 -0
- package/comandos/swl/nuevo-proyecto.md +154 -0
- package/comandos/swl/planear-fase.md +221 -0
- package/comandos/swl/plugins.md +256 -0
- package/comandos/swl/reflect-skills.md +125 -0
- package/comandos/swl/release.md +217 -0
- package/comandos/swl/revisar-impacto.md +206 -0
- package/comandos/swl/revisar.md +330 -0
- package/comandos/swl/salud.md +363 -0
- package/comandos/swl/sesiones.md +200 -0
- package/comandos/swl/skill-search.md +113 -0
- package/comandos/swl/verificar.md +585 -0
- package/comandos/swl/wiki.md +620 -0
- package/contextos/dev.md +32 -0
- package/contextos/research.md +30 -0
- package/contextos/review.md +31 -0
- package/habilidades/accesibilidad-a11y/SKILL.md +201 -0
- package/habilidades/accesibilidad-a11y/evals/evals.json +56 -0
- package/habilidades/accesibilidad-a11y/recursos/ejemplos-y-checklist-completo.md +441 -0
- package/habilidades/agent-browser/SKILL.md +218 -0
- package/habilidades/agentes-como-servicio/SKILL.md +218 -0
- package/habilidades/ai-runtime-security/SKILL.md +273 -0
- package/habilidades/angular-avanzado/SKILL.md +164 -0
- package/habilidades/angular-avanzado/recursos/ejemplos-avanzados.md +219 -0
- package/habilidades/angular-moderno/SKILL.md +186 -0
- package/habilidades/angular-moderno/evals/evals.json +45 -0
- package/habilidades/angular-moderno/recursos/ejemplos-avanzados.md +106 -0
- package/habilidades/api-rest-diseno/SKILL.md +191 -0
- package/habilidades/api-rest-diseno/recursos/openapi-template.yaml +506 -0
- package/habilidades/api-rest-diseno/recursos/referencia-api.md +140 -0
- package/habilidades/aprendizaje-continuo/SKILL.md +151 -0
- package/habilidades/aprendizaje-continuo/evals/evals.json +53 -0
- package/habilidades/aprendizaje-continuo/recursos/referencia-instintos.md +290 -0
- package/habilidades/async-python/SKILL.md +149 -0
- package/habilidades/async-python/evals/evals.json +47 -0
- package/habilidades/async-python/recursos/patrones-y-ejemplos-completos.md +292 -0
- package/habilidades/auth-patrones/.evolved.json +9 -0
- package/habilidades/auth-patrones/SKILL.md +413 -0
- package/habilidades/auth-patrones/recursos/implementaciones-completas.md +229 -0
- package/habilidades/auto-evolucion-protocolo/SKILL.md +276 -0
- package/habilidades/auto-evolucion-protocolo/evals/evals.json +55 -0
- package/habilidades/auto-evolucion-protocolo/recursos/referencia-completa.md +145 -0
- package/habilidades/autoresearch/SKILL.md +268 -0
- package/habilidades/autoresearch/evals/evals.json +41 -0
- package/habilidades/autoresearch/recursos/checklist-template.md +191 -0
- package/habilidades/autoresearch/scripts/calcular-score.js +88 -0
- package/habilidades/azure-cloud/SKILL.md +308 -0
- package/habilidades/azure-cloud/recursos/aks.md +327 -0
- package/habilidades/backend-mcp-servidor/SKILL.md +270 -0
- package/habilidades/backend-production-resilience/SKILL.md +288 -0
- package/habilidades/brainstorming/SKILL.md +295 -0
- package/habilidades/brainstorming/recursos/componentes-html.md +247 -0
- package/habilidades/build-errors-cpp/SKILL.md +270 -0
- package/habilidades/build-errors-csharp/SKILL.md +265 -0
- package/habilidades/build-errors-go/SKILL.md +306 -0
- package/habilidades/build-errors-java/SKILL.md +278 -0
- package/habilidades/build-errors-kotlin/SKILL.md +303 -0
- package/habilidades/build-errors-nextjs/SKILL.md +312 -0
- package/habilidades/build-errors-php/SKILL.md +270 -0
- package/habilidades/build-errors-python/SKILL.md +292 -0
- package/habilidades/build-errors-rust/SKILL.md +284 -0
- package/habilidades/build-errors-swift/SKILL.md +272 -0
- package/habilidades/build-errors-typescript/SKILL.md +369 -0
- package/habilidades/checklist-calidad/SKILL.md +271 -0
- package/habilidades/checklist-calidad/recursos/quality-report-template.md +148 -0
- package/habilidades/checklist-seguridad/SKILL.md +285 -0
- package/habilidades/checkpoints-verificacion/SKILL.md +298 -0
- package/habilidades/checkpoints-verificacion/recursos/checkpoint-templates.md +360 -0
- package/habilidades/ci-cd-pipelines/SKILL.md +157 -0
- package/habilidades/ci-cd-pipelines/recursos/github-actions-template.yaml +403 -0
- package/habilidades/ci-cd-pipelines/recursos/pipelines-completos.md +487 -0
- package/habilidades/cloud-aws/SKILL.md +142 -0
- package/habilidades/cloud-aws/recursos/servicios-aws-referencia.md +321 -0
- package/habilidades/compactacion-contexto/SKILL.md +247 -0
- package/habilidades/contenedores-docker/SKILL.md +137 -0
- package/habilidades/contenedores-docker/recursos/dockerfile-template.dockerfile +160 -0
- package/habilidades/contenedores-docker/recursos/ejemplos-y-configuraciones.md +327 -0
- package/habilidades/context-builder/SKILL.md +170 -0
- package/habilidades/control-profundidad/SKILL.md +128 -0
- package/habilidades/csharp-experto/SKILL.md +322 -0
- package/habilidades/csharp-patrones/SKILL.md +316 -0
- package/habilidades/csharp-testing/SKILL.md +286 -0
- package/habilidades/css-moderno/SKILL.md +166 -0
- package/habilidades/css-moderno/evals/evals.json +43 -0
- package/habilidades/css-moderno/recursos/ejemplos-y-patrones-completos.md +337 -0
- package/habilidades/datos-etl/SKILL.md +129 -0
- package/habilidades/datos-etl/recursos/implementaciones-completas.md +322 -0
- package/habilidades/dbml-experto/SKILL.md +339 -0
- package/habilidades/dbml-experto/evals/evals.json +56 -0
- package/habilidades/dependencias-auditoria/SKILL.md +320 -0
- package/habilidades/deprecacion-migracion/SKILL.md +169 -0
- package/habilidades/deprecacion-migracion/recursos/implementaciones-completas.md +220 -0
- package/habilidades/design-tokens/SKILL.md +158 -0
- package/habilidades/design-tokens/recursos/tokens-y-configuracion.md +363 -0
- package/habilidades/devsecops-pipeline-security/SKILL.md +309 -0
- package/habilidades/diagrama-arquitectura/SKILL.md +165 -0
- package/habilidades/diagrama-arquitectura/assets/template.html +276 -0
- package/habilidades/discutir-fase/SKILL.md +188 -0
- package/habilidades/diseno-herramientas-agente/SKILL.md +199 -0
- package/habilidades/diseno-responsivo/SKILL.md +186 -0
- package/habilidades/diseno-responsivo/recursos/ejemplos-layouts.md +156 -0
- package/habilidades/django-experto/SKILL.md +205 -0
- package/habilidades/django-experto/recursos/async-django.md +390 -0
- package/habilidades/django-experto/recursos/drf-patrones.md +438 -0
- package/habilidades/django-experto/recursos/orm-avanzado.md +382 -0
- package/habilidades/django-experto/recursos/referencia-completa.md +188 -0
- package/habilidades/django-experto/recursos/testing-django.md +415 -0
- package/habilidades/doc-sync/SKILL.md +280 -0
- package/habilidades/drift-detection/SKILL.md +179 -0
- package/habilidades/ejecutar-fase/SKILL.md +468 -0
- package/habilidades/estilo-sin-ai-isms/SKILL.md +775 -0
- package/habilidades/estilo-sin-ai-isms/evals/evals.json +63 -0
- package/habilidades/estilo-sin-ai-isms/scripts/detectar_aiisms.py +500 -0
- package/habilidades/estructura-proyecto-claude/SKILL.md +215 -0
- package/habilidades/estructura-proyecto-claude/recursos/claude-md-template.md +261 -0
- package/habilidades/estructura-proyecto-claude/recursos/configuracion-y-extensiones.md +176 -0
- package/habilidades/estructura-proyecto-claude/recursos/frontmatter-y-hooks-referencia.md +289 -0
- package/habilidades/estructura-proyecto-claude/recursos/mcp-json-template.json +77 -0
- package/habilidades/estructura-proyecto-claude/recursos/variantes-por-stack.md +177 -0
- package/habilidades/evaluacion-agentes/SKILL.md +314 -0
- package/habilidades/event-driven/SKILL.md +153 -0
- package/habilidades/event-driven/recursos/implementaciones-completas.md +423 -0
- package/habilidades/extraccion-documentos/SKILL.md +221 -0
- package/habilidades/extractor-de-aprendizajes/.evolved.json +9 -0
- package/habilidades/extractor-de-aprendizajes/SKILL.md +311 -0
- package/habilidades/extractor-de-aprendizajes/evals/evals.json +55 -0
- package/habilidades/fastapi-experto/SKILL.md +221 -0
- package/habilidades/fastapi-experto/recursos/async-patterns.md +438 -0
- package/habilidades/fastapi-experto/recursos/dependency-injection.md +330 -0
- package/habilidades/fastapi-experto/recursos/referencia-completa.md +79 -0
- package/habilidades/fastapi-experto/recursos/testing-httpx.md +420 -0
- package/habilidades/filament-admin/SKILL.md +290 -0
- package/habilidades/frontend-avanzado/SKILL.md +257 -0
- package/habilidades/frontend-avanzado/recursos/apis-nativas-ejemplos.md +341 -0
- package/habilidades/gcp-cloud/SKILL.md +260 -0
- package/habilidades/gcp-cloud/recursos/gke.md +234 -0
- package/habilidades/gcp-cloud/recursos/terraform-gcp.md +307 -0
- package/habilidades/generacion-mermaid/SKILL.md +229 -0
- package/habilidades/git-worktrees-paralelo/SKILL.md +270 -0
- package/habilidades/go-experto/SKILL.md +305 -0
- package/habilidades/go-patrones/SKILL.md +299 -0
- package/habilidades/go-testing/SKILL.md +291 -0
- package/habilidades/graphql-experto/SKILL.md +323 -0
- package/habilidades/guardrail-semantico/SKILL.md +282 -0
- package/habilidades/harness-claude-code/SKILL.md +299 -0
- package/habilidades/iam-secretos/SKILL.md +265 -0
- package/habilidades/iam-secretos/recursos/implementaciones-completas.md +356 -0
- package/habilidades/infra-github-actions/SKILL.md +166 -0
- package/habilidades/instalar-sistema/.evolved.json +9 -0
- package/habilidades/instalar-sistema/SKILL.md +221 -0
- package/habilidades/java-experto/SKILL.md +290 -0
- package/habilidades/java-patrones/SKILL.md +275 -0
- package/habilidades/java-testing/SKILL.md +288 -0
- package/habilidades/kotlin-compose/SKILL.md +278 -0
- package/habilidades/kotlin-compose/recursos/animaciones-performance.md +93 -0
- package/habilidades/kotlin-experto/SKILL.md +318 -0
- package/habilidades/kotlin-testing/SKILL.md +267 -0
- package/habilidades/kotlin-testing/recursos/testing-avanzado.md +74 -0
- package/habilidades/kubernetes-orquestacion/SKILL.md +152 -0
- package/habilidades/kubernetes-orquestacion/recursos/manifiestos-completos.md +452 -0
- package/habilidades/langchain-langraph/SKILL.md +386 -0
- package/habilidades/langchain-langraph/recursos/evaluacion-rag.md +321 -0
- package/habilidades/langchain-langraph/recursos/rag-maturity-model.md +225 -0
- package/habilidades/langchain-langraph/recursos/vectorstores.md +306 -0
- package/habilidades/legacy-code-rescue/SKILL.md +267 -0
- package/habilidades/likec4-experto/SKILL.md +412 -0
- package/habilidades/likec4-experto/evals/evals.json +69 -0
- package/habilidades/manejo-errores/.evolved.json +9 -0
- package/habilidades/manejo-errores/SKILL.md +407 -0
- package/habilidades/manejo-errores/recursos/implementaciones-completas.md +248 -0
- package/habilidades/mapear-codebase/SKILL.md +275 -0
- package/habilidades/memoria-busqueda/SKILL.md +194 -0
- package/habilidades/memoria-busqueda/evals/evals.json +44 -0
- package/habilidades/meta-skills-estandar/SKILL.md +298 -0
- package/habilidades/meta-skills-estandar/recursos/anti-patrones-y-leyes.md +205 -0
- package/habilidades/meta-skills-estandar/recursos/frameworks-seguridad.md +107 -0
- package/habilidades/meta-skills-estandar/recursos/idiomas-framework.md +60 -0
- package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -0
- package/habilidades/microservicios/SKILL.md +155 -0
- package/habilidades/microservicios/recursos/patrones-y-ejemplos-completos.md +325 -0
- package/habilidades/mobile-flutter/SKILL.md +199 -0
- package/habilidades/mobile-flutter/recursos/ejemplos-completos.md +319 -0
- package/habilidades/mobile-react-native/SKILL.md +176 -0
- package/habilidades/mobile-react-native/recursos/ejemplos-completos.md +216 -0
- package/habilidades/mongodb-experto/SKILL.md +302 -0
- package/habilidades/monitoring-alertas/SKILL.md +201 -0
- package/habilidades/monitoring-alertas/recursos/instrumentacion-y-alertas.md +301 -0
- package/habilidades/nestjs-experto/SKILL.md +307 -0
- package/habilidades/nestjs-experto/recursos/guards-interceptors.md +339 -0
- package/habilidades/nestjs-experto/recursos/modulos-di.md +287 -0
- package/habilidades/nestjs-experto/recursos/testing-nestjs.md +354 -0
- package/habilidades/nextjs-experto/SKILL.md +335 -0
- package/habilidades/nextjs-patrones/SKILL.md +303 -0
- package/habilidades/nextjs-testing/SKILL.md +331 -0
- package/habilidades/node-experto/.evolved.json +9 -0
- package/habilidades/node-experto/SKILL.md +266 -0
- package/habilidades/node-experto/recursos/patrones-completos.md +283 -0
- package/habilidades/notificaciones-multicanal/SKILL.md +159 -0
- package/habilidades/notificaciones-multicanal/recursos/config-template.json +115 -0
- package/habilidades/notificaciones-multicanal/recursos/configuracion-y-templates.md +303 -0
- package/habilidades/nuevo-proyecto/SKILL.md +204 -0
- package/habilidades/orquestacion-async/SKILL.md +303 -0
- package/habilidades/paid-media-tracking/SKILL.md +269 -0
- package/habilidades/paid-media-tracking/recursos/auditoria-tracking.md +220 -0
- package/habilidades/paid-media-tracking/recursos/google-ads-api.md +215 -0
- package/habilidades/patrones-python/SKILL.md +228 -0
- package/habilidades/patrones-python/evals/evals.json +56 -0
- package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -0
- package/habilidades/patrones-python/recursos/referencia-completa.md +202 -0
- package/habilidades/perfil-usuario/SKILL.md +200 -0
- package/habilidades/perfil-usuario/evals/evals.json +55 -0
- package/habilidades/performance-baseline/SKILL.md +297 -0
- package/habilidades/php-experto/SKILL.md +291 -0
- package/habilidades/php-patrones/SKILL.md +306 -0
- package/habilidades/php-testing/SKILL.md +280 -0
- package/habilidades/planear-fase/SKILL.md +269 -0
- package/habilidades/postgresql-experto/SKILL.md +151 -0
- package/habilidades/postgresql-experto/evals/evals.json +53 -0
- package/habilidades/postgresql-experto/recursos/referencia-completa.md +215 -0
- package/habilidades/prevencion-racionalizacion/SKILL.md +175 -0
- package/habilidades/prevencion-sobreingenieria/SKILL.md +323 -0
- package/habilidades/privacy-memoria/SKILL.md +141 -0
- package/habilidades/privacy-memoria/evals/evals.json +43 -0
- package/habilidades/prompt-engineering/SKILL.md +518 -0
- package/habilidades/prompt-engineering/recursos/patrones-avanzados.md +467 -0
- package/habilidades/rag-arquitectura/SKILL.md +338 -0
- package/habilidades/rails-experto/SKILL.md +237 -0
- package/habilidades/rails-experto/recursos/active-record.md +260 -0
- package/habilidades/rails-experto/recursos/hotwire-turbo.md +293 -0
- package/habilidades/rails-experto/recursos/testing-rspec.md +362 -0
- package/habilidades/react-experto/SKILL.md +209 -0
- package/habilidades/react-experto/evals/evals.json +55 -0
- package/habilidades/react-experto/recursos/patrones-y-ejemplos-completos.md +240 -0
- package/habilidades/react-optimizacion/SKILL.md +174 -0
- package/habilidades/react-optimizacion/recursos/patrones-avanzados.md +138 -0
- package/habilidades/redis-experto/SKILL.md +305 -0
- package/habilidades/release-semver/.evolved.json +9 -0
- package/habilidades/release-semver/SKILL.md +248 -0
- package/habilidades/release-semver/scripts/generar-changelog.sh +238 -0
- package/habilidades/rust-experto/SKILL.md +400 -0
- package/habilidades/rust-patrones/SKILL.md +296 -0
- package/habilidades/rust-testing/SKILL.md +311 -0
- package/habilidades/seguridad-skills-ia/SKILL.md +262 -0
- package/habilidades/sql-optimizacion/SKILL.md +200 -0
- package/habilidades/sql-optimizacion/evals/evals.json +54 -0
- package/habilidades/sql-optimizacion/recursos/patrones-sql-avanzados.md +131 -0
- package/habilidades/sre-patrones/SKILL.md +333 -0
- package/habilidades/sre-patrones/recursos/chaos-engineering.md +241 -0
- package/habilidades/sre-patrones/recursos/oncall-design.md +236 -0
- package/habilidades/stripe-pagos/SKILL.md +550 -0
- package/habilidades/stripe-pagos/recursos/errores-reintentos.md +390 -0
- package/habilidades/stripe-pagos/recursos/stripe-connect.md +290 -0
- package/habilidades/structured-outputs/SKILL.md +343 -0
- package/habilidades/swift-experto/SKILL.md +320 -0
- package/habilidades/swift-experto/recursos/keychain-y-wrappers.md +110 -0
- package/habilidades/swift-patrones/SKILL.md +313 -0
- package/habilidades/swift-patrones/recursos/tca-ejemplo-completo.md +113 -0
- package/habilidades/swift-testing/SKILL.md +254 -0
- package/habilidades/swift-testing/recursos/xcuitest-planes.md +143 -0
- package/habilidades/swl-dashboard/SKILL.md +370 -0
- package/habilidades/swl-markitdown/SKILL.md +285 -0
- package/habilidades/swl-markitdown/evals/evals.json +52 -0
- package/habilidades/swl-revisar-impacto/SKILL.md +233 -0
- package/habilidades/tailwind-experto/SKILL.md +240 -0
- package/habilidades/tailwind-experto/recursos/referencia-completa.md +184 -0
- package/habilidades/tdd-workflow/SKILL.md +293 -0
- package/habilidades/terraform-experto/SKILL.md +321 -0
- package/habilidades/testing-python/SKILL.md +340 -0
- package/habilidades/testing-python/recursos/ejemplos-completos.md +167 -0
- package/habilidades/threat-model-lite/SKILL.md +246 -0
- package/habilidades/tracing-processor/SKILL.md +212 -0
- package/habilidades/tracking-measurement/SKILL.md +239 -0
- package/habilidades/tracking-measurement/recursos/consent-mode.md +231 -0
- package/habilidades/tracking-measurement/recursos/gtm-datalayer.md +216 -0
- package/habilidades/tracking-measurement/recursos/meta-capi.md +262 -0
- package/habilidades/typescript-avanzado/SKILL.md +144 -0
- package/habilidades/typescript-avanzado/evals/evals.json +55 -0
- package/habilidades/typescript-avanzado/recursos/patrones-y-ejemplos-completos.md +298 -0
- package/habilidades/typescript-diagnosticos/SKILL.md +513 -0
- package/habilidades/ux-diseno/SKILL.md +116 -0
- package/habilidades/ux-diseno/evals/evals.json +43 -0
- package/habilidades/ux-diseno/recursos/patrones-ux-referencia.md +214 -0
- package/habilidades/validacion-ci-sistema/SKILL.md +136 -0
- package/habilidades/validacion-ci-sistema/recursos/validadores-completos.md +369 -0
- package/habilidades/validacion-ci-sistema/scripts/validar-sistema.sh +286 -0
- package/habilidades/verificacion-evidencia/SKILL.md +160 -0
- package/habilidades/verificar-trabajo/SKILL.md +303 -0
- package/habilidades/verificar-trabajo/recursos/plantilla-verificacion.md +60 -0
- package/habilidades/wiki-conocimiento/SKILL.md +276 -0
- package/habilidades/wireframes-flujos/SKILL.md +212 -0
- package/habilidades/wireframes-flujos/recursos/referencia-completa.md +192 -0
- package/habilidades/workflow-claude-code/SKILL.md +260 -0
- package/habilidades/workflow-claude-code/recursos/referencia-completa.md +109 -0
- package/hooks/_run-hook.sh +57 -0
- package/hooks/actualizar-perfil-usuario.js +364 -0
- package/hooks/agente-lifecycle.js +71 -0
- package/hooks/aiisms-detector.js +173 -0
- package/hooks/audit-trail.js +204 -0
- package/hooks/auto-background.js +97 -0
- package/hooks/auto-consolidacion.js +178 -0
- package/hooks/auto-evolucion.js +666 -0
- package/hooks/auto-restaurar-settings.js +360 -0
- package/hooks/calidad-pre-commit.js +929 -0
- package/hooks/calidad-typescript.js +511 -0
- package/hooks/captura-feedback-usuario.js +148 -0
- package/hooks/check-update.js +211 -0
- package/hooks/clasificador-mensajes.js +271 -0
- package/hooks/degradacion-instintos.js +272 -0
- package/hooks/escaneo-secretos.js +389 -0
- package/hooks/extraccion-aprendizajes.js +763 -0
- package/hooks/grafo-contexto.js +129 -0
- package/hooks/graph-update.js +67 -0
- package/hooks/guardrail-modelo.js +247 -0
- package/hooks/inbox-aviso.js +75 -0
- package/hooks/inyeccion-contexto.js +246 -0
- package/hooks/lib/abort-registry.js +214 -0
- package/hooks/lib/agent-backend.js +210 -0
- package/hooks/lib/agent-comms.js +263 -0
- package/hooks/lib/agent-issue-codes.js +284 -0
- package/hooks/lib/agent-matcher.js +189 -0
- package/hooks/lib/async-hook-registry.js +252 -0
- package/hooks/lib/atomic-write.js +130 -0
- package/hooks/lib/auto-consolidator.js +335 -0
- package/hooks/lib/canary-skills.js +187 -0
- package/hooks/lib/consolidation-lock.js +291 -0
- package/hooks/lib/context-builder.js +430 -0
- package/hooks/lib/context-compressor.js +657 -0
- package/hooks/lib/convergence-detector.js +105 -0
- package/hooks/lib/delegation-tracker.js +198 -0
- package/hooks/lib/detectar-package-manager.js +423 -0
- package/hooks/lib/edit-accumulator.js +171 -0
- package/hooks/lib/error-classifier.js +308 -0
- package/hooks/lib/event-bus.js +112 -0
- package/hooks/lib/evolution-tracker.js +442 -0
- package/hooks/lib/execution-state.js +316 -0
- package/hooks/lib/fingerprint-id.js +135 -0
- package/hooks/lib/gateway-notify.js +116 -0
- package/hooks/lib/graph-security.js +75 -0
- package/hooks/lib/guardrail-metrics.js +202 -0
- package/hooks/lib/hook-circuit-breaker.js +206 -0
- package/hooks/lib/loop-detector.js +267 -0
- package/hooks/lib/mcp-health.js +184 -0
- package/hooks/lib/mcp-pool.js +436 -0
- package/hooks/lib/memory-search.js +506 -0
- package/hooks/lib/merkle-audit.js +96 -0
- package/hooks/lib/model-router.js +222 -0
- package/hooks/lib/normalize-error.js +324 -0
- package/hooks/lib/normalize-input.js +65 -0
- package/hooks/lib/nudge-tracker.js +306 -0
- package/hooks/lib/otlp-exporter.js +365 -0
- package/hooks/lib/performance-marks.js +239 -0
- package/hooks/lib/privacy-filter.js +128 -0
- package/hooks/lib/prompt-injection-scanner.js +209 -0
- package/hooks/lib/provenance-tracker.js +183 -0
- package/hooks/lib/rate-limit-tracker.js +253 -0
- package/hooks/lib/reflect-classifier.js +164 -0
- package/hooks/lib/resource-quota.js +122 -0
- package/hooks/lib/retry-jitter.js +165 -0
- package/hooks/lib/risk-engine.js +368 -0
- package/hooks/lib/run-log.js +408 -0
- package/hooks/lib/session-fts.js +379 -0
- package/hooks/lib/session-store.js +293 -0
- package/hooks/lib/singleton-guard.js +159 -0
- package/hooks/lib/skill-auditor.js +588 -0
- package/hooks/lib/sync-status.js +228 -0
- package/hooks/lib/taint-tracker.js +107 -0
- package/hooks/lib/task-service.js +295 -0
- package/hooks/lib/tech-skills-map.js +146 -0
- package/hooks/lib/telegram-cliente.js +159 -0
- package/hooks/lib/telegram-config.js +170 -0
- package/hooks/lib/token-budget.js +156 -0
- package/hooks/lib/token-estimator.js +420 -0
- package/hooks/lib/toon-compressor.js +245 -0
- package/hooks/lib/usage-model.js +183 -0
- package/hooks/lib/variable-resolver.js +230 -0
- package/hooks/linea-estado.js +324 -0
- package/hooks/metricas-evolucion.js +209 -0
- package/hooks/monitor-contexto.js +325 -0
- package/hooks/notificacion-sesion-stop.js +198 -0
- package/hooks/notificacion-telegram-notification.js +4 -0
- package/hooks/notificacion-telegram-subagent.js +4 -0
- package/hooks/notificacion-telegram.js +267 -0
- package/hooks/preservar-estado-pre-compact.js +150 -0
- package/hooks/proteccion-rutas.js +366 -0
- package/hooks/registro-turnos.js +209 -0
- package/hooks/resumen-sesion.js +249 -0
- package/hooks/risk-scoring.js +323 -0
- package/hooks/rotar-audit-auto.js +122 -0
- package/hooks/sugerir-regenerar-inventario.js +170 -0
- package/hooks/telemetria-agentes.js +167 -0
- package/hooks/tracking-costos.js +688 -0
- package/instintos/global.yaml +8 -0
- package/instintos/perfil-usuario.yaml +53 -0
- package/instintos/prompt-appendices.yaml +57 -0
- package/instintos/proyecto.yaml +372 -0
- package/manifiestos/gateway-config.json +77 -0
- package/manifiestos/handoff-context.json +223 -0
- package/manifiestos/hook-profiles.json +44 -0
- package/manifiestos/hooks-config.json +360 -0
- package/manifiestos/modulos.json +1173 -0
- package/manifiestos/perfiles.json +404 -0
- package/package.json +86 -0
- package/plantillas/ESTADO.md +109 -0
- package/plantillas/HOJA-RUTA.md +143 -0
- package/plantillas/PROYECTO.md +122 -0
- package/plantillas/REQUISITOS.md +132 -0
- package/plantillas/auditor-veto-template.md +105 -0
- package/plantillas/github-workflows/README.md +47 -0
- package/plantillas/github-workflows/release-please.yml +44 -0
- package/plantillas/github-workflows/swl-ci.yml +107 -0
- package/plantillas/github-workflows/swl-security.yml +51 -0
- package/plantillas/mcp-mineru.json +13 -0
- package/plantillas/research/ARQUITECTURA.md +220 -0
- package/plantillas/research/FUNCIONALIDADES.md +175 -0
- package/plantillas/research/RESUMEN.md +165 -0
- package/plantillas/research/STACK.md +233 -0
- package/plantillas/research/TRAMPAS.md +299 -0
- package/plantillas/skill-evals-template.json +44 -0
- package/plugin.json +343 -0
- package/reglas/accesibilidad.md +269 -0
- package/reglas/api-diseno.md +400 -0
- package/reglas/arquitectura.md +352 -0
- package/reglas/brevedad-output.md +124 -0
- package/reglas/cloud-infra.md +247 -0
- package/reglas/docs.md +245 -0
- package/reglas/estilo-codigo.md +201 -0
- package/reglas/git-workflow.md +245 -0
- package/reglas/gobernanza.md +271 -0
- package/reglas/harness-claude-code.md +213 -0
- package/reglas/hooks.md +186 -0
- package/reglas/lenguajes/csharp/estilo-codigo.md +231 -0
- package/reglas/lenguajes/csharp/hooks.md +281 -0
- package/reglas/lenguajes/csharp/patrones.md +226 -0
- package/reglas/lenguajes/csharp/seguridad.md +258 -0
- package/reglas/lenguajes/csharp/testing.md +176 -0
- package/reglas/lenguajes/go/estilo-codigo.md +195 -0
- package/reglas/lenguajes/go/hooks.md +249 -0
- package/reglas/lenguajes/go/patrones.md +249 -0
- package/reglas/lenguajes/go/seguridad.md +225 -0
- package/reglas/lenguajes/go/testing.md +272 -0
- package/reglas/lenguajes/java/estilo-codigo.md +217 -0
- package/reglas/lenguajes/java/hooks.md +251 -0
- package/reglas/lenguajes/java/patrones.md +226 -0
- package/reglas/lenguajes/java/seguridad.md +233 -0
- package/reglas/lenguajes/java/testing.md +238 -0
- package/reglas/lenguajes/kotlin/estilo-codigo.md +208 -0
- package/reglas/lenguajes/kotlin/hooks.md +245 -0
- package/reglas/lenguajes/kotlin/patrones.md +201 -0
- package/reglas/lenguajes/kotlin/seguridad.md +202 -0
- package/reglas/lenguajes/kotlin/testing.md +236 -0
- package/reglas/lenguajes/nextjs/estilo-codigo.md +175 -0
- package/reglas/lenguajes/nextjs/hooks.md +186 -0
- package/reglas/lenguajes/nextjs/patrones.md +225 -0
- package/reglas/lenguajes/nextjs/seguridad.md +216 -0
- package/reglas/lenguajes/nextjs/testing.md +193 -0
- package/reglas/lenguajes/php/estilo-codigo.md +228 -0
- package/reglas/lenguajes/php/hooks.md +165 -0
- package/reglas/lenguajes/php/patrones.md +233 -0
- package/reglas/lenguajes/php/seguridad.md +186 -0
- package/reglas/lenguajes/php/testing.md +205 -0
- package/reglas/lenguajes/rust/estilo-codigo.md +207 -0
- package/reglas/lenguajes/rust/hooks.md +240 -0
- package/reglas/lenguajes/rust/patrones.md +250 -0
- package/reglas/lenguajes/rust/seguridad.md +221 -0
- package/reglas/lenguajes/rust/testing.md +194 -0
- package/reglas/lenguajes/swift/estilo-codigo.md +238 -0
- package/reglas/lenguajes/swift/hooks.md +257 -0
- package/reglas/lenguajes/swift/patrones.md +235 -0
- package/reglas/lenguajes/swift/seguridad.md +248 -0
- package/reglas/lenguajes/swift/testing.md +242 -0
- package/reglas/markitdown.md +60 -0
- package/reglas/memoria-consolidada.md +209 -0
- package/reglas/patrones.md +225 -0
- package/reglas/performance.md +195 -0
- package/reglas/pruebas.md +159 -0
- package/reglas/seguridad-agentes.md +351 -0
- package/reglas/seguridad.md +151 -0
- package/reglas/skills-estandar.md +373 -0
- package/reglas/testing.md +193 -0
- package/schemas/agent-contract.json +176 -0
- package/schemas/agent-frontmatter.schema.json +149 -0
- package/schemas/agent-message.schema.json +53 -0
- package/schemas/agent-output-implementacion.schema.json +85 -0
- package/schemas/agent-output-planificacion.schema.json +113 -0
- package/schemas/agent-output-review.schema.json +78 -0
- package/schemas/diary-entry.schema.json +80 -0
- package/schemas/hook-profiles.schema.json +39 -0
- package/schemas/hooks-config.schema.json +74 -0
- package/schemas/instinct.schema.json +115 -0
- package/schemas/modulos.schema.json +29 -0
- package/schemas/perfiles.schema.json +28 -0
- package/schemas/plugin.schema.json +64 -0
- package/schemas/skill-evals.schema.json +95 -0
- package/schemas/skill-frontmatter.schema.json +170 -0
- package/scripts/actualizar.js +145 -0
- package/scripts/audit-skills.sh +78 -0
- package/scripts/auditar-agentes-gaps.js +149 -0
- package/scripts/auditar-cobertura-frameworks.js +241 -0
- package/scripts/auditar-skills-gaps.js +206 -0
- package/scripts/bootstrap-instintos.js +259 -0
- package/scripts/check-update.js +109 -0
- package/scripts/comandos/agents.js +105 -0
- package/scripts/comandos/info.js +108 -0
- package/scripts/comandos/install-asistido.js +186 -0
- package/scripts/comandos/skills.js +211 -0
- package/scripts/configurar-branch-protection.js +418 -0
- package/scripts/daemon-swl.py +388 -0
- package/scripts/desinstalar.js +130 -0
- package/scripts/doctor.js +559 -0
- package/scripts/field-report.js +199 -0
- package/scripts/generar-inventario.js +317 -0
- package/scripts/inbox-tmux-inject.js +161 -0
- package/scripts/inferir-herramientas-permitidas.js +586 -0
- package/scripts/inicializar.js +133 -0
- package/scripts/instalador.js +1031 -0
- package/scripts/instalar-git-hook.js +122 -0
- package/scripts/lib/agp-frontmatter.js +222 -0
- package/scripts/lib/append-con-marcadores.js +199 -0
- package/scripts/lib/artefactos-python.js +43 -0
- package/scripts/lib/audit-query.js +221 -0
- package/scripts/lib/autostart-linux.js +347 -0
- package/scripts/lib/autostart-macos.js +360 -0
- package/scripts/lib/autostart-windows.js +307 -0
- package/scripts/lib/budget-enforcer.js +252 -0
- package/scripts/lib/claude-sessions.js +285 -0
- package/scripts/lib/configurar-ci.js +380 -0
- package/scripts/lib/console-span-exporter.js +92 -0
- package/scripts/lib/contadores-inventario.js +217 -0
- package/scripts/lib/dashboard-widgets.js +290 -0
- package/scripts/lib/detectar-runtime.js +279 -0
- package/scripts/lib/detectar-stack.js +187 -0
- package/scripts/lib/diary-entry.js +234 -0
- package/scripts/lib/drift-detector.js +545 -0
- package/scripts/lib/estado.js +124 -0
- package/scripts/lib/gestor-componentes.js +243 -0
- package/scripts/lib/gitignore-manifest.js +305 -0
- package/scripts/lib/graph-analyze.py +556 -0
- package/scripts/lib/graph-builder.py +485 -0
- package/scripts/lib/graph-cluster.py +259 -0
- package/scripts/lib/health-row.js +168 -0
- package/scripts/lib/hooks-settings.js +789 -0
- package/scripts/lib/manifiestos.js +138 -0
- package/scripts/lib/mc-client.js +137 -0
- package/scripts/lib/notificaciones-telegram.js +1107 -0
- package/scripts/lib/npm-version.js +261 -0
- package/scripts/lib/paquetes-conocidos.js +50 -0
- package/scripts/lib/preservar-usuario.js +586 -0
- package/scripts/lib/prompt-builder.js +264 -0
- package/scripts/lib/resolver-externo.js +332 -0
- package/scripts/lib/schedule-parser.js +305 -0
- package/scripts/lib/scoring-instintos.js +240 -0
- package/scripts/lib/seguridad.js +160 -0
- package/scripts/lib/selector-interactivo.js +152 -0
- package/scripts/lib/semantic-search.js +242 -0
- package/scripts/lib/skill-discovery.js +234 -0
- package/scripts/lib/skill-metrics.js +246 -0
- package/scripts/lib/skill-normalizer.js +112 -0
- package/scripts/lib/skills-hub.js +340 -0
- package/scripts/lib/span-schema.js +134 -0
- package/scripts/lib/tool-cost-analyzer.js +255 -0
- package/scripts/lib/tracing-processor-interface.js +286 -0
- package/scripts/lib/transformadores/base.js +80 -0
- package/scripts/lib/transformadores/claude.js +124 -0
- package/scripts/lib/transformadores/codex.js +115 -0
- package/scripts/lib/transformadores/copilot.js +106 -0
- package/scripts/lib/transformadores/gemini.js +74 -0
- package/scripts/lib/transformadores/index.js +35 -0
- package/scripts/lib/transformadores/opencode.js +75 -0
- package/scripts/lib/ui.js +259 -0
- package/scripts/limpiar-artefactos-python.js +131 -0
- package/scripts/mcp-orchestrator.py +386 -0
- package/scripts/mcp-pool-manager.py +352 -0
- package/scripts/mcp-telemetry.py +378 -0
- package/scripts/poblar-evolvable.js +226 -0
- package/scripts/publicar.js +287 -0
- package/scripts/reflect-skills.js +403 -0
- package/scripts/rotar-audit-logs.js +185 -0
- package/scripts/run-skill-evals.js +242 -0
- package/scripts/smoke-test.js +374 -0
- package/scripts/token-analysis.py +471 -0
- package/scripts/validar-manifest.js +195 -0
- package/scripts/validar-memoria.js +321 -0
- package/scripts/validar-tests-aislamiento.js +184 -0
- package/scripts/validar-tokens-test.js +208 -0
- package/scripts/validar.js +147 -0
- package/scripts/validate-markdown.py +339 -0
- package/scripts/validate-skills.py +385 -0
- package/scripts/vendor/claude-usage/README.md +116 -0
- package/scripts/vendor/claude-usage/cli.py +334 -0
- package/scripts/vendor/claude-usage/dashboard.py +795 -0
- package/scripts/vendor/claude-usage/scanner.py +467 -0
- package/scripts/vendor/markitdown/cli.py +194 -0
- package/scripts/verificar-evolucion.js +289 -0
- package/scripts/verificar-release.js +494 -0
|
@@ -0,0 +1,666 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Hook: auto-evolucion.js
|
|
6
|
+
* Tipo: SubagentStop (async: true — fire-and-forget)
|
|
7
|
+
*
|
|
8
|
+
* Registra la terminación de cada subagente SWL en un log JSONL y, cuando
|
|
9
|
+
* un mismo agente acumula suficientes señales de baja calidad o alta fricción,
|
|
10
|
+
* emite un nudge por stderr sugiriendo invocar /swl:evolucionar.
|
|
11
|
+
*
|
|
12
|
+
* Señales registradas por agente:
|
|
13
|
+
* - status:failed — el agente terminó con error
|
|
14
|
+
* - trivial — ≤5 tool calls (se descarta del análisis)
|
|
15
|
+
* - tool_calls — número de tool calls totales
|
|
16
|
+
* - duracion_ms — tiempo de ejecución
|
|
17
|
+
* - tipo_fallo — clasificación del fallo (AGP Reflect, Zhang 2026 §3.2.2)
|
|
18
|
+
* uno de: bad_output_format, tool_error, timeout,
|
|
19
|
+
* schema_violation, task_incomplete, unknown
|
|
20
|
+
*
|
|
21
|
+
* Umbrales (configurables vía env):
|
|
22
|
+
* SWL_AUTOEVOL_FALLOS (default 3) — fallos en 14 días → nudge
|
|
23
|
+
* SWL_AUTOEVOL_RUNS (default 10) — corridas en 14 días → sugerir revisión
|
|
24
|
+
* SWL_AUTOEVOL_THROTTLE_H (default 24) — horas entre nudges del mismo agente
|
|
25
|
+
* SWL_DRIFT_THROTTLE_H (default 24) — horas entre chequeos globales de drift
|
|
26
|
+
* SWL_DRIFT_DISABLED=1 — desactiva por completo el drift-detector
|
|
27
|
+
*
|
|
28
|
+
* El hook nunca bloquea (siempre exit 0). Zero-dependencies.
|
|
29
|
+
* NO invoca agentes — solo registra y sugiere.
|
|
30
|
+
*
|
|
31
|
+
* convergence-detector (REC-D03): invocado en cada SubagentStop cuando hay ≥6
|
|
32
|
+
* invocaciones de herramienta en las trazas del día para el agente actual.
|
|
33
|
+
* Emite un nudge tipo 'loop' independiente del gate de fallos/volumen.
|
|
34
|
+
*
|
|
35
|
+
* drift-detector (REC-D07): invocado con throttle propio (SWL_DRIFT_THROTTLE_H)
|
|
36
|
+
* sobre los últimos JSONL de trazas. Opt-out via SWL_DRIFT_DISABLED=1.
|
|
37
|
+
*
|
|
38
|
+
* @see hooks/auto-consolidacion.js (patrón anti-spam con flag)
|
|
39
|
+
* @see agentes/auto-evolucion-swl.md (agente que se invoca tras el nudge)
|
|
40
|
+
*/
|
|
41
|
+
|
|
42
|
+
const fs = require('fs');
|
|
43
|
+
const path = require('path');
|
|
44
|
+
|
|
45
|
+
let nudgeTracker;
|
|
46
|
+
try {
|
|
47
|
+
nudgeTracker = require('./lib/nudge-tracker');
|
|
48
|
+
} catch {
|
|
49
|
+
nudgeTracker = null;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
let reflect;
|
|
53
|
+
try {
|
|
54
|
+
reflect = require('./lib/reflect-classifier');
|
|
55
|
+
} catch {
|
|
56
|
+
reflect = null;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
let convergenceDetector;
|
|
60
|
+
try {
|
|
61
|
+
convergenceDetector = require('./lib/convergence-detector');
|
|
62
|
+
} catch {
|
|
63
|
+
convergenceDetector = null;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
let driftDetector;
|
|
67
|
+
try {
|
|
68
|
+
driftDetector = require('../scripts/lib/drift-detector');
|
|
69
|
+
} catch {
|
|
70
|
+
driftDetector = null;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
let singletonGuard;
|
|
74
|
+
try {
|
|
75
|
+
singletonGuard = require('./lib/singleton-guard');
|
|
76
|
+
} catch {
|
|
77
|
+
singletonGuard = null;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// ---------------------------------------------------------------------------
|
|
81
|
+
// Configuración
|
|
82
|
+
// ---------------------------------------------------------------------------
|
|
83
|
+
|
|
84
|
+
const UMBRAL_FALLOS = parseInt(process.env.SWL_AUTOEVOL_FALLOS || '3', 10);
|
|
85
|
+
const UMBRAL_RUNS = parseInt(process.env.SWL_AUTOEVOL_RUNS || '10', 10);
|
|
86
|
+
const THROTTLE_HORAS = parseInt(process.env.SWL_AUTOEVOL_THROTTLE_H || '24', 10);
|
|
87
|
+
const DRIFT_THROTTLE_HORAS = parseInt(process.env.SWL_DRIFT_THROTTLE_H || '24', 10);
|
|
88
|
+
const VENTANA_DIAS = 14;
|
|
89
|
+
const MIN_TOOL_CALLS = 5; // por debajo se considera trivial
|
|
90
|
+
const MIN_INVOCACIONES_LOOP = 6; // mínimo para que el ratio de convergencia sea significativo
|
|
91
|
+
|
|
92
|
+
const DIR_AUTOEVOL = path.join(process.cwd(), '.planning', 'auto-evolucion');
|
|
93
|
+
const DIR_TRACES = path.join(process.cwd(), '.planning', 'traces');
|
|
94
|
+
const LOG_PATH = path.join(DIR_AUTOEVOL, 'agentes.jsonl');
|
|
95
|
+
const NUDGES_PATH = path.join(DIR_AUTOEVOL, 'nudges.json');
|
|
96
|
+
|
|
97
|
+
// ---------------------------------------------------------------------------
|
|
98
|
+
// Utilidades de filesystem (zero-deps)
|
|
99
|
+
// ---------------------------------------------------------------------------
|
|
100
|
+
|
|
101
|
+
function ensureDir(dir) {
|
|
102
|
+
try { fs.mkdirSync(dir, { recursive: true }); } catch { /* ignore */ }
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
function append(obj) {
|
|
106
|
+
ensureDir(DIR_AUTOEVOL);
|
|
107
|
+
try {
|
|
108
|
+
fs.appendFileSync(LOG_PATH, JSON.stringify(obj) + '\n', 'utf8');
|
|
109
|
+
} catch { /* ignore */ }
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
function leerNudges() {
|
|
113
|
+
try {
|
|
114
|
+
return JSON.parse(fs.readFileSync(NUDGES_PATH, 'utf8'));
|
|
115
|
+
} catch {
|
|
116
|
+
return {};
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
function escribirNudges(obj) {
|
|
121
|
+
ensureDir(DIR_AUTOEVOL);
|
|
122
|
+
try {
|
|
123
|
+
fs.writeFileSync(NUDGES_PATH, JSON.stringify(obj, null, 2), 'utf8');
|
|
124
|
+
} catch { /* ignore */ }
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Lee las últimas N entradas del JSONL (sin cargar todo si es grande).
|
|
129
|
+
* Para el tamaño esperado (log de agentes) una lectura simple es suficiente.
|
|
130
|
+
*/
|
|
131
|
+
function leerLog() {
|
|
132
|
+
try {
|
|
133
|
+
const raw = fs.readFileSync(LOG_PATH, 'utf8');
|
|
134
|
+
return raw
|
|
135
|
+
.split(/\r?\n/)
|
|
136
|
+
.filter(Boolean)
|
|
137
|
+
.map(l => { try { return JSON.parse(l); } catch { return null; } })
|
|
138
|
+
.filter(Boolean);
|
|
139
|
+
} catch {
|
|
140
|
+
return [];
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// ---------------------------------------------------------------------------
|
|
145
|
+
// Clasificador de tipo_fallo — vive en hooks/lib/reflect-classifier.js
|
|
146
|
+
// (AGP Reflect, Zhang 2026 §3.2.2)
|
|
147
|
+
// ---------------------------------------------------------------------------
|
|
148
|
+
|
|
149
|
+
function clasificarTipoFallo(data, response) {
|
|
150
|
+
if (reflect) return reflect.clasificarTipoFallo(data, response);
|
|
151
|
+
return null; // fallback silencioso: si falta el modulo, el hook no bloquea
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// ---------------------------------------------------------------------------
|
|
155
|
+
// Análisis de payload de SubagentStop
|
|
156
|
+
// ---------------------------------------------------------------------------
|
|
157
|
+
|
|
158
|
+
/**
|
|
159
|
+
* Extrae la info relevante del payload que Claude Code pasa al hook.
|
|
160
|
+
* Campos esperados (best-effort; varían según harness):
|
|
161
|
+
* - session_id
|
|
162
|
+
* - tool_name (= "Agent")
|
|
163
|
+
* - tool_input.subagent_type (ej. "implementador-swl")
|
|
164
|
+
* - tool_response / tool_result
|
|
165
|
+
* - duration_ms (si el harness lo provee)
|
|
166
|
+
*/
|
|
167
|
+
function analizarPayload(data) {
|
|
168
|
+
const toolName = String(data.tool_name || data.tool?.name || '');
|
|
169
|
+
if (toolName !== 'Agent') return null;
|
|
170
|
+
|
|
171
|
+
const subagentType =
|
|
172
|
+
data.tool_input?.subagent_type ||
|
|
173
|
+
data.tool?.input?.subagent_type ||
|
|
174
|
+
data.subagent_type ||
|
|
175
|
+
'desconocido';
|
|
176
|
+
|
|
177
|
+
// Solo trackeamos agentes SWL (los demás no tenemos potestad de evolucionar)
|
|
178
|
+
if (!/-swl$/.test(subagentType)) return null;
|
|
179
|
+
|
|
180
|
+
const response = data.tool_response || data.tool_result || {};
|
|
181
|
+
const status = response.is_error || response.status === 'failed' ? 'failed' : 'success';
|
|
182
|
+
|
|
183
|
+
// Número de tool calls — lo estimamos por longitud de mensajes si está disponible
|
|
184
|
+
const mensajes = response.messages || response.mensajes || [];
|
|
185
|
+
const toolCalls = Array.isArray(mensajes)
|
|
186
|
+
? mensajes.filter(m => m && (m.type === 'tool_use' || m.role === 'tool')).length
|
|
187
|
+
: 0;
|
|
188
|
+
|
|
189
|
+
const duracionMs =
|
|
190
|
+
data.duration_ms ||
|
|
191
|
+
response.duration_ms ||
|
|
192
|
+
response.uso?.duration_ms ||
|
|
193
|
+
null;
|
|
194
|
+
|
|
195
|
+
const tipoFallo = status === 'failed'
|
|
196
|
+
? clasificarTipoFallo(data, response)
|
|
197
|
+
: null;
|
|
198
|
+
|
|
199
|
+
return {
|
|
200
|
+
ts: new Date().toISOString(),
|
|
201
|
+
sessionId: String(data.session_id || 'default'),
|
|
202
|
+
agente: subagentType,
|
|
203
|
+
status,
|
|
204
|
+
toolCalls,
|
|
205
|
+
trivial: toolCalls > 0 && toolCalls < MIN_TOOL_CALLS,
|
|
206
|
+
duracionMs,
|
|
207
|
+
tipo_fallo: tipoFallo,
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
// ---------------------------------------------------------------------------
|
|
212
|
+
// Gates para emitir nudge
|
|
213
|
+
// ---------------------------------------------------------------------------
|
|
214
|
+
|
|
215
|
+
function filtrarVentana(entradas, dias) {
|
|
216
|
+
const limite = Date.now() - dias * 24 * 3600 * 1000;
|
|
217
|
+
return entradas.filter(e => {
|
|
218
|
+
const t = Date.parse(e.ts);
|
|
219
|
+
return Number.isFinite(t) && t >= limite;
|
|
220
|
+
});
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
function resumenPorAgente(entradasVentana) {
|
|
224
|
+
const resumen = {};
|
|
225
|
+
for (const e of entradasVentana) {
|
|
226
|
+
if (!resumen[e.agente]) {
|
|
227
|
+
resumen[e.agente] = { runs: 0, fallos: 0, triviales: 0, fallos_por_tipo: {} };
|
|
228
|
+
}
|
|
229
|
+
const r = resumen[e.agente];
|
|
230
|
+
if (e.trivial) r.triviales++;
|
|
231
|
+
else {
|
|
232
|
+
r.runs++;
|
|
233
|
+
if (e.status === 'failed') {
|
|
234
|
+
r.fallos++;
|
|
235
|
+
const tipo = e.tipo_fallo || 'unknown';
|
|
236
|
+
r.fallos_por_tipo[tipo] = (r.fallos_por_tipo[tipo] || 0) + 1;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
return resumen;
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
/**
|
|
244
|
+
* Determina el tipo_fallo dominante y retorna un diagnóstico estructurado
|
|
245
|
+
* listo para propagar al nudge. Delega al módulo reflect-classifier.
|
|
246
|
+
*/
|
|
247
|
+
function construirDiagnosis(resumenAgente) {
|
|
248
|
+
if (reflect) return reflect.construirDiagnosis(resumenAgente);
|
|
249
|
+
return null;
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
function debeEmitirNudge(agente, resumen, nudges) {
|
|
253
|
+
const r = resumen[agente];
|
|
254
|
+
if (!r) return null;
|
|
255
|
+
|
|
256
|
+
const ultimoNudge = nudges[agente];
|
|
257
|
+
if (ultimoNudge) {
|
|
258
|
+
const horas = (Date.now() - Date.parse(ultimoNudge)) / 3600000;
|
|
259
|
+
if (horas < THROTTLE_HORAS) return null;
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
if (r.fallos >= UMBRAL_FALLOS) {
|
|
263
|
+
return {
|
|
264
|
+
razon: 'fallos',
|
|
265
|
+
detalle: `${r.fallos} fallos en ${VENTANA_DIAS} días`,
|
|
266
|
+
};
|
|
267
|
+
}
|
|
268
|
+
if (r.runs >= UMBRAL_RUNS) {
|
|
269
|
+
return {
|
|
270
|
+
razon: 'volumen',
|
|
271
|
+
detalle: `${r.runs} corridas no triviales en ${VENTANA_DIAS} días`,
|
|
272
|
+
};
|
|
273
|
+
}
|
|
274
|
+
return null;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// ---------------------------------------------------------------------------
|
|
278
|
+
// Convergence detector — gate de loop (REC-D03)
|
|
279
|
+
// ---------------------------------------------------------------------------
|
|
280
|
+
|
|
281
|
+
/**
|
|
282
|
+
* Lee las invocaciones de herramienta del agente desde los JSONL de trazas.
|
|
283
|
+
*
|
|
284
|
+
* El formato nativo de otlp-exporter es:
|
|
285
|
+
* { traceId, spanId, nombre, inicio, fin, duracionMs, estado, atributos }
|
|
286
|
+
* donde `nombre` es tipo "agent:nombre-swl" o "skill:nombre-skill".
|
|
287
|
+
*
|
|
288
|
+
* Retorna array de { tool: string } compatible con calcularScoreConvergencia.
|
|
289
|
+
* Nunca lanza — retorna [] ante cualquier falla.
|
|
290
|
+
*
|
|
291
|
+
* @param {string} agente - Nombre del agente SWL (sin prefijo, ej: "implementador-swl")
|
|
292
|
+
* @param {number} [diasAtras] - Cuántos días de traces revisar (1 = solo hoy, 2 = hoy y ayer)
|
|
293
|
+
* @returns {Array<{tool: string}>}
|
|
294
|
+
*/
|
|
295
|
+
function leerInvocacionesDelAgente(agente, diasAtras = 1) {
|
|
296
|
+
const invocaciones = [];
|
|
297
|
+
try {
|
|
298
|
+
const ahora = new Date();
|
|
299
|
+
for (let i = 0; i < diasAtras; i++) {
|
|
300
|
+
const fecha = new Date(ahora.getTime() - i * 24 * 3600 * 1000)
|
|
301
|
+
.toISOString()
|
|
302
|
+
.slice(0, 10);
|
|
303
|
+
const rutaJsonl = path.join(DIR_TRACES, `${fecha}.jsonl`);
|
|
304
|
+
if (!fs.existsSync(rutaJsonl)) continue;
|
|
305
|
+
|
|
306
|
+
let contenido;
|
|
307
|
+
try {
|
|
308
|
+
contenido = fs.readFileSync(rutaJsonl, 'utf8');
|
|
309
|
+
} catch {
|
|
310
|
+
continue;
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
const lineas = contenido.split(/\r?\n/).filter(Boolean);
|
|
314
|
+
for (const linea of lineas) {
|
|
315
|
+
let evento;
|
|
316
|
+
try {
|
|
317
|
+
evento = JSON.parse(linea);
|
|
318
|
+
} catch {
|
|
319
|
+
continue;
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
// El campo nombre puede ser "agent:nombre-swl", "skill:nombre", etc.
|
|
323
|
+
// Filtramos eventos asociados al agente actual.
|
|
324
|
+
const nombre = String(evento.nombre || '');
|
|
325
|
+
const esDelAgente =
|
|
326
|
+
nombre === `agent:${agente}` ||
|
|
327
|
+
nombre.includes(agente) ||
|
|
328
|
+
(evento.atributos?.agente === agente) ||
|
|
329
|
+
(evento.atributos?.agent === agente);
|
|
330
|
+
|
|
331
|
+
if (!esDelAgente) continue;
|
|
332
|
+
|
|
333
|
+
// Extraemos el nombre del tool/skill del evento como la herramienta invocada.
|
|
334
|
+
// En el formato OTLP-lite el "tool" es el nombre del span.
|
|
335
|
+
const tool = nombre || 'unknown';
|
|
336
|
+
invocaciones.push({ tool });
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
} catch {
|
|
340
|
+
// Fallback silencioso — el hook no debe crashear por traces inexistentes
|
|
341
|
+
}
|
|
342
|
+
return invocaciones;
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
/**
|
|
346
|
+
* Determina si hay throttle activo para el nudge de loop del agente.
|
|
347
|
+
*
|
|
348
|
+
* Usa la clave "<agente>::loop" en el objeto nudges con el mismo
|
|
349
|
+
* throttle configurado en THROTTLE_HORAS.
|
|
350
|
+
*
|
|
351
|
+
* @param {string} agente
|
|
352
|
+
* @param {object} nudges
|
|
353
|
+
* @returns {boolean} true si debe omitir el nudge de loop
|
|
354
|
+
*/
|
|
355
|
+
function estaThrottledLoop(agente, nudges) {
|
|
356
|
+
const clave = `${agente}::loop`;
|
|
357
|
+
const ultimo = nudges[clave];
|
|
358
|
+
if (!ultimo) return false;
|
|
359
|
+
const horas = (Date.now() - Date.parse(ultimo)) / 3600000;
|
|
360
|
+
return horas < THROTTLE_HORAS;
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
// ---------------------------------------------------------------------------
|
|
364
|
+
// Drift detector automático (REC-D07)
|
|
365
|
+
// ---------------------------------------------------------------------------
|
|
366
|
+
|
|
367
|
+
/**
|
|
368
|
+
* Retorna las rutas de los últimos N archivos JSONL de traces ordenados
|
|
369
|
+
* por fecha descendente (más reciente primero).
|
|
370
|
+
*
|
|
371
|
+
* @param {number} [max=5]
|
|
372
|
+
* @returns {string[]}
|
|
373
|
+
*/
|
|
374
|
+
function rutasTracesRecientes(max) {
|
|
375
|
+
const n = max || 5;
|
|
376
|
+
try {
|
|
377
|
+
if (!fs.existsSync(DIR_TRACES)) return [];
|
|
378
|
+
const archivos = fs.readdirSync(DIR_TRACES)
|
|
379
|
+
.filter(f => /^\d{4}-\d{2}-\d{2}\.jsonl$/.test(f))
|
|
380
|
+
.sort()
|
|
381
|
+
.reverse()
|
|
382
|
+
.slice(0, n);
|
|
383
|
+
return archivos.map(f => path.join(DIR_TRACES, f));
|
|
384
|
+
} catch {
|
|
385
|
+
return [];
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
/**
|
|
390
|
+
* Ejecuta el drift-detector de forma automática con throttle propio.
|
|
391
|
+
*
|
|
392
|
+
* Respeta la clave "__drift__::global" en el objeto nudges para evitar
|
|
393
|
+
* correr en cada invocación de subagente (sería costoso).
|
|
394
|
+
*
|
|
395
|
+
* Opt-out completo con SWL_DRIFT_DISABLED=1.
|
|
396
|
+
*
|
|
397
|
+
* @param {object} nudges - Objeto de nudges leído de disco (se muta in-place)
|
|
398
|
+
*/
|
|
399
|
+
function ejecutarDriftAutomatico(nudges) {
|
|
400
|
+
if (process.env.SWL_DRIFT_DISABLED === '1') return;
|
|
401
|
+
if (!driftDetector) return;
|
|
402
|
+
|
|
403
|
+
// Throttle: no correr si ya se corrió en las últimas DRIFT_THROTTLE_HORAS
|
|
404
|
+
const claveThrottle = '__drift__::global';
|
|
405
|
+
const ultimoDrift = nudges[claveThrottle];
|
|
406
|
+
if (ultimoDrift) {
|
|
407
|
+
const horas = (Date.now() - Date.parse(ultimoDrift)) / 3600000;
|
|
408
|
+
if (horas < DRIFT_THROTTLE_HORAS) return;
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
// Actualizar throttle inmediatamente para que invocaciones concurrentes no dupliquen
|
|
412
|
+
nudges[claveThrottle] = new Date().toISOString();
|
|
413
|
+
|
|
414
|
+
try {
|
|
415
|
+
const rutas = rutasTracesRecientes(5);
|
|
416
|
+
if (rutas.length === 0) return;
|
|
417
|
+
|
|
418
|
+
// Usar la traza más reciente como fuente principal de drift.
|
|
419
|
+
// detectarDrift acepta una sola rutaJsonl — pasamos el archivo más reciente.
|
|
420
|
+
const resultado = driftDetector.detectarDrift({
|
|
421
|
+
rutaJsonl: rutas[0],
|
|
422
|
+
ventanaDias: 7,
|
|
423
|
+
baselineSemanasBase: 4,
|
|
424
|
+
umbralPct: 10,
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
if (!resultado) return;
|
|
428
|
+
if (resultado.estado_global !== 'warn' && resultado.estado_global !== 'critico') return;
|
|
429
|
+
|
|
430
|
+
// Filtrar solo las métricas que driftearon
|
|
431
|
+
const driftsSig = (resultado.drifts || []).filter(d => d.estado !== 'ok');
|
|
432
|
+
if (driftsSig.length === 0) return;
|
|
433
|
+
|
|
434
|
+
// Emitir nudge por stderr
|
|
435
|
+
const lineasDrift = driftsSig.map(d =>
|
|
436
|
+
` • ${d.metrica}: ${d.driftPct > 0 ? '+' : ''}${d.driftPct}% (${d.estado})`
|
|
437
|
+
);
|
|
438
|
+
const mensajeDrift = [
|
|
439
|
+
'',
|
|
440
|
+
`[auto-evolución] Drift detectado en trazas de agentes (estado: ${resultado.estado_global}).`,
|
|
441
|
+
' Métricas con variación significativa:',
|
|
442
|
+
...lineasDrift,
|
|
443
|
+
' Considera revisar el ciclo de evolución:',
|
|
444
|
+
' /swl:evolucionar',
|
|
445
|
+
'',
|
|
446
|
+
].join('\n');
|
|
447
|
+
process.stderr.write(mensajeDrift);
|
|
448
|
+
|
|
449
|
+
// Registrar en nudge-tracker
|
|
450
|
+
if (nudgeTracker) {
|
|
451
|
+
try {
|
|
452
|
+
// Drift crítico → repair/high; drift warn → repair/medium
|
|
453
|
+
const esCritico = resultado.estado_global === 'critico';
|
|
454
|
+
nudgeTracker.emit({
|
|
455
|
+
kind: 'drift-detectado',
|
|
456
|
+
target: '__global__',
|
|
457
|
+
message: mensajeDrift.trim(),
|
|
458
|
+
source: 'hooks/auto-evolucion.js',
|
|
459
|
+
mutation_category: 'repair',
|
|
460
|
+
risk_level: esCritico ? 'high' : 'medium',
|
|
461
|
+
data: {
|
|
462
|
+
estado_global: resultado.estado_global,
|
|
463
|
+
drifts: driftsSig,
|
|
464
|
+
},
|
|
465
|
+
});
|
|
466
|
+
} catch { /* silencioso */ }
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
} catch {
|
|
470
|
+
// Fallback silencioso — el drift no debe bloquear el hook principal
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
// ---------------------------------------------------------------------------
|
|
475
|
+
// Entrypoint
|
|
476
|
+
// ---------------------------------------------------------------------------
|
|
477
|
+
|
|
478
|
+
// Solo registrar listeners de stdin en modo CLI. Cuando se importa como
|
|
479
|
+
// módulo (tests de integración), los listeners mantendrían el event loop
|
|
480
|
+
// vivo hasta que stdin emita 'end' — evento que nunca llega cuando el
|
|
481
|
+
// runner hereda stdin del proceso padre — y la suite de tests cuelga
|
|
482
|
+
// hasta el timeout. El hook real invocado por Claude Code siempre se
|
|
483
|
+
// ejecuta como require.main, así que el CLI behavior se preserva intacto.
|
|
484
|
+
if (require.main === module) {
|
|
485
|
+
let inputRaw = '';
|
|
486
|
+
process.stdin.on('data', chunk => { inputRaw += chunk; });
|
|
487
|
+
|
|
488
|
+
process.stdin.on('end', () => {
|
|
489
|
+
// Singleton guard: si otra instancia del hook está corriendo (poco probable
|
|
490
|
+
// en SubagentStop pero posible con eventos concurrentes), salir silenciosamente.
|
|
491
|
+
// Lock stale tras 5 min (el hook normalmente corre en < 1s).
|
|
492
|
+
if (singletonGuard && !singletonGuard.adquirir('auto-evolucion', { staleMs: 300000 })) {
|
|
493
|
+
return;
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
try {
|
|
497
|
+
const data = JSON.parse(inputRaw);
|
|
498
|
+
const entrada = analizarPayload(data);
|
|
499
|
+
if (!entrada) { if (singletonGuard) singletonGuard.liberar('auto-evolucion'); return; }
|
|
500
|
+
|
|
501
|
+
// 1. Registrar en el log JSONL
|
|
502
|
+
append(entrada);
|
|
503
|
+
|
|
504
|
+
// 2. Si fue trivial, no analizamos nudges
|
|
505
|
+
if (entrada.trivial) { if (singletonGuard) singletonGuard.liberar('auto-evolucion'); return; }
|
|
506
|
+
|
|
507
|
+
// 3. Cargar histórico en ventana y evaluar gates
|
|
508
|
+
const historico = leerLog();
|
|
509
|
+
const ventana = filtrarVentana(historico, VENTANA_DIAS);
|
|
510
|
+
const resumen = resumenPorAgente(ventana);
|
|
511
|
+
|
|
512
|
+
const nudges = leerNudges();
|
|
513
|
+
const decision = debeEmitirNudge(entrada.agente, resumen, nudges);
|
|
514
|
+
|
|
515
|
+
// 4. Gate de loop (convergence-detector, REC-D03)
|
|
516
|
+
let loopInfo = null;
|
|
517
|
+
if (convergenceDetector) {
|
|
518
|
+
try {
|
|
519
|
+
const invocaciones = leerInvocacionesDelAgente(entrada.agente, 1);
|
|
520
|
+
if (invocaciones.length >= MIN_INVOCACIONES_LOOP) {
|
|
521
|
+
loopInfo = convergenceDetector.detectarLoop({ invocaciones });
|
|
522
|
+
}
|
|
523
|
+
} catch { /* fallback silencioso */ }
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
// Mapeo razón → (mutation_category, risk_level) según Evolver GEP.
|
|
527
|
+
// repair: corrige regresión observable; optimize: mejora sin bug; innovate: capacidad nueva.
|
|
528
|
+
const clasificarRazon = (razon) => {
|
|
529
|
+
if (razon === 'fallos' || razon === 'loop') return { mutation_category: 'repair', risk_level: 'high' };
|
|
530
|
+
if (razon === 'volumen') return { mutation_category: 'optimize', risk_level: 'medium' };
|
|
531
|
+
if (razon === 'drift') return { mutation_category: 'repair', risk_level: 'medium' };
|
|
532
|
+
return { mutation_category: undefined, risk_level: undefined };
|
|
533
|
+
};
|
|
534
|
+
|
|
535
|
+
// 5. Emitir nudge de loop si aplica (gate independiente del de fallos/volumen)
|
|
536
|
+
if (loopInfo?.estaEnLoop && !estaThrottledLoop(entrada.agente, nudges)) {
|
|
537
|
+
const mensajeLoop = [
|
|
538
|
+
'',
|
|
539
|
+
`[auto-evolución] Posible loop detectado en "${entrada.agente}".`,
|
|
540
|
+
` Ratio calls/tools: ${Math.round(loopInfo.ratio * 100) / 100} (umbral ${loopInfo.umbralUsado})`,
|
|
541
|
+
` Invocaciones: ${loopInfo.totalCalls} Tools únicos: ${loopInfo.uniqueTools}`,
|
|
542
|
+
` Considera revisar el diseño del agente:`,
|
|
543
|
+
` /swl:evolucionar ${entrada.agente}`,
|
|
544
|
+
'',
|
|
545
|
+
].join('\n');
|
|
546
|
+
|
|
547
|
+
process.stderr.write(mensajeLoop);
|
|
548
|
+
|
|
549
|
+
// Registrar throttle del loop por separado
|
|
550
|
+
nudges[`${entrada.agente}::loop`] = new Date().toISOString();
|
|
551
|
+
|
|
552
|
+
if (nudgeTracker) {
|
|
553
|
+
try {
|
|
554
|
+
const clasif = clasificarRazon('loop');
|
|
555
|
+
nudgeTracker.emit({
|
|
556
|
+
kind: 'auto-evolucion',
|
|
557
|
+
target: entrada.agente,
|
|
558
|
+
message: mensajeLoop.trim(),
|
|
559
|
+
source: 'hooks/auto-evolucion.js',
|
|
560
|
+
mutation_category: clasif.mutation_category,
|
|
561
|
+
risk_level: clasif.risk_level,
|
|
562
|
+
data: {
|
|
563
|
+
razon: 'loop',
|
|
564
|
+
ratio: loopInfo.ratio,
|
|
565
|
+
totalCalls: loopInfo.totalCalls,
|
|
566
|
+
uniqueTools: loopInfo.uniqueTools,
|
|
567
|
+
umbralUsado: loopInfo.umbralUsado,
|
|
568
|
+
},
|
|
569
|
+
});
|
|
570
|
+
} catch { /* silencioso */ }
|
|
571
|
+
}
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
// 6. Si no hay decision de nudge principal, pasar al drift y salir
|
|
575
|
+
if (!decision) {
|
|
576
|
+
escribirNudges(nudges);
|
|
577
|
+
ejecutarDriftAutomatico(nudges);
|
|
578
|
+
escribirNudges(nudges);
|
|
579
|
+
if (singletonGuard) singletonGuard.liberar('auto-evolucion');
|
|
580
|
+
return;
|
|
581
|
+
}
|
|
582
|
+
|
|
583
|
+
// 7. Construir diagnosis semántico (AGP Reflect)
|
|
584
|
+
const diagnosis = construirDiagnosis(resumen[entrada.agente]);
|
|
585
|
+
|
|
586
|
+
// 8. Emitir nudge principal (stderr, visible al modelo como contexto)
|
|
587
|
+
const lineas = [
|
|
588
|
+
'',
|
|
589
|
+
`[auto-evolución] Agente "${entrada.agente}" acumuló señales para evolución.`,
|
|
590
|
+
` Motivo: ${decision.razon} — ${decision.detalle}.`,
|
|
591
|
+
];
|
|
592
|
+
if (diagnosis) {
|
|
593
|
+
lineas.push(` Tipo de fallo dominante: ${diagnosis.tipo_fallo} (${diagnosis.conteo_dominante}/${diagnosis.total_fallos} fallos).`);
|
|
594
|
+
lineas.push(` Hipotesis de mejora: ${diagnosis.hint_mejora}.`);
|
|
595
|
+
}
|
|
596
|
+
// Incluir info de loop en el nudge principal si también está activo
|
|
597
|
+
if (loopInfo?.estaEnLoop) {
|
|
598
|
+
lineas.push(` Además: loop detectado (ratio ${Math.round(loopInfo.ratio * 100) / 100}, umbral ${loopInfo.umbralUsado}).`);
|
|
599
|
+
}
|
|
600
|
+
lineas.push(` Considera invocar:`);
|
|
601
|
+
lineas.push(` /swl:evolucionar ${entrada.agente}`);
|
|
602
|
+
lineas.push(` (o el agente auto-evolucion-swl con foco en "${entrada.agente}")`);
|
|
603
|
+
lineas.push('');
|
|
604
|
+
const mensaje = lineas.join('\n');
|
|
605
|
+
process.stderr.write(mensaje);
|
|
606
|
+
|
|
607
|
+
// 9. Registrar timestamp del nudge para throttle
|
|
608
|
+
nudges[entrada.agente] = new Date().toISOString();
|
|
609
|
+
escribirNudges(nudges);
|
|
610
|
+
|
|
611
|
+
// 10. Registrar en nudge-tracker para medir tasa de acción
|
|
612
|
+
if (nudgeTracker) {
|
|
613
|
+
try {
|
|
614
|
+
const clasif = clasificarRazon(decision.razon);
|
|
615
|
+
nudgeTracker.emit({
|
|
616
|
+
kind: 'auto-evolucion',
|
|
617
|
+
target: entrada.agente,
|
|
618
|
+
message: mensaje.trim(),
|
|
619
|
+
source: 'hooks/auto-evolucion.js',
|
|
620
|
+
mutation_category: clasif.mutation_category,
|
|
621
|
+
risk_level: clasif.risk_level,
|
|
622
|
+
data: {
|
|
623
|
+
razon: decision.razon,
|
|
624
|
+
detalle: decision.detalle,
|
|
625
|
+
diagnosis,
|
|
626
|
+
loop: loopInfo?.estaEnLoop ? loopInfo : null,
|
|
627
|
+
},
|
|
628
|
+
});
|
|
629
|
+
} catch { /* silencioso */ }
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
// 11. Drift automático (REC-D07) al final
|
|
633
|
+
ejecutarDriftAutomatico(nudges);
|
|
634
|
+
escribirNudges(nudges);
|
|
635
|
+
|
|
636
|
+
} catch (err) {
|
|
637
|
+
// Hook nunca bloquea, pero deja traza diagnóstica para no perder
|
|
638
|
+
// regresiones silenciosas en el ciclo de auto-evolución.
|
|
639
|
+
process.stderr.write(`[auto-evolucion] error interno: ${String(err.message || err).slice(0, 120)}\n`);
|
|
640
|
+
} finally {
|
|
641
|
+
if (singletonGuard) singletonGuard.liberar('auto-evolucion');
|
|
642
|
+
}
|
|
643
|
+
});
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
// ---------------------------------------------------------------------------
|
|
647
|
+
// Exports para tests unitarios
|
|
648
|
+
// (el entrypoint de stdin solo corre cuando se ejecuta directamente)
|
|
649
|
+
// ---------------------------------------------------------------------------
|
|
650
|
+
|
|
651
|
+
if (require.main !== module) {
|
|
652
|
+
module.exports = {
|
|
653
|
+
leerInvocacionesDelAgente,
|
|
654
|
+
ejecutarDriftAutomatico,
|
|
655
|
+
estaThrottledLoop,
|
|
656
|
+
rutasTracesRecientes,
|
|
657
|
+
// Exponer para tests de integración completos
|
|
658
|
+
_internals: {
|
|
659
|
+
analizarPayload,
|
|
660
|
+
filtrarVentana,
|
|
661
|
+
resumenPorAgente,
|
|
662
|
+
debeEmitirNudge,
|
|
663
|
+
construirDiagnosis,
|
|
664
|
+
},
|
|
665
|
+
};
|
|
666
|
+
}
|