@saulwade/swl-ses 1.5.0 → 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 (134) hide show
  1. package/CLAUDE.md +19 -2
  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 +225 -1
  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 +105 -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 +150 -4
  37. package/habilidades/testing-python/SKILL.md +340 -340
  38. package/habilidades/verificar-trabajo/SKILL.md +8 -3
  39. package/habilidades/web-fetcher-routing/SKILL.md +75 -75
  40. package/hooks/check-update.js +31 -3
  41. package/hooks/claudemd-bloat-detector.js +161 -161
  42. package/hooks/lib/agent-routing.js +107 -107
  43. package/hooks/lib/auto-consolidator.js +335 -335
  44. package/hooks/lib/error-classifier.js +308 -308
  45. package/hooks/lib/merkle-audit.js +96 -96
  46. package/hooks/lib/provenance-tracker.js +191 -191
  47. package/hooks/lib/rate-limit-tracker.js +253 -253
  48. package/hooks/lib/resource-quota.js +122 -122
  49. package/hooks/lib/retry-jitter.js +165 -165
  50. package/hooks/lib/security-net.js +201 -201
  51. package/hooks/lib/skill-auditor.js +588 -588
  52. package/hooks/lib/sync-status.js +228 -228
  53. package/hooks/lib/taint-tracker.js +107 -107
  54. package/hooks/lib/text-similarity.js +241 -241
  55. package/hooks/lib/toon-compressor.js +245 -245
  56. package/hooks/registro-turnos.js +209 -209
  57. package/hooks/sugerir-regenerar-inventario.js +170 -170
  58. package/hooks/validar-formato-post-subagente.js +140 -140
  59. package/hooks/validar-memoria-hook.js +218 -218
  60. package/instintos/prompt-appendices.yaml +57 -57
  61. package/manifiestos/agent-output-schemas.json +57 -57
  62. package/manifiestos/modulos.json +1324 -1321
  63. package/manifiestos/skills-lock.json +1114 -1114
  64. package/package.json +2 -2
  65. package/plantillas/auditor-veto-template.md +105 -105
  66. package/plantillas/github-workflows/README.md +47 -47
  67. package/plantillas/github-workflows/release-please.yml +44 -44
  68. package/plantillas/github-workflows/swl-ci.yml +107 -107
  69. package/plantillas/github-workflows/swl-security.yml +51 -51
  70. package/plugin.json +353 -351
  71. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  72. package/reglas/arreglar-al-detectar.md +147 -147
  73. package/reglas/fragmentos-compartidos.md +152 -152
  74. package/reglas/harness-claude-code.md +213 -213
  75. package/reglas/registro-componentes-nuevos.md +192 -0
  76. package/reglas/usar-context7.md +226 -226
  77. package/schemas/diary-entry.schema.json +80 -80
  78. package/scripts/actualizar.js +110 -1
  79. package/scripts/audit-tools/audit-history.js +330 -330
  80. package/scripts/audit-tools/bundle-tracker.js +290 -290
  81. package/scripts/audit-tools/canary-monitor.js +352 -352
  82. package/scripts/audit-tools/code-profiler.js +605 -605
  83. package/scripts/audit-tools/dep-doctor.js +320 -320
  84. package/scripts/audit-tools/env-validator.js +206 -206
  85. package/scripts/audit-tools/lib/fs-walk.js +48 -48
  86. package/scripts/audit-tools/lib/output.js +23 -23
  87. package/scripts/audit-tools/migration-checker.js +392 -392
  88. package/scripts/audit-tools/pentest-scanner.js +1436 -1436
  89. package/scripts/benchmark-memoria.js +167 -167
  90. package/scripts/configurar-branch-protection.js +418 -418
  91. package/scripts/derivar-feature-list.js +489 -489
  92. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  93. package/scripts/doctor.js +58 -4
  94. package/scripts/field-report.js +199 -199
  95. package/scripts/generar-checklists-consolidados.js +273 -273
  96. package/scripts/generar-inventario.js +420 -420
  97. package/scripts/generar-matriz-lenguajes.js +271 -271
  98. package/scripts/lib/artefactos-python.js +43 -43
  99. package/scripts/lib/benchmark-metrics.js +160 -160
  100. package/scripts/lib/budget-enforcer.js +252 -252
  101. package/scripts/lib/configurar-ci.js +380 -380
  102. package/scripts/lib/contadores-inventario.js +217 -217
  103. package/scripts/lib/detectar-stack-detallado.js +307 -307
  104. package/scripts/lib/diary-entry.js +234 -234
  105. package/scripts/lib/eval-metrics-store.js +218 -218
  106. package/scripts/lib/eval-quality.js +171 -171
  107. package/scripts/lib/eval-schemas.js +144 -144
  108. package/scripts/lib/eval-self-correct.js +106 -106
  109. package/scripts/lib/eval-validator.js +185 -185
  110. package/scripts/lib/expandir-targets.js +71 -71
  111. package/scripts/lib/jaccard-similarity.js +98 -98
  112. package/scripts/lib/longmemeval-runner.js +125 -125
  113. package/scripts/lib/mcp_config.py +127 -0
  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/lib/toml-merge.js +204 -204
  121. package/scripts/lib/transformadores/codex.js +375 -375
  122. package/scripts/lib/transformadores/cursor.js +359 -359
  123. package/scripts/limpiar-artefactos-python.js +131 -131
  124. package/scripts/mcp-orchestrator.py +8 -18
  125. package/scripts/mcp-pool-manager.py +12 -23
  126. package/scripts/mcp-server/README.md +170 -170
  127. package/scripts/mcp-server/auth.js +105 -105
  128. package/scripts/mcp-server/cache.js +106 -106
  129. package/scripts/mcp-server/telemetry.js +78 -78
  130. package/scripts/migrar-csv-a-array.js +168 -168
  131. package/scripts/migrar-fase-dominio.js +201 -201
  132. package/scripts/publicar.js +511 -511
  133. package/scripts/run-eval.js +141 -141
  134. package/scripts/validar-userland-vacio.js +110 -110
@@ -1,147 +1,147 @@
1
- # Patrones de Estado Acoplado — Language-Agnostic
2
-
3
- 9 patrones de estado acoplado frecuentes en sistemas Python, TypeScript, Go, Rust, Java y C#.
4
-
5
- ---
6
-
7
- ## Patrón 1: Balance ↔ Caché derivada
8
-
9
- **Descripción**: un valor base (balance de cuenta, stock de inventario, crédito disponible) tiene una caché calculada a partir de él (porcentaje disponible, categoría de riesgo, tier de cliente).
10
-
11
- **Invariante**: la caché debe reflejar el balance actual, no el del momento en que se calculó.
12
-
13
- **Paths de mutación riesgosos**: cualquier operación que modifica el balance directamente sin pasar por el flujo que actualiza la caché.
14
-
15
- **Trigger típico**: función de ajuste manual, importación masiva, o corrección administrativa que escribe el balance sin llamar al servicio que recalcula la caché.
16
-
17
- **Consecuencia**: decisiones que leen la caché (¿aplica descuento? ¿otorgar crédito?) usan datos obsoletos.
18
-
19
- ---
20
-
21
- ## Patrón 2: Registro principal ↔ Índice secundario
22
-
23
- **Descripción**: una tabla o colección tiene un índice secundario (índice invertido, tabla de búsqueda rápida, lista de IDs por categoría) que debe estar sincronizado con la tabla principal.
24
-
25
- **Invariante**: toda entrada en la tabla principal tiene exactamente una entrada correspondiente en el índice secundario (o exactamente ninguna, según el diseño).
26
-
27
- **Paths de mutación riesgosos**: delete de la tabla principal sin delete del índice; insert sin insert correspondiente; update que cambia la clave del índice sin actualizar el índice.
28
-
29
- **Trigger típico**: función de "purga" o "limpieza" que borra de la tabla principal pero olvida el índice. O migración que popula la tabla principal pero no regenera el índice.
30
-
31
- **Consecuencia**: búsquedas por índice devuelven resultados incorrectos (IDs que ya no existen, o registros que no aparecen en búsqueda aunque existan).
32
-
33
- ---
34
-
35
- ## Patrón 3: Contador agregado ↔ Suma de partes
36
-
37
- **Descripción**: un contador de totales (total de pedidos, total de puntos, total de items activos) debe ser igual a la suma de los contadores individuales.
38
-
39
- **Invariante**: `total == sum(individuales)` siempre.
40
-
41
- **Paths de mutación riesgosos**: operaciones que modifican un contador individual sin ajustar el total. Frecuente en operaciones de batch donde el total se actualiza una sola vez al final pero un error parcial deja los individuales en estado inconsistente.
42
-
43
- **Trigger típico**: transacción fallida a mitad que actualiza N de M elementos pero no revierte el total; o función que incrementa directamente el conteo individual sin notificar al agregador.
44
-
45
- **Consecuencia**: reportes y dashboards muestran cifras incorrectas; límites y cuotas se aplican con base en datos erróneos.
46
-
47
- ---
48
-
49
- ## Patrón 4: Sesión / Permiso derivado ↔ Recurso de origen
50
-
51
- **Descripción**: un objeto de sesión, token, o permiso derivado almacena un snapshot de los permisos o atributos del actor al momento de su creación. El actor puede cambiar después.
52
-
53
- **Invariante**: depende del diseño. Si los permisos se evalúan en tiempo de solicitud, el objeto derivado debe reflejar el estado actual del actor. Si se usan permisos fijados al momento de creación, el invariante es documentado explícitamente.
54
-
55
- **Paths de mutación riesgosos**: revocar permisos al actor sin invalidar sesiones activas; cambiar el rol del actor sin forzar re-autenticación; eliminar al actor sin expirar sus tokens.
56
-
57
- **Trigger típico**: función de "deshabilitar cuenta" que setea `is_active=False` en el usuario pero no invalida los tokens JWT o sesiones existentes.
58
-
59
- **Consecuencia**: actor deshabilitado sigue operando hasta que el token expira naturalmente.
60
-
61
- ---
62
-
63
- ## Patrón 5: Posición ↔ Factor de salud cacheado
64
-
65
- **Descripción**: un recurso con múltiples atributos (posición de deuda: monto + colateral + fecha + valor de mercado del colateral) tiene un factor calculado (ratio LTV, score de riesgo, estado de salud) que se cachea por rendimiento.
66
-
67
- **Invariante**: el factor cacheado debe recalcularse cada vez que cualquiera de sus inputs cambia.
68
-
69
- **Paths de mutación riesgosos**: actualizar cualquier input del cálculo sin invalidar o recalcular el factor cacheado.
70
-
71
- **Trigger típico**: función que actualiza el colateral sin llamar a `recalcular_factor()`. O función de ajuste de precio que modifica el valor de mercado sin propagar la invalidación de caché.
72
-
73
- **Consecuencia**: decisiones de riesgo (¿emitir alerta? ¿bloquear operación?) usan un factor obsoleto.
74
-
75
- ---
76
-
77
- ## Patrón 6: Replica / Proyección ↔ Origen CDC
78
-
79
- **Descripción**: un sistema mantiene una replica o proyección de datos de origen para acelerar lecturas (tabla denormalizada, proyección CQRS, replica de lectura, caché materializada). El origen puede cambiar vía CDC (Change Data Capture), eventos de dominio, o sincronización periódica.
80
-
81
- **Invariante**: la replica debe converger al estado del origen dentro de la ventana de latencia acordada.
82
-
83
- **Paths de mutación riesgosos**: escrituras directas al origen que no emiten el evento de dominio necesario para actualizar la replica; fallos en el consumer de eventos que quedan silenciosos; migraciones que modifican el origen sin reprocesar los eventos para actualizar la replica.
84
-
85
- **Trigger típico**: endpoint de "corrección de datos" que escribe directo a la BD origen sin publicar el evento de dominio correspondiente.
86
-
87
- **Consecuencia**: lecturas de la replica devuelven datos obsoletos indefinidamente.
88
-
89
- ---
90
-
91
- ## Patrón 7: Inventario total ↔ Items en almacén (por ubicación)
92
-
93
- **Descripción**: el inventario total de un SKU es la suma de los items en cada almacén o ubicación. Se mantiene desnormalizado por rendimiento.
94
-
95
- **Invariante**: `inventario_total[SKU] == sum(items_por_ubicacion[ubicacion][SKU] for ubicacion in todas)`
96
-
97
- **Paths de mutación riesgosos**: movimiento entre almacenes que decrementa una ubicación sin incrementar la otra; recepción en almacén sin incrementar el total; ajuste de inventario físico que corrige la ubicación sin propagar al total.
98
-
99
- **Trigger típico**: función de "ajuste de cierre" que actualiza el stock físico de un almacén sin pasar por la capa de negocio que mantiene el total sincronizado.
100
-
101
- **Consecuencia**: órdenes de compra se emiten sobre stock inexistente; o stock disponible no se vende por parecer agotado.
102
-
103
- ---
104
-
105
- ## Patrón 8: Estado de workflow ↔ Timestamps de auditoría
106
-
107
- **Descripción**: una entidad con workflow (pedido, solicitud, tarea) almacena su estado actual y los timestamps de cada transición. El timestamp de la última transición debe corresponder al estado actual.
108
-
109
- **Invariante**: `estado_actual` siempre tiene un timestamp asociado que refleja cuándo se alcanzó ese estado.
110
-
111
- **Paths de mutación riesgosos**: funciones que cambian el estado sin registrar el timestamp; o funciones que registran el timestamp sin actualizar el estado.
112
-
113
- **Trigger típico**: función de "forzar estado" en panel de administración que escribe `estado = 'COMPLETADO'` directamente sin pasar por la máquina de estados que registra `completado_en = now()`.
114
-
115
- **Consecuencia**: reportes de tiempo de ciclo son incorrectos; SLAs no se calculan bien; auditorías de cumplimiento fallan.
116
-
117
- ---
118
-
119
- ## Patrón 9: Configuración global ↔ Snapshot por instancia
120
-
121
- **Descripción**: hay una configuración global (tasa de impuesto, precio de tarifa, límite de política) y objetos que almacenan el valor que tenía la configuración en el momento de su creación (tasa de impuesto aplicada a esta factura, precio de tarifa al momento de la reserva).
122
-
123
- **Invariante**: el snapshot por instancia debe preservar el valor histórico, nunca actualizarse cuando cambia la configuración global. Pero si hay un bug en la creación, el snapshot puede haber capturado un valor incorrecto.
124
-
125
- **Paths de mutación riesgosos**: función que no captura el snapshot al crear la instancia y usa la configuración global en tiempo de lectura; o función de retroactiva que sobreescribe snapshots históricos con el valor actual de la configuración.
126
-
127
- **Trigger típico**: migración de datos que "normaliza" snapshots históricos al valor actual de la configuración global, destruyendo la trazabilidad.
128
-
129
- **Consecuencia**: recálculos retroactivos producen cifras incorrectas; auditorías de facturación no reproducen los valores originales.
130
-
131
- ---
132
-
133
- ## Patrones de enmascaramiento
134
-
135
- Código defensivo que oculta invariantes rotos en lugar de detectarlos:
136
-
137
- | Patrón | Código típico | Por qué es señal de alerta |
138
- |--------|--------------|---------------------------|
139
- | Ternario de clamp | `a > b ? a - b : 0` | Si el invariante se mantuviera, `a` nunca sería menor que `b` |
140
- | Try/catch vacío | `try { ... } catch {}` | El revert por estado roto se captura y se ignora |
141
- | Early exit en cero | `if value == 0: return` | Omite el cómputo cuando el estado roto produce cero |
142
- | Cap con min() | `min(calculado, disponible)` | El over-counting es el bug; `min()` solo evita el error visible |
143
- | Fallback a default | `valor = d.get(k, 0)` | Si la key debería existir pero fue eliminada sin limpiar el par, el default enmascara el dato faltante |
144
-
145
- Cuando se encuentra cualquiera de estos patrones en código que involucra dos valores acoplados, rastrear si el patrón existe porque el invariante ya estaba roto en algún path de mutación.
146
-
147
- <!-- Adaptado de nemesis-auditor-main bajo MIT License (https://github.com/0xiehnnkta/nemesis-auditor) -->
1
+ # Patrones de Estado Acoplado — Language-Agnostic
2
+
3
+ 9 patrones de estado acoplado frecuentes en sistemas Python, TypeScript, Go, Rust, Java y C#.
4
+
5
+ ---
6
+
7
+ ## Patrón 1: Balance ↔ Caché derivada
8
+
9
+ **Descripción**: un valor base (balance de cuenta, stock de inventario, crédito disponible) tiene una caché calculada a partir de él (porcentaje disponible, categoría de riesgo, tier de cliente).
10
+
11
+ **Invariante**: la caché debe reflejar el balance actual, no el del momento en que se calculó.
12
+
13
+ **Paths de mutación riesgosos**: cualquier operación que modifica el balance directamente sin pasar por el flujo que actualiza la caché.
14
+
15
+ **Trigger típico**: función de ajuste manual, importación masiva, o corrección administrativa que escribe el balance sin llamar al servicio que recalcula la caché.
16
+
17
+ **Consecuencia**: decisiones que leen la caché (¿aplica descuento? ¿otorgar crédito?) usan datos obsoletos.
18
+
19
+ ---
20
+
21
+ ## Patrón 2: Registro principal ↔ Índice secundario
22
+
23
+ **Descripción**: una tabla o colección tiene un índice secundario (índice invertido, tabla de búsqueda rápida, lista de IDs por categoría) que debe estar sincronizado con la tabla principal.
24
+
25
+ **Invariante**: toda entrada en la tabla principal tiene exactamente una entrada correspondiente en el índice secundario (o exactamente ninguna, según el diseño).
26
+
27
+ **Paths de mutación riesgosos**: delete de la tabla principal sin delete del índice; insert sin insert correspondiente; update que cambia la clave del índice sin actualizar el índice.
28
+
29
+ **Trigger típico**: función de "purga" o "limpieza" que borra de la tabla principal pero olvida el índice. O migración que popula la tabla principal pero no regenera el índice.
30
+
31
+ **Consecuencia**: búsquedas por índice devuelven resultados incorrectos (IDs que ya no existen, o registros que no aparecen en búsqueda aunque existan).
32
+
33
+ ---
34
+
35
+ ## Patrón 3: Contador agregado ↔ Suma de partes
36
+
37
+ **Descripción**: un contador de totales (total de pedidos, total de puntos, total de items activos) debe ser igual a la suma de los contadores individuales.
38
+
39
+ **Invariante**: `total == sum(individuales)` siempre.
40
+
41
+ **Paths de mutación riesgosos**: operaciones que modifican un contador individual sin ajustar el total. Frecuente en operaciones de batch donde el total se actualiza una sola vez al final pero un error parcial deja los individuales en estado inconsistente.
42
+
43
+ **Trigger típico**: transacción fallida a mitad que actualiza N de M elementos pero no revierte el total; o función que incrementa directamente el conteo individual sin notificar al agregador.
44
+
45
+ **Consecuencia**: reportes y dashboards muestran cifras incorrectas; límites y cuotas se aplican con base en datos erróneos.
46
+
47
+ ---
48
+
49
+ ## Patrón 4: Sesión / Permiso derivado ↔ Recurso de origen
50
+
51
+ **Descripción**: un objeto de sesión, token, o permiso derivado almacena un snapshot de los permisos o atributos del actor al momento de su creación. El actor puede cambiar después.
52
+
53
+ **Invariante**: depende del diseño. Si los permisos se evalúan en tiempo de solicitud, el objeto derivado debe reflejar el estado actual del actor. Si se usan permisos fijados al momento de creación, el invariante es documentado explícitamente.
54
+
55
+ **Paths de mutación riesgosos**: revocar permisos al actor sin invalidar sesiones activas; cambiar el rol del actor sin forzar re-autenticación; eliminar al actor sin expirar sus tokens.
56
+
57
+ **Trigger típico**: función de "deshabilitar cuenta" que setea `is_active=False` en el usuario pero no invalida los tokens JWT o sesiones existentes.
58
+
59
+ **Consecuencia**: actor deshabilitado sigue operando hasta que el token expira naturalmente.
60
+
61
+ ---
62
+
63
+ ## Patrón 5: Posición ↔ Factor de salud cacheado
64
+
65
+ **Descripción**: un recurso con múltiples atributos (posición de deuda: monto + colateral + fecha + valor de mercado del colateral) tiene un factor calculado (ratio LTV, score de riesgo, estado de salud) que se cachea por rendimiento.
66
+
67
+ **Invariante**: el factor cacheado debe recalcularse cada vez que cualquiera de sus inputs cambia.
68
+
69
+ **Paths de mutación riesgosos**: actualizar cualquier input del cálculo sin invalidar o recalcular el factor cacheado.
70
+
71
+ **Trigger típico**: función que actualiza el colateral sin llamar a `recalcular_factor()`. O función de ajuste de precio que modifica el valor de mercado sin propagar la invalidación de caché.
72
+
73
+ **Consecuencia**: decisiones de riesgo (¿emitir alerta? ¿bloquear operación?) usan un factor obsoleto.
74
+
75
+ ---
76
+
77
+ ## Patrón 6: Replica / Proyección ↔ Origen CDC
78
+
79
+ **Descripción**: un sistema mantiene una replica o proyección de datos de origen para acelerar lecturas (tabla denormalizada, proyección CQRS, replica de lectura, caché materializada). El origen puede cambiar vía CDC (Change Data Capture), eventos de dominio, o sincronización periódica.
80
+
81
+ **Invariante**: la replica debe converger al estado del origen dentro de la ventana de latencia acordada.
82
+
83
+ **Paths de mutación riesgosos**: escrituras directas al origen que no emiten el evento de dominio necesario para actualizar la replica; fallos en el consumer de eventos que quedan silenciosos; migraciones que modifican el origen sin reprocesar los eventos para actualizar la replica.
84
+
85
+ **Trigger típico**: endpoint de "corrección de datos" que escribe directo a la BD origen sin publicar el evento de dominio correspondiente.
86
+
87
+ **Consecuencia**: lecturas de la replica devuelven datos obsoletos indefinidamente.
88
+
89
+ ---
90
+
91
+ ## Patrón 7: Inventario total ↔ Items en almacén (por ubicación)
92
+
93
+ **Descripción**: el inventario total de un SKU es la suma de los items en cada almacén o ubicación. Se mantiene desnormalizado por rendimiento.
94
+
95
+ **Invariante**: `inventario_total[SKU] == sum(items_por_ubicacion[ubicacion][SKU] for ubicacion in todas)`
96
+
97
+ **Paths de mutación riesgosos**: movimiento entre almacenes que decrementa una ubicación sin incrementar la otra; recepción en almacén sin incrementar el total; ajuste de inventario físico que corrige la ubicación sin propagar al total.
98
+
99
+ **Trigger típico**: función de "ajuste de cierre" que actualiza el stock físico de un almacén sin pasar por la capa de negocio que mantiene el total sincronizado.
100
+
101
+ **Consecuencia**: órdenes de compra se emiten sobre stock inexistente; o stock disponible no se vende por parecer agotado.
102
+
103
+ ---
104
+
105
+ ## Patrón 8: Estado de workflow ↔ Timestamps de auditoría
106
+
107
+ **Descripción**: una entidad con workflow (pedido, solicitud, tarea) almacena su estado actual y los timestamps de cada transición. El timestamp de la última transición debe corresponder al estado actual.
108
+
109
+ **Invariante**: `estado_actual` siempre tiene un timestamp asociado que refleja cuándo se alcanzó ese estado.
110
+
111
+ **Paths de mutación riesgosos**: funciones que cambian el estado sin registrar el timestamp; o funciones que registran el timestamp sin actualizar el estado.
112
+
113
+ **Trigger típico**: función de "forzar estado" en panel de administración que escribe `estado = 'COMPLETADO'` directamente sin pasar por la máquina de estados que registra `completado_en = now()`.
114
+
115
+ **Consecuencia**: reportes de tiempo de ciclo son incorrectos; SLAs no se calculan bien; auditorías de cumplimiento fallan.
116
+
117
+ ---
118
+
119
+ ## Patrón 9: Configuración global ↔ Snapshot por instancia
120
+
121
+ **Descripción**: hay una configuración global (tasa de impuesto, precio de tarifa, límite de política) y objetos que almacenan el valor que tenía la configuración en el momento de su creación (tasa de impuesto aplicada a esta factura, precio de tarifa al momento de la reserva).
122
+
123
+ **Invariante**: el snapshot por instancia debe preservar el valor histórico, nunca actualizarse cuando cambia la configuración global. Pero si hay un bug en la creación, el snapshot puede haber capturado un valor incorrecto.
124
+
125
+ **Paths de mutación riesgosos**: función que no captura el snapshot al crear la instancia y usa la configuración global en tiempo de lectura; o función de retroactiva que sobreescribe snapshots históricos con el valor actual de la configuración.
126
+
127
+ **Trigger típico**: migración de datos que "normaliza" snapshots históricos al valor actual de la configuración global, destruyendo la trazabilidad.
128
+
129
+ **Consecuencia**: recálculos retroactivos producen cifras incorrectas; auditorías de facturación no reproducen los valores originales.
130
+
131
+ ---
132
+
133
+ ## Patrones de enmascaramiento
134
+
135
+ Código defensivo que oculta invariantes rotos en lugar de detectarlos:
136
+
137
+ | Patrón | Código típico | Por qué es señal de alerta |
138
+ |--------|--------------|---------------------------|
139
+ | Ternario de clamp | `a > b ? a - b : 0` | Si el invariante se mantuviera, `a` nunca sería menor que `b` |
140
+ | Try/catch vacío | `try { ... } catch {}` | El revert por estado roto se captura y se ignora |
141
+ | Early exit en cero | `if value == 0: return` | Omite el cómputo cuando el estado roto produce cero |
142
+ | Cap con min() | `min(calculado, disponible)` | El over-counting es el bug; `min()` solo evita el error visible |
143
+ | Fallback a default | `valor = d.get(k, 0)` | Si la key debería existir pero fue eliminada sin limpiar el par, el default enmascara el dato faltante |
144
+
145
+ Cuando se encuentra cualquiera de estos patrones en código que involucra dos valores acoplados, rastrear si el patrón existe porque el invariante ya estaba roto en algún path de mutación.
146
+
147
+ <!-- Adaptado de nemesis-auditor-main bajo MIT License (https://github.com/0xiehnnkta/nemesis-auditor) -->
@@ -1,12 +1,12 @@
1
1
  ---
2
2
  name: tdd-workflow
3
3
  description: Flujo completo de Test-Driven Development. Ciclo RED (el test falla) → GREEN (implementación mínima) → REFACTOR (limpieza). Incluye cobertura mínima obligatoria, tests de frontera, factories, fixtures y estrategias para diferentes tipos de código (APIs, services, componentes Angular).
4
- version: "1.0.2"
4
+ version: "1.0.4"
5
5
  evolved: true
6
- evolved-from: "1.0.1"
7
- evolved-at: "2026-05-14"
6
+ evolved-from: "1.0.3"
7
+ evolved-at: "2026-05-16"
8
8
  evolved-by: "aprender"
9
- evolved-note: "Patrón reloj inyectable (parámetro `ahora` con default Date.now()) para tests deterministas sin freezegun/jest.useFakeTimers validado en 3 módulos sesión webhook Opción C"
9
+ evolved-note: "v1.0.3: gotcha cwd cacheado al require(). v1.0.4: silenced tests por race en path único compartido + anti-patrón `if (X) assert(Y)` sin else. Origen PR #30 v1.5.2 (swl-ses)."
10
10
  herramientasPermitidas: [Read, Bash]
11
11
  evolvable: true # default para skill estandar
12
12
  exclusiones:
@@ -330,3 +330,149 @@ assert.equal(b.consumir(5, T0 + 5000), true); // 5 seg después: 5 tokens
330
330
  Aplica también a tests de clock skew (tiempo retrocede por NTP): pasar `T0 - 1000` y validar que la lógica no rompe. Origen: rate-limit-ip.js + webhook-dedup.js sesión 2026-05-13.
331
331
 
332
332
  **Tests nombrados por feature (`test_emitir_factura_exitosa`) pierden poder regresivo; nombrados por causa raíz (`test_repository_no_usa_columna_inexistente_p_monto`) detectan regresiones específicas sin reproducción manual** [CONFIRMADO en SIGM Opción C F1.4]: cuando se descubre un bug por una causa raíz concreta (typo en nombre de columna SQL, omisión de `selectinload`, mock que devuelve dict en vez de objeto, schema obsoleto), el test de regresión que se escribe debe llevar el nombre de la causa, no del feature afectado. Caso real: durante F1.4 de SIGM, el repository de pagos referenciaba `p.monto` cuando la columna se llamaba `p.monto_pagado`; el test escrito como `test_repository_no_usa_columna_inexistente_p_monto` falló inmediatamente en la siguiente sesión cuando otro agente reintrodujo el typo, sin necesidad de reproducir el escenario de negocio (emitir cobro real, verificar respuesta). Causa: los nombres orientados a feature (`test_pago_exitoso`) son ambiguos sobre QUÉ falla — si el test falla, el desarrollador debe diagnosticar; los nombres orientados a causa raíz (`test_X_no_usa_Y`, `test_query_incluye_selectinload_Z`, `test_service_devuelve_dict_no_objeto`) son auto-diagnósticos. Fix: para cada bug que cueste >30 min diagnosticar, escribir UN test adicional cuyo nombre describa la condición técnica violada, no el escenario de negocio. Convención: `test_<componente>_<condicion_tecnica>` o `test_<componente>_no_<anti_patron>`. Estos tests son tu segunda línea de defensa contra regresiones de la misma causa raíz, complementarios a los tests de comportamiento.
333
+
334
+ **`process.cwd()` cacheado al `require()` rompe tests con `process.chdir(sandbox)`** [PATRÓN GENÉRICO TESTING CLI]: scripts Node exportables que leen `process.cwd()` en el scope del módulo (al cargar) congelan el cwd al directorio de invocación. Los tests que crean sandboxes con `fs.mkdtempSync()` y luego `process.chdir(sandbox)` no afectan al cwd cacheado — el script sigue leyendo del cwd original y los assertions fallan con paths inesperados. Caso real (swl-ses `scripts/derivar-feature-list.js` 2026-05-15): la función `enriquecerDesdeFases(fases)` leía `const CWD = process.cwd()` calculado al `require()`; 2 tests con `process.chdir(sandbox)` retornaron `[]` en lugar de detectar el PLAN.md fixture. Causa: el constante se evaluó cuando el `node --test` cargó el módulo desde el cwd del proyecto, no desde el sandbox del test individual. Fix obligatorio: funciones exportables deben aceptar `cwd` como parámetro opcional con fallback dinámico (`function fn(args, opciones = {}) { const cwd = opciones.cwd || process.cwd(); ... }`). El código de producción no cambia (sin args extras), pero los tests pueden inyectar el cwd correcto. Aplica también a Python (`def fn(args, cwd: str | None = None): cwd = cwd or os.getcwd()`) y a cualquier lenguaje con tests que usen chdir.
335
+
336
+ ```js
337
+ // MAL — cwd cacheado al require, tests con process.chdir() fallan
338
+ const CWD = process.cwd();
339
+ const PLANNING_DIR = path.join(CWD, '.planning');
340
+
341
+ function enriquecerDesdeFases(fases) {
342
+ const archivos = fs.readdirSync(path.join(PLANNING_DIR, 'fases')); // cwd congelado
343
+ // ...
344
+ }
345
+
346
+ // BIEN — cwd dinámico con parámetro opcional para tests
347
+ function enriquecerDesdeFases(fases, opciones = {}) {
348
+ const cwd = opciones.cwd || process.cwd(); // recalcula al llamar
349
+ const archivos = fs.readdirSync(path.join(cwd, '.planning', 'fases'));
350
+ // ...
351
+ }
352
+
353
+ // En el test:
354
+ const sandbox = fs.mkdtempSync(path.join(os.tmpdir(), 'test-'));
355
+ fs.mkdirSync(path.join(sandbox, '.planning', 'fases'), { recursive: true });
356
+ // Opción A: pasar cwd explícito (recomendado)
357
+ const r = enriquecerDesdeFases([], { cwd: sandbox });
358
+ // Opción B: process.chdir() — solo funciona con cwd dinámico
359
+ process.chdir(sandbox);
360
+ const r2 = enriquecerDesdeFases([]);
361
+ ```
362
+
363
+ ---
364
+
365
+ ## Gotcha: silenced tests por race condition sobre estado compartido
366
+
367
+ ### El anti-patrón
368
+
369
+ ```javascript
370
+ // MAL — assertion condicional dentro de if que puede ser false por race
371
+ const FLAG = path.join(os.tmpdir(), 'mi-app.json');
372
+
373
+ test('flag sin contenido emite warning', () => {
374
+ borrarFlag();
375
+ const res = correrSubproceso();
376
+ if (fs.existsSync(FLAG)) { // ← otro test paralelo creó el flag
377
+ assert.match(res.stdout, /WARN/); // ← NUNCA se ejecuta si el if es false
378
+ }
379
+ // sin else → test PASA sin haber validado nada
380
+ });
381
+
382
+ // El test "verde" no significa "pasó" — significa "no falló ninguna assertion".
383
+ // Si la assertion vive dentro de un `if (race)`, una race favorable la salta
384
+ // y el test es vacío.
385
+ ```
386
+
387
+ ### Por qué pasa
388
+
389
+ `node:test` paraleliza **archivos** `.test.js` por default (no tests dentro
390
+ del mismo archivo). Si dos archivos tocan el mismo path único de filesystem
391
+ (`/tmp/foo.json`, lockfiles, sockets), las operaciones se intercalan no
392
+ deterministamente. Patrones típicos:
393
+
394
+ - Archivo A: `borrarFlag()` → spawn subprocess → assert
395
+ - Archivo B: spawn subprocess → `crearFlag()` durante A → assert de A condicionado falla
396
+
397
+ ### Patrones correctos
398
+
399
+ **Patrón 1 — Aislamiento por path único** (recomendado):
400
+
401
+ ```javascript
402
+ const env = { ...process.env };
403
+
404
+ // Path único por test usando mkdtempSync — sin contención
405
+ const dir = fs.mkdtempSync(path.join(os.tmpdir(), 'mi-app-test-'));
406
+ env.MI_APP_FLAG_PATH = path.join(dir, 'flag.json');
407
+
408
+ const res = spawnSync('node', [BIN], { env, ... });
409
+
410
+ // Ahora el assert es incondicional — el path es del test, no compartido
411
+ assert.match(res.stdout, /WARN/);
412
+ ```
413
+
414
+ Requiere que el SUT (System Under Test) honre una env var para override
415
+ del path. Si no la honra, agregar el override es parte del fix.
416
+
417
+ **Patrón 2 — Serialización forzada** (cuando el path es hardcoded):
418
+
419
+ ```bash
420
+ # Forzar --test-concurrency=1 en la suite completa
421
+ node --test --test-concurrency=1 tests/
422
+ ```
423
+
424
+ Tradeoff: tests más lentos pero deterministas. Aceptable si el aislamiento
425
+ no es factible (legacy code).
426
+
427
+ **Patrón 3 — assertions incondicionales** con setup determinista:
428
+
429
+ ```javascript
430
+ // MAL
431
+ if (fs.existsSync(FLAG)) assert.match(...)
432
+
433
+ // BIEN — setup garantiza la precondición, assertion no se salta
434
+ escribirFlag({ ... });
435
+ assert.ok(fs.existsSync(FLAG), 'precondición del test'); // ← assertion sobre el setup
436
+ const res = correrSubproceso();
437
+ assert.match(res.stdout, /WARN/); // ← assertion incondicional sobre el resultado
438
+ ```
439
+
440
+ ### Anti-patrón: `if (X) assert(Y)` sin `else`
441
+
442
+ ```javascript
443
+ // MAL — un test que pasa silenciosamente cuando X es false
444
+ test('hace algo', () => {
445
+ const algo = obtenerAlgo();
446
+ if (algo) { // ← race u otra fuente de no-determinismo
447
+ assert.equal(algo.valor, 42);
448
+ }
449
+ // sin else → veredicto "pass" sin haber validado nada
450
+ });
451
+
452
+ // BIEN — el setup garantiza la precondición o el test falla explícito
453
+ test('hace algo', () => {
454
+ const algo = obtenerAlgo();
455
+ assert.ok(algo, 'precondición: obtenerAlgo debe devolver valor');
456
+ assert.equal(algo.valor, 42);
457
+ });
458
+ ```
459
+
460
+ **Regla**: una assertion dentro de un `if` sin `else` es **un test que
461
+ puede pasar sin validar nada**. Estos "silenced tests" son la peor clase
462
+ de falsa cobertura: el reporter dice "pass" y nadie revisa el código
463
+ hasta que un bug llega a producción.
464
+
465
+ ### Detección
466
+
467
+ - Buscar `if (` dentro de cuerpos de `test(...)`/`it(...)` sin `else { fail() }`
468
+ o `else { assert(...) }` correspondiente.
469
+ - Si el cuerpo del `if` contiene `assert.*`, considerarlo silenced test
470
+ hasta que se demuestre que el `if` no puede ser false en ningún escenario.
471
+
472
+ ### Origen
473
+
474
+ Detectado en sesión 2026-05-16 del proyecto swl-ses (PR #30): tests del
475
+ flag `swl-ses-update-check.json` compartido entre dos archivos `.test.js`
476
+ paralelos. El test "sin flag → debe advertir" pasaba en CI cuando otro
477
+ archivo creaba el flag, sin ejecutar ninguna assertion. Fix: env var
478
+ `SWL_UPDATE_FLAG_PATH` para aislamiento + assertions incondicionales.