@saulwade/swl-ses 1.3.7 → 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (129) hide show
  1. package/CLAUDE.md +12 -4
  2. package/README.md +1 -1
  3. package/bin/swl-mcp-server.js +187 -187
  4. package/bin/swl-webhook-server.js +198 -0
  5. package/comandos/swl/.evolved.json +22 -22
  6. package/comandos/swl/adoptar-proyecto.md +21 -1
  7. package/comandos/swl/claudemd.md +14 -1
  8. package/comandos/swl/contribuir.md +233 -233
  9. package/comandos/swl/exportar-vault.md +207 -7
  10. package/comandos/swl/nuevo-proyecto.md +24 -2
  11. package/gateway/adapters/base.js +109 -0
  12. package/gateway/adapters/discord.js +167 -0
  13. package/gateway/adapters/email.js +221 -0
  14. package/gateway/adapters/slack.js +192 -0
  15. package/gateway/adapters/telegram.js +183 -0
  16. package/gateway/adapters/webhook.js +113 -0
  17. package/gateway/adapters/whatsapp.js +214 -0
  18. package/gateway/agent-executor.js +322 -0
  19. package/gateway/command-relay.js +271 -0
  20. package/gateway/cron/jobs.js +263 -0
  21. package/gateway/cron/scheduler.js +322 -0
  22. package/gateway/cron/store.js +335 -0
  23. package/gateway/index.js +320 -0
  24. package/gateway/lib/event-channel.js +191 -0
  25. package/gateway/session.js +131 -0
  26. package/gateway/webhook-server.js +324 -0
  27. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  28. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  29. package/habilidades/build-errors-nextjs/SKILL.md +55 -1
  30. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  31. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  32. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  33. package/habilidades/eval-framework/SKILL.md +212 -212
  34. package/habilidades/extractor-de-aprendizajes/SKILL.md +24 -10
  35. package/habilidades/harness-claude-code/SKILL.md +299 -299
  36. package/habilidades/infra-github-actions/SKILL.md +166 -166
  37. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  38. package/habilidades/manejo-errores/.evolved.json +8 -8
  39. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  40. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  41. package/habilidades/nextjs-testing/SKILL.md +89 -5
  42. package/habilidades/node-experto/SKILL.md +37 -1
  43. package/habilidades/patrones-python/SKILL.md +229 -229
  44. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  45. package/habilidades/planear-fase/SKILL.md +319 -319
  46. package/habilidades/react-experto/SKILL.md +45 -4
  47. package/habilidades/release-semver/.evolved.json +8 -8
  48. package/habilidades/swl-claudemd/SKILL.md +15 -1
  49. package/habilidades/tdd-workflow/SKILL.md +36 -4
  50. package/habilidades/testing-python/SKILL.md +340 -340
  51. package/hooks/claudemd-bloat-detector.js +161 -161
  52. package/hooks/inyeccion-contexto.js +8 -3
  53. package/hooks/lib/agent-routing.js +107 -107
  54. package/hooks/lib/auto-consolidator.js +335 -335
  55. package/hooks/lib/error-classifier.js +308 -308
  56. package/hooks/lib/merkle-audit.js +96 -96
  57. package/hooks/lib/provenance-tracker.js +191 -191
  58. package/hooks/lib/rate-limit-ip.js +177 -0
  59. package/hooks/lib/rate-limit-tracker.js +253 -253
  60. package/hooks/lib/resource-quota.js +122 -122
  61. package/hooks/lib/retry-jitter.js +165 -165
  62. package/hooks/lib/skill-auditor.js +588 -588
  63. package/hooks/lib/sync-status.js +228 -228
  64. package/hooks/lib/taint-tracker.js +107 -107
  65. package/hooks/lib/text-similarity.js +241 -241
  66. package/hooks/lib/toon-compressor.js +245 -245
  67. package/hooks/lib/webhook-dedup.js +184 -0
  68. package/hooks/lib/webhook-verify.js +123 -0
  69. package/hooks/proteccion-rutas.js +120 -15
  70. package/hooks/registro-turnos.js +209 -209
  71. package/hooks/sugerir-regenerar-inventario.js +170 -170
  72. package/hooks/validar-formato-post-subagente.js +140 -140
  73. package/hooks/validar-memoria-hook.js +218 -218
  74. package/instintos/prompt-appendices.yaml +57 -57
  75. package/manifiestos/agent-output-schemas.json +57 -57
  76. package/manifiestos/modulos.json +1 -0
  77. package/manifiestos/skills-lock.json +37 -37
  78. package/package.json +5 -3
  79. package/plantillas/auditor-veto-template.md +105 -105
  80. package/plantillas/github-workflows/README.md +47 -47
  81. package/plantillas/github-workflows/release-please.yml +44 -44
  82. package/plantillas/github-workflows/swl-ci.yml +107 -107
  83. package/plantillas/github-workflows/swl-security.yml +51 -51
  84. package/plugin.json +1 -1
  85. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  86. package/reglas/arreglar-al-detectar.md +147 -147
  87. package/reglas/fragmentos-compartidos.md +152 -152
  88. package/reglas/harness-claude-code.md +213 -213
  89. package/reglas/usar-context7.md +226 -226
  90. package/reglas/usar-sistema-swl.md +251 -0
  91. package/schemas/diary-entry.schema.json +80 -80
  92. package/scripts/benchmark-memoria.js +167 -167
  93. package/scripts/comandos/skills.js +251 -2
  94. package/scripts/configurar-branch-protection.js +418 -418
  95. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  96. package/scripts/field-report.js +199 -199
  97. package/scripts/generar-checklists-consolidados.js +273 -273
  98. package/scripts/generar-inventario.js +420 -420
  99. package/scripts/generar-matriz-lenguajes.js +271 -271
  100. package/scripts/lib/artefactos-python.js +43 -43
  101. package/scripts/lib/benchmark-metrics.js +160 -160
  102. package/scripts/lib/budget-enforcer.js +252 -252
  103. package/scripts/lib/configurar-ci.js +380 -380
  104. package/scripts/lib/contadores-inventario.js +217 -217
  105. package/scripts/lib/detectar-stack-detallado.js +307 -307
  106. package/scripts/lib/diary-entry.js +234 -234
  107. package/scripts/lib/eval-metrics-store.js +218 -218
  108. package/scripts/lib/eval-quality.js +171 -171
  109. package/scripts/lib/eval-schemas.js +144 -144
  110. package/scripts/lib/eval-self-correct.js +106 -106
  111. package/scripts/lib/eval-validator.js +185 -185
  112. package/scripts/lib/jaccard-similarity.js +98 -98
  113. package/scripts/lib/longmemeval-runner.js +125 -125
  114. package/scripts/lib/npm-version.js +261 -261
  115. package/scripts/lib/paquetes-conocidos.js +50 -50
  116. package/scripts/lib/prompt-builder.js +264 -264
  117. package/scripts/lib/rrf-fusion.js +175 -175
  118. package/scripts/lib/scoring-instintos.js +277 -277
  119. package/scripts/lib/semantic-search.js +252 -252
  120. package/scripts/limpiar-artefactos-python.js +131 -131
  121. package/scripts/mcp-server/README.md +128 -128
  122. package/scripts/mcp-server/handlers.js +206 -206
  123. package/scripts/migrar-csv-a-array.js +168 -168
  124. package/scripts/migrar-fase-dominio.js +201 -201
  125. package/scripts/publicar.js +511 -511
  126. package/scripts/run-eval.js +141 -141
  127. package/scripts/validar-manifest.js +195 -195
  128. package/scripts/validar-userland-vacio.js +110 -110
  129. package/scripts/verificar-release.js +110 -0
@@ -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
  }