@saulwade/swl-ses 1.3.8 → 1.4.1

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.
Files changed (148) hide show
  1. package/CLAUDE.md +15 -6
  2. package/README.md +15 -14
  3. package/agentes/nemesis-auditor-swl.md +161 -0
  4. package/bin/swl-mcp-server.js +187 -187
  5. package/bin/swl-webhook-server.js +198 -0
  6. package/comandos/swl/.evolved.json +22 -22
  7. package/comandos/swl/adoptar-proyecto.md +21 -1
  8. package/comandos/swl/claudemd.md +14 -1
  9. package/comandos/swl/contribuir.md +233 -233
  10. package/comandos/swl/exportar-vault.md +108 -0
  11. package/comandos/swl/nemesis.md +122 -0
  12. package/comandos/swl/nuevo-proyecto.md +24 -2
  13. package/comandos/swl/salud.md +34 -0
  14. package/comandos/swl/verificar.md +45 -0
  15. package/gateway/adapters/base.js +109 -0
  16. package/gateway/adapters/discord.js +167 -0
  17. package/gateway/adapters/email.js +221 -0
  18. package/gateway/adapters/slack.js +192 -0
  19. package/gateway/adapters/telegram.js +183 -0
  20. package/gateway/adapters/webhook.js +113 -0
  21. package/gateway/adapters/whatsapp.js +214 -0
  22. package/gateway/agent-executor.js +322 -0
  23. package/gateway/command-relay.js +271 -0
  24. package/gateway/cron/jobs.js +263 -0
  25. package/gateway/cron/scheduler.js +322 -0
  26. package/gateway/cron/store.js +335 -0
  27. package/gateway/index.js +320 -0
  28. package/gateway/lib/event-channel.js +191 -0
  29. package/gateway/session.js +131 -0
  30. package/gateway/webhook-server.js +324 -0
  31. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  32. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  33. package/habilidades/build-errors-nextjs/SKILL.md +55 -1
  34. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  35. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  36. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  37. package/habilidades/eval-framework/SKILL.md +212 -212
  38. package/habilidades/extractor-de-aprendizajes/SKILL.md +20 -10
  39. package/habilidades/feynman-auditor-swl/SKILL.md +123 -0
  40. package/habilidades/feynman-auditor-swl/recursos/preguntas-language-agnostic.md +108 -0
  41. package/habilidades/harness-claude-code/SKILL.md +299 -299
  42. package/habilidades/infra-github-actions/SKILL.md +166 -166
  43. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  44. package/habilidades/manejo-errores/.evolved.json +8 -8
  45. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  46. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  47. package/habilidades/nextjs-testing/SKILL.md +89 -5
  48. package/habilidades/node-experto/SKILL.md +37 -1
  49. package/habilidades/patrones-python/SKILL.md +229 -229
  50. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  51. package/habilidades/planear-fase/SKILL.md +319 -319
  52. package/habilidades/react-experto/SKILL.md +45 -4
  53. package/habilidades/release-semver/.evolved.json +8 -8
  54. package/habilidades/state-inconsistency-auditor-swl/SKILL.md +166 -0
  55. package/habilidades/state-inconsistency-auditor-swl/recursos/coupled-state-patterns.md +147 -0
  56. package/habilidades/tdd-workflow/SKILL.md +36 -4
  57. package/habilidades/testing-python/SKILL.md +340 -340
  58. package/habilidades/web-fetcher-routing/SKILL.md +75 -0
  59. package/hooks/claudemd-bloat-detector.js +161 -161
  60. package/hooks/inyeccion-contexto.js +8 -3
  61. package/hooks/lib/agent-routing.js +107 -107
  62. package/hooks/lib/auto-consolidator.js +335 -335
  63. package/hooks/lib/error-classifier.js +308 -308
  64. package/hooks/lib/merkle-audit.js +96 -96
  65. package/hooks/lib/provenance-tracker.js +191 -191
  66. package/hooks/lib/rate-limit-ip.js +177 -0
  67. package/hooks/lib/rate-limit-tracker.js +253 -253
  68. package/hooks/lib/resource-quota.js +122 -122
  69. package/hooks/lib/retry-jitter.js +165 -165
  70. package/hooks/lib/security-net.js +201 -0
  71. package/hooks/lib/skill-auditor.js +588 -588
  72. package/hooks/lib/sync-status.js +228 -228
  73. package/hooks/lib/taint-tracker.js +107 -107
  74. package/hooks/lib/text-similarity.js +241 -241
  75. package/hooks/lib/toon-compressor.js +245 -245
  76. package/hooks/lib/webhook-dedup.js +184 -0
  77. package/hooks/lib/webhook-verify.js +123 -0
  78. package/hooks/proteccion-rutas.js +120 -15
  79. package/hooks/registro-turnos.js +209 -209
  80. package/hooks/sugerir-regenerar-inventario.js +170 -170
  81. package/hooks/validar-formato-post-subagente.js +140 -140
  82. package/hooks/validar-memoria-hook.js +218 -218
  83. package/instintos/prompt-appendices.yaml +57 -57
  84. package/manifiestos/agent-output-schemas.json +57 -57
  85. package/manifiestos/modulos.json +31 -0
  86. package/manifiestos/skills-lock.json +1114 -1093
  87. package/package.json +6 -4
  88. package/plantillas/auditor-veto-template.md +105 -105
  89. package/plantillas/github-workflows/README.md +47 -47
  90. package/plantillas/github-workflows/release-please.yml +44 -44
  91. package/plantillas/github-workflows/swl-ci.yml +107 -107
  92. package/plantillas/github-workflows/swl-security.yml +51 -51
  93. package/plugin.json +2 -2
  94. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  95. package/reglas/arreglar-al-detectar.md +147 -147
  96. package/reglas/fragmentos-compartidos.md +152 -152
  97. package/reglas/harness-claude-code.md +213 -213
  98. package/reglas/usar-context7.md +226 -226
  99. package/reglas/usar-sistema-swl.md +251 -0
  100. package/schemas/diary-entry.schema.json +80 -80
  101. package/scripts/audit-tools/audit-history.js +330 -0
  102. package/scripts/audit-tools/bundle-tracker.js +290 -0
  103. package/scripts/audit-tools/canary-monitor.js +352 -0
  104. package/scripts/audit-tools/code-profiler.js +605 -0
  105. package/scripts/audit-tools/dep-doctor.js +320 -0
  106. package/scripts/audit-tools/env-validator.js +206 -0
  107. package/scripts/audit-tools/lib/fs-walk.js +48 -0
  108. package/scripts/audit-tools/lib/output.js +23 -0
  109. package/scripts/audit-tools/migration-checker.js +392 -0
  110. package/scripts/audit-tools/pentest-scanner.js +1436 -0
  111. package/scripts/benchmark-memoria.js +167 -167
  112. package/scripts/comandos/skills.js +251 -2
  113. package/scripts/configurar-branch-protection.js +418 -418
  114. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  115. package/scripts/field-report.js +199 -199
  116. package/scripts/generar-checklists-consolidados.js +273 -273
  117. package/scripts/generar-inventario.js +420 -420
  118. package/scripts/generar-matriz-lenguajes.js +271 -271
  119. package/scripts/lib/artefactos-python.js +43 -43
  120. package/scripts/lib/benchmark-metrics.js +160 -160
  121. package/scripts/lib/budget-enforcer.js +252 -252
  122. package/scripts/lib/configurar-ci.js +380 -380
  123. package/scripts/lib/contadores-inventario.js +217 -217
  124. package/scripts/lib/detectar-stack-detallado.js +307 -307
  125. package/scripts/lib/diary-entry.js +234 -234
  126. package/scripts/lib/eval-metrics-store.js +218 -218
  127. package/scripts/lib/eval-quality.js +171 -171
  128. package/scripts/lib/eval-schemas.js +144 -144
  129. package/scripts/lib/eval-self-correct.js +106 -106
  130. package/scripts/lib/eval-validator.js +185 -185
  131. package/scripts/lib/jaccard-similarity.js +98 -98
  132. package/scripts/lib/longmemeval-runner.js +125 -125
  133. package/scripts/lib/npm-version.js +261 -261
  134. package/scripts/lib/paquetes-conocidos.js +50 -50
  135. package/scripts/lib/prompt-builder.js +264 -264
  136. package/scripts/lib/rrf-fusion.js +175 -175
  137. package/scripts/lib/scoring-instintos.js +277 -277
  138. package/scripts/lib/semantic-search.js +252 -252
  139. package/scripts/limpiar-artefactos-python.js +131 -131
  140. package/scripts/mcp-server/README.md +128 -128
  141. package/scripts/mcp-server/handlers.js +206 -206
  142. package/scripts/migrar-csv-a-array.js +168 -168
  143. package/scripts/migrar-fase-dominio.js +201 -201
  144. package/scripts/publicar.js +511 -511
  145. package/scripts/run-eval.js +141 -141
  146. package/scripts/validar-manifest.js +195 -195
  147. package/scripts/validar-userland-vacio.js +110 -110
  148. package/scripts/verificar-release.js +110 -0
@@ -1,335 +1,335 @@
1
- 'use strict';
2
-
3
- /**
4
- * Auto-Consolidator — Consolidación automática de aprendizajes sin intervención humana.
5
- *
6
- * Cierra el ciclo: captura automática (extraccion-aprendizajes.js) →
7
- * consolidación automática (este módulo) → aplicación (instintos/skills).
8
- *
9
- * Criterios conservadores para operar sin supervisión:
10
- * - Solo promueve aprendizajes con ≥3 confirmaciones cruzadas
11
- * - Solo degrada con ≥3 contradicciones
12
- * - Solo poda entradas con >30 días sin referencia
13
- * - Nunca modifica CLAUDE.md ni skills (eso requiere /swl:aprender manual)
14
- * - Solo opera sobre APRENDIZAJES.md e instintos/proyecto.yaml
15
- *
16
- * Zero dependencias externas.
17
- *
18
- * @module hooks/lib/auto-consolidator
19
- */
20
-
21
- const fs = require('fs');
22
- const path = require('path');
23
-
24
- // ---------------------------------------------------------------------------
25
- // Constantes
26
- // ---------------------------------------------------------------------------
27
-
28
- /** Días sin referencia para considerar un aprendizaje como candidato a poda. */
29
- const DIAS_PODA = 30;
30
-
31
- /** Mínimo de confirmaciones para promover un aprendizaje. */
32
- const MIN_CONFIRMACIONES_PROMOCION = 3;
33
-
34
- /** Mínimo de contradicciones para degradar un aprendizaje. */
35
- const MIN_CONTRADICCIONES_DEGRADACION = 3;
36
-
37
- /** Máximo de entradas automáticas a procesar por consolidación. */
38
- const MAX_ENTRADAS_POR_CICLO = 50;
39
-
40
- /** Patrón para detectar entradas automáticas del hook extraccion-aprendizajes. */
41
- const PATRON_ENTRADA_AUTO = /^## \[(\d{4}-\d{2}-\d{2})\] ([\w\u00C0-\u024F-]+) — (.+)$/;
42
-
43
- /** Patrón para detectar confirmaciones. */
44
- const PATRON_CONFIRMADO = /\[CONFIRMADO(?:\s+x(\d+))?\]/;
45
-
46
- /** Patrón para detectar contradicciones. */
47
- const PATRON_CONTRADICHO = /\[CONTRADICHO(?::\s*\d{4}-\d{2}-\d{2})?\]/g;
48
-
49
- /** Patrón para detectar degradados. */
50
- const PATRON_DEGRADADO = /\[DEGRADADO\]/;
51
-
52
- // ---------------------------------------------------------------------------
53
- // Parsing de APRENDIZAJES.md
54
- // ---------------------------------------------------------------------------
55
-
56
- /**
57
- * @typedef {object} EntradaAprendizaje
58
- * @property {string} fecha - YYYY-MM-DD
59
- * @property {string} tipo - anti-patrón, patrón, gotcha, decisión, bug-fix, descubrimiento
60
- * @property {string} titulo - Descripción corta
61
- * @property {string} contenido - Contenido completo de la entrada
62
- * @property {number} confirmaciones - Número de confirmaciones
63
- * @property {number} contradicciones - Número de contradicciones
64
- * @property {boolean} degradado - Si ya fue marcado como degradado
65
- * @property {number} lineaInicio - Línea donde empieza en el archivo
66
- * @property {number} lineaFin - Línea donde termina
67
- */
68
-
69
- /**
70
- * Parsea APRENDIZAJES.md en entradas estructuradas.
71
- *
72
- * @param {string} contenido
73
- * @returns {EntradaAprendizaje[]}
74
- */
75
- function parsearAprendizajes(contenido) {
76
- const lineas = contenido.split('\n');
77
- const entradas = [];
78
- let actual = null;
79
-
80
- for (let i = 0; i < lineas.length; i++) {
81
- const match = lineas[i].match(PATRON_ENTRADA_AUTO);
82
- if (match) {
83
- if (actual) {
84
- actual.lineaFin = i - 1;
85
- actual.contenido = lineas.slice(actual.lineaInicio, i).join('\n');
86
- entradas.push(actual);
87
- }
88
-
89
- const [, fecha, tipo, titulo] = match;
90
-
91
- // Contar confirmaciones
92
- const textoHastaAhora = lineas[i];
93
- const confMatch = textoHastaAhora.match(PATRON_CONFIRMADO);
94
- const confirmaciones = confMatch ? parseInt(confMatch[1] || '1', 10) : 0;
95
-
96
- actual = {
97
- fecha,
98
- tipo,
99
- titulo,
100
- contenido: '',
101
- confirmaciones,
102
- contradicciones: 0,
103
- degradado: PATRON_DEGRADADO.test(lineas[i]),
104
- lineaInicio: i,
105
- lineaFin: i,
106
- };
107
- } else if (actual) {
108
- // Buscar confirmaciones y contradicciones en el contenido
109
- const confMatch = lineas[i].match(PATRON_CONFIRMADO);
110
- if (confMatch) {
111
- actual.confirmaciones = Math.max(actual.confirmaciones, parseInt(confMatch[1] || '1', 10));
112
- }
113
- const contMatches = lineas[i].match(PATRON_CONTRADICHO);
114
- if (contMatches) actual.contradicciones += contMatches.length;
115
- if (PATRON_DEGRADADO.test(lineas[i])) actual.degradado = true;
116
- }
117
- }
118
-
119
- if (actual) {
120
- actual.lineaFin = lineas.length - 1;
121
- actual.contenido = lineas.slice(actual.lineaInicio).join('\n');
122
- entradas.push(actual);
123
- }
124
-
125
- return entradas;
126
- }
127
-
128
- // ---------------------------------------------------------------------------
129
- // Deduplicación
130
- // ---------------------------------------------------------------------------
131
-
132
- /**
133
- * Detecta entradas duplicadas por similitud de título.
134
- * Usa comparación simple normalizada (lowercase, sin acentos, sin puntuación).
135
- *
136
- * @param {EntradaAprendizaje[]} entradas
137
- * @returns {number[]} Índices de entradas duplicadas (las más antiguas se eliminan)
138
- */
139
- function detectarDuplicados(entradas) {
140
- const duplicados = [];
141
- const vistos = new Map(); // título normalizado → índice de la versión más reciente
142
-
143
- for (let i = 0; i < entradas.length; i++) {
144
- const normalizado = entradas[i].titulo
145
- .toLowerCase()
146
- .normalize('NFD').replace(/[\u0300-\u036f]/g, '')
147
- .replace(/[^a-z0-9\s]/g, '')
148
- .replace(/\s+/g, ' ')
149
- .trim();
150
-
151
- if (vistos.has(normalizado)) {
152
- // Mantener la más reciente (mayor fecha) o la más confirmada
153
- const prevIdx = vistos.get(normalizado);
154
- const prev = entradas[prevIdx];
155
- const curr = entradas[i];
156
-
157
- // Priorizar por confirmaciones, luego por fecha
158
- const currGana = curr.confirmaciones > prev.confirmaciones
159
- || (curr.confirmaciones === prev.confirmaciones && curr.fecha > prev.fecha);
160
- if (currGana) {
161
- duplicados.push(prevIdx);
162
- vistos.set(normalizado, i);
163
- } else {
164
- duplicados.push(i);
165
- }
166
- } else {
167
- vistos.set(normalizado, i);
168
- }
169
- }
170
-
171
- return duplicados;
172
- }
173
-
174
- // ---------------------------------------------------------------------------
175
- // Poda
176
- // ---------------------------------------------------------------------------
177
-
178
- /**
179
- * Identifica entradas candidatas a poda (>30 días sin actualización, sin confirmaciones).
180
- *
181
- * @param {EntradaAprendizaje[]} entradas
182
- * @returns {number[]} Índices de entradas a podar
183
- */
184
- function detectarObsoletos(entradas) {
185
- const ahora = new Date();
186
- const obsoletos = [];
187
-
188
- for (let i = 0; i < entradas.length; i++) {
189
- const e = entradas[i];
190
-
191
- // Nunca podar entradas confirmadas
192
- if (e.confirmaciones >= 2) continue;
193
-
194
- // Nunca podar secciones curadas manualmente (no tienen fecha de hook)
195
- if (!e.fecha) continue;
196
-
197
- const fechaEntrada = new Date(e.fecha);
198
- const diasDesde = (ahora - fechaEntrada) / (1000 * 60 * 60 * 24);
199
-
200
- // Podar si: >30 días + 0 confirmaciones + no degradado (ya marcado)
201
- if (diasDesde > DIAS_PODA && e.confirmaciones === 0 && !e.degradado) {
202
- obsoletos.push(i);
203
- }
204
- }
205
-
206
- return obsoletos;
207
- }
208
-
209
- // ---------------------------------------------------------------------------
210
- // Consolidación
211
- // ---------------------------------------------------------------------------
212
-
213
- /**
214
- * Ejecuta la consolidación automática sobre APRENDIZAJES.md.
215
- *
216
- * Operaciones (en orden):
217
- * 1. Parsear entradas
218
- * 2. Detectar y eliminar duplicados
219
- * 3. Podar obsoletos (>30 días sin confirmación)
220
- * 4. Marcar degradados (≥3 contradicciones)
221
- *
222
- * NO hace (requiere /swl:aprender manual):
223
- * - Promover a CLAUDE.md (tipo A)
224
- * - Crear skills nuevos (tipo C)
225
- * - Modificar comandos (tipo D)
226
- *
227
- * @param {string} baseDir - Raíz del proyecto
228
- * @returns {{ ejecutado: boolean, detalles: object }}
229
- */
230
- function consolidar(baseDir) {
231
- const ruta = path.join(baseDir, '.planning', 'APRENDIZAJES.md');
232
-
233
- if (!fs.existsSync(ruta)) {
234
- return { ejecutado: false, detalles: { razon: 'archivo no existe' } };
235
- }
236
-
237
- const contenidoOriginal = fs.readFileSync(ruta, 'utf8');
238
- const entradas = parsearAprendizajes(contenidoOriginal);
239
-
240
- if (entradas.length === 0) {
241
- return { ejecutado: false, detalles: { razon: 'sin entradas para consolidar' } };
242
- }
243
-
244
- // Limitar procesamiento por ciclo
245
- const aConsolidar = entradas.slice(-MAX_ENTRADAS_POR_CICLO);
246
-
247
- // Detectar duplicados
248
- const duplicados = detectarDuplicados(aConsolidar);
249
-
250
- // Detectar obsoletos
251
- const obsoletos = detectarObsoletos(aConsolidar);
252
-
253
- // Detectar degradables (≥3 contradicciones sin marcar)
254
- const degradables = [];
255
- for (let i = 0; i < aConsolidar.length; i++) {
256
- if (aConsolidar[i].contradicciones >= MIN_CONTRADICCIONES_DEGRADACION && !aConsolidar[i].degradado) {
257
- degradables.push(i);
258
- }
259
- }
260
-
261
- // Si no hay nada que hacer, salir
262
- const totalAcciones = duplicados.length + obsoletos.length + degradables.length;
263
- if (totalAcciones === 0) {
264
- return {
265
- ejecutado: false,
266
- detalles: {
267
- razon: 'sin acciones necesarias',
268
- entradasAnalizadas: aConsolidar.length,
269
- },
270
- };
271
- }
272
-
273
- // Construir índice de líneas a eliminar y entradas a modificar
274
- const lineasAEliminar = new Set();
275
- const indicesAEliminar = new Set([...duplicados, ...obsoletos]);
276
-
277
- for (const idx of indicesAEliminar) {
278
- const e = aConsolidar[idx];
279
- for (let l = e.lineaInicio; l <= e.lineaFin; l++) {
280
- lineasAEliminar.add(l);
281
- }
282
- }
283
-
284
- // Reconstruir archivo
285
- const lineas = contenidoOriginal.split('\n');
286
- const nuevasLineas = [];
287
-
288
- for (let i = 0; i < lineas.length; i++) {
289
- if (lineasAEliminar.has(i)) continue;
290
-
291
- // Marcar degradables
292
- let linea = lineas[i];
293
- for (const idx of degradables) {
294
- const e = aConsolidar[idx];
295
- if (i === e.lineaInicio && !PATRON_DEGRADADO.test(linea)) {
296
- linea = linea + ' [DEGRADADO]';
297
- }
298
- }
299
-
300
- nuevasLineas.push(linea);
301
- }
302
-
303
- // Limpiar líneas vacías consecutivas (más de 2)
304
- const resultado = nuevasLineas.join('\n').replace(/\n{4,}/g, '\n\n\n');
305
-
306
- // Escribir resultado
307
- fs.writeFileSync(ruta, resultado, 'utf8');
308
-
309
- return {
310
- ejecutado: true,
311
- detalles: {
312
- entradasAnalizadas: aConsolidar.length,
313
- duplicadosEliminados: duplicados.length,
314
- obsoletoPodados: obsoletos.length,
315
- degradados: degradables.length,
316
- lineasAntes: lineas.length,
317
- lineasDespues: resultado.split('\n').length,
318
- },
319
- };
320
- }
321
-
322
- // ---------------------------------------------------------------------------
323
- // Exports
324
- // ---------------------------------------------------------------------------
325
-
326
- module.exports = {
327
- consolidar,
328
- parsearAprendizajes,
329
- detectarDuplicados,
330
- detectarObsoletos,
331
- DIAS_PODA,
332
- MIN_CONFIRMACIONES_PROMOCION,
333
- MIN_CONTRADICCIONES_DEGRADACION,
334
- MAX_ENTRADAS_POR_CICLO,
335
- };
1
+ 'use strict';
2
+
3
+ /**
4
+ * Auto-Consolidator — Consolidación automática de aprendizajes sin intervención humana.
5
+ *
6
+ * Cierra el ciclo: captura automática (extraccion-aprendizajes.js) →
7
+ * consolidación automática (este módulo) → aplicación (instintos/skills).
8
+ *
9
+ * Criterios conservadores para operar sin supervisión:
10
+ * - Solo promueve aprendizajes con ≥3 confirmaciones cruzadas
11
+ * - Solo degrada con ≥3 contradicciones
12
+ * - Solo poda entradas con >30 días sin referencia
13
+ * - Nunca modifica CLAUDE.md ni skills (eso requiere /swl:aprender manual)
14
+ * - Solo opera sobre APRENDIZAJES.md e instintos/proyecto.yaml
15
+ *
16
+ * Zero dependencias externas.
17
+ *
18
+ * @module hooks/lib/auto-consolidator
19
+ */
20
+
21
+ const fs = require('fs');
22
+ const path = require('path');
23
+
24
+ // ---------------------------------------------------------------------------
25
+ // Constantes
26
+ // ---------------------------------------------------------------------------
27
+
28
+ /** Días sin referencia para considerar un aprendizaje como candidato a poda. */
29
+ const DIAS_PODA = 30;
30
+
31
+ /** Mínimo de confirmaciones para promover un aprendizaje. */
32
+ const MIN_CONFIRMACIONES_PROMOCION = 3;
33
+
34
+ /** Mínimo de contradicciones para degradar un aprendizaje. */
35
+ const MIN_CONTRADICCIONES_DEGRADACION = 3;
36
+
37
+ /** Máximo de entradas automáticas a procesar por consolidación. */
38
+ const MAX_ENTRADAS_POR_CICLO = 50;
39
+
40
+ /** Patrón para detectar entradas automáticas del hook extraccion-aprendizajes. */
41
+ const PATRON_ENTRADA_AUTO = /^## \[(\d{4}-\d{2}-\d{2})\] ([\w\u00C0-\u024F-]+) — (.+)$/;
42
+
43
+ /** Patrón para detectar confirmaciones. */
44
+ const PATRON_CONFIRMADO = /\[CONFIRMADO(?:\s+x(\d+))?\]/;
45
+
46
+ /** Patrón para detectar contradicciones. */
47
+ const PATRON_CONTRADICHO = /\[CONTRADICHO(?::\s*\d{4}-\d{2}-\d{2})?\]/g;
48
+
49
+ /** Patrón para detectar degradados. */
50
+ const PATRON_DEGRADADO = /\[DEGRADADO\]/;
51
+
52
+ // ---------------------------------------------------------------------------
53
+ // Parsing de APRENDIZAJES.md
54
+ // ---------------------------------------------------------------------------
55
+
56
+ /**
57
+ * @typedef {object} EntradaAprendizaje
58
+ * @property {string} fecha - YYYY-MM-DD
59
+ * @property {string} tipo - anti-patrón, patrón, gotcha, decisión, bug-fix, descubrimiento
60
+ * @property {string} titulo - Descripción corta
61
+ * @property {string} contenido - Contenido completo de la entrada
62
+ * @property {number} confirmaciones - Número de confirmaciones
63
+ * @property {number} contradicciones - Número de contradicciones
64
+ * @property {boolean} degradado - Si ya fue marcado como degradado
65
+ * @property {number} lineaInicio - Línea donde empieza en el archivo
66
+ * @property {number} lineaFin - Línea donde termina
67
+ */
68
+
69
+ /**
70
+ * Parsea APRENDIZAJES.md en entradas estructuradas.
71
+ *
72
+ * @param {string} contenido
73
+ * @returns {EntradaAprendizaje[]}
74
+ */
75
+ function parsearAprendizajes(contenido) {
76
+ const lineas = contenido.split('\n');
77
+ const entradas = [];
78
+ let actual = null;
79
+
80
+ for (let i = 0; i < lineas.length; i++) {
81
+ const match = lineas[i].match(PATRON_ENTRADA_AUTO);
82
+ if (match) {
83
+ if (actual) {
84
+ actual.lineaFin = i - 1;
85
+ actual.contenido = lineas.slice(actual.lineaInicio, i).join('\n');
86
+ entradas.push(actual);
87
+ }
88
+
89
+ const [, fecha, tipo, titulo] = match;
90
+
91
+ // Contar confirmaciones
92
+ const textoHastaAhora = lineas[i];
93
+ const confMatch = textoHastaAhora.match(PATRON_CONFIRMADO);
94
+ const confirmaciones = confMatch ? parseInt(confMatch[1] || '1', 10) : 0;
95
+
96
+ actual = {
97
+ fecha,
98
+ tipo,
99
+ titulo,
100
+ contenido: '',
101
+ confirmaciones,
102
+ contradicciones: 0,
103
+ degradado: PATRON_DEGRADADO.test(lineas[i]),
104
+ lineaInicio: i,
105
+ lineaFin: i,
106
+ };
107
+ } else if (actual) {
108
+ // Buscar confirmaciones y contradicciones en el contenido
109
+ const confMatch = lineas[i].match(PATRON_CONFIRMADO);
110
+ if (confMatch) {
111
+ actual.confirmaciones = Math.max(actual.confirmaciones, parseInt(confMatch[1] || '1', 10));
112
+ }
113
+ const contMatches = lineas[i].match(PATRON_CONTRADICHO);
114
+ if (contMatches) actual.contradicciones += contMatches.length;
115
+ if (PATRON_DEGRADADO.test(lineas[i])) actual.degradado = true;
116
+ }
117
+ }
118
+
119
+ if (actual) {
120
+ actual.lineaFin = lineas.length - 1;
121
+ actual.contenido = lineas.slice(actual.lineaInicio).join('\n');
122
+ entradas.push(actual);
123
+ }
124
+
125
+ return entradas;
126
+ }
127
+
128
+ // ---------------------------------------------------------------------------
129
+ // Deduplicación
130
+ // ---------------------------------------------------------------------------
131
+
132
+ /**
133
+ * Detecta entradas duplicadas por similitud de título.
134
+ * Usa comparación simple normalizada (lowercase, sin acentos, sin puntuación).
135
+ *
136
+ * @param {EntradaAprendizaje[]} entradas
137
+ * @returns {number[]} Índices de entradas duplicadas (las más antiguas se eliminan)
138
+ */
139
+ function detectarDuplicados(entradas) {
140
+ const duplicados = [];
141
+ const vistos = new Map(); // título normalizado → índice de la versión más reciente
142
+
143
+ for (let i = 0; i < entradas.length; i++) {
144
+ const normalizado = entradas[i].titulo
145
+ .toLowerCase()
146
+ .normalize('NFD').replace(/[\u0300-\u036f]/g, '')
147
+ .replace(/[^a-z0-9\s]/g, '')
148
+ .replace(/\s+/g, ' ')
149
+ .trim();
150
+
151
+ if (vistos.has(normalizado)) {
152
+ // Mantener la más reciente (mayor fecha) o la más confirmada
153
+ const prevIdx = vistos.get(normalizado);
154
+ const prev = entradas[prevIdx];
155
+ const curr = entradas[i];
156
+
157
+ // Priorizar por confirmaciones, luego por fecha
158
+ const currGana = curr.confirmaciones > prev.confirmaciones
159
+ || (curr.confirmaciones === prev.confirmaciones && curr.fecha > prev.fecha);
160
+ if (currGana) {
161
+ duplicados.push(prevIdx);
162
+ vistos.set(normalizado, i);
163
+ } else {
164
+ duplicados.push(i);
165
+ }
166
+ } else {
167
+ vistos.set(normalizado, i);
168
+ }
169
+ }
170
+
171
+ return duplicados;
172
+ }
173
+
174
+ // ---------------------------------------------------------------------------
175
+ // Poda
176
+ // ---------------------------------------------------------------------------
177
+
178
+ /**
179
+ * Identifica entradas candidatas a poda (>30 días sin actualización, sin confirmaciones).
180
+ *
181
+ * @param {EntradaAprendizaje[]} entradas
182
+ * @returns {number[]} Índices de entradas a podar
183
+ */
184
+ function detectarObsoletos(entradas) {
185
+ const ahora = new Date();
186
+ const obsoletos = [];
187
+
188
+ for (let i = 0; i < entradas.length; i++) {
189
+ const e = entradas[i];
190
+
191
+ // Nunca podar entradas confirmadas
192
+ if (e.confirmaciones >= 2) continue;
193
+
194
+ // Nunca podar secciones curadas manualmente (no tienen fecha de hook)
195
+ if (!e.fecha) continue;
196
+
197
+ const fechaEntrada = new Date(e.fecha);
198
+ const diasDesde = (ahora - fechaEntrada) / (1000 * 60 * 60 * 24);
199
+
200
+ // Podar si: >30 días + 0 confirmaciones + no degradado (ya marcado)
201
+ if (diasDesde > DIAS_PODA && e.confirmaciones === 0 && !e.degradado) {
202
+ obsoletos.push(i);
203
+ }
204
+ }
205
+
206
+ return obsoletos;
207
+ }
208
+
209
+ // ---------------------------------------------------------------------------
210
+ // Consolidación
211
+ // ---------------------------------------------------------------------------
212
+
213
+ /**
214
+ * Ejecuta la consolidación automática sobre APRENDIZAJES.md.
215
+ *
216
+ * Operaciones (en orden):
217
+ * 1. Parsear entradas
218
+ * 2. Detectar y eliminar duplicados
219
+ * 3. Podar obsoletos (>30 días sin confirmación)
220
+ * 4. Marcar degradados (≥3 contradicciones)
221
+ *
222
+ * NO hace (requiere /swl:aprender manual):
223
+ * - Promover a CLAUDE.md (tipo A)
224
+ * - Crear skills nuevos (tipo C)
225
+ * - Modificar comandos (tipo D)
226
+ *
227
+ * @param {string} baseDir - Raíz del proyecto
228
+ * @returns {{ ejecutado: boolean, detalles: object }}
229
+ */
230
+ function consolidar(baseDir) {
231
+ const ruta = path.join(baseDir, '.planning', 'APRENDIZAJES.md');
232
+
233
+ if (!fs.existsSync(ruta)) {
234
+ return { ejecutado: false, detalles: { razon: 'archivo no existe' } };
235
+ }
236
+
237
+ const contenidoOriginal = fs.readFileSync(ruta, 'utf8');
238
+ const entradas = parsearAprendizajes(contenidoOriginal);
239
+
240
+ if (entradas.length === 0) {
241
+ return { ejecutado: false, detalles: { razon: 'sin entradas para consolidar' } };
242
+ }
243
+
244
+ // Limitar procesamiento por ciclo
245
+ const aConsolidar = entradas.slice(-MAX_ENTRADAS_POR_CICLO);
246
+
247
+ // Detectar duplicados
248
+ const duplicados = detectarDuplicados(aConsolidar);
249
+
250
+ // Detectar obsoletos
251
+ const obsoletos = detectarObsoletos(aConsolidar);
252
+
253
+ // Detectar degradables (≥3 contradicciones sin marcar)
254
+ const degradables = [];
255
+ for (let i = 0; i < aConsolidar.length; i++) {
256
+ if (aConsolidar[i].contradicciones >= MIN_CONTRADICCIONES_DEGRADACION && !aConsolidar[i].degradado) {
257
+ degradables.push(i);
258
+ }
259
+ }
260
+
261
+ // Si no hay nada que hacer, salir
262
+ const totalAcciones = duplicados.length + obsoletos.length + degradables.length;
263
+ if (totalAcciones === 0) {
264
+ return {
265
+ ejecutado: false,
266
+ detalles: {
267
+ razon: 'sin acciones necesarias',
268
+ entradasAnalizadas: aConsolidar.length,
269
+ },
270
+ };
271
+ }
272
+
273
+ // Construir índice de líneas a eliminar y entradas a modificar
274
+ const lineasAEliminar = new Set();
275
+ const indicesAEliminar = new Set([...duplicados, ...obsoletos]);
276
+
277
+ for (const idx of indicesAEliminar) {
278
+ const e = aConsolidar[idx];
279
+ for (let l = e.lineaInicio; l <= e.lineaFin; l++) {
280
+ lineasAEliminar.add(l);
281
+ }
282
+ }
283
+
284
+ // Reconstruir archivo
285
+ const lineas = contenidoOriginal.split('\n');
286
+ const nuevasLineas = [];
287
+
288
+ for (let i = 0; i < lineas.length; i++) {
289
+ if (lineasAEliminar.has(i)) continue;
290
+
291
+ // Marcar degradables
292
+ let linea = lineas[i];
293
+ for (const idx of degradables) {
294
+ const e = aConsolidar[idx];
295
+ if (i === e.lineaInicio && !PATRON_DEGRADADO.test(linea)) {
296
+ linea = linea + ' [DEGRADADO]';
297
+ }
298
+ }
299
+
300
+ nuevasLineas.push(linea);
301
+ }
302
+
303
+ // Limpiar líneas vacías consecutivas (más de 2)
304
+ const resultado = nuevasLineas.join('\n').replace(/\n{4,}/g, '\n\n\n');
305
+
306
+ // Escribir resultado
307
+ fs.writeFileSync(ruta, resultado, 'utf8');
308
+
309
+ return {
310
+ ejecutado: true,
311
+ detalles: {
312
+ entradasAnalizadas: aConsolidar.length,
313
+ duplicadosEliminados: duplicados.length,
314
+ obsoletoPodados: obsoletos.length,
315
+ degradados: degradables.length,
316
+ lineasAntes: lineas.length,
317
+ lineasDespues: resultado.split('\n').length,
318
+ },
319
+ };
320
+ }
321
+
322
+ // ---------------------------------------------------------------------------
323
+ // Exports
324
+ // ---------------------------------------------------------------------------
325
+
326
+ module.exports = {
327
+ consolidar,
328
+ parsearAprendizajes,
329
+ detectarDuplicados,
330
+ detectarObsoletos,
331
+ DIAS_PODA,
332
+ MIN_CONFIRMACIONES_PROMOCION,
333
+ MIN_CONTRADICCIONES_DEGRADACION,
334
+ MAX_ENTRADAS_POR_CICLO,
335
+ };