@saulwade/swl-ses 1.3.4 → 1.3.7
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 +2 -2
- package/README.md +34 -34
- package/bin/swl-mcp-server.js +187 -187
- package/bin/swl-ses.js +4 -62
- package/comandos/swl/.evolved.json +22 -22
- package/comandos/swl/adoptar-proyecto.md +207 -207
- package/comandos/swl/contribuir.md +233 -233
- 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/extractor-de-aprendizajes/SKILL.md +321 -321
- 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/swl-claudemd/SKILL.md +220 -220
- package/habilidades/testing-python/SKILL.md +340 -340
- package/hooks/claudemd-bloat-detector.js +161 -161
- package/hooks/extraccion-aprendizajes.js +19 -12
- 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/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/skills-lock.json +1093 -1093
- 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 +1 -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/benchmark-memoria.js +167 -167
- package/scripts/comandos/info.js +1 -1
- package/scripts/configurar-branch-protection.js +418 -418
- package/scripts/detectar-aprendizajes-duplicados.js +151 -151
- package/scripts/doctor.js +77 -3
- 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/inicializar.js +2 -2
- package/scripts/instalador.js +40 -3
- package/scripts/instalar-git-hook.js +2 -2
- 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/gitignore-manifest.js +1 -1
- package/scripts/lib/jaccard-similarity.js +98 -98
- package/scripts/lib/longmemeval-runner.js +125 -125
- package/scripts/lib/npm-version.js +261 -261
- package/scripts/lib/paquetes-conocidos.js +50 -50
- package/scripts/lib/parsear-opciones.js +136 -0
- 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/lib/transformadores/claude.js +200 -200
- package/scripts/lib/transformadores/codex.js +1 -1
- package/scripts/lib/transformadores/copilot.js +1 -1
- package/scripts/lib/transformadores/gemini.js +1 -1
- package/scripts/lib/transformadores/opencode.js +1 -1
- 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 +195 -195
- package/scripts/validar-userland-vacio.js +110 -110
- package/scripts/verificar-release.js +5 -1
|
@@ -1,267 +1,267 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: legacy-code-rescue
|
|
3
|
-
description: >
|
|
4
|
-
Técnicas para trabajar con código heredado sin tests, siguiendo la metodología
|
|
5
|
-
de Michael Feathers ("Working Effectively with Legacy Code"). Cubre
|
|
6
|
-
identificación de seams, characterization tests para capturar comportamiento
|
|
7
|
-
real (no el deseado), patrones para introducir cambios sin romper
|
|
8
|
-
(sprout method, sprout class, wrap method, wrap class), técnicas para romper
|
|
9
|
-
dependencias problemáticas, y orden de prioridad cuando se hereda código
|
|
10
|
-
sin cobertura. Cargar cuando se herede código sin tests, cuando un módulo
|
|
11
|
-
crítico tenga cobertura <30%, o cuando se planee refactoring de un sistema
|
|
12
|
-
heredado.
|
|
13
|
-
version: "1.0.0"
|
|
14
|
-
evolved: false
|
|
15
|
-
herramientasPermitidas: [Read]
|
|
16
|
-
exclusiones:
|
|
17
|
-
- "No cargar para refactoring de código moderno con buena cobertura — si los tests existen y cubren el área a refactorear, basta con `tdd-workflow`. Este skill es específico para código sin red de seguridad."
|
|
18
|
-
- "No cargar para migración entre versiones de framework — usar `deprecacion-migracion`. Este skill cubre rescate de código heredado en general, no actualizaciones de stack."
|
|
19
|
-
- "No cargar para deprecación planificada de features — usar `deprecacion-migracion`. Este skill cubre el problema inverso: código que NO se quiere deprecar pero hay que cambiar con seguridad."
|
|
20
|
-
- "No cargar para optimización de rendimiento de código moderno — los patrones aquí (especialmente characterization tests) priorizan seguridad sobre velocidad. Si la prioridad es perf, usar `performance-baseline` y profilers."
|
|
21
|
-
evolvable: true
|
|
22
|
-
---
|
|
23
|
-
|
|
24
|
-
# Legacy Code Rescue
|
|
25
|
-
|
|
26
|
-
Adaptado de "Working Effectively with Legacy Code" (Michael Feathers, 2004).
|
|
27
|
-
Definición operativa de Feathers: **legacy code es código sin tests**, sin
|
|
28
|
-
importar su edad o calidad aparente.
|
|
29
|
-
|
|
30
|
-
## Cuándo cargar
|
|
31
|
-
|
|
32
|
-
- Se hereda un módulo crítico con cobertura <30% y hay que cambiarlo.
|
|
33
|
-
- Se diagnostica un bug en código que nadie ha tocado en meses/años.
|
|
34
|
-
- Se planea un refactor de un sistema sin red de seguridad de tests.
|
|
35
|
-
- Se necesita agregar una feature en código que actualmente no se puede
|
|
36
|
-
testear de forma aislada.
|
|
37
|
-
- Se reciben quejas tipo "tengo miedo de tocar X porque puede romper Y".
|
|
38
|
-
|
|
39
|
-
## Cuándo NO cargar
|
|
40
|
-
|
|
41
|
-
- El código tiene cobertura ≥80% y los tests pasan — usar `tdd-workflow`.
|
|
42
|
-
- Se está migrando entre versiones de framework — usar `deprecacion-migracion`.
|
|
43
|
-
- Se está deprecando una feature intencionalmente — usar `deprecacion-migracion`.
|
|
44
|
-
- El objetivo es optimizar rendimiento — los patrones aquí priorizan seguridad
|
|
45
|
-
sobre velocidad; usar `performance-baseline` y profilers.
|
|
46
|
-
|
|
47
|
-
## Directiva primaria
|
|
48
|
-
|
|
49
|
-
**Antes de cambiar código heredado, capturar lo que hace**, no lo que se
|
|
50
|
-
supone que debe hacer. El comportamiento actual (incluidos sus bugs) es
|
|
51
|
-
la fuente de verdad — algún caller probablemente depende de cada peculiaridad.
|
|
52
|
-
|
|
53
|
-
Orden estricto:
|
|
54
|
-
1. **Caracterizar** el comportamiento actual con tests.
|
|
55
|
-
2. **Romper dependencias** lo mínimo necesario para que los tests sean posibles.
|
|
56
|
-
3. **Cambiar** el código.
|
|
57
|
-
4. **Refactorizar** ya con red de seguridad.
|
|
58
|
-
|
|
59
|
-
NO empezar refactorizando "porque el código está feo". Sin tests, cada cambio
|
|
60
|
-
es una apuesta.
|
|
61
|
-
|
|
62
|
-
---
|
|
63
|
-
|
|
64
|
-
## Identificar seams (costuras)
|
|
65
|
-
|
|
66
|
-
Un **seam** es un punto donde el comportamiento del programa puede cambiarse
|
|
67
|
-
sin editar el código original. Los seams son la palanca para introducir
|
|
68
|
-
tests sin romper nada.
|
|
69
|
-
|
|
70
|
-
### Tipos de seams
|
|
71
|
-
|
|
72
|
-
| Tipo | Cuándo aplica | Cómo se activa |
|
|
73
|
-
|------|---------------|----------------|
|
|
74
|
-
| **Object seam** | OOP con polimorfismo | Inyectar una subclase/mock que herede |
|
|
75
|
-
| **Preprocessing seam** | C/C++ con `#define` | Macros o preprocesador |
|
|
76
|
-
| **Link seam** | Lenguajes compilados con linker | Sustituir librería en build |
|
|
77
|
-
| **Method seam** (Python/JS) | Lenguajes dinámicos | Monkeypatching, importlib |
|
|
78
|
-
|
|
79
|
-
En Python/TypeScript modernos, el object seam es el predominante:
|
|
80
|
-
inyectar dependencias en lugar de instanciarlas dentro del constructor.
|
|
81
|
-
|
|
82
|
-
### Identificar seams en código existente
|
|
83
|
-
|
|
84
|
-
Buscar:
|
|
85
|
-
- `new SomeClass()` dentro de constructores → object seam latente al refactorizar.
|
|
86
|
-
- Llamadas estáticas a clases globales → buscar wrappers o usar dependency injection.
|
|
87
|
-
- Imports de módulos con efectos secundarios → candidato a method seam.
|
|
88
|
-
- Funciones puras escondidas dentro de métodos largos → extraer es seam barato.
|
|
89
|
-
|
|
90
|
-
---
|
|
91
|
-
|
|
92
|
-
## Characterization tests — capturar el comportamiento real
|
|
93
|
-
|
|
94
|
-
Un **characterization test** documenta lo que el código HACE, no lo que
|
|
95
|
-
debería hacer. Son la red de seguridad mínima antes de cualquier cambio.
|
|
96
|
-
|
|
97
|
-
### Receta
|
|
98
|
-
|
|
99
|
-
1. Tomar el código y ejecutarlo con un input concreto.
|
|
100
|
-
2. Capturar el output exacto (incluido errores, side effects, state final).
|
|
101
|
-
3. Escribir un test que assert ese output específico.
|
|
102
|
-
4. Repetir con varios inputs representativos: caso típico, edge cases
|
|
103
|
-
conocidos, inputs malformados, valores extremos.
|
|
104
|
-
5. Cuando el test pasa, **es la verdad** — incluso si revela un bug.
|
|
105
|
-
|
|
106
|
-
```python
|
|
107
|
-
# Ejemplo: función legacy sin tests
|
|
108
|
-
def calcular_descuento(precio, tipo_cliente):
|
|
109
|
-
# ... 80 líneas de if/else, redondeos extraños, edge cases ocultos
|
|
110
|
-
return total
|
|
111
|
-
|
|
112
|
-
# Characterization test — captura el comportamiento ACTUAL
|
|
113
|
-
def test_caracterizacion_descuento():
|
|
114
|
-
# Input típico
|
|
115
|
-
assert calcular_descuento(100, "VIP") == 85.0
|
|
116
|
-
# Edge case observado: tipo_cliente vacío
|
|
117
|
-
assert calcular_descuento(100, "") == 100.0
|
|
118
|
-
# Bug aparente, pero callers pueden depender de él
|
|
119
|
-
assert calcular_descuento(0, "VIP") == 0.0
|
|
120
|
-
# Negativos: el código devuelve negativos sin validar — documentarlo
|
|
121
|
-
assert calcular_descuento(-50, "VIP") == -42.5
|
|
122
|
-
```
|
|
123
|
-
|
|
124
|
-
### Reglas para characterization tests
|
|
125
|
-
|
|
126
|
-
- **No "arreglar" bugs** mientras se caracteriza — eso viene después.
|
|
127
|
-
- Si el comportamiento parece incorrecto, escribir un test que lo
|
|
128
|
-
documente Y abrir un ticket separado para discutir el cambio.
|
|
129
|
-
- Apuntar a cubrir los flujos que se van a tocar — no toda la base de
|
|
130
|
-
código.
|
|
131
|
-
- Tests rápidos > tests perfectos. Un test feo que corre es mejor que
|
|
132
|
-
uno elegante que no se escribió.
|
|
133
|
-
|
|
134
|
-
---
|
|
135
|
-
|
|
136
|
-
## Patrones para cambio seguro
|
|
137
|
-
|
|
138
|
-
Cuando hay que agregar comportamiento sin tocar código existente.
|
|
139
|
-
|
|
140
|
-
### Sprout Method
|
|
141
|
-
|
|
142
|
-
Si necesitas lógica nueva dentro de un método legacy difícil de testear,
|
|
143
|
-
**escribe la lógica nueva como método aparte** (con tests propios) y
|
|
144
|
-
llámalo desde el método legacy.
|
|
145
|
-
|
|
146
|
-
```python
|
|
147
|
-
# Antes — quieres agregar validación al método legacy
|
|
148
|
-
def procesar_orden(orden):
|
|
149
|
-
# 200 líneas legacy intocables
|
|
150
|
-
...
|
|
151
|
-
|
|
152
|
-
# Después — la nueva validación vive aparte y SÍ es testeable
|
|
153
|
-
def procesar_orden(orden):
|
|
154
|
-
# 200 líneas legacy intocables
|
|
155
|
-
if not _validar_disponibilidad_inventario(orden): # nuevo
|
|
156
|
-
return ResultadoOrden.fallo("sin stock")
|
|
157
|
-
...
|
|
158
|
-
|
|
159
|
-
def _validar_disponibilidad_inventario(orden):
|
|
160
|
-
"""Nuevo método con tests propios (no toca el legacy)."""
|
|
161
|
-
return all(item.cantidad <= item.producto.stock for item in orden.items)
|
|
162
|
-
```
|
|
163
|
-
|
|
164
|
-
### Sprout Class
|
|
165
|
-
|
|
166
|
-
Cuando la lógica nueva es suficientemente grande, va en una **clase nueva**
|
|
167
|
-
en lugar de un método. Misma idea: lógica nueva con tests, llamada desde
|
|
168
|
-
el legacy.
|
|
169
|
-
|
|
170
|
-
### Wrap Method
|
|
171
|
-
|
|
172
|
-
Si necesitas ejecutar algo ANTES o DESPUÉS de un método legacy, renombra
|
|
173
|
-
el original a algo interno (`_procesar_orden_original`) y crea uno nuevo
|
|
174
|
-
con la firma original que llame al wrap + el original.
|
|
175
|
-
|
|
176
|
-
```python
|
|
177
|
-
def procesar_orden(orden): # firma pública intacta
|
|
178
|
-
_registrar_intento(orden) # nueva pre-acción, testeable
|
|
179
|
-
resultado = _procesar_orden_original(orden)
|
|
180
|
-
_registrar_resultado(resultado) # nueva post-acción, testeable
|
|
181
|
-
return resultado
|
|
182
|
-
|
|
183
|
-
def _procesar_orden_original(orden):
|
|
184
|
-
# 200 líneas legacy intocables, ahora son privadas
|
|
185
|
-
...
|
|
186
|
-
```
|
|
187
|
-
|
|
188
|
-
### Wrap Class
|
|
189
|
-
|
|
190
|
-
Para wrap de un sistema más grande: una clase nueva con la misma interfaz
|
|
191
|
-
delega al legacy y agrega comportamiento alrededor (decorator pattern).
|
|
192
|
-
|
|
193
|
-
---
|
|
194
|
-
|
|
195
|
-
## Romper dependencias
|
|
196
|
-
|
|
197
|
-
Algunos legacy methods son intratables por dependencias:
|
|
198
|
-
- Conexiones a BD/red en el constructor.
|
|
199
|
-
- Singletons globales que se acceden en cualquier parte.
|
|
200
|
-
- Llamadas a `time.now()`, `random()`, filesystem.
|
|
201
|
-
|
|
202
|
-
Técnicas (en orden de invasividad creciente):
|
|
203
|
-
|
|
204
|
-
1. **Subclass and Override Method**: hereda de la clase legacy, sobreescribe
|
|
205
|
-
el método problemático con un mock para los tests.
|
|
206
|
-
2. **Extract Interface / Extract Class**: extrae una interfaz pequeña con
|
|
207
|
-
solo los métodos que el código usa, e inyecta una implementación de prueba.
|
|
208
|
-
3. **Parameterize Method/Constructor**: inyecta como parámetro lo que antes
|
|
209
|
-
se obtenía globalmente (clock, random, IO).
|
|
210
|
-
4. **Replace Function with Function Pointer / Strategy**: cambiar funciones
|
|
211
|
-
estáticas por callables inyectables.
|
|
212
|
-
|
|
213
|
-
Aplicar la técnica MENOS invasiva que resuelva el problema. Cuanto más
|
|
214
|
-
invasiva, más riesgo de romper algo durante el rescate.
|
|
215
|
-
|
|
216
|
-
---
|
|
217
|
-
|
|
218
|
-
## Anti-patrones
|
|
219
|
-
|
|
220
|
-
- **Refactorizar antes de tener characterization tests**: cualquier cambio
|
|
221
|
-
estructural sin red de seguridad es una apuesta.
|
|
222
|
-
- **"Mejorar" el comportamiento mientras se caracteriza**: si parece un
|
|
223
|
-
bug, documéntalo en un ticket y déjalo intacto en el characterization
|
|
224
|
-
test. Cambios funcionales van en commits separados con su propia revisión.
|
|
225
|
-
- **Apuntar a 100% de cobertura del módulo entero**: el objetivo es
|
|
226
|
-
cubrir lo que se va a tocar. Cobertura completa puede tomar meses;
|
|
227
|
-
cubrir el slice de cambio toma horas.
|
|
228
|
-
- **Borrar código "muerto" sin verificar**: en legacy, algo que parece
|
|
229
|
-
muerto puede ser invocado por reflection o cron. Buscar referencias
|
|
230
|
-
con `grep -r` antes de borrar.
|
|
231
|
-
- **Sustituir "todo" con un mock**: si el test pasa solo porque mockeaste
|
|
232
|
-
todas las dependencias, ya no testea nada. Mockear solo lo necesario
|
|
233
|
-
para aislar el código bajo prueba.
|
|
234
|
-
|
|
235
|
-
---
|
|
236
|
-
|
|
237
|
-
## Gotchas / Errores comunes no obvios
|
|
238
|
-
|
|
239
|
-
- **Characterization test "perfecto" que se rompe en CI por orden no determinista**: el código legacy depende del orden de iteración de un dict (Python <3.7) o del timezone del servidor. Causa: comportamiento dependiente de entorno. Solución: capturar la dependencia (forzar timezone UTC en el test, ordenar collections antes de assert) y documentar la dependencia ambiental como bug a discutir.
|
|
240
|
-
- **Sprout Method que sí toca el legacy "solo un poquito"**: el desarrollador agrega un `if condicion_nueva` dentro del legacy en lugar de extraer. Causa: parece más simple. Solución: si tocas el legacy, ya no es sprout; o caracterizas primero, o aplicas wrap method que NO modifica el cuerpo original.
|
|
241
|
-
- **Inyectar dependencia con un default que llama al global**: `def __init__(self, db=None): self.db = db or get_global_db()`. Causa: querer mantener compatibilidad con callers viejos. Solución: el default escondido invalida la inyección — los tests acaban tocando el global. Aceptar que romper la firma es parte del rescate; agregar wrappers de compatibilidad fuera de la clase si es necesario.
|
|
242
|
-
- **Cobertura como métrica engañosa**: 80% de cobertura de líneas pero los asserts solo verifican que "no lanza excepción". Causa: tests escritos para subir el número, no para capturar comportamiento. Solución: cada characterization test debe tener al menos un assert sobre el output o side effect concreto, no solo `assert resultado is not None`.
|
|
243
|
-
- **Mock de la BD que devuelve datos perfectos donde el legacy maneja datos sucios**: el código legacy tiene defensas contra datos inconsistentes que el mock no reproduce. Causa: el mock idealiza la realidad. Solución: el mock debe reproducir la sucidad observada en producción (NULLs inesperados, encodings raros, duplicados) — caracterizar la entrada real es parte del rescate.
|
|
244
|
-
|
|
245
|
-
## Orden de prioridad cuando se hereda código
|
|
246
|
-
|
|
247
|
-
1. **Estabilizar**: identificar el módulo más crítico, escribir
|
|
248
|
-
characterization tests sobre los flujos que se van a tocar pronto.
|
|
249
|
-
2. **Aislar**: romper dependencias problemáticas con seams para que el
|
|
250
|
-
módulo crítico pueda testearse en isolation.
|
|
251
|
-
3. **Cubrir**: agregar tests sobre código nuevo (Sprout/Wrap) y sobre
|
|
252
|
-
regresiones detectadas.
|
|
253
|
-
4. **Cambiar**: hacer el cambio funcional pedido, ya con red de seguridad.
|
|
254
|
-
5. **Refactorizar**: mejorar la estructura ya que está cubierta. NO antes.
|
|
255
|
-
|
|
256
|
-
NO refactorices código que no piensas tocar. La cobertura no es un fin
|
|
257
|
-
en sí mismo — es palanca para cambios futuros.
|
|
258
|
-
|
|
259
|
-
## Checklist antes de tocar código heredado
|
|
260
|
-
|
|
261
|
-
- [ ] ¿Existen characterization tests sobre la zona de cambio?
|
|
262
|
-
- [ ] Si no: ¿se escribieron antes de cualquier modificación?
|
|
263
|
-
- [ ] ¿Las dependencias bloqueantes (DB, red, time, random) están aisladas?
|
|
264
|
-
- [ ] ¿El cambio funcional va en un commit separado del refactor estructural?
|
|
265
|
-
- [ ] ¿Se documentaron en tickets los bugs detectados durante la caracterización?
|
|
266
|
-
- [ ] ¿La cobertura nueva es del slice tocado, no del módulo completo?
|
|
267
|
-
- [ ] ¿El sprout/wrap se queda fuera del cuerpo legacy original?
|
|
1
|
+
---
|
|
2
|
+
name: legacy-code-rescue
|
|
3
|
+
description: >
|
|
4
|
+
Técnicas para trabajar con código heredado sin tests, siguiendo la metodología
|
|
5
|
+
de Michael Feathers ("Working Effectively with Legacy Code"). Cubre
|
|
6
|
+
identificación de seams, characterization tests para capturar comportamiento
|
|
7
|
+
real (no el deseado), patrones para introducir cambios sin romper
|
|
8
|
+
(sprout method, sprout class, wrap method, wrap class), técnicas para romper
|
|
9
|
+
dependencias problemáticas, y orden de prioridad cuando se hereda código
|
|
10
|
+
sin cobertura. Cargar cuando se herede código sin tests, cuando un módulo
|
|
11
|
+
crítico tenga cobertura <30%, o cuando se planee refactoring de un sistema
|
|
12
|
+
heredado.
|
|
13
|
+
version: "1.0.0"
|
|
14
|
+
evolved: false
|
|
15
|
+
herramientasPermitidas: [Read]
|
|
16
|
+
exclusiones:
|
|
17
|
+
- "No cargar para refactoring de código moderno con buena cobertura — si los tests existen y cubren el área a refactorear, basta con `tdd-workflow`. Este skill es específico para código sin red de seguridad."
|
|
18
|
+
- "No cargar para migración entre versiones de framework — usar `deprecacion-migracion`. Este skill cubre rescate de código heredado en general, no actualizaciones de stack."
|
|
19
|
+
- "No cargar para deprecación planificada de features — usar `deprecacion-migracion`. Este skill cubre el problema inverso: código que NO se quiere deprecar pero hay que cambiar con seguridad."
|
|
20
|
+
- "No cargar para optimización de rendimiento de código moderno — los patrones aquí (especialmente characterization tests) priorizan seguridad sobre velocidad. Si la prioridad es perf, usar `performance-baseline` y profilers."
|
|
21
|
+
evolvable: true
|
|
22
|
+
---
|
|
23
|
+
|
|
24
|
+
# Legacy Code Rescue
|
|
25
|
+
|
|
26
|
+
Adaptado de "Working Effectively with Legacy Code" (Michael Feathers, 2004).
|
|
27
|
+
Definición operativa de Feathers: **legacy code es código sin tests**, sin
|
|
28
|
+
importar su edad o calidad aparente.
|
|
29
|
+
|
|
30
|
+
## Cuándo cargar
|
|
31
|
+
|
|
32
|
+
- Se hereda un módulo crítico con cobertura <30% y hay que cambiarlo.
|
|
33
|
+
- Se diagnostica un bug en código que nadie ha tocado en meses/años.
|
|
34
|
+
- Se planea un refactor de un sistema sin red de seguridad de tests.
|
|
35
|
+
- Se necesita agregar una feature en código que actualmente no se puede
|
|
36
|
+
testear de forma aislada.
|
|
37
|
+
- Se reciben quejas tipo "tengo miedo de tocar X porque puede romper Y".
|
|
38
|
+
|
|
39
|
+
## Cuándo NO cargar
|
|
40
|
+
|
|
41
|
+
- El código tiene cobertura ≥80% y los tests pasan — usar `tdd-workflow`.
|
|
42
|
+
- Se está migrando entre versiones de framework — usar `deprecacion-migracion`.
|
|
43
|
+
- Se está deprecando una feature intencionalmente — usar `deprecacion-migracion`.
|
|
44
|
+
- El objetivo es optimizar rendimiento — los patrones aquí priorizan seguridad
|
|
45
|
+
sobre velocidad; usar `performance-baseline` y profilers.
|
|
46
|
+
|
|
47
|
+
## Directiva primaria
|
|
48
|
+
|
|
49
|
+
**Antes de cambiar código heredado, capturar lo que hace**, no lo que se
|
|
50
|
+
supone que debe hacer. El comportamiento actual (incluidos sus bugs) es
|
|
51
|
+
la fuente de verdad — algún caller probablemente depende de cada peculiaridad.
|
|
52
|
+
|
|
53
|
+
Orden estricto:
|
|
54
|
+
1. **Caracterizar** el comportamiento actual con tests.
|
|
55
|
+
2. **Romper dependencias** lo mínimo necesario para que los tests sean posibles.
|
|
56
|
+
3. **Cambiar** el código.
|
|
57
|
+
4. **Refactorizar** ya con red de seguridad.
|
|
58
|
+
|
|
59
|
+
NO empezar refactorizando "porque el código está feo". Sin tests, cada cambio
|
|
60
|
+
es una apuesta.
|
|
61
|
+
|
|
62
|
+
---
|
|
63
|
+
|
|
64
|
+
## Identificar seams (costuras)
|
|
65
|
+
|
|
66
|
+
Un **seam** es un punto donde el comportamiento del programa puede cambiarse
|
|
67
|
+
sin editar el código original. Los seams son la palanca para introducir
|
|
68
|
+
tests sin romper nada.
|
|
69
|
+
|
|
70
|
+
### Tipos de seams
|
|
71
|
+
|
|
72
|
+
| Tipo | Cuándo aplica | Cómo se activa |
|
|
73
|
+
|------|---------------|----------------|
|
|
74
|
+
| **Object seam** | OOP con polimorfismo | Inyectar una subclase/mock que herede |
|
|
75
|
+
| **Preprocessing seam** | C/C++ con `#define` | Macros o preprocesador |
|
|
76
|
+
| **Link seam** | Lenguajes compilados con linker | Sustituir librería en build |
|
|
77
|
+
| **Method seam** (Python/JS) | Lenguajes dinámicos | Monkeypatching, importlib |
|
|
78
|
+
|
|
79
|
+
En Python/TypeScript modernos, el object seam es el predominante:
|
|
80
|
+
inyectar dependencias en lugar de instanciarlas dentro del constructor.
|
|
81
|
+
|
|
82
|
+
### Identificar seams en código existente
|
|
83
|
+
|
|
84
|
+
Buscar:
|
|
85
|
+
- `new SomeClass()` dentro de constructores → object seam latente al refactorizar.
|
|
86
|
+
- Llamadas estáticas a clases globales → buscar wrappers o usar dependency injection.
|
|
87
|
+
- Imports de módulos con efectos secundarios → candidato a method seam.
|
|
88
|
+
- Funciones puras escondidas dentro de métodos largos → extraer es seam barato.
|
|
89
|
+
|
|
90
|
+
---
|
|
91
|
+
|
|
92
|
+
## Characterization tests — capturar el comportamiento real
|
|
93
|
+
|
|
94
|
+
Un **characterization test** documenta lo que el código HACE, no lo que
|
|
95
|
+
debería hacer. Son la red de seguridad mínima antes de cualquier cambio.
|
|
96
|
+
|
|
97
|
+
### Receta
|
|
98
|
+
|
|
99
|
+
1. Tomar el código y ejecutarlo con un input concreto.
|
|
100
|
+
2. Capturar el output exacto (incluido errores, side effects, state final).
|
|
101
|
+
3. Escribir un test que assert ese output específico.
|
|
102
|
+
4. Repetir con varios inputs representativos: caso típico, edge cases
|
|
103
|
+
conocidos, inputs malformados, valores extremos.
|
|
104
|
+
5. Cuando el test pasa, **es la verdad** — incluso si revela un bug.
|
|
105
|
+
|
|
106
|
+
```python
|
|
107
|
+
# Ejemplo: función legacy sin tests
|
|
108
|
+
def calcular_descuento(precio, tipo_cliente):
|
|
109
|
+
# ... 80 líneas de if/else, redondeos extraños, edge cases ocultos
|
|
110
|
+
return total
|
|
111
|
+
|
|
112
|
+
# Characterization test — captura el comportamiento ACTUAL
|
|
113
|
+
def test_caracterizacion_descuento():
|
|
114
|
+
# Input típico
|
|
115
|
+
assert calcular_descuento(100, "VIP") == 85.0
|
|
116
|
+
# Edge case observado: tipo_cliente vacío
|
|
117
|
+
assert calcular_descuento(100, "") == 100.0
|
|
118
|
+
# Bug aparente, pero callers pueden depender de él
|
|
119
|
+
assert calcular_descuento(0, "VIP") == 0.0
|
|
120
|
+
# Negativos: el código devuelve negativos sin validar — documentarlo
|
|
121
|
+
assert calcular_descuento(-50, "VIP") == -42.5
|
|
122
|
+
```
|
|
123
|
+
|
|
124
|
+
### Reglas para characterization tests
|
|
125
|
+
|
|
126
|
+
- **No "arreglar" bugs** mientras se caracteriza — eso viene después.
|
|
127
|
+
- Si el comportamiento parece incorrecto, escribir un test que lo
|
|
128
|
+
documente Y abrir un ticket separado para discutir el cambio.
|
|
129
|
+
- Apuntar a cubrir los flujos que se van a tocar — no toda la base de
|
|
130
|
+
código.
|
|
131
|
+
- Tests rápidos > tests perfectos. Un test feo que corre es mejor que
|
|
132
|
+
uno elegante que no se escribió.
|
|
133
|
+
|
|
134
|
+
---
|
|
135
|
+
|
|
136
|
+
## Patrones para cambio seguro
|
|
137
|
+
|
|
138
|
+
Cuando hay que agregar comportamiento sin tocar código existente.
|
|
139
|
+
|
|
140
|
+
### Sprout Method
|
|
141
|
+
|
|
142
|
+
Si necesitas lógica nueva dentro de un método legacy difícil de testear,
|
|
143
|
+
**escribe la lógica nueva como método aparte** (con tests propios) y
|
|
144
|
+
llámalo desde el método legacy.
|
|
145
|
+
|
|
146
|
+
```python
|
|
147
|
+
# Antes — quieres agregar validación al método legacy
|
|
148
|
+
def procesar_orden(orden):
|
|
149
|
+
# 200 líneas legacy intocables
|
|
150
|
+
...
|
|
151
|
+
|
|
152
|
+
# Después — la nueva validación vive aparte y SÍ es testeable
|
|
153
|
+
def procesar_orden(orden):
|
|
154
|
+
# 200 líneas legacy intocables
|
|
155
|
+
if not _validar_disponibilidad_inventario(orden): # nuevo
|
|
156
|
+
return ResultadoOrden.fallo("sin stock")
|
|
157
|
+
...
|
|
158
|
+
|
|
159
|
+
def _validar_disponibilidad_inventario(orden):
|
|
160
|
+
"""Nuevo método con tests propios (no toca el legacy)."""
|
|
161
|
+
return all(item.cantidad <= item.producto.stock for item in orden.items)
|
|
162
|
+
```
|
|
163
|
+
|
|
164
|
+
### Sprout Class
|
|
165
|
+
|
|
166
|
+
Cuando la lógica nueva es suficientemente grande, va en una **clase nueva**
|
|
167
|
+
en lugar de un método. Misma idea: lógica nueva con tests, llamada desde
|
|
168
|
+
el legacy.
|
|
169
|
+
|
|
170
|
+
### Wrap Method
|
|
171
|
+
|
|
172
|
+
Si necesitas ejecutar algo ANTES o DESPUÉS de un método legacy, renombra
|
|
173
|
+
el original a algo interno (`_procesar_orden_original`) y crea uno nuevo
|
|
174
|
+
con la firma original que llame al wrap + el original.
|
|
175
|
+
|
|
176
|
+
```python
|
|
177
|
+
def procesar_orden(orden): # firma pública intacta
|
|
178
|
+
_registrar_intento(orden) # nueva pre-acción, testeable
|
|
179
|
+
resultado = _procesar_orden_original(orden)
|
|
180
|
+
_registrar_resultado(resultado) # nueva post-acción, testeable
|
|
181
|
+
return resultado
|
|
182
|
+
|
|
183
|
+
def _procesar_orden_original(orden):
|
|
184
|
+
# 200 líneas legacy intocables, ahora son privadas
|
|
185
|
+
...
|
|
186
|
+
```
|
|
187
|
+
|
|
188
|
+
### Wrap Class
|
|
189
|
+
|
|
190
|
+
Para wrap de un sistema más grande: una clase nueva con la misma interfaz
|
|
191
|
+
delega al legacy y agrega comportamiento alrededor (decorator pattern).
|
|
192
|
+
|
|
193
|
+
---
|
|
194
|
+
|
|
195
|
+
## Romper dependencias
|
|
196
|
+
|
|
197
|
+
Algunos legacy methods son intratables por dependencias:
|
|
198
|
+
- Conexiones a BD/red en el constructor.
|
|
199
|
+
- Singletons globales que se acceden en cualquier parte.
|
|
200
|
+
- Llamadas a `time.now()`, `random()`, filesystem.
|
|
201
|
+
|
|
202
|
+
Técnicas (en orden de invasividad creciente):
|
|
203
|
+
|
|
204
|
+
1. **Subclass and Override Method**: hereda de la clase legacy, sobreescribe
|
|
205
|
+
el método problemático con un mock para los tests.
|
|
206
|
+
2. **Extract Interface / Extract Class**: extrae una interfaz pequeña con
|
|
207
|
+
solo los métodos que el código usa, e inyecta una implementación de prueba.
|
|
208
|
+
3. **Parameterize Method/Constructor**: inyecta como parámetro lo que antes
|
|
209
|
+
se obtenía globalmente (clock, random, IO).
|
|
210
|
+
4. **Replace Function with Function Pointer / Strategy**: cambiar funciones
|
|
211
|
+
estáticas por callables inyectables.
|
|
212
|
+
|
|
213
|
+
Aplicar la técnica MENOS invasiva que resuelva el problema. Cuanto más
|
|
214
|
+
invasiva, más riesgo de romper algo durante el rescate.
|
|
215
|
+
|
|
216
|
+
---
|
|
217
|
+
|
|
218
|
+
## Anti-patrones
|
|
219
|
+
|
|
220
|
+
- **Refactorizar antes de tener characterization tests**: cualquier cambio
|
|
221
|
+
estructural sin red de seguridad es una apuesta.
|
|
222
|
+
- **"Mejorar" el comportamiento mientras se caracteriza**: si parece un
|
|
223
|
+
bug, documéntalo en un ticket y déjalo intacto en el characterization
|
|
224
|
+
test. Cambios funcionales van en commits separados con su propia revisión.
|
|
225
|
+
- **Apuntar a 100% de cobertura del módulo entero**: el objetivo es
|
|
226
|
+
cubrir lo que se va a tocar. Cobertura completa puede tomar meses;
|
|
227
|
+
cubrir el slice de cambio toma horas.
|
|
228
|
+
- **Borrar código "muerto" sin verificar**: en legacy, algo que parece
|
|
229
|
+
muerto puede ser invocado por reflection o cron. Buscar referencias
|
|
230
|
+
con `grep -r` antes de borrar.
|
|
231
|
+
- **Sustituir "todo" con un mock**: si el test pasa solo porque mockeaste
|
|
232
|
+
todas las dependencias, ya no testea nada. Mockear solo lo necesario
|
|
233
|
+
para aislar el código bajo prueba.
|
|
234
|
+
|
|
235
|
+
---
|
|
236
|
+
|
|
237
|
+
## Gotchas / Errores comunes no obvios
|
|
238
|
+
|
|
239
|
+
- **Characterization test "perfecto" que se rompe en CI por orden no determinista**: el código legacy depende del orden de iteración de un dict (Python <3.7) o del timezone del servidor. Causa: comportamiento dependiente de entorno. Solución: capturar la dependencia (forzar timezone UTC en el test, ordenar collections antes de assert) y documentar la dependencia ambiental como bug a discutir.
|
|
240
|
+
- **Sprout Method que sí toca el legacy "solo un poquito"**: el desarrollador agrega un `if condicion_nueva` dentro del legacy en lugar de extraer. Causa: parece más simple. Solución: si tocas el legacy, ya no es sprout; o caracterizas primero, o aplicas wrap method que NO modifica el cuerpo original.
|
|
241
|
+
- **Inyectar dependencia con un default que llama al global**: `def __init__(self, db=None): self.db = db or get_global_db()`. Causa: querer mantener compatibilidad con callers viejos. Solución: el default escondido invalida la inyección — los tests acaban tocando el global. Aceptar que romper la firma es parte del rescate; agregar wrappers de compatibilidad fuera de la clase si es necesario.
|
|
242
|
+
- **Cobertura como métrica engañosa**: 80% de cobertura de líneas pero los asserts solo verifican que "no lanza excepción". Causa: tests escritos para subir el número, no para capturar comportamiento. Solución: cada characterization test debe tener al menos un assert sobre el output o side effect concreto, no solo `assert resultado is not None`.
|
|
243
|
+
- **Mock de la BD que devuelve datos perfectos donde el legacy maneja datos sucios**: el código legacy tiene defensas contra datos inconsistentes que el mock no reproduce. Causa: el mock idealiza la realidad. Solución: el mock debe reproducir la sucidad observada en producción (NULLs inesperados, encodings raros, duplicados) — caracterizar la entrada real es parte del rescate.
|
|
244
|
+
|
|
245
|
+
## Orden de prioridad cuando se hereda código
|
|
246
|
+
|
|
247
|
+
1. **Estabilizar**: identificar el módulo más crítico, escribir
|
|
248
|
+
characterization tests sobre los flujos que se van a tocar pronto.
|
|
249
|
+
2. **Aislar**: romper dependencias problemáticas con seams para que el
|
|
250
|
+
módulo crítico pueda testearse en isolation.
|
|
251
|
+
3. **Cubrir**: agregar tests sobre código nuevo (Sprout/Wrap) y sobre
|
|
252
|
+
regresiones detectadas.
|
|
253
|
+
4. **Cambiar**: hacer el cambio funcional pedido, ya con red de seguridad.
|
|
254
|
+
5. **Refactorizar**: mejorar la estructura ya que está cubierta. NO antes.
|
|
255
|
+
|
|
256
|
+
NO refactorices código que no piensas tocar. La cobertura no es un fin
|
|
257
|
+
en sí mismo — es palanca para cambios futuros.
|
|
258
|
+
|
|
259
|
+
## Checklist antes de tocar código heredado
|
|
260
|
+
|
|
261
|
+
- [ ] ¿Existen characterization tests sobre la zona de cambio?
|
|
262
|
+
- [ ] Si no: ¿se escribieron antes de cualquier modificación?
|
|
263
|
+
- [ ] ¿Las dependencias bloqueantes (DB, red, time, random) están aisladas?
|
|
264
|
+
- [ ] ¿El cambio funcional va en un commit separado del refactor estructural?
|
|
265
|
+
- [ ] ¿Se documentaron en tickets los bugs detectados durante la caracterización?
|
|
266
|
+
- [ ] ¿La cobertura nueva es del slice tocado, no del módulo completo?
|
|
267
|
+
- [ ] ¿El sprout/wrap se queda fuera del cuerpo legacy original?
|
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
{
|
|
2
|
-
"SKILL.md": {
|
|
3
|
-
"evolved": true,
|
|
4
|
-
"evolvedFrom": "1.1.0",
|
|
5
|
-
"evolvedAt": "2026-05-02",
|
|
6
|
-
"evolvedBy": "aprender",
|
|
7
|
-
"evolvedNote": "Gotcha nueva: try/catch require para módulos opcionales (15 archivos migrados)"
|
|
8
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"SKILL.md": {
|
|
3
|
+
"evolved": true,
|
|
4
|
+
"evolvedFrom": "1.1.0",
|
|
5
|
+
"evolvedAt": "2026-05-02",
|
|
6
|
+
"evolvedBy": "aprender",
|
|
7
|
+
"evolvedNote": "Gotcha nueva: try/catch require para módulos opcionales (15 archivos migrados)"
|
|
8
|
+
}
|
|
9
9
|
}
|