@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.
- package/CLAUDE.md +196 -196
- package/README.md +578 -578
- package/agentes/orquestador-swl.md +37 -1
- package/comandos/swl/adoptar-proyecto.md +28 -0
- package/comandos/swl/aprender.md +142 -3
- package/comandos/swl/claudemd.md +81 -1
- package/comandos/swl/ejecutar-fase.md +14 -0
- package/comandos/swl/nuevo-proyecto.md +29 -0
- package/habilidades/prevencion-sobreingenieria/SKILL.md +9 -5
- package/habilidades/prevencion-sobreingenieria/recursos/EXAMPLES.md +580 -0
- package/habilidades/swl-claudemd/SKILL.md +224 -2
- package/hooks/claudemd-duplicacion-detector.js +170 -0
- package/manifiestos/hooks-config.json +9 -0
- package/manifiestos/modulos.json +6 -1
- package/manifiestos/skills-lock.json +8 -8
- package/package.json +92 -92
- package/plugin.json +371 -371
- package/reglas/sin-duplicacion-reglas-globales.md +182 -0
- package/reglas/verificar-citas-temporales.md +139 -0
- package/scripts/auditar-claudemd.js +107 -1
- package/scripts/lib/detector-reglas-duplicadas.js +220 -0
- package/scripts/lib/reglas-globales-conocidas.json +112 -0
|
@@ -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.
|