@saulwade/swl-ses 1.6.7 → 1.7.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.
@@ -0,0 +1,580 @@
1
+ # Ejemplos MAL→BIEN — prevención de sobre-ingeniería
2
+
3
+ Catálogo de 9 ejemplos pareados (código que falla / código correcto) para los
4
+ 4 principios del skill. Cada uno se mapea a un anti-patrón ya documentado en
5
+ `SKILL.md` y a un escenario operativo de SWL.
6
+
7
+ Adaptado de
8
+ [andrej-karpathy-skills](https://github.com/forrestchang/andrej-karpathy-skills)
9
+ (MIT) al ecosistema SWL — español de México, escenarios concretos del sistema.
10
+
11
+ **Cómo usar este archivo**: cuando un agente SWL (`implementador-swl`,
12
+ `revisor-codigo-swl`, agentes de stack) detecta el anti-patrón en su sesión,
13
+ puede citar `recursos/EXAMPLES.md § <ejemplo>` como evidencia observable en
14
+ lugar de invocar la regla abstracta.
15
+
16
+ ---
17
+
18
+ ## Principio 1 — Pensar antes de codificar
19
+
20
+ ### Ejemplo 1.1 — Supuestos ocultos sobre alcance
21
+
22
+ **Solicitud del usuario**: *"Agrega una feature para exportar datos de usuarios."*
23
+
24
+ #### ❌ MAL — asume sin preguntar
25
+
26
+ ```python
27
+ def exportar_usuarios(formato='json'):
28
+ """Exporta todos los usuarios a un archivo."""
29
+ usuarios = Usuario.query.all()
30
+
31
+ if formato == 'json':
32
+ with open('usuarios.json', 'w') as f:
33
+ json.dump([u.to_dict() for u in usuarios], f)
34
+ elif formato == 'csv':
35
+ with open('usuarios.csv', 'w') as f:
36
+ writer = csv.DictWriter(f, fieldnames=['id', 'email', 'nombre'])
37
+ writer.writeheader()
38
+ writer.writerows([u.to_dict() for u in usuarios])
39
+
40
+ return f"Exportados {len(usuarios)} usuarios"
41
+ ```
42
+
43
+ **Problemas**:
44
+ - Asume que se deben exportar TODOS los usuarios (¿paginación? ¿privacidad?)
45
+ - Asume ubicación del archivo sin preguntar
46
+ - Asume campos a incluir (algunos podrían ser sensibles)
47
+ - Hardcodea fieldnames del CSV sin revisar la estructura real
48
+
49
+ #### ✅ BIEN — explicita supuestos antes de implementar
50
+
51
+ ```
52
+ Antes de implementar, necesito clarificar:
53
+
54
+ 1. **Alcance**: ¿exportar todos los usuarios o un subconjunto filtrado?
55
+ (implicaciones de privacidad)
56
+ 2. **Formato**: cuando dices "exportar", ¿te refieres a:
57
+ - Descarga de archivo en el navegador?
58
+ - Job en background con notificación por email?
59
+ - Endpoint de API que devuelve los datos?
60
+ 3. **Campos**: ¿qué campos del usuario? (algunos pueden ser sensibles —
61
+ datos personales, hash de contraseña, tokens)
62
+ 4. **Volumen**: ¿cuántos usuarios típicamente? (afecta el enfoque)
63
+
64
+ Enfoque más simple: agregar endpoint de API que devuelve JSON paginado.
65
+ Para exportación a archivo necesito más detalle.
66
+
67
+ ¿Cuál prefieres?
68
+ ```
69
+
70
+ **Aplicación SWL**: este patrón está documentado en el `Principio 1` del SKILL.md y
71
+ es exactamente lo que `/swl:discutir-fase` resuelve antes de que llegue a
72
+ `implementador-swl`.
73
+
74
+ ---
75
+
76
+ ### Ejemplo 1.2 — Múltiples interpretaciones, elegir en silencio
77
+
78
+ **Solicitud del usuario**: *"Haz la búsqueda más rápida."*
79
+
80
+ #### ❌ MAL — elige una interpretación y construye 200 LOC
81
+
82
+ ```python
83
+ @lru_cache(maxsize=1000)
84
+ async def buscar(query: str) -> list[Resultado]:
85
+ # ... 200 líneas: cache, índices, async, connection pooling ...
86
+ ```
87
+
88
+ #### ✅ BIEN — presenta interpretaciones con costo estimado
89
+
90
+ ```
91
+ "Más rápida" puede significar tres cosas distintas:
92
+
93
+ 1. **Latencia menor por request** (de 500ms a <100ms)
94
+ - Agregar índices de BD
95
+ - Cachear queries frecuentes
96
+ - Esfuerzo estimado: 2h
97
+
98
+ 2. **Más throughput concurrente**
99
+ - async/await + connection pooling
100
+ - Esfuerzo estimado: 4h
101
+
102
+ 3. **Velocidad percibida mejor** (UX)
103
+ - Mostrar resultados parciales antes
104
+ - Carga progresiva
105
+ - Esfuerzo estimado: 3h
106
+
107
+ La búsqueda actual tarda ~500ms en queries típicas.
108
+ ¿Cuál importa más para tu caso de uso?
109
+ ```
110
+
111
+ **Aplicación SWL**: cuando el usuario invoca `/swl:planear-fase` con instrucción
112
+ vaga, el `planificador-swl` debe responder así antes de generar PLAN.md.
113
+
114
+ ---
115
+
116
+ ## Principio 2 — Simplicidad primero
117
+
118
+ ### Ejemplo 2.1 — Abstracción prematura (Strategy pattern para un caso de uso)
119
+
120
+ **Solicitud del usuario**: *"Agrega una función para calcular descuento."*
121
+
122
+ #### ❌ MAL — Strategy + Factory + Config para una multiplicación
123
+
124
+ ```python
125
+ from abc import ABC, abstractmethod
126
+ from dataclasses import dataclass
127
+
128
+ class EstrategiaDescuento(ABC):
129
+ @abstractmethod
130
+ def calcular(self, monto: float) -> float: ...
131
+
132
+ class DescuentoPorcentual(EstrategiaDescuento):
133
+ def __init__(self, porcentaje: float):
134
+ self.porcentaje = porcentaje
135
+ def calcular(self, monto: float) -> float:
136
+ return monto * (self.porcentaje / 100)
137
+
138
+ class DescuentoFijo(EstrategiaDescuento):
139
+ def __init__(self, monto_fijo: float):
140
+ self.monto_fijo = monto_fijo
141
+ def calcular(self, monto: float) -> float:
142
+ return min(self.monto_fijo, monto)
143
+
144
+ @dataclass
145
+ class ConfigDescuento:
146
+ estrategia: EstrategiaDescuento
147
+ compra_minima: float = 0.0
148
+ descuento_maximo: float = float('inf')
149
+
150
+ class CalculadoraDescuento:
151
+ def __init__(self, config: ConfigDescuento):
152
+ self.config = config
153
+ def aplicar(self, monto: float) -> float:
154
+ if monto < self.config.compra_minima:
155
+ return 0
156
+ descuento = self.config.estrategia.calcular(monto)
157
+ return min(descuento, self.config.descuento_maximo)
158
+
159
+ # 30+ líneas de setup para una multiplicación
160
+ ```
161
+
162
+ #### ✅ BIEN — función directa
163
+
164
+ ```python
165
+ def calcular_descuento(monto: float, porcentaje: float) -> float:
166
+ """Calcula el descuento. porcentaje en rango 0-100."""
167
+ return monto * (porcentaje / 100)
168
+
169
+ # Uso
170
+ descuento = calcular_descuento(100.0, 10.0) # 10.0
171
+ ```
172
+
173
+ **Cuándo agregar complejidad**: solo cuando aparezca el segundo o tercer tipo
174
+ de descuento. Si la necesidad llega después, refactorizar entonces.
175
+
176
+ **Aplicación SWL**: regla SKILL.md "Tres usos concretos justifican una
177
+ abstracción; menos de tres, no". Este ejemplo es la materialización del umbral.
178
+
179
+ ---
180
+
181
+ ### Ejemplo 2.2 — Features especulativas no solicitadas
182
+
183
+ **Solicitud del usuario**: *"Guarda las preferencias del usuario en la base de datos."*
184
+
185
+ #### ❌ MAL — cache, validator, merge, notificación
186
+
187
+ ```python
188
+ class GestorPreferencias:
189
+ def __init__(self, db, cache=None, validator=None):
190
+ self.db = db
191
+ self.cache = cache or CacheEnMemoria()
192
+ self.validator = validator or ValidadorDefault()
193
+
194
+ def guardar(self, usuario_id: int, prefs: dict,
195
+ merge: bool = True,
196
+ validar: bool = True,
197
+ notificar: bool = False) -> bool:
198
+ if validar:
199
+ errores = self.validator.validar(prefs)
200
+ if errores:
201
+ raise ValidationError(errores)
202
+
203
+ existentes = self.obtener(usuario_id) if merge else {}
204
+ if merge:
205
+ existentes.update(prefs)
206
+ prefs = existentes
207
+
208
+ self.db.execute(
209
+ "UPDATE usuarios SET preferencias = ? WHERE id = ?",
210
+ (json.dumps(prefs), usuario_id)
211
+ )
212
+
213
+ if self.cache:
214
+ self.cache.set(f"prefs:{usuario_id}", prefs)
215
+
216
+ if notificar:
217
+ self.notificar_cambio(usuario_id, prefs)
218
+
219
+ return True
220
+ ```
221
+
222
+ #### ✅ BIEN — solo lo que pidieron
223
+
224
+ ```python
225
+ def guardar_preferencias(db, usuario_id: int, preferencias: dict):
226
+ """Guarda las preferencias del usuario en la base de datos."""
227
+ db.execute(
228
+ "UPDATE usuarios SET preferencias = ? WHERE id = ?",
229
+ (json.dumps(preferencias), usuario_id)
230
+ )
231
+ ```
232
+
233
+ **Agregar después si aparece la necesidad**:
234
+ - Caching → cuando aparezca problema de performance medible
235
+ - Validación → cuando lleguen datos malformados a la BD
236
+ - Merging → cuando emerja el requisito explícito
237
+
238
+ **Aplicación SWL**: este es exactamente el escenario donde `revisor-codigo-swl`
239
+ debe emitir veto item DRY/sobre-ingeniería contra commits que agregan
240
+ parámetros opcionales sin requisito documentado.
241
+
242
+ ---
243
+
244
+ ## Principio 3 — Cambios quirúrgicos
245
+
246
+ ### Ejemplo 3.1 — Drive-by refactoring (refactor de oportunidad)
247
+
248
+ **Solicitud del usuario**: *"Corrige el bug donde los emails vacíos rompen el validador."*
249
+
250
+ #### ❌ MAL — corrige el bug Y "mejora" todo lo demás
251
+
252
+ ```diff
253
+ def validar_usuario(datos_usuario):
254
+ - # Verificar formato de email
255
+ - if not datos_usuario.get('email'):
256
+ + """Valida los datos del usuario."""
257
+ + email = datos_usuario.get('email', '').strip()
258
+ +
259
+ + # Validar email
260
+ + if not email:
261
+ raise ValueError("Email requerido")
262
+ -
263
+ - # Validación básica de email
264
+ - if '@' not in datos_usuario['email']:
265
+ + if '@' not in email or '.' not in email.split('@')[1]:
266
+ raise ValueError("Email inválido")
267
+
268
+ - # Verificar nombre de usuario
269
+ - if not datos_usuario.get('nombre_usuario'):
270
+ + # Validar nombre de usuario
271
+ + nombre_usuario = datos_usuario.get('nombre_usuario', '').strip()
272
+ + if not nombre_usuario:
273
+ raise ValueError("Nombre de usuario requerido")
274
+ + if len(nombre_usuario) < 3:
275
+ + raise ValueError("Nombre de usuario muy corto")
276
+ + if not nombre_usuario.isalnum():
277
+ + raise ValueError("Nombre de usuario debe ser alfanumérico")
278
+
279
+ return True
280
+ ```
281
+
282
+ **Problemas**:
283
+ - "Mejoró" la validación de email más allá del fix
284
+ - Agregó validación de username que nadie pidió
285
+ - Cambió comentarios
286
+ - Agregó docstring
287
+
288
+ #### ✅ BIEN — solo el fix necesario
289
+
290
+ ```diff
291
+ def validar_usuario(datos_usuario):
292
+ # Verificar formato de email
293
+ - if not datos_usuario.get('email'):
294
+ + email = datos_usuario.get('email', '')
295
+ + if not email or not email.strip():
296
+ raise ValueError("Email requerido")
297
+
298
+ # Validación básica de email
299
+ - if '@' not in datos_usuario['email']:
300
+ + if '@' not in email:
301
+ raise ValueError("Email inválido")
302
+
303
+ # Verificar nombre de usuario
304
+ if not datos_usuario.get('nombre_usuario'):
305
+ raise ValueError("Nombre de usuario requerido")
306
+
307
+ return True
308
+ ```
309
+
310
+ **Solo cambió**: las líneas específicas que corrigen el manejo del email vacío.
311
+
312
+ **Aplicación SWL**: regla del SKILL.md "Cada línea cambiada debe trazar
313
+ directamente al requisito del usuario". `revisor-codigo-swl` revisa cada línea
314
+ del diff contra esta regla.
315
+
316
+ ---
317
+
318
+ ### Ejemplo 3.2 — Style drift (cambiar estilo no solicitado)
319
+
320
+ **Solicitud del usuario**: *"Agrega logging a la función `subir_archivo`."*
321
+
322
+ #### ❌ MAL — reformatea todo: comillas, type hints, espaciado
323
+
324
+ ```diff
325
+ - def subir_archivo(ruta, destino):
326
+ + def subir_archivo(ruta: str, destino: str) -> bool:
327
+ + """Sube archivo al destino con logging."""
328
+ + logger.info(f"Subiendo {ruta} a {destino}")
329
+ +
330
+ try:
331
+ - with open(ruta, 'rb') as f:
332
+ - datos = f.read()
333
+ -
334
+ - response = requests.post(destino, files={'file': datos})
335
+ -
336
+ - if response.status_code == 200:
337
+ - return True
338
+ - else:
339
+ - return False
340
+ + with open(ruta, "rb") as f:
341
+ + datos = f.read()
342
+ + response = requests.post(destino, files={"file": datos})
343
+ + exito = response.status_code == 200
344
+ +
345
+ + if exito:
346
+ + logger.info(f"Subida exitosa: {ruta}")
347
+ + else:
348
+ + logger.error(f"Subida falló: {ruta}, status={response.status_code}")
349
+ +
350
+ + return exito
351
+ except Exception as e:
352
+ - print(f"Error: {e}")
353
+ + logger.exception(f"Error subiendo {ruta}: {e}")
354
+ return False
355
+ ```
356
+
357
+ **Problemas**:
358
+ - Cambió comillas de simples a dobles
359
+ - Agregó type hints que nadie pidió
360
+ - Agregó docstring
361
+ - Reformateó espaciado
362
+ - Cambió la lógica del return booleano
363
+
364
+ #### ✅ BIEN — respeta estilo existente
365
+
366
+ ```diff
367
+ + import logging
368
+ +
369
+ + logger = logging.getLogger(__name__)
370
+ +
371
+ def subir_archivo(ruta, destino):
372
+ + logger.info(f'Iniciando subida: {ruta}')
373
+ try:
374
+ with open(ruta, 'rb') as f:
375
+ datos = f.read()
376
+
377
+ response = requests.post(destino, files={'file': datos})
378
+
379
+ if response.status_code == 200:
380
+ + logger.info(f'Subida exitosa: {ruta}')
381
+ return True
382
+ else:
383
+ + logger.error(f'Subida falló: {ruta}, status={response.status_code}')
384
+ return False
385
+ except Exception as e:
386
+ - print(f"Error: {e}")
387
+ + logger.exception(f'Error subiendo: {ruta}')
388
+ return False
389
+ ```
390
+
391
+ **Respetó**: comillas simples, ausencia de type hints, patrón booleano existente,
392
+ estilo de espaciado.
393
+
394
+ **Aplicación SWL**: este caso es señal de alerta en el "Resumen de señales de
395
+ alerta" del SKILL.md ("Se modificaron archivos fuera del scope"). El hook
396
+ pre-commit y `revisor-codigo-swl` deben marcarlo.
397
+
398
+ ---
399
+
400
+ ## Principio 4 — Ejecución orientada a metas
401
+
402
+ ### Ejemplo 4.1 — Plan vago vs criterios verificables
403
+
404
+ **Solicitud del usuario**: *"Arregla el sistema de autenticación."*
405
+
406
+ #### ❌ MAL — plan genérico sin criterios de éxito
407
+
408
+ ```
409
+ Voy a arreglar el sistema de autenticación así:
410
+ 1. Revisar el código
411
+ 2. Identificar problemas
412
+ 3. Hacer mejoras
413
+ 4. Probar los cambios
414
+
415
+ [procede a hacer cambios sin criterio de éxito claro]
416
+ ```
417
+
418
+ #### ✅ BIEN — pide la meta concreta, luego plan verificable
419
+
420
+ ```
421
+ Para "arreglar autenticación" necesito definir el criterio de éxito.
422
+ ¿Qué problema específico estamos resolviendo?
423
+
424
+ Si el problema es "los usuarios quedan logueados después de cambiar contraseña":
425
+
426
+ Plan:
427
+ 1. Test: cambiar contraseña → verificar que la sesión vieja queda invalidada
428
+ Verificación: el test FALLA (reproduce el bug)
429
+
430
+ 2. Implementación: invalidar sesiones al cambiar contraseña
431
+ Verificación: el test PASA
432
+
433
+ 3. Casos edge: múltiples sesiones activas, cambios concurrentes
434
+ Verificación: tests adicionales PASAN
435
+
436
+ 4. Sin regresión: tests de auth existentes siguen pasando
437
+ Verificación: suite completa en verde
438
+
439
+ Cobertura actual de auth: [N%]
440
+ ¿Cuál es el problema específico que ves?
441
+ ```
442
+
443
+ **Aplicación SWL**: `Skill("verificar-trabajo")` aplica este patrón con 4
444
+ niveles goal-backward (EXISTE → SUSTANTIVO → CONECTADO → DATOS_FLUYEN). El
445
+ ejemplo materializa el nivel "DATOS_FLUYEN" con tests verificables.
446
+
447
+ ---
448
+
449
+ ### Ejemplo 4.2 — Multi-paso con verificación incremental
450
+
451
+ **Solicitud del usuario**: *"Agrega rate limiting a la API."*
452
+
453
+ #### ❌ MAL — implementación completa en un solo commit de 300 LOC
454
+
455
+ ```python
456
+ # Implementa rate limiting completo con Redis, múltiples estrategias,
457
+ # sistema de configuración y monitoreo en un solo commit de 300 líneas.
458
+ # Sin pasos de verificación claros.
459
+ ```
460
+
461
+ #### ✅ BIEN — incremental con verificación por paso
462
+
463
+ ```
464
+ Plan para rate limiting:
465
+
466
+ 1. Rate limiting básico en memoria (un solo endpoint)
467
+ Verificación:
468
+ - Test: 100 requests → primeros 10 pasan, resto recibe 429
469
+ - Manual: curl al endpoint 11 veces, ver error de rate limit
470
+
471
+ 2. Extraer a middleware (aplicar a todos los endpoints)
472
+ Verificación:
473
+ - Test: rate limits aplican a /usuarios y /posts
474
+ - Tests existentes de endpoints siguen pasando
475
+
476
+ 3. Backend Redis (para multi-instancia)
477
+ Verificación:
478
+ - Test: rate limit persiste entre restarts de la app
479
+ - Test: dos instancias comparten el contador
480
+
481
+ 4. Configuración (rates por endpoint)
482
+ Verificación:
483
+ - Test: /search permite 10/min, /usuarios permite 100/min
484
+ - Archivo de config parseado correctamente
485
+
486
+ Cada paso es verificable y desplegable independientemente.
487
+ ¿Empezamos con el paso 1?
488
+ ```
489
+
490
+ **Aplicación SWL**: este patrón es el core de `/swl:ejecutar-fase` con commits
491
+ atómicos por tarea. Cada paso del plan corresponde a un commit verificable.
492
+
493
+ ---
494
+
495
+ ### Ejemplo 4.3 — Test-first: reproducir antes de corregir
496
+
497
+ **Solicitud del usuario**: *"El ordenamiento se rompe cuando hay scores duplicados."*
498
+
499
+ #### ❌ MAL — corrige sin reproducir primero
500
+
501
+ ```python
502
+ # Cambia inmediatamente la lógica de sort sin confirmar el bug
503
+ def ordenar_scores(scores):
504
+ return sorted(scores, key=lambda x: (-x['score'], x['nombre']))
505
+ ```
506
+
507
+ #### ✅ BIEN — reproduce con test, luego corrige
508
+
509
+ ```python
510
+ # 1. Primero: test que reproduce el problema
511
+ def test_ordenar_con_scores_duplicados():
512
+ """Verifica el ordenamiento cuando varios items tienen el mismo score."""
513
+ scores = [
514
+ {'nombre': 'Ana', 'score': 100},
515
+ {'nombre': 'Bruno', 'score': 100},
516
+ {'nombre': 'Carlos', 'score': 90},
517
+ ]
518
+
519
+ resultado = ordenar_scores(scores)
520
+
521
+ # Bug: el orden es no determinista para duplicados
522
+ # Corriendo este test varias veces debería ser consistente
523
+ assert resultado[0]['score'] == 100
524
+ assert resultado[1]['score'] == 100
525
+ assert resultado[2]['score'] == 90
526
+
527
+ # Verificación: corre el test 10 veces → falla con orden inconsistente
528
+
529
+ # 2. Ahora sí, fix con sort estable
530
+ def ordenar_scores(scores):
531
+ """Ordena por score descendente, luego nombre ascendente para empates."""
532
+ return sorted(scores, key=lambda x: (-x['score'], x['nombre']))
533
+
534
+ # Verificación: test pasa consistentemente
535
+ ```
536
+
537
+ **Aplicación SWL**: regla del SKILL.md "Test primero para bugs". `tdd-qa-swl`
538
+ con `Skill("tdd-workflow")` aplica este ciclo RED→GREEN→REFACTOR.
539
+
540
+ ---
541
+
542
+ ## Resumen de anti-patrones
543
+
544
+ | Principio | Anti-patrón | Fix |
545
+ |---|---|---|
546
+ | Pensar antes | Asume en silencio formato, campos, alcance | Listar supuestos explícitamente, preguntar |
547
+ | Simplicidad | Strategy pattern para un solo cálculo | Una función hasta que la complejidad sea real |
548
+ | Quirúrgico | Reformatea comillas, agrega type hints mientras corrige bug | Solo las líneas que arreglan el bug reportado |
549
+ | Orientado a metas | "Voy a revisar y mejorar el código" | "Test que reproduce X → hacerlo pasar → sin regresiones" |
550
+
551
+ ---
552
+
553
+ ## Insight clave
554
+
555
+ Los ejemplos "sobre-complicados" no son obviamente incorrectos: siguen design
556
+ patterns y buenas prácticas. El problema es el **timing**: agregan complejidad
557
+ antes de que se necesite, lo cual:
558
+
559
+ - Hace el código más difícil de entender
560
+ - Introduce más bugs
561
+ - Toma más tiempo implementar
562
+ - Es más difícil de probar
563
+
564
+ Las versiones "simples" son:
565
+ - Más fáciles de entender
566
+ - Más rápidas de implementar
567
+ - Más fáciles de probar
568
+ - Refactorizables después cuando la complejidad realmente se necesite
569
+
570
+ > **Buen código es código que resuelve el problema de hoy de forma simple,
571
+ > no el problema de mañana de forma prematura.**
572
+
573
+ ---
574
+
575
+ ## Origen y licencia
576
+
577
+ Adaptado de [andrej-karpathy-skills](https://github.com/forrestchang/andrej-karpathy-skills)
578
+ (MIT License, Andrej Karpathy + Forrest Chang). El skill SWL
579
+ `prevencion-sobreingenieria` consolida los 4 principios + sección propia
580
+ "Variables residuales post-refactor" + calibración para tasks triviales.