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