@saulwade/swl-ses 2.0.0 → 2.2.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 (97) hide show
  1. package/CLAUDE.md +196 -196
  2. package/README.md +579 -579
  3. package/agentes/_propose-step.md +90 -0
  4. package/agentes/implementador-swl.md +2 -0
  5. package/agentes/orquestador-swl.md +2 -0
  6. package/agentes/perfilador-usuario-swl.md +14 -1
  7. package/bin/swl-ses.js +64 -1
  8. package/comandos/swl/adoptar-proyecto.md +258 -255
  9. package/comandos/swl/aprender.md +828 -840
  10. package/comandos/swl/aprobar-plan.md +26 -37
  11. package/comandos/swl/autoresearch.md +12 -14
  12. package/comandos/swl/briefing.md +119 -0
  13. package/comandos/swl/checkpoint.md +10 -15
  14. package/comandos/swl/claudemd.md +239 -234
  15. package/comandos/swl/compactar.md +29 -2
  16. package/comandos/swl/configurar-ci.md +20 -19
  17. package/comandos/swl/cron.md +10 -12
  18. package/comandos/swl/discutir-fase.md +8 -5
  19. package/comandos/swl/ejecutar-fase.md +15 -2
  20. package/comandos/swl/evolucionar.md +6 -11
  21. package/comandos/swl/inbox.md +10 -10
  22. package/comandos/swl/modelo.md +7 -9
  23. package/comandos/swl/notificaciones.md +19 -116
  24. package/comandos/swl/nuevo-proyecto.md +205 -205
  25. package/comandos/swl/planear-fase.md +5 -3
  26. package/comandos/swl/release.md +46 -0
  27. package/comandos/swl/status.md +333 -279
  28. package/comandos/swl/verificar.md +817 -812
  29. package/habilidades/changelog-generator/scripts/parse-commits.js +6 -4
  30. package/habilidades/ejecutar-fase/SKILL.md +541 -518
  31. package/habilidades/planear-fase/SKILL.md +3 -2
  32. package/habilidades/swl-claudemd/SKILL.md +10 -6
  33. package/habilidades/tdd-workflow/SKILL.md +715 -713
  34. package/habilidades/validacion-ci-sistema/SKILL.md +17 -1
  35. package/hooks/calidad-pre-commit.js +5 -1
  36. package/hooks/check-update.js +39 -1
  37. package/hooks/lib/autonomia.js +208 -0
  38. package/hooks/lib/briefing.js +474 -0
  39. package/hooks/lib/propose-step.js +358 -0
  40. package/hooks/session-briefing.js +98 -0
  41. package/hooks/telemetria-skill-routing.js +100 -0
  42. package/instintos/autonomia.yaml +27 -0
  43. package/llms.txt +4 -4
  44. package/manifiestos/hooks-config.json +18 -0
  45. package/manifiestos/modulos.json +25 -3
  46. package/manifiestos/skills-lock.json +17 -17
  47. package/package.json +93 -93
  48. package/plugin.json +371 -371
  49. package/reglas/analizar-directorios-antes-de-escribir.md +228 -0
  50. package/reglas/consultar-vault-primero.md +195 -0
  51. package/reglas/debatir-antes-de-aceptar.md +158 -0
  52. package/reglas/git-coauthor.md +100 -0
  53. package/reglas/monitor-ci.md +309 -0
  54. package/reglas/registro-componentes-nuevos.md +38 -10
  55. package/reglas/sesiones-paralelas.md +180 -0
  56. package/reglas/usar-code-review-graph.md +155 -0
  57. package/reglas/verificar-citas-normativas.md +548 -0
  58. package/scripts/auditar-claudemd.js +38 -0
  59. package/scripts/cli/aprobar-plan.js +73 -0
  60. package/scripts/cli/briefing.js +23 -0
  61. package/scripts/cli/ciclo-evolucion.js +26 -0
  62. package/scripts/cli/configurar-ci.js +40 -0
  63. package/scripts/cli/derivar-feature-list.js +25 -0
  64. package/scripts/cli/detectar-host.js +27 -0
  65. package/scripts/cli/diary-entry.js +69 -0
  66. package/scripts/cli/execution-state.js +18 -0
  67. package/scripts/cli/gateway-notify.js +41 -0
  68. package/scripts/cli/liberar-fase.js +42 -0
  69. package/scripts/cli/loop-telemetry.js +125 -0
  70. package/scripts/cli/mark-evolved.js +56 -0
  71. package/scripts/cli/metricas-dora.js +26 -0
  72. package/scripts/cli/near-duplicate.js +55 -0
  73. package/scripts/cli/notificaciones.js +123 -0
  74. package/scripts/cli/propose-step.js +29 -0
  75. package/scripts/cli/schedule-parse.js +19 -0
  76. package/scripts/cli/sugerir-modelo.js +20 -0
  77. package/scripts/cli/verificar-plan.js +36 -0
  78. package/scripts/cli/verificar-trazabilidad.js +35 -0
  79. package/scripts/derivar-feature-list.js +1 -0
  80. package/scripts/instalador.js +52 -6
  81. package/scripts/lib/auditar-invocaciones-comandos.js +104 -0
  82. package/scripts/lib/ci-reader.js +193 -0
  83. package/scripts/lib/detectar-host-swl.js +175 -0
  84. package/scripts/lib/evidencia-release.js +322 -0
  85. package/scripts/lib/gate-hooks-requires.js +249 -0
  86. package/scripts/lib/gate-licencias.js +212 -0
  87. package/scripts/lib/git-metricas.js +257 -0
  88. package/scripts/lib/metricas-dora.js +204 -0
  89. package/scripts/lib/resolver-plan-fase.js +37 -0
  90. package/scripts/tui/ejecutores.js +1 -1
  91. package/scripts/validar-manifest.js +92 -1
  92. package/scripts/validar.js +13 -0
  93. package/scripts/verificar-evolucion.js +54 -4
  94. package/scripts/verificar-release.js +102 -0
  95. package/scripts/verificar-trazabilidad.js +12 -6
  96. package/reglas/arquitectura.evolved.json +0 -7
  97. package/reglas/seguridad.evolved.json +0 -7
@@ -0,0 +1,548 @@
1
+ # Regla: Verificar citas verificables antes de escribirlas o aceptarlas
2
+
3
+ Esta regla es OBLIGATORIA y aplica a todo contenido que Claude genere o acepte
4
+ como válido en cualquier proyecto del usuario donde aparezca una cita
5
+ verificable. Cubre dos familias del mismo principio:
6
+
7
+ 1. **Citas normativas / legales / documentales**: artículo de ley o reglamento,
8
+ sección de un estándar, número de RFC/ISO, página de documento oficial,
9
+ referencia a un acuerdo institucional, nombre de un programa o catálogo
10
+ oficial, fecha de promulgación, o atribución de fundamento normativo a una
11
+ entidad de datos.
12
+ 2. **Citas técnicas archivo:línea**: cualquier afirmación que cite una
13
+ ubicación específica del codebase (`path/al/archivo.py:123`,
14
+ `module/file.ts:45-90`, `schema.sql:438`) en reportes de auditoría,
15
+ análisis de bugs, hallazgos de seguridad, code review, post-mortems,
16
+ reportes de revisión externa, o conclusiones que tú mismo escribas
17
+ acerca del código.
18
+
19
+ Las dos familias comparten el mismo problema y la misma defensa: el agente
20
+ o el reporte afirma evidencia específica, y esa evidencia puede inventarse,
21
+ trasladarse mal o estar desactualizada. El antídoto es siempre el mismo:
22
+ **`grep` / `Read` sobre la fuente real antes de aceptar o re-emitir la cita**.
23
+
24
+ ---
25
+
26
+ ## Principio
27
+
28
+ > Antes de escribir una cita verificable (normativa o técnica archivo:línea)
29
+ > en código, UI, documentación, reportes, comentarios, docstrings o
30
+ > conclusiones, **debes verificarla con `grep` o `Read` sobre la fuente real
31
+ > del proyecto**. Si no puedes encontrar evidencia textual de la cita, NO la
32
+ > escribas. Una cita plausible que suena correcta NO es una cita verificada.
33
+ >
34
+ > El mismo principio aplica cuando ACEPTAS una cita ajena: un reporte de
35
+ > otro agente, un hallazgo de auditoría, una afirmación del usuario sobre
36
+ > "el bug está en X:42". Si vas a actuar sobre la cita (priorizarla,
37
+ > remediarla, reproducirla en tu propio output), verifícala primero contra
38
+ > la fuente. Aceptar sin verificar es propagar la confabulación.
39
+
40
+ Las normas mexicanas (RINEMAABMS, LGCG, LAASSP, CONAC, Constitución, leyes
41
+ generales) y los estándares técnicos (RFC, ISO, OWASP, WCAG) son tentadores
42
+ para confabular contextos plausibles. **Las citas archivo:línea generadas
43
+ por otros agentes son tentadoras para aceptar sin verificar** — vienen con
44
+ apariencia de evidencia "ya hecha". Plausible ≠ verificado. El costo de
45
+ inventar o propagar una cita es pérdida de confianza del usuario y debug
46
+ innecesario; el costo de verificar es un `grep` o `Read` de 2 segundos.
47
+
48
+ ---
49
+
50
+ ## Qué cuenta como "cita verificable"
51
+
52
+ ### Familia 1 — citas normativas / legales / documentales
53
+
54
+ OBLIGATORIO verificar antes de escribir:
55
+
56
+ - Artículo numerado de una norma: "Art. 23 RINEMAABMS", "Art. 30 LGCG",
57
+ "Art. 134 CPEUM", "fracción IV del Art. 17", etc.
58
+ - Número de RFC / ISO / IEEE: "RFC 7231", "ISO/IEC 27001", "IEEE 802.11n".
59
+ - Sección de un estándar: "OWASP Top 10 A03", "WCAG 2.1 SC 1.4.3", "NIST
60
+ SP 800-63".
61
+ - Nombre de catálogo, programa o acuerdo oficial: "PAAASINE-INE", "PAF",
62
+ "Acuerdo INE/CG2024/15".
63
+ - Atribución de fundamento normativo a un dato/tabla/módulo: "según
64
+ Art. X de la norma Y".
65
+ - Fechas y números de páginas de documentos oficiales: "actualización
66
+ julio 2017, 71 páginas".
67
+ - Citas textuales entre comillas atribuidas a un documento concreto.
68
+
69
+ ### Familia 2 — citas técnicas archivo:línea
70
+
71
+ OBLIGATORIO verificar antes de escribir o de aceptar como base para acción:
72
+
73
+ - Ubicación específica de un bug, hallazgo o evidencia:
74
+ `auth/service.py:191-243`, `schema.sql:438-451`, `repository.py:1260`.
75
+ - Atribución de un fragmento de código a un archivo: "la función `foo`
76
+ en `bar/baz.ts:42` hace X".
77
+ - Cita textual de líneas de código incluidas en un reporte: si el reporte
78
+ pega 4 líneas como evidencia, esas 4 líneas deben coincidir letra por
79
+ letra con el archivo en el HEAD actual del repo.
80
+ - Citas dentro de reportes de auditoría externa, nemesis, code review,
81
+ análisis de seguridad, post-mortems, RCAs, hallazgos consolidados
82
+ (incluyendo los que tú mismo emites).
83
+ - Conclusiones derivadas de una cita: "es bug porque X:42 ignora Y" —
84
+ hay que confirmar primero que X:42 realmente hace lo que el reporte
85
+ dice.
86
+ - Citas en commit messages, PR descriptions o issues que pretenden ser
87
+ evidencia accionable.
88
+
89
+ NO requiere verificación previa:
90
+
91
+ - Conceptos genéricos del dominio sin atribución a artículo específico
92
+ ("auditoría interna", "contratación pública", "expediente de
93
+ contratación").
94
+ - Conocimiento técnico estable sin cita numérica ("SQL injection",
95
+ "OAuth2 flow", "JWT").
96
+ - Descripciones funcionales del propio código que estás editando en
97
+ ese momento.
98
+ - Referencias a archivos sin número de línea cuando solo se discute la
99
+ existencia, no el contenido ("el módulo `auth/` está implementado").
100
+
101
+ ---
102
+
103
+ ## Cómo verificar — protocolo obligatorio
104
+
105
+ ### Paso 1: identificar la fuente potencial
106
+
107
+ Antes de escribir la cita, identifica de dónde podría salir:
108
+
109
+ - Seeds del proyecto: `grep -rn "TEXTO_CITA" backend/app/seeds/`
110
+ - Modelos / docstrings: `grep -rn "TEXTO_CITA" backend/app/modules/`
111
+ - Documentación normativa: `grep -rn "TEXTO_CITA" referencia/normativa/`
112
+ - ADRs: `grep -rn "TEXTO_CITA" docs/adr/`
113
+ - README, CHANGELOG, planning: `grep -rn "TEXTO_CITA" docs/`
114
+
115
+ ### Paso 2: ejecutar la búsqueda
116
+
117
+ Usar `grep` con el patrón exacto de lo que vas a escribir:
118
+
119
+ ```bash
120
+ # Verificar "Art. 23 RINEMAABMS"
121
+ grep -rin "art.*23.*rinemaabms\|rinemaabms.*art.*23" path/
122
+
123
+ # Verificar nombre de catálogo
124
+ grep -rin "clasif" referencia/normativa/
125
+
126
+ # Verificar número RFC
127
+ grep -rin "RFC 7231" .
128
+ ```
129
+
130
+ ### Paso 3: leer la fuente original
131
+
132
+ Si grep devuelve coincidencias, abrir el archivo con `Read` y verificar
133
+ que el contexto sí respalda la cita. Una coincidencia léxica no es prueba
134
+ suficiente — el contexto puede ser opuesto al que asumes.
135
+
136
+ ### Paso 4: decidir
137
+
138
+ - **Si la fuente confirma la cita**: escríbela con el contenido que dice
139
+ textualmente el origen.
140
+ - **Si la fuente dice algo parecido pero distinto**: usa el wording real
141
+ del origen, no el que recuerdes de memoria.
142
+ - **Si no hay fuente**: NO escribas la cita. Sustituye por descripción
143
+ neutra sin atribución, o pide al usuario la cita exacta.
144
+
145
+ ---
146
+
147
+ ## Protocolo para citas archivo:línea en reportes
148
+
149
+ Cuando trabajas con un reporte que cita ubicaciones del codebase (auditoría
150
+ nemesis, hallazgos de revisor-seguridad-swl, output de revisor-codigo-swl,
151
+ informe externo, RCA, post-mortem, hallazgo del usuario):
152
+
153
+ ### Paso 1: clasificar las citas
154
+
155
+ Lista cada `archivo:línea` mencionada y clasifícala:
156
+
157
+ - **Crítica accionable** — bug que se va a remediar, hallazgo que se va a
158
+ priorizar, evidencia que se va a citar en un commit o PR. OBLIGATORIO
159
+ verificar.
160
+ - **Contextual** — mención de paso en un párrafo sin que ninguna acción
161
+ dependa de su exactitud. Verificación opcional pero recomendada al
162
+ primer hallazgo crítico que la cite indirectamente.
163
+
164
+ ### Paso 2: muestreo mínimo verificable
165
+
166
+ Si el reporte tiene N citas críticas, verificar como mínimo:
167
+
168
+ - **Las top 4-6 acciones priorizadas** del reporte (P1 / críticas).
169
+ - **Cualquier cita con número exacto de líneas** (`:191-243`, `:438-451`)
170
+ — porque exactitud específica es la más propensa a confabulación.
171
+ - **Cualquier hallazgo etiquetado "regresión", "latente" o "extra"** —
172
+ porque suelen detectarse de carambola y son los más frágiles.
173
+ - **Cualquier cita que contradiga decisiones documentadas** — un reporte
174
+ que dice "esto está mal" sobre algo que un ADR / APRENDIZAJES dice
175
+ "esto se decidió así" requiere verificación dura antes de actuar.
176
+
177
+ ### Paso 3: verificar con `Read` y `grep`, no asumir
178
+
179
+ Para cada cita crítica:
180
+
181
+ ```bash
182
+ # Verificar que las líneas existen y dicen lo que el reporte dice
183
+ Read file_path="<archivo>" offset=<inicio-2> limit=<rango+4>
184
+
185
+ # Buscar evidencia complementaria (¿hay más sitios con el mismo patrón?)
186
+ grep -rn "<patrón>" backend/app/ --include="*.py"
187
+ ```
188
+
189
+ Una afirmación del tipo "X:42 ignora la validación" debe verificarse
190
+ leyendo X:42 directamente. Si el reporte cita 4 líneas como evidencia,
191
+ las 4 líneas deben coincidir con el archivo en el HEAD actual.
192
+
193
+ ### Paso 4: detectar gaps del reporte
194
+
195
+ Aprovechar la verificación para detectar lo que el reporte NO dice:
196
+
197
+ - **Sweep por patrón**: si el reporte cita un bug en `X.py:1260` por usar
198
+ `u.col_inexistente`, hacer `grep -rn "u\.col_inexistente"` en TODO el
199
+ módulo para descartar regresiones adicionales. Los reportes suelen
200
+ parar en el primer hallazgo del patrón.
201
+ - **Búsqueda inversa**: si el reporte dice "RESUELTO en commit Y",
202
+ verificar que el commit realmente toca el archivo citado.
203
+ - **Evidencia complementaria**: si el reporte cita una columna SQL,
204
+ verificar el `CREATE TABLE` del schema correspondiente. La doble
205
+ evidencia (código + schema) refuerza o refuta el hallazgo.
206
+
207
+ ### Paso 5: separar lo verificado de lo no verificado
208
+
209
+ Al re-emitir conclusiones a partir del reporte:
210
+
211
+ - **Marca explícitamente qué citas verificaste personalmente** vs. qué
212
+ citas reproducís sin re-verificar.
213
+ - **NO presentes el reporte como si fuera tu análisis directo** si no
214
+ lo verificaste. Es propagación de cita ajena, no análisis propio.
215
+ - **Si el reporte tiene 11 acciones y solo verificaste 6**, dilo así.
216
+ "Verifiqué 6/11; las 5 restantes asumo correctas por consistencia
217
+ metodológica con las que sí verifiqué" es honesto. "Las 11 acciones
218
+ están confirmadas" cuando solo viste 6 es propagación deshonesta.
219
+
220
+ ---
221
+
222
+ ## Pattern de redacción cuando NO hay cita verificada
223
+
224
+ En vez de inventar fundamento normativo:
225
+
226
+ - "Documento oficial INE, actualización julio 2017" (cita textual del seed).
227
+ - "Catálogo administrativo interno" (descripción neutra).
228
+ - "Definido por el equipo de proyecto" (atribución honesta sin invento).
229
+ - Sin cualquier cita ni atribución (preferible a inventar).
230
+
231
+ NUNCA:
232
+
233
+ - "Art. X de Y" sin verificación.
234
+ - "Catálogo oficial Z" sin verificación.
235
+ - "Conforme a la sección N del estándar" sin verificación.
236
+
237
+ ---
238
+
239
+ ## Anti-patrones explícitos
240
+
241
+ ### Confabulación plausible
242
+
243
+ Inventar una cita que suena correcta porque el contexto sugiere que
244
+ podría existir. Ejemplo real (SIGAF, 2026-05-13): asocié "Art. 23
245
+ RINEMAABMS" como fundamento del Clasificador de Gasto del INE porque
246
+ RINEMAABMS regula contratación INE y el Art. 23 trata adquisiciones —
247
+ plausible pero falso. El seed `07_clasificador_gasto.sql` no menciona
248
+ ningún artículo, solo cita "Documento oficial INE, actualización julio
249
+ 2017, 71 páginas".
250
+
251
+ ### Asociación incorrecta entre instrumento y catálogo
252
+
253
+ Confundir un programa de planeación con un catálogo de clasificación.
254
+ Ejemplo: presentar "PAAASINE" (Programa Anual de Adquisiciones del INE,
255
+ instrumento de planeación) como si fuera el "catálogo oficial" del
256
+ Clasificador de Gasto. Son cosas distintas en el dominio.
257
+
258
+ ### Cita por intuición de memoria del modelo
259
+
260
+ Atribuir contenido a un artículo o estándar porque "lo aprendí en el
261
+ entrenamiento". El conocimiento de entrenamiento sobre normas mexicanas
262
+ específicas, especialmente las del INE/OIC y reglamentos administrativos,
263
+ es alto en confabulación. Trata cada cita normativa como si NO lo supieras
264
+ hasta que el grep al proyecto lo confirme.
265
+
266
+ ### Cita por inferencia transitiva
267
+
268
+ "Si la norma X regula Y, entonces la cita debe ser Art. N de X". No.
269
+ La transición lógica no es prueba documental.
270
+
271
+ ### Cita "para dar peso"
272
+
273
+ Inventar fundamento normativo para hacer ver la UI/docs como más
274
+ profesional o jurídicamente sólida. Una cita falsa es peor que ninguna
275
+ cita — destruye la confianza del usuario en TODO lo demás que escribiste.
276
+
277
+ ### Aceptar reporte ajeno sin re-verificar evidencia
278
+
279
+ Recibir un reporte de auditoría / nemesis / revisor con citas
280
+ archivo:línea y proceder a priorizar, planear remediación o re-emitir
281
+ las conclusiones SIN haber hecho `Read` ni `grep` sobre las citas
282
+ críticas. El reporte puede:
283
+
284
+ - Tener una cita correcta en archivo pero incorrecta en número de línea
285
+ (el código se movió post-reporte).
286
+ - Tener una cita totalmente inventada por confabulación del agente que
287
+ lo produjo.
288
+ - Estar desactualizado: el bug citado ya se cerró en un commit posterior.
289
+ - Confundir dos archivos similares (`router.py` de módulo A con
290
+ `router.py` de módulo B).
291
+
292
+ Propagar el reporte como si fuera tu análisis directo es deshonesto y
293
+ multiplica el costo del error. La verificación de las top 4-6 citas
294
+ críticas toma 5-10 minutos y separa lo verificado de lo asumido.
295
+
296
+ ### Sweep parcial cuando el patrón sugiere más sitios
297
+
298
+ Verificar la cita exacta del reporte (`X.py:1260`) y detenerse ahí
299
+ cuando el bug es **un patrón** (uso de columna inexistente, llamada a
300
+ función deprecada, falta de validación). Los reportes de auditoría
301
+ suelen parar en el primer hallazgo del patrón. Si verificaste que
302
+ `u.col_inexistente` está mal en `X.py:1260`, ejecuta `grep -rn` sobre
303
+ toda la base para encontrar los hermanos antes de marcar el hallazgo
304
+ como cerrado.
305
+
306
+ ### "El reporte ya lo verificó"
307
+
308
+ Falacia. Un reporte declara haber verificado, no demuestra haber
309
+ verificado. Si vas a actuar sobre la cita, tu firma vale lo que
310
+ verificaste personalmente — no lo que el reporte afirma haber hecho.
311
+
312
+ ---
313
+
314
+ ## Excepciones legítimas
315
+
316
+ NO aplica la regla cuando:
317
+
318
+ 1. **El usuario dictó la cita explícitamente**: si el usuario escribe
319
+ "agrega que esto es por el Art. 23 RINEMAABMS", asume que el usuario
320
+ verificó. Pero si la cita parece dudosa, repregunta.
321
+ 2. **La cita ya existe en el codebase verificado**: si encuentras "Art.
322
+ 23 RINEMAABMS" ya escrito en un docstring del modelo, puedes
323
+ reutilizarlo en UI sin re-verificar la fuente externa (asumes que
324
+ alguien antes la verificó).
325
+ 3. **Conceptos canónicos sin atribución específica**: "auditoría",
326
+ "expediente", "contratación" — no requieren cita.
327
+ 4. **Citas de standards técnicos universalmente conocidos sin número**:
328
+ "OWASP", "WCAG" sin sub-versión. Si pones "WCAG 2.1 SC 1.4.3" sí
329
+ aplica la regla.
330
+ 5. **Cita archivo:línea producida por TU MISMO en ESTE turno** sobre un
331
+ archivo que acabas de leer con `Read` en este mismo turno. La
332
+ evidencia es directa, no necesita re-verificación.
333
+ 6. **Reporte ajeno que solo vas a leer / archivar / referenciar como
334
+ contexto histórico** sin tomar acción. La verificación se activa
335
+ cuando vas a priorizar, remediar, propagar o concluir a partir del
336
+ reporte.
337
+
338
+ ---
339
+
340
+ ## Familia 2 extendida — reportes COMPACTACION.md y status reports SWL
341
+
342
+ Los reportes generados por `/swl:compactar` (Delta vN en `.planning/COMPACTACION.md`),
343
+ `/swl:checkpoint` (continue-here.md), y similares **son también sub-productos
344
+ verificables**. El agente que los escribió puede sintetizar afirmaciones cuantitativas
345
+ sin re-verificar contra `git`/`pytest`/`grep`, y el agente que los lee después
346
+ hereda los errores si los acepta como base para decisiones.
347
+
348
+ ### Cuándo aplica
349
+
350
+ OBLIGATORIO verificar antes de tomar acción sobre afirmaciones cuantitativas
351
+ extraídas de:
352
+
353
+ - `COMPACTACION.md` Delta vN (la última sección de cierre de sesión).
354
+ - `RESUMEN.md` de fase (conteos de archivos, commits, slices).
355
+ - `ESTADO.md` (DT activas, OP pendientes, fases completadas).
356
+ - `continue-here.md` (commits pendientes, archivos sin terminar).
357
+ - Reportes de auditoría con cifras agregadas ("N hallazgos críticos resueltos").
358
+
359
+ ### Verificaciones rápidas obligatorias
360
+
361
+ Cuando un reporte declara una cifra accionable, ejecutar la primitiva equivalente
362
+ ANTES de basar decisiones en ella:
363
+
364
+ | Afirmación del reporte | Verificación |
365
+ |---|---|
366
+ | "N commits sin push" | `git log origin/main..HEAD --oneline \| wc -l` |
367
+ | "N archivos modificados" | `git diff --stat HEAD~M..HEAD \| tail -1` |
368
+ | "N DT activas" | `grep -cE "^### (DT\|DA\|OP)-" .planning/DEUDAS.md` (sección Activas) |
369
+ | "N tests verdes" | output reciente de `pytest --co -q` o último run de CI |
370
+ | "Fase N completada" | revisar `NN-CIERRE.md` y commits asociados, no solo el reporte |
371
+ | "Cobertura X%" | re-ejecutar `coverage report` si la cifra es relevante para decisión |
372
+
373
+ ### Caso real (origen — sesión SIGM 2026-05-21)
374
+
375
+ El reporte v10 de COMPACTACION.md afirmó:
376
+
377
+ > "Trabajo pendiente mapeado: 11 commits acumulados sin push"
378
+
379
+ Verificación con `git log origin/main..HEAD --oneline | wc -l` retornó **2**.
380
+ Los otros 9 commits ya estaban en `origin/main`; el fetch falló por red pero la
381
+ referencia local seguía válida. Si el siguiente turno hubiera aceptado el dato
382
+ como cierto, habría "esperado créditos GH para pushear 11" sin necesidad.
383
+
384
+ Lección: el reporte de compactación es un artefacto de prosa con cifras — NO
385
+ es la fuente de verdad. La fuente es el filesystem + git + el output de las
386
+ herramientas. Tres segundos de `git log origin/main..HEAD` previenen propagación
387
+ de error.
388
+
389
+ ### Anti-patrón específico
390
+
391
+ - **Aceptar el reporte v10 (o vN) tras `/compact` como base de decisiones**: el
392
+ reporte fue escrito por el agente al cerrar contexto, sin re-verificar. Tras
393
+ la compactación, el agente NUEVO lo lee con confianza por defecto y propaga
394
+ cualquier desalineación. Aplicar Familia 2: verificar 2-3 cifras críticas
395
+ del reporte contra git/pytest/grep antes de planificar la siguiente acción.
396
+
397
+ ---
398
+
399
+ ## Familia 2 extendida — comentarios temporales en código
400
+
401
+ (Absorbe la regla `verificar-citas-temporales.md`, unificada aquí el 2026-06-11
402
+ porque su propio texto se declaraba extensión de esta Familia 2.)
403
+
404
+ > Un comentario en código que afirma alineación con otro commit, versión o
405
+ > contrato (`// Alineado con commits X+Y`, `// Refactorizado a vN.M.0`,
406
+ > `# Sincronizado con backend`, `<!-- Coherente con schema X -->`) **NO es
407
+ > verificación** — es afirmación temporal sin prueba. Antes de actuar sobre el
408
+ > comentario como si fuera evidencia, verificar contra la fuente real con
409
+ > `grep` / `Read` / `git show`.
410
+
411
+ El comentario puede haberse desactualizado silenciosamente si el commit
412
+ referenciado se revirtió, nunca se aplicó, o el refactor citado cambió después.
413
+
414
+ **Cuándo aplica**: comentarios que afirman alineación cross-stack o con commits,
415
+ versiones, schemas. NO aplica a comentarios de intención, `TODO`/`FIXME` con
416
+ ticket, ni explicaciones del POR QUÉ.
417
+
418
+ **Cómo verificar**: identificar QUÉ se afirma alineado → `git show <SHA> --
419
+ <archivo>` o `grep` del campo clave en ambos stacks → comparar campo por campo.
420
+ Si hay drift, NO confiar en el comentario: actualizarlo o eliminarlo en el mismo
421
+ commit (un comentario stale es peor que ninguno).
422
+
423
+ **Alternativas seguras** al comentario temporal: test de contrato en CI (en vez
424
+ de `// Alineado con commits X+Y`), versión semántica + lock (en vez de
425
+ `// Refactorizado a vN`), tipos generados/codegen (en vez de `# Sincronizado con
426
+ backend`), validador en pre-commit (en vez de `<!-- Coherente con schema -->`).
427
+ Si el comentario es la única opción, incluir **fecha y SHA explícitos** y la
428
+ condición de re-verificación.
429
+
430
+ **Caso de origen** (SIGAF 2026-05-22): `contrato.models.ts:2` decía "Alineados
431
+ con el backend refactorizado (commits 284df74 + 5869ac9)" pero el backend nunca
432
+ aplicó ese refactor; Pydantic con `extra=ignore` descartaba 5 campos en
433
+ silencio y los contratos se crearon sin monto ni vigencia durante semanas. El
434
+ comentario funcionó como falso anclaje de confianza para los reviewers.
435
+
436
+ **Anti-patrones**: confiar en el comentario en code review sin `git show`;
437
+ propagarlo en refactors (viaja con el copy-paste hasta volverse falso); no
438
+ actualizarlo al revertir el commit citado; usarlo como "prueba" en debates
439
+ técnicos.
440
+
441
+ ---
442
+
443
+ ## Origen de esta regla
444
+
445
+ ### Origen Familia 1 — citas normativas (versión inicial)
446
+
447
+ Sesión 2026-05-13, proyecto SIGAF. Durante implementación del refactor
448
+ ADR-076 F4 del form de Nuevo Expediente, agregué como fundamento del
449
+ catálogo "Clasificador por objeto y tipo de gasto para el INE" la cita
450
+ "Art. 23 RINEMAABMS · catálogo oficial PAAASINE-INE" sin haberla
451
+ verificado en ningún archivo del proyecto. Era plausible (RINEMAABMS
452
+ regula contratación INE; PAAASINE es el programa anual de adquisiciones)
453
+ pero ambas atribuciones eran falsas:
454
+
455
+ - El modelo `CatClasificadorGasto` y el seed `07_clasificador_gasto.sql`
456
+ no mencionan ningún artículo del RINEMAABMS.
457
+ - El seed cita textualmente "Documento oficial INE, actualización julio
458
+ 2017, 71 páginas" como fuente — sin asociarlo a artículo legal.
459
+ - PAAASINE es un instrumento de planeación, no un catálogo de
460
+ clasificación. Conceptualmente distintos.
461
+
462
+ El usuario detectó la alucinación al verificar el SQL de referencia y
463
+ pidió analizarla. La regla se promueve a global porque el patrón puede
464
+ repetirse en cualquier proyecto del usuario que toque normas mexicanas o
465
+ estándares técnicos numerados — el sesgo de confabulación plausible no es
466
+ específico de SIGAF.
467
+
468
+ Commit donde se corrigió la alucinación original: `f2b025a` en SIGAF.
469
+
470
+ ### Extensión Familia 2 — citas archivo:línea en reportes
471
+
472
+ Sesión 2026-05-16, proyecto SIGM. El usuario pidió analizar un reporte
473
+ de auditoría nemesis con 16 hallazgos (F-01 a F-16) más 3 hallazgos
474
+ heredados (H5.A1, H5.A2, H3.1), todos con citas `archivo:línea`
475
+ específicas (`auth/service.py:191-243`, `dependencies.py:73-78`,
476
+ `database/init/init.sh`, `catastro/repository.py:1260`, etc.).
477
+
478
+ Detecté que aceptar el reporte sin verificar habría sido propagación
479
+ de cita ajena. Apliqué el principio de la regla original adaptado al
480
+ dominio técnico: cada cita crítica priorizada (top 6) se verificó con
481
+ `Read` + `grep` contra el código real antes de re-emitir conclusiones.
482
+
483
+ Resultado: 6/6 hallazgos top verificados al 100%, con **evidencia
484
+ adicional** que el reporte no había detallado:
485
+
486
+ - F-04 lockout: además del backend que no escribe `intentos_fallidos`,
487
+ el schema declara `CHECK (intentos_fallidos >= 0 AND intentos_fallidos
488
+ <= 10)` — el tope de 10 es invariante de la BD, doble confirmación.
489
+ - Regresión `u.nombre_completo`: el reporte citó 2 sitios; la
490
+ verificación con `grep -n "u.nombre_completo"` reveló que **podían
491
+ haber más** y el reporte no había hecho sweep global.
492
+
493
+ Esa última observación se promueve a sub-regla operativa (Paso 4
494
+ del protocolo archivo:línea: detectar gaps del reporte con sweep por
495
+ patrón).
496
+
497
+ La adaptación se fusiona en esta regla en lugar de crear una nueva
498
+ porque el principio es idéntico — solo cambia el dominio de la cita
499
+ (normativa vs. archivo:línea). Mantener una sola regla evita
500
+ duplicación operativa y refuerza que el origen del riesgo es siempre el
501
+ mismo: una afirmación con apariencia de evidencia que en realidad no
502
+ se verificó.
503
+
504
+ ### Extensión Familia 2 — reportes COMPACTACION.md (2026-05-21)
505
+
506
+ Sesión SIGM 2026-05-21. Tras `/swl:compactar` v10, el reporte declaró
507
+ "11 commits acumulados sin push". Verificación con
508
+ `git log origin/main..HEAD --oneline | wc -l` retornó **2**. El reporte se
509
+ escribió desde la perspectiva del agente al cerrar contexto, sin re-verificar
510
+ contra `origin/main` (que sigue local actualizada incluso con `git fetch`
511
+ fallando por red).
512
+
513
+ La adaptación se incorpora como sub-sección "Familia 2 extendida — reportes
514
+ COMPACTACION.md y status reports SWL" arriba, no como Familia 3 separada. El
515
+ principio es el mismo: afirmación cuantitativa en un reporte = sub-producto
516
+ verificable. Cambia el dominio (compactación SWL vs. nemesis auditorías), no
517
+ el protocolo.
518
+
519
+ Lección operativa: las 3 cifras que más se desalinean tras un `/compact` son
520
+ "commits sin push" (depende de fetch reciente), "DT activas" (depende del
521
+ último Edit de DEUDAS.md vs lo que la prosa del reporte describe) y "tests
522
+ verdes" (depende del último run de CI vs lo que el agente recuerda). Verificar
523
+ esas 3 antes de decidir siguiente acción.
524
+
525
+ ---
526
+
527
+ ## Checklist antes de escribir una cita verificable (Familia 1)
528
+
529
+ - [ ] ¿La cita tiene un número, artículo, fracción o referencia
530
+ específica?
531
+ - [ ] ¿Ejecuté `grep` o `Read` sobre la fuente real del proyecto?
532
+ - [ ] ¿La fuente respalda textualmente la cita?
533
+ - [ ] Si no respalda: ¿sustituí por descripción neutra o eliminé la cita?
534
+ - [ ] Si no encontré fuente y el usuario no la dictó: ¿pregunté al
535
+ usuario por la cita exacta antes de inventarla?
536
+
537
+ ## Checklist antes de aceptar o re-emitir una cita archivo:línea (Familia 2)
538
+
539
+ - [ ] ¿Identifiqué cuáles citas del reporte son críticas accionables vs.
540
+ contextuales?
541
+ - [ ] ¿Hice `Read` sobre cada cita crítica priorizada (top 4-6)?
542
+ - [ ] ¿El archivo:línea citado dice textualmente lo que el reporte afirma?
543
+ - [ ] ¿Hice sweep por patrón (`grep -rn`) para detectar hermanos del bug
544
+ que el reporte no listó?
545
+ - [ ] ¿Verifiqué evidencia complementaria (schema, ADRs, commits citados)
546
+ antes de aceptar el hallazgo?
547
+ - [ ] Al re-emitir conclusiones: ¿marqué explícitamente qué citas
548
+ verifiqué vs. cuáles reproduzco sin verificar?
@@ -86,6 +86,28 @@ function ubicarClaudeMd(dir = process.cwd()) {
86
86
  return null;
87
87
  }
88
88
 
89
+ // Detecta @-includes con path RELATIVO al CLAUDE.md cuyo destino no existe.
90
+ // Solo evalúa rutas relativas al proyecto — omite `~/` y rutas absolutas
91
+ // porque su existencia depende del entorno de instalación, no del repo.
92
+ // Causa raíz del bug downstream: plantillas que emitían `@reglas/...` cuando
93
+ // las reglas se instalan en `.claude/rules/` (no en `reglas/` del proyecto).
94
+ function detectarReferenciasRotas(contenido, rutaClaudeMd) {
95
+ const baseDir = path.dirname(rutaClaudeMd);
96
+ // Primer char [A-Za-z0-9_.] excluye `~` y `/` → solo captura rutas relativas.
97
+ const re = /@([A-Za-z0-9_.][A-Za-z0-9_./\-]*\.md)\b/g;
98
+ const rotas = [];
99
+ const vistos = new Set();
100
+ let m;
101
+ while ((m = re.exec(contenido)) !== null) {
102
+ const ref = m[1];
103
+ if (vistos.has(ref)) continue;
104
+ vistos.add(ref);
105
+ if (ref.startsWith('~') || path.isAbsolute(ref)) continue;
106
+ if (!fs.existsSync(path.resolve(baseDir, ref))) rotas.push(ref);
107
+ }
108
+ return rotas;
109
+ }
110
+
89
111
  function auditar(rutaClaudeMd) {
90
112
  if (!rutaClaudeMd || !fs.existsSync(rutaClaudeMd)) {
91
113
  return {
@@ -150,6 +172,17 @@ function auditar(rutaClaudeMd) {
150
172
  }
151
173
  }
152
174
 
175
+ // 4b. @references rotos: @-includes con path relativo cuyo destino no existe.
176
+ const referenciasRotas = detectarReferenciasRotas(contenido, rutaClaudeMd);
177
+ for (const ref of referenciasRotas) {
178
+ hallazgos.push({
179
+ severidad: 'WARN',
180
+ regla: 'at-reference-rota',
181
+ mensaje: `@-reference rota: \`@${ref}\` no existe (resuelto relativo al CLAUDE.md)`,
182
+ sugerencia: 'Corregir la ruta o, si es una regla global auto-cargada (p. ej. usar-sistema-swl), eliminar el @-include y dejar solo mención de texto',
183
+ });
184
+ }
185
+
153
186
  // 5. Placeholders sin reemplazar
154
187
  const matches = [...contenido.matchAll(PLACEHOLDERS)];
155
188
  if (matches.length > 0) {
@@ -237,6 +270,7 @@ function auditar(rutaClaudeMd) {
237
270
  secciones_presentes: SECCIONES_CANONICAS.filter(s => s.regex.test(contenido)).map(s => s.nombre),
238
271
  secciones_ausentes: seccionesAusentes.map(s => s.nombre),
239
272
  tiene_at_references: /@[a-zA-Z][a-zA-Z0-9_\-./]+\.md/.test(contenido),
273
+ at_references_rotas: referenciasRotas,
240
274
  tiene_referencia_karpathy: tieneReferenciaKarpathy,
241
275
  es_project_level: esProjectLevel,
242
276
  duplicaciones_reglas_globales: {
@@ -375,6 +409,9 @@ function imprimirReporte(resultado) {
375
409
  console.log(` - Secciones ausentes: ${m.secciones_ausentes.join(', ')}`);
376
410
  }
377
411
  console.log(` - @references: ${m.tiene_at_references ? 'sí' : 'no'}`);
412
+ if (m.at_references_rotas && m.at_references_rotas.length > 0) {
413
+ console.log(` - @references rotas: ${m.at_references_rotas.length} (${m.at_references_rotas.join(', ')})`);
414
+ }
378
415
  if (m.es_project_level) {
379
416
  console.log(` - Referencia Karpathy: ${m.tiene_referencia_karpathy ? 'sí' : 'no'}`);
380
417
  }
@@ -436,6 +473,7 @@ module.exports = {
436
473
  ubicarClaudeMd,
437
474
  detectarBulletsGigantes,
438
475
  detectarReferenciaKarpathy,
476
+ detectarReferenciasRotas,
439
477
  esRutaUserLevel,
440
478
  main,
441
479
  MAX_LINES,