@saulwade/swl-ses 1.3.7 → 1.4.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.
Files changed (129) hide show
  1. package/CLAUDE.md +12 -4
  2. package/README.md +1 -1
  3. package/bin/swl-mcp-server.js +187 -187
  4. package/bin/swl-webhook-server.js +198 -0
  5. package/comandos/swl/.evolved.json +22 -22
  6. package/comandos/swl/adoptar-proyecto.md +21 -1
  7. package/comandos/swl/claudemd.md +14 -1
  8. package/comandos/swl/contribuir.md +233 -233
  9. package/comandos/swl/exportar-vault.md +207 -7
  10. package/comandos/swl/nuevo-proyecto.md +24 -2
  11. package/gateway/adapters/base.js +109 -0
  12. package/gateway/adapters/discord.js +167 -0
  13. package/gateway/adapters/email.js +221 -0
  14. package/gateway/adapters/slack.js +192 -0
  15. package/gateway/adapters/telegram.js +183 -0
  16. package/gateway/adapters/webhook.js +113 -0
  17. package/gateway/adapters/whatsapp.js +214 -0
  18. package/gateway/agent-executor.js +322 -0
  19. package/gateway/command-relay.js +271 -0
  20. package/gateway/cron/jobs.js +263 -0
  21. package/gateway/cron/scheduler.js +322 -0
  22. package/gateway/cron/store.js +335 -0
  23. package/gateway/index.js +320 -0
  24. package/gateway/lib/event-channel.js +191 -0
  25. package/gateway/session.js +131 -0
  26. package/gateway/webhook-server.js +324 -0
  27. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  28. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  29. package/habilidades/build-errors-nextjs/SKILL.md +55 -1
  30. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  31. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  32. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  33. package/habilidades/eval-framework/SKILL.md +212 -212
  34. package/habilidades/extractor-de-aprendizajes/SKILL.md +24 -10
  35. package/habilidades/harness-claude-code/SKILL.md +299 -299
  36. package/habilidades/infra-github-actions/SKILL.md +166 -166
  37. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  38. package/habilidades/manejo-errores/.evolved.json +8 -8
  39. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  40. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  41. package/habilidades/nextjs-testing/SKILL.md +89 -5
  42. package/habilidades/node-experto/SKILL.md +37 -1
  43. package/habilidades/patrones-python/SKILL.md +229 -229
  44. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  45. package/habilidades/planear-fase/SKILL.md +319 -319
  46. package/habilidades/react-experto/SKILL.md +45 -4
  47. package/habilidades/release-semver/.evolved.json +8 -8
  48. package/habilidades/swl-claudemd/SKILL.md +15 -1
  49. package/habilidades/tdd-workflow/SKILL.md +36 -4
  50. package/habilidades/testing-python/SKILL.md +340 -340
  51. package/hooks/claudemd-bloat-detector.js +161 -161
  52. package/hooks/inyeccion-contexto.js +8 -3
  53. package/hooks/lib/agent-routing.js +107 -107
  54. package/hooks/lib/auto-consolidator.js +335 -335
  55. package/hooks/lib/error-classifier.js +308 -308
  56. package/hooks/lib/merkle-audit.js +96 -96
  57. package/hooks/lib/provenance-tracker.js +191 -191
  58. package/hooks/lib/rate-limit-ip.js +177 -0
  59. package/hooks/lib/rate-limit-tracker.js +253 -253
  60. package/hooks/lib/resource-quota.js +122 -122
  61. package/hooks/lib/retry-jitter.js +165 -165
  62. package/hooks/lib/skill-auditor.js +588 -588
  63. package/hooks/lib/sync-status.js +228 -228
  64. package/hooks/lib/taint-tracker.js +107 -107
  65. package/hooks/lib/text-similarity.js +241 -241
  66. package/hooks/lib/toon-compressor.js +245 -245
  67. package/hooks/lib/webhook-dedup.js +184 -0
  68. package/hooks/lib/webhook-verify.js +123 -0
  69. package/hooks/proteccion-rutas.js +120 -15
  70. package/hooks/registro-turnos.js +209 -209
  71. package/hooks/sugerir-regenerar-inventario.js +170 -170
  72. package/hooks/validar-formato-post-subagente.js +140 -140
  73. package/hooks/validar-memoria-hook.js +218 -218
  74. package/instintos/prompt-appendices.yaml +57 -57
  75. package/manifiestos/agent-output-schemas.json +57 -57
  76. package/manifiestos/modulos.json +1 -0
  77. package/manifiestos/skills-lock.json +37 -37
  78. package/package.json +5 -3
  79. package/plantillas/auditor-veto-template.md +105 -105
  80. package/plantillas/github-workflows/README.md +47 -47
  81. package/plantillas/github-workflows/release-please.yml +44 -44
  82. package/plantillas/github-workflows/swl-ci.yml +107 -107
  83. package/plantillas/github-workflows/swl-security.yml +51 -51
  84. package/plugin.json +1 -1
  85. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  86. package/reglas/arreglar-al-detectar.md +147 -147
  87. package/reglas/fragmentos-compartidos.md +152 -152
  88. package/reglas/harness-claude-code.md +213 -213
  89. package/reglas/usar-context7.md +226 -226
  90. package/reglas/usar-sistema-swl.md +251 -0
  91. package/schemas/diary-entry.schema.json +80 -80
  92. package/scripts/benchmark-memoria.js +167 -167
  93. package/scripts/comandos/skills.js +251 -2
  94. package/scripts/configurar-branch-protection.js +418 -418
  95. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  96. package/scripts/field-report.js +199 -199
  97. package/scripts/generar-checklists-consolidados.js +273 -273
  98. package/scripts/generar-inventario.js +420 -420
  99. package/scripts/generar-matriz-lenguajes.js +271 -271
  100. package/scripts/lib/artefactos-python.js +43 -43
  101. package/scripts/lib/benchmark-metrics.js +160 -160
  102. package/scripts/lib/budget-enforcer.js +252 -252
  103. package/scripts/lib/configurar-ci.js +380 -380
  104. package/scripts/lib/contadores-inventario.js +217 -217
  105. package/scripts/lib/detectar-stack-detallado.js +307 -307
  106. package/scripts/lib/diary-entry.js +234 -234
  107. package/scripts/lib/eval-metrics-store.js +218 -218
  108. package/scripts/lib/eval-quality.js +171 -171
  109. package/scripts/lib/eval-schemas.js +144 -144
  110. package/scripts/lib/eval-self-correct.js +106 -106
  111. package/scripts/lib/eval-validator.js +185 -185
  112. package/scripts/lib/jaccard-similarity.js +98 -98
  113. package/scripts/lib/longmemeval-runner.js +125 -125
  114. package/scripts/lib/npm-version.js +261 -261
  115. package/scripts/lib/paquetes-conocidos.js +50 -50
  116. package/scripts/lib/prompt-builder.js +264 -264
  117. package/scripts/lib/rrf-fusion.js +175 -175
  118. package/scripts/lib/scoring-instintos.js +277 -277
  119. package/scripts/lib/semantic-search.js +252 -252
  120. package/scripts/limpiar-artefactos-python.js +131 -131
  121. package/scripts/mcp-server/README.md +128 -128
  122. package/scripts/mcp-server/handlers.js +206 -206
  123. package/scripts/migrar-csv-a-array.js +168 -168
  124. package/scripts/migrar-fase-dominio.js +201 -201
  125. package/scripts/publicar.js +511 -511
  126. package/scripts/run-eval.js +141 -141
  127. package/scripts/validar-manifest.js +195 -195
  128. package/scripts/validar-userland-vacio.js +110 -110
  129. 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
+ };