@saulwade/swl-ses 1.5.1 → 1.5.2

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 (133) hide show
  1. package/CLAUDE.md +225 -209
  2. package/README.md +561 -561
  3. package/agentes/arquitecto-swl.md +33 -1
  4. package/agentes/nemesis-auditor-swl.md +59 -19
  5. package/bin/swl-mcp-server.js +214 -214
  6. package/comandos/swl/.evolved.json +22 -22
  7. package/comandos/swl/contribuir.md +233 -233
  8. package/comandos/swl/nemesis.md +230 -56
  9. package/gateway/lib/event-channel.js +191 -191
  10. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  11. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  12. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  13. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  14. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  15. package/habilidades/ejecutar-task-iterativo/SKILL.md +278 -278
  16. package/habilidades/eval-framework/SKILL.md +212 -212
  17. package/habilidades/feynman-auditor-swl/SKILL.md +123 -123
  18. package/habilidades/feynman-auditor-swl/recursos/preguntas-language-agnostic.md +108 -108
  19. package/habilidades/harness-claude-code/SKILL.md +299 -299
  20. package/habilidades/infra-github-actions/SKILL.md +166 -166
  21. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  22. package/habilidades/manejo-errores/.evolved.json +8 -8
  23. package/habilidades/meta-skills-estandar/SKILL.md +207 -4
  24. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  25. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  26. package/habilidades/nemesis-evaluacion-json/SKILL.md +266 -0
  27. package/habilidades/nemesis-redistribuir/SKILL.md +341 -0
  28. package/habilidades/node-experto/SKILL.md +94 -4
  29. package/habilidades/patrones-python/SKILL.md +229 -229
  30. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  31. package/habilidades/planear-fase/SKILL.md +319 -319
  32. package/habilidades/protocolo-revision-swl/SKILL.md +350 -276
  33. package/habilidades/release-semver/.evolved.json +8 -8
  34. package/habilidades/state-inconsistency-auditor-swl/SKILL.md +166 -166
  35. package/habilidades/state-inconsistency-auditor-swl/recursos/coupled-state-patterns.md +147 -147
  36. package/habilidades/tdd-workflow/SKILL.md +121 -4
  37. package/habilidades/testing-python/SKILL.md +340 -340
  38. package/habilidades/web-fetcher-routing/SKILL.md +75 -75
  39. package/hooks/check-update.js +31 -3
  40. package/hooks/claudemd-bloat-detector.js +161 -161
  41. package/hooks/lib/agent-routing.js +107 -107
  42. package/hooks/lib/auto-consolidator.js +335 -335
  43. package/hooks/lib/error-classifier.js +308 -308
  44. package/hooks/lib/merkle-audit.js +96 -96
  45. package/hooks/lib/provenance-tracker.js +191 -191
  46. package/hooks/lib/rate-limit-tracker.js +253 -253
  47. package/hooks/lib/resource-quota.js +122 -122
  48. package/hooks/lib/retry-jitter.js +165 -165
  49. package/hooks/lib/security-net.js +201 -201
  50. package/hooks/lib/skill-auditor.js +588 -588
  51. package/hooks/lib/sync-status.js +228 -228
  52. package/hooks/lib/taint-tracker.js +107 -107
  53. package/hooks/lib/text-similarity.js +241 -241
  54. package/hooks/lib/toon-compressor.js +245 -245
  55. package/hooks/registro-turnos.js +209 -209
  56. package/hooks/sugerir-regenerar-inventario.js +170 -170
  57. package/hooks/validar-formato-post-subagente.js +140 -140
  58. package/hooks/validar-memoria-hook.js +218 -218
  59. package/instintos/prompt-appendices.yaml +57 -57
  60. package/manifiestos/agent-output-schemas.json +57 -57
  61. package/manifiestos/modulos.json +1324 -1321
  62. package/manifiestos/skills-lock.json +1114 -1114
  63. package/package.json +2 -2
  64. package/plantillas/auditor-veto-template.md +105 -105
  65. package/plantillas/github-workflows/README.md +47 -47
  66. package/plantillas/github-workflows/release-please.yml +44 -44
  67. package/plantillas/github-workflows/swl-ci.yml +107 -107
  68. package/plantillas/github-workflows/swl-security.yml +51 -51
  69. package/plugin.json +353 -351
  70. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  71. package/reglas/arreglar-al-detectar.md +147 -147
  72. package/reglas/fragmentos-compartidos.md +152 -152
  73. package/reglas/harness-claude-code.md +213 -213
  74. package/reglas/registro-componentes-nuevos.md +192 -0
  75. package/reglas/usar-context7.md +226 -226
  76. package/schemas/diary-entry.schema.json +80 -80
  77. package/scripts/actualizar.js +110 -1
  78. package/scripts/audit-tools/audit-history.js +330 -330
  79. package/scripts/audit-tools/bundle-tracker.js +290 -290
  80. package/scripts/audit-tools/canary-monitor.js +352 -352
  81. package/scripts/audit-tools/code-profiler.js +605 -605
  82. package/scripts/audit-tools/dep-doctor.js +320 -320
  83. package/scripts/audit-tools/env-validator.js +206 -206
  84. package/scripts/audit-tools/lib/fs-walk.js +48 -48
  85. package/scripts/audit-tools/lib/output.js +23 -23
  86. package/scripts/audit-tools/migration-checker.js +392 -392
  87. package/scripts/audit-tools/pentest-scanner.js +1436 -1436
  88. package/scripts/benchmark-memoria.js +167 -167
  89. package/scripts/configurar-branch-protection.js +418 -418
  90. package/scripts/derivar-feature-list.js +489 -489
  91. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  92. package/scripts/doctor.js +27 -0
  93. package/scripts/field-report.js +199 -199
  94. package/scripts/generar-checklists-consolidados.js +273 -273
  95. package/scripts/generar-inventario.js +420 -420
  96. package/scripts/generar-matriz-lenguajes.js +271 -271
  97. package/scripts/lib/artefactos-python.js +43 -43
  98. package/scripts/lib/benchmark-metrics.js +160 -160
  99. package/scripts/lib/budget-enforcer.js +252 -252
  100. package/scripts/lib/configurar-ci.js +380 -380
  101. package/scripts/lib/contadores-inventario.js +217 -217
  102. package/scripts/lib/detectar-stack-detallado.js +307 -307
  103. package/scripts/lib/diary-entry.js +234 -234
  104. package/scripts/lib/eval-metrics-store.js +218 -218
  105. package/scripts/lib/eval-quality.js +171 -171
  106. package/scripts/lib/eval-schemas.js +144 -144
  107. package/scripts/lib/eval-self-correct.js +106 -106
  108. package/scripts/lib/eval-validator.js +185 -185
  109. package/scripts/lib/expandir-targets.js +71 -71
  110. package/scripts/lib/jaccard-similarity.js +98 -98
  111. package/scripts/lib/longmemeval-runner.js +125 -125
  112. package/scripts/lib/mcp_config.py +127 -0
  113. package/scripts/lib/npm-version.js +261 -261
  114. package/scripts/lib/paquetes-conocidos.js +50 -50
  115. package/scripts/lib/prompt-builder.js +264 -264
  116. package/scripts/lib/rrf-fusion.js +175 -175
  117. package/scripts/lib/scoring-instintos.js +277 -277
  118. package/scripts/lib/semantic-search.js +252 -252
  119. package/scripts/lib/toml-merge.js +204 -204
  120. package/scripts/lib/transformadores/codex.js +375 -375
  121. package/scripts/lib/transformadores/cursor.js +359 -359
  122. package/scripts/limpiar-artefactos-python.js +131 -131
  123. package/scripts/mcp-orchestrator.py +8 -18
  124. package/scripts/mcp-pool-manager.py +12 -23
  125. package/scripts/mcp-server/README.md +170 -170
  126. package/scripts/mcp-server/auth.js +105 -105
  127. package/scripts/mcp-server/cache.js +106 -106
  128. package/scripts/mcp-server/telemetry.js +78 -78
  129. package/scripts/migrar-csv-a-array.js +168 -168
  130. package/scripts/migrar-fase-dominio.js +201 -201
  131. package/scripts/publicar.js +511 -511
  132. package/scripts/run-eval.js +141 -141
  133. package/scripts/validar-userland-vacio.js +110 -110
@@ -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
+ };