@saulwade/swl-ses 1.4.1 → 1.4.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.
- package/CLAUDE.md +1 -1
- package/README.md +1 -1
- package/agentes/nemesis-auditor-swl.md +161 -161
- package/bin/swl-mcp-server.js +187 -187
- package/comandos/swl/.evolved.json +22 -22
- package/comandos/swl/contribuir.md +233 -233
- package/comandos/swl/nemesis.md +122 -122
- package/gateway/lib/event-channel.js +191 -191
- package/habilidades/backend-production-resilience/SKILL.md +288 -288
- package/habilidades/benchmark-memoria/SKILL.md +186 -186
- package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
- package/habilidades/doubt-driven-review/SKILL.md +171 -171
- package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
- package/habilidades/eval-framework/SKILL.md +212 -212
- package/habilidades/feynman-auditor-swl/SKILL.md +123 -123
- package/habilidades/feynman-auditor-swl/recursos/preguntas-language-agnostic.md +108 -108
- package/habilidades/harness-claude-code/SKILL.md +299 -299
- package/habilidades/infra-github-actions/SKILL.md +166 -166
- package/habilidades/legacy-code-rescue/SKILL.md +267 -267
- package/habilidades/manejo-errores/.evolved.json +8 -8
- package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
- package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
- package/habilidades/patrones-python/SKILL.md +229 -229
- package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
- package/habilidades/planear-fase/SKILL.md +319 -319
- package/habilidades/release-semver/.evolved.json +8 -8
- package/habilidades/state-inconsistency-auditor-swl/SKILL.md +166 -166
- package/habilidades/state-inconsistency-auditor-swl/recursos/coupled-state-patterns.md +147 -147
- package/habilidades/testing-python/SKILL.md +340 -340
- package/habilidades/web-fetcher-routing/SKILL.md +75 -75
- package/hooks/claudemd-bloat-detector.js +161 -161
- package/hooks/lib/agent-routing.js +107 -107
- package/hooks/lib/auto-consolidator.js +335 -335
- package/hooks/lib/error-classifier.js +308 -308
- package/hooks/lib/merkle-audit.js +96 -96
- package/hooks/lib/provenance-tracker.js +191 -191
- package/hooks/lib/rate-limit-tracker.js +253 -253
- package/hooks/lib/resource-quota.js +122 -122
- package/hooks/lib/retry-jitter.js +165 -165
- package/hooks/lib/security-net.js +201 -201
- package/hooks/lib/skill-auditor.js +588 -588
- package/hooks/lib/sync-status.js +228 -228
- package/hooks/lib/taint-tracker.js +107 -107
- package/hooks/lib/text-similarity.js +241 -241
- package/hooks/lib/toon-compressor.js +245 -245
- package/hooks/registro-turnos.js +209 -209
- package/hooks/sugerir-regenerar-inventario.js +170 -170
- package/hooks/validar-formato-post-subagente.js +140 -140
- package/hooks/validar-memoria-hook.js +218 -218
- package/instintos/prompt-appendices.yaml +57 -57
- package/manifiestos/agent-output-schemas.json +57 -57
- package/manifiestos/modulos.json +11 -6
- package/manifiestos/perfiles.json +2 -1
- package/manifiestos/skills-lock.json +1114 -1114
- package/package.json +1 -1
- package/plantillas/auditor-veto-template.md +105 -105
- package/plantillas/github-workflows/README.md +47 -47
- package/plantillas/github-workflows/release-please.yml +44 -44
- package/plantillas/github-workflows/swl-ci.yml +107 -107
- package/plantillas/github-workflows/swl-security.yml +51 -51
- package/plugin.json +9 -1
- package/reglas/analisis-previo-tareas-grandes.md +172 -172
- package/reglas/arreglar-al-detectar.md +147 -147
- package/reglas/fragmentos-compartidos.md +152 -152
- package/reglas/harness-claude-code.md +213 -213
- package/reglas/usar-context7.md +226 -226
- package/schemas/diary-entry.schema.json +80 -80
- package/scripts/audit-tools/audit-history.js +330 -330
- package/scripts/audit-tools/bundle-tracker.js +290 -290
- package/scripts/audit-tools/canary-monitor.js +352 -352
- package/scripts/audit-tools/code-profiler.js +605 -605
- package/scripts/audit-tools/dep-doctor.js +320 -320
- package/scripts/audit-tools/env-validator.js +206 -206
- package/scripts/audit-tools/lib/fs-walk.js +48 -48
- package/scripts/audit-tools/lib/output.js +23 -23
- package/scripts/audit-tools/migration-checker.js +392 -392
- package/scripts/audit-tools/pentest-scanner.js +1436 -1436
- package/scripts/benchmark-memoria.js +167 -167
- package/scripts/configurar-branch-protection.js +418 -418
- package/scripts/detectar-aprendizajes-duplicados.js +151 -151
- package/scripts/field-report.js +199 -199
- package/scripts/generar-checklists-consolidados.js +273 -273
- package/scripts/generar-inventario.js +420 -420
- package/scripts/generar-matriz-lenguajes.js +271 -271
- package/scripts/lib/artefactos-python.js +43 -43
- package/scripts/lib/benchmark-metrics.js +160 -160
- package/scripts/lib/budget-enforcer.js +252 -252
- package/scripts/lib/configurar-ci.js +380 -380
- package/scripts/lib/contadores-inventario.js +217 -217
- package/scripts/lib/detectar-stack-detallado.js +307 -307
- package/scripts/lib/diary-entry.js +234 -234
- package/scripts/lib/eval-metrics-store.js +218 -218
- package/scripts/lib/eval-quality.js +171 -171
- package/scripts/lib/eval-schemas.js +144 -144
- package/scripts/lib/eval-self-correct.js +106 -106
- package/scripts/lib/eval-validator.js +185 -185
- package/scripts/lib/jaccard-similarity.js +98 -98
- package/scripts/lib/longmemeval-runner.js +125 -125
- package/scripts/lib/manifiestos.js +42 -1
- package/scripts/lib/npm-version.js +261 -261
- package/scripts/lib/paquetes-conocidos.js +50 -50
- package/scripts/lib/prompt-builder.js +264 -264
- package/scripts/lib/rrf-fusion.js +175 -175
- package/scripts/lib/scoring-instintos.js +277 -277
- package/scripts/lib/semantic-search.js +252 -252
- package/scripts/limpiar-artefactos-python.js +131 -131
- package/scripts/mcp-server/README.md +128 -128
- package/scripts/mcp-server/handlers.js +206 -206
- package/scripts/migrar-csv-a-array.js +168 -168
- package/scripts/migrar-fase-dominio.js +201 -201
- package/scripts/publicar.js +511 -511
- package/scripts/run-eval.js +141 -141
- package/scripts/validar-manifest.js +231 -195
- package/scripts/validar-userland-vacio.js +110 -110
|
@@ -1,166 +1,166 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: state-inconsistency-auditor-swl
|
|
3
|
-
description: >
|
|
4
|
-
Encuentra bugs donde una operación muta un fragmento de estado acoplado sin
|
|
5
|
-
actualizar su contraparte dependiente, causando corrupción silenciosa o fallos
|
|
6
|
-
en operaciones posteriores. Mapea sistemáticamente pares de estado acoplado,
|
|
7
|
-
todos sus paths de mutación, y los gaps donde un lado se actualiza sin el otro.
|
|
8
|
-
Language-agnostic (Python, TS, Go, Rust, Java, C#). Cargar cuando se sospeche
|
|
9
|
-
estado desincronizado: caché obsoleta, índices secundarios huérfanos, contadores
|
|
10
|
-
que no coinciden con la suma de sus partes, o permisos derivados que no reflejan
|
|
11
|
-
el estado de origen.
|
|
12
|
-
---
|
|
13
|
-
|
|
14
|
-
# Cuándo cargar
|
|
15
|
-
|
|
16
|
-
- Sospechas que un caché, índice secundario, o contador agregado está desincronizado.
|
|
17
|
-
- Una función actualiza la tabla principal pero no el índice derivado.
|
|
18
|
-
- Hay paths de "emergencia" o "admin" que modifican estado sin pasar por el flujo normal.
|
|
19
|
-
- Nemesis está corriendo su Pasada 2.
|
|
20
|
-
|
|
21
|
-
# Cuándo NO cargar
|
|
22
|
-
|
|
23
|
-
- Para búsqueda de vulnerabilidades conocidas (inyección SQL, XSS) — usar `revisor-seguridad-swl`.
|
|
24
|
-
- Para revisión de lógica de función individual sin estado acoplado — usar `feynman-auditor-swl`.
|
|
25
|
-
- Para módulos sin persistencia de estado (transformaciones funcionales puras, utilería).
|
|
26
|
-
|
|
27
|
-
---
|
|
28
|
-
|
|
29
|
-
# State Inconsistency Auditor
|
|
30
|
-
|
|
31
|
-
Encuentra bugs donde una operación actualiza State A sin actualizar State B, cuando el invariante del sistema exige que ambos cambien juntos.
|
|
32
|
-
|
|
33
|
-
Los patrones detallados de estado acoplado están en [recursos/coupled-state-patterns.md](recursos/coupled-state-patterns.md).
|
|
34
|
-
|
|
35
|
-
---
|
|
36
|
-
|
|
37
|
-
## El patrón abstracto
|
|
38
|
-
|
|
39
|
-
Todo sistema mantiene **PARES DE ESTADO ACOPLADO**: dos o más valores de almacenamiento que deben mantener una relación (un invariante) entre sí. Cuando cualquier operación cambia un lado del par sin ajustar el otro, el invariante se rompe. Operaciones posteriores que leen ambos valores producen resultados incorrectos.
|
|
40
|
-
|
|
41
|
-
---
|
|
42
|
-
|
|
43
|
-
## Reglas del auditor
|
|
44
|
-
|
|
45
|
-
```
|
|
46
|
-
REGLA 0: MAPEAR ANTES DE CAZAR
|
|
47
|
-
Nunca empezar a revisar funciones sin tener el mapa completo de
|
|
48
|
-
dependencias de estado acoplado. No se puede encontrar una actualización
|
|
49
|
-
faltante si no se sabe qué actualizaciones son necesarias.
|
|
50
|
-
|
|
51
|
-
REGLA 1: TODOS LOS PATHS DE MUTACIÓN IMPORTAN
|
|
52
|
-
Una variable de estado puede modificarse desde 5 funciones distintas.
|
|
53
|
-
Las 5 deben actualizar el estado acoplado. Si 4 lo hacen y 1 no — ese es el bug.
|
|
54
|
-
|
|
55
|
-
REGLA 2: LAS OPERACIONES PARCIALES SON LA FUENTE #1
|
|
56
|
-
Las eliminaciones completas (eliminar todo) generalmente resetean todo el estado
|
|
57
|
-
correctamente. Las operaciones parciales (reducir en X) frecuentemente olvidan
|
|
58
|
-
reducir proporcionalmente el estado acoplado.
|
|
59
|
-
|
|
60
|
-
REGLA 3: COMPARAR PATHS PARALELOS
|
|
61
|
-
Si transferir() y eliminar() ambos reducen un balance, AMBOS deben actualizar
|
|
62
|
-
el mismo conjunto de estado acoplado. Si uno lo hace y el otro no — hallazgo.
|
|
63
|
-
|
|
64
|
-
REGLA 4: EL CÓDIGO DEFENSIVO ENMASCARA BUGS
|
|
65
|
-
Código como `valor > minimo ? valor - minimo : 0` o `min(calculado, disponible)`
|
|
66
|
-
oculta invariantes rotos silenciosamente. Son señales de alerta, no protecciones.
|
|
67
|
-
|
|
68
|
-
REGLA 5: SOLO HALLAZGOS BASADOS EN EVIDENCIA
|
|
69
|
-
Todo hallazgo debe incluir: el par acoplado, la operación que lo rompe,
|
|
70
|
-
una secuencia de trigger concreta, y la consecuencia observable.
|
|
71
|
-
```
|
|
72
|
-
|
|
73
|
-
---
|
|
74
|
-
|
|
75
|
-
## Proceso
|
|
76
|
-
|
|
77
|
-
### Fase 1: Mapear pares de estado acoplado
|
|
78
|
-
|
|
79
|
-
Para cada variable de almacenamiento, preguntar: **"¿Qué otros valores de almacenamiento deben cambiar cuando este cambia?"**
|
|
80
|
-
|
|
81
|
-
Construir un mapa de dependencias. Ver patrones comunes en `recursos/coupled-state-patterns.md`.
|
|
82
|
-
|
|
83
|
-
Salida de Fase 1: **Mapa de Dependencias de Estado Acoplado**.
|
|
84
|
-
|
|
85
|
-
### Fase 2: Matriz de mutación
|
|
86
|
-
|
|
87
|
-
Para cada variable identificada en Fase 1, listar TODAS las funciones y paths de código que la modifican. Incluir: escrituras directas, incrementos/decrementos, eliminaciones, mutaciones indirectas (llamadas internas), triggers externos (callbacks, hooks).
|
|
88
|
-
|
|
89
|
-
Marcar con `???` toda entrada donde no se ha confirmado si la contraparte acoplada también se actualiza.
|
|
90
|
-
|
|
91
|
-
Los `???` son los **objetivos primarios de auditoría**.
|
|
92
|
-
|
|
93
|
-
### Fase 3: Cross-check — el corazón de la auditoría
|
|
94
|
-
|
|
95
|
-
Para cada par (operación, variable de estado) de Fase 2:
|
|
96
|
-
|
|
97
|
-
> "Esta operación modifica State A. ¿También actualiza todo el estado acoplado que depende de A?"
|
|
98
|
-
|
|
99
|
-
Verificar específicamente:
|
|
100
|
-
- Eliminación completa (A → 0): ¿se resetea/limpia todo el estado acoplado?
|
|
101
|
-
- Eliminación parcial (A decrece): ¿se reduce proporcionalmente todo el estado acoplado?
|
|
102
|
-
- Incremento (A crece): ¿se incrementa proporcionalmente todo el estado acoplado?
|
|
103
|
-
- Transferencia (A se mueve entre entidades): ¿el estado acoplado se mueve también?
|
|
104
|
-
- Eliminación de entrada de colección: ¿la entrada del par también se elimina?
|
|
105
|
-
- Operación en lote: ¿el estado acoplado se actualiza por iteración o solo una vez?
|
|
106
|
-
|
|
107
|
-
**Si algún path actualiza A sin actualizar su estado acoplado → HALLAZGO.**
|
|
108
|
-
|
|
109
|
-
### Fase 4: Orden de operaciones dentro de funciones
|
|
110
|
-
|
|
111
|
-
Muchas funciones realizan múltiples cambios de estado en secuencia. Rastrear el orden exacto y preguntar en cada paso: "¿Después de este paso, todos los pares acoplados siguen siendo consistentes?"
|
|
112
|
-
|
|
113
|
-
Bugs de orden comunes:
|
|
114
|
-
- Calcular resultado ANTES de actualizar el índice → resultado usa estado obsoleto
|
|
115
|
-
- Leer caché DESPUÉS de modificar el origen → validación usa datos viejos
|
|
116
|
-
- Emitir evento con valores viejos DESPUÉS del cambio de estado → sistemas externos se desincronizarán
|
|
117
|
-
|
|
118
|
-
### Fase 5: Comparar paths paralelos
|
|
119
|
-
|
|
120
|
-
Encontrar operaciones que logran resultados similares por paths distintos. Para cada grupo, comparar: ¿TODOS los paths actualizan el mismo estado acoplado?
|
|
121
|
-
|
|
122
|
-
### Fase 6: Rastrear journeys multi-paso
|
|
123
|
-
|
|
124
|
-
Simular secuencias donde un actor interactúa múltiples veces:
|
|
125
|
-
|
|
126
|
-
1. Actor inicializa un recurso (estado inicializado)
|
|
127
|
-
2. Tiempo pasa / estado externo evoluciona
|
|
128
|
-
3. Actor hace modificación PARCIAL (el estado acoplado puede romperse aquí)
|
|
129
|
-
4. Más tiempo pasa
|
|
130
|
-
5. Actor hace otra operación que lee el estado acoplado
|
|
131
|
-
|
|
132
|
-
En el paso 5: ¿el estado acoplado sigue siendo válido dado el cambio parcial del paso 3?
|
|
133
|
-
|
|
134
|
-
### Fase 7: Patrones de enmascaramiento
|
|
135
|
-
|
|
136
|
-
El código defensivo puede ocultar invariantes rotos. Ver patrones en `recursos/coupled-state-patterns.md`.
|
|
137
|
-
|
|
138
|
-
Señal: si un ternario del tipo `a > b ? a - b : 0` está en un cálculo que involucra dos valores acoplados, preguntar: ¿por qué `a` podría ser menor que `b`? Si el invariante se mantuviera, no podría.
|
|
139
|
-
|
|
140
|
-
---
|
|
141
|
-
|
|
142
|
-
## Verificación (obligatoria para C/A/M)
|
|
143
|
-
|
|
144
|
-
**Método A: Rastreo de código** — leer la función exacta, rastrear todas las llamadas internas, confirmar que no existe actualización oculta al estado acoplado.
|
|
145
|
-
|
|
146
|
-
**Método B: Test de PoC** — escribir un test que ejecute la secuencia de trigger y afirme que el estado es inconsistente después de la operación.
|
|
147
|
-
|
|
148
|
-
**Método C: Híbrido** — rastreo + PoC para hallazgos que abarcan múltiples módulos.
|
|
149
|
-
|
|
150
|
-
**Falsos positivos frecuentes:**
|
|
151
|
-
- Reconciliación oculta en un hook `_before_save` / `_after_update` / middleware.
|
|
152
|
-
- Evaluación diferida intencional: el estado acoplado se reconcilia en la próxima lectura.
|
|
153
|
-
- Asimetría de diseño documentada: los dos estados no son realmente acoplados como se asumió.
|
|
154
|
-
|
|
155
|
-
---
|
|
156
|
-
|
|
157
|
-
## Adaptación por lenguaje
|
|
158
|
-
|
|
159
|
-
| Concepto | Python | TypeScript | Go | Rust | Java | C# |
|
|
160
|
-
|---------|--------|------------|-----|------|------|-----|
|
|
161
|
-
| Almacenamiento | atributos / BD | propiedades / BD | campos struct / BD | campos struct | campos clase | propiedades |
|
|
162
|
-
| Colección clave-valor | `dict` / `HashMap` Redis | `Map` / objeto | `map[K]V` | `HashMap` | `Map<K,V>` | `Dictionary<K,V>` |
|
|
163
|
-
| Eliminar entrada | `del d[k]` / `pop` | `delete obj[k]` | `delete(m, k)` | `map.remove(&k)` | `map.remove(k)` | `dict.Remove(k)` |
|
|
164
|
-
| Hook post-mutación | `@receiver` / signal | middleware / hook | middleware | `impl Drop` / hook | `@PostUpdate` | event handler |
|
|
165
|
-
|
|
166
|
-
<!-- Adaptado de nemesis-auditor-main bajo MIT License (https://github.com/0xiehnnkta/nemesis-auditor) -->
|
|
1
|
+
---
|
|
2
|
+
name: state-inconsistency-auditor-swl
|
|
3
|
+
description: >
|
|
4
|
+
Encuentra bugs donde una operación muta un fragmento de estado acoplado sin
|
|
5
|
+
actualizar su contraparte dependiente, causando corrupción silenciosa o fallos
|
|
6
|
+
en operaciones posteriores. Mapea sistemáticamente pares de estado acoplado,
|
|
7
|
+
todos sus paths de mutación, y los gaps donde un lado se actualiza sin el otro.
|
|
8
|
+
Language-agnostic (Python, TS, Go, Rust, Java, C#). Cargar cuando se sospeche
|
|
9
|
+
estado desincronizado: caché obsoleta, índices secundarios huérfanos, contadores
|
|
10
|
+
que no coinciden con la suma de sus partes, o permisos derivados que no reflejan
|
|
11
|
+
el estado de origen.
|
|
12
|
+
---
|
|
13
|
+
|
|
14
|
+
# Cuándo cargar
|
|
15
|
+
|
|
16
|
+
- Sospechas que un caché, índice secundario, o contador agregado está desincronizado.
|
|
17
|
+
- Una función actualiza la tabla principal pero no el índice derivado.
|
|
18
|
+
- Hay paths de "emergencia" o "admin" que modifican estado sin pasar por el flujo normal.
|
|
19
|
+
- Nemesis está corriendo su Pasada 2.
|
|
20
|
+
|
|
21
|
+
# Cuándo NO cargar
|
|
22
|
+
|
|
23
|
+
- Para búsqueda de vulnerabilidades conocidas (inyección SQL, XSS) — usar `revisor-seguridad-swl`.
|
|
24
|
+
- Para revisión de lógica de función individual sin estado acoplado — usar `feynman-auditor-swl`.
|
|
25
|
+
- Para módulos sin persistencia de estado (transformaciones funcionales puras, utilería).
|
|
26
|
+
|
|
27
|
+
---
|
|
28
|
+
|
|
29
|
+
# State Inconsistency Auditor
|
|
30
|
+
|
|
31
|
+
Encuentra bugs donde una operación actualiza State A sin actualizar State B, cuando el invariante del sistema exige que ambos cambien juntos.
|
|
32
|
+
|
|
33
|
+
Los patrones detallados de estado acoplado están en [recursos/coupled-state-patterns.md](recursos/coupled-state-patterns.md).
|
|
34
|
+
|
|
35
|
+
---
|
|
36
|
+
|
|
37
|
+
## El patrón abstracto
|
|
38
|
+
|
|
39
|
+
Todo sistema mantiene **PARES DE ESTADO ACOPLADO**: dos o más valores de almacenamiento que deben mantener una relación (un invariante) entre sí. Cuando cualquier operación cambia un lado del par sin ajustar el otro, el invariante se rompe. Operaciones posteriores que leen ambos valores producen resultados incorrectos.
|
|
40
|
+
|
|
41
|
+
---
|
|
42
|
+
|
|
43
|
+
## Reglas del auditor
|
|
44
|
+
|
|
45
|
+
```
|
|
46
|
+
REGLA 0: MAPEAR ANTES DE CAZAR
|
|
47
|
+
Nunca empezar a revisar funciones sin tener el mapa completo de
|
|
48
|
+
dependencias de estado acoplado. No se puede encontrar una actualización
|
|
49
|
+
faltante si no se sabe qué actualizaciones son necesarias.
|
|
50
|
+
|
|
51
|
+
REGLA 1: TODOS LOS PATHS DE MUTACIÓN IMPORTAN
|
|
52
|
+
Una variable de estado puede modificarse desde 5 funciones distintas.
|
|
53
|
+
Las 5 deben actualizar el estado acoplado. Si 4 lo hacen y 1 no — ese es el bug.
|
|
54
|
+
|
|
55
|
+
REGLA 2: LAS OPERACIONES PARCIALES SON LA FUENTE #1
|
|
56
|
+
Las eliminaciones completas (eliminar todo) generalmente resetean todo el estado
|
|
57
|
+
correctamente. Las operaciones parciales (reducir en X) frecuentemente olvidan
|
|
58
|
+
reducir proporcionalmente el estado acoplado.
|
|
59
|
+
|
|
60
|
+
REGLA 3: COMPARAR PATHS PARALELOS
|
|
61
|
+
Si transferir() y eliminar() ambos reducen un balance, AMBOS deben actualizar
|
|
62
|
+
el mismo conjunto de estado acoplado. Si uno lo hace y el otro no — hallazgo.
|
|
63
|
+
|
|
64
|
+
REGLA 4: EL CÓDIGO DEFENSIVO ENMASCARA BUGS
|
|
65
|
+
Código como `valor > minimo ? valor - minimo : 0` o `min(calculado, disponible)`
|
|
66
|
+
oculta invariantes rotos silenciosamente. Son señales de alerta, no protecciones.
|
|
67
|
+
|
|
68
|
+
REGLA 5: SOLO HALLAZGOS BASADOS EN EVIDENCIA
|
|
69
|
+
Todo hallazgo debe incluir: el par acoplado, la operación que lo rompe,
|
|
70
|
+
una secuencia de trigger concreta, y la consecuencia observable.
|
|
71
|
+
```
|
|
72
|
+
|
|
73
|
+
---
|
|
74
|
+
|
|
75
|
+
## Proceso
|
|
76
|
+
|
|
77
|
+
### Fase 1: Mapear pares de estado acoplado
|
|
78
|
+
|
|
79
|
+
Para cada variable de almacenamiento, preguntar: **"¿Qué otros valores de almacenamiento deben cambiar cuando este cambia?"**
|
|
80
|
+
|
|
81
|
+
Construir un mapa de dependencias. Ver patrones comunes en `recursos/coupled-state-patterns.md`.
|
|
82
|
+
|
|
83
|
+
Salida de Fase 1: **Mapa de Dependencias de Estado Acoplado**.
|
|
84
|
+
|
|
85
|
+
### Fase 2: Matriz de mutación
|
|
86
|
+
|
|
87
|
+
Para cada variable identificada en Fase 1, listar TODAS las funciones y paths de código que la modifican. Incluir: escrituras directas, incrementos/decrementos, eliminaciones, mutaciones indirectas (llamadas internas), triggers externos (callbacks, hooks).
|
|
88
|
+
|
|
89
|
+
Marcar con `???` toda entrada donde no se ha confirmado si la contraparte acoplada también se actualiza.
|
|
90
|
+
|
|
91
|
+
Los `???` son los **objetivos primarios de auditoría**.
|
|
92
|
+
|
|
93
|
+
### Fase 3: Cross-check — el corazón de la auditoría
|
|
94
|
+
|
|
95
|
+
Para cada par (operación, variable de estado) de Fase 2:
|
|
96
|
+
|
|
97
|
+
> "Esta operación modifica State A. ¿También actualiza todo el estado acoplado que depende de A?"
|
|
98
|
+
|
|
99
|
+
Verificar específicamente:
|
|
100
|
+
- Eliminación completa (A → 0): ¿se resetea/limpia todo el estado acoplado?
|
|
101
|
+
- Eliminación parcial (A decrece): ¿se reduce proporcionalmente todo el estado acoplado?
|
|
102
|
+
- Incremento (A crece): ¿se incrementa proporcionalmente todo el estado acoplado?
|
|
103
|
+
- Transferencia (A se mueve entre entidades): ¿el estado acoplado se mueve también?
|
|
104
|
+
- Eliminación de entrada de colección: ¿la entrada del par también se elimina?
|
|
105
|
+
- Operación en lote: ¿el estado acoplado se actualiza por iteración o solo una vez?
|
|
106
|
+
|
|
107
|
+
**Si algún path actualiza A sin actualizar su estado acoplado → HALLAZGO.**
|
|
108
|
+
|
|
109
|
+
### Fase 4: Orden de operaciones dentro de funciones
|
|
110
|
+
|
|
111
|
+
Muchas funciones realizan múltiples cambios de estado en secuencia. Rastrear el orden exacto y preguntar en cada paso: "¿Después de este paso, todos los pares acoplados siguen siendo consistentes?"
|
|
112
|
+
|
|
113
|
+
Bugs de orden comunes:
|
|
114
|
+
- Calcular resultado ANTES de actualizar el índice → resultado usa estado obsoleto
|
|
115
|
+
- Leer caché DESPUÉS de modificar el origen → validación usa datos viejos
|
|
116
|
+
- Emitir evento con valores viejos DESPUÉS del cambio de estado → sistemas externos se desincronizarán
|
|
117
|
+
|
|
118
|
+
### Fase 5: Comparar paths paralelos
|
|
119
|
+
|
|
120
|
+
Encontrar operaciones que logran resultados similares por paths distintos. Para cada grupo, comparar: ¿TODOS los paths actualizan el mismo estado acoplado?
|
|
121
|
+
|
|
122
|
+
### Fase 6: Rastrear journeys multi-paso
|
|
123
|
+
|
|
124
|
+
Simular secuencias donde un actor interactúa múltiples veces:
|
|
125
|
+
|
|
126
|
+
1. Actor inicializa un recurso (estado inicializado)
|
|
127
|
+
2. Tiempo pasa / estado externo evoluciona
|
|
128
|
+
3. Actor hace modificación PARCIAL (el estado acoplado puede romperse aquí)
|
|
129
|
+
4. Más tiempo pasa
|
|
130
|
+
5. Actor hace otra operación que lee el estado acoplado
|
|
131
|
+
|
|
132
|
+
En el paso 5: ¿el estado acoplado sigue siendo válido dado el cambio parcial del paso 3?
|
|
133
|
+
|
|
134
|
+
### Fase 7: Patrones de enmascaramiento
|
|
135
|
+
|
|
136
|
+
El código defensivo puede ocultar invariantes rotos. Ver patrones en `recursos/coupled-state-patterns.md`.
|
|
137
|
+
|
|
138
|
+
Señal: si un ternario del tipo `a > b ? a - b : 0` está en un cálculo que involucra dos valores acoplados, preguntar: ¿por qué `a` podría ser menor que `b`? Si el invariante se mantuviera, no podría.
|
|
139
|
+
|
|
140
|
+
---
|
|
141
|
+
|
|
142
|
+
## Verificación (obligatoria para C/A/M)
|
|
143
|
+
|
|
144
|
+
**Método A: Rastreo de código** — leer la función exacta, rastrear todas las llamadas internas, confirmar que no existe actualización oculta al estado acoplado.
|
|
145
|
+
|
|
146
|
+
**Método B: Test de PoC** — escribir un test que ejecute la secuencia de trigger y afirme que el estado es inconsistente después de la operación.
|
|
147
|
+
|
|
148
|
+
**Método C: Híbrido** — rastreo + PoC para hallazgos que abarcan múltiples módulos.
|
|
149
|
+
|
|
150
|
+
**Falsos positivos frecuentes:**
|
|
151
|
+
- Reconciliación oculta en un hook `_before_save` / `_after_update` / middleware.
|
|
152
|
+
- Evaluación diferida intencional: el estado acoplado se reconcilia en la próxima lectura.
|
|
153
|
+
- Asimetría de diseño documentada: los dos estados no son realmente acoplados como se asumió.
|
|
154
|
+
|
|
155
|
+
---
|
|
156
|
+
|
|
157
|
+
## Adaptación por lenguaje
|
|
158
|
+
|
|
159
|
+
| Concepto | Python | TypeScript | Go | Rust | Java | C# |
|
|
160
|
+
|---------|--------|------------|-----|------|------|-----|
|
|
161
|
+
| Almacenamiento | atributos / BD | propiedades / BD | campos struct / BD | campos struct | campos clase | propiedades |
|
|
162
|
+
| Colección clave-valor | `dict` / `HashMap` Redis | `Map` / objeto | `map[K]V` | `HashMap` | `Map<K,V>` | `Dictionary<K,V>` |
|
|
163
|
+
| Eliminar entrada | `del d[k]` / `pop` | `delete obj[k]` | `delete(m, k)` | `map.remove(&k)` | `map.remove(k)` | `dict.Remove(k)` |
|
|
164
|
+
| Hook post-mutación | `@receiver` / signal | middleware / hook | middleware | `impl Drop` / hook | `@PostUpdate` | event handler |
|
|
165
|
+
|
|
166
|
+
<!-- Adaptado de nemesis-auditor-main bajo MIT License (https://github.com/0xiehnnkta/nemesis-auditor) -->
|
|
@@ -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) -->
|