@saulwade/swl-ses 1.6.7 → 1.6.8

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 CHANGED
@@ -1,4 +1,4 @@
1
- # CLAUDE.md — @saulwade/swl-ses v1.6.7
1
+ # CLAUDE.md — @saulwade/swl-ses v1.6.8
2
2
 
3
3
  ## Reglas de máxima prioridad (aplican SIEMPRE, sin excepción)
4
4
 
@@ -8,8 +8,8 @@ Todo contenido generado DEBE ser en español de México: respuestas, código com
8
8
  ### Uso obligatorio del sistema SWL
9
9
  Toda tarea DEBE usar el sistema SWL completo: agentes especializados, habilidades, hooks y comandos `/swl:*`. NO hacer trabajo directo que un agente SWL especializado haría mejor. Para tareas complejas: `orquestador-swl`. Para implementación: `implementador-swl` o el agente de stack. Para debugging: `depurador-swl`. Para revisión: `revisor-codigo-swl`. Para planificación: `planificador-swl`. Cargar skills con `Skill("nombre")` antes de implementar.
10
10
 
11
- ### Investigar antes de editar
12
- Investigar el codebase ANTES de editar. NUNCA modificar código que no se ha leído primero. Leer el archivo completo, entender el contexto, y solo entonces hacer cambios.
11
+ ### Cuatro principios de implementación (Karpathy)
12
+ Antes de implementar, refactorizar o corregir bugs: (1) **pensar antes de codificar** (no asumir en silencio), (2) **simplicidad primero** (sin abstracciones especulativas), (3) **cambios quirúrgicos** (leer archivo completo antes de editar, no refactor de oportunidad), (4) **ejecución orientada a metas** (criterios verificables, test que reproduce bugs antes del fix). Tabla operativa + mapeo a agentes SWL en `@docs/karpathy-principios.md`. Detalle + 9 ejemplos MAL→BIEN: `Skill("prevencion-sobreingenieria")` + `recursos/EXAMPLES.md`.
13
13
 
14
14
  ### Lectura de documentos Office y Jupyter
15
15
  Cuando necesites leer el **contenido** de un archivo `.docx`, `.xlsx`, `.xls`, `.pptx` o `.ipynb`, NUNCA uses el Read tool directamente (no soporta esos formatos). Usa:
package/README.md CHANGED
@@ -1,4 +1,4 @@
1
- # swl-ses v1.6.7
1
+ # swl-ses v1.6.8
2
2
 
3
3
  > El paquete anterior `@saulwadeleon/swl-software-engineering-system` está deprecado. Migrar a `@saulwade/swl-ses` (npmjs.org canónico) o `@saul-wade/swl-ses` (mirror en GitHub Packages) — el CLI `swl-ses` no cambia.
4
4
 
@@ -128,6 +128,14 @@ y genera `./CLAUDE.md` mínimo con:
128
128
  operacional de uso del sistema SWL al inicio de cada sesión. Si el proyecto
129
129
  tiene otras reglas locales (`@reglas/seguridad.md`, `@reglas/arquitectura.md`,
130
130
  etc.), agregarlas debajo.
131
+ - Sección **Reglas de máxima prioridad** con sub-sección obligatoria
132
+ **Cuatro principios de implementación (Karpathy)**:
133
+ ```markdown
134
+ ### Cuatro principios de implementación (Karpathy)
135
+ Antes de implementar, refactorizar o corregir bugs: (1) **pensar antes de codificar** (no asumir en silencio), (2) **simplicidad primero** (sin abstracciones especulativas), (3) **cambios quirúrgicos** (leer archivo completo antes de editar, no refactor de oportunidad), (4) **ejecución orientada a metas** (criterios verificables, test que reproduce bugs antes del fix). Detalle + 9 ejemplos MAL→BIEN: `Skill("prevencion-sobreingenieria")` + `recursos/EXAMPLES.md`.
136
+ ```
137
+ Esta sub-sección es estándar SWL y cubre los gaps cognitivos más
138
+ frecuentes del agente sin agregar reglas específicas del proyecto.
131
139
  - Sección **Stack** poblada (lenguaje, framework, ORM, package manager)
132
140
  - Sección **Comandos** poblada (npm scripts detectados o comandos típicos)
133
141
  - Sección **Code style** vacía con placeholders
@@ -137,9 +145,23 @@ y genera `./CLAUDE.md` mínimo con:
137
145
 
138
146
  Si CLAUDE.md ya existe, **NO lo sobreescribe**. Sugiere correr
139
147
  `/swl:claudemd refactor` para mejorar el actual. Si detecta que el CLAUDE.md
140
- existente NO incluye `@reglas/usar-sistema-swl.md`, emitir hallazgo WARN
148
+ existente NO incluye `@reglas/usar-sistema-swl.md` o NO menciona los cuatro
149
+ principios Karpathy (o `prevencion-sobreingenieria`), emitir hallazgo WARN
141
150
  recomendando agregarlo manualmente.
142
151
 
152
+ ### audit — checks específicos sobre Karpathy
153
+
154
+ El subcomando `audit` verifica además de las dimensiones estándar:
155
+
156
+ - ¿El CLAUDE.md menciona los cuatro principios Karpathy (texto literal
157
+ "Karpathy" o nombres de los 4 principios) o referencia
158
+ `prevencion-sobreingenieria` con `Skill(...)` o `@reglas/`?
159
+
160
+ Si NO los menciona y el archivo tiene >50 líneas: emitir hallazgo WARN
161
+ recomendando agregar la sub-sección compacta (4 líneas) como guía de
162
+ implementación. NO emite WARN para CLAUDE.md mínimos (<50 LOC) ni para
163
+ archivos `~/.claude/CLAUDE.md` user-level (esos siguen otro contrato).
164
+
143
165
  ## Variables de entorno
144
166
 
145
167
  Ver `@docs/variables-entorno.md` sección "Calidad de CLAUDE.md":
@@ -7,12 +7,12 @@ description: >
7
7
  implementar features, refactorizar código o resolver bugs — especialmente
8
8
  cuando el agente tiende a agregar abstracciones especulativas o tocar código
9
9
  fuera del alcance solicitado.
10
- version: "1.1.0"
10
+ version: "1.2.0"
11
11
  evolved: true
12
- evolved-from: "1.0.0"
13
- evolved-at: "2026-04-24"
14
- evolved-by: "aprender"
15
- evolved-note: "Sección nueva: variables residuales post-refactor son code smell (F841 con ruff)"
12
+ evolved-from: "1.1.0"
13
+ evolved-at: "2026-05-22"
14
+ evolved-by: "absorcion-temp-karpathy"
15
+ evolved-note: "Recurso nuevo: recursos/EXAMPLES.md con 9 ejemplos MAL→BIEN adaptados del repo andrej-karpathy-skills (MIT). Cierra gap de regla skills-estandar.md § Convención EXAMPLES.md."
16
16
  herramientasPermitidas: [Read, Grep]
17
17
  toolsRequeridos: Read, Grep, Glob
18
18
  exclusiones:
@@ -27,6 +27,10 @@ evolvable: true # default para skill estandar
27
27
  Skill derivado de los principios de Andrej Karpathy sobre errores comunes de LLMs
28
28
  al generar código, adaptado al ecosistema SWL.
29
29
 
30
+ > **Para 9 ejemplos MAL→BIEN lado a lado con código real**, ver
31
+ > [recursos/EXAMPLES.md](recursos/EXAMPLES.md). Cada ejemplo está mapeado a un
32
+ > escenario operativo concreto de SWL (qué agente debe citarlo, en qué flujo).
33
+
30
34
  ## Cuándo cargar
31
35
 
32
36
  - Antes de implementar cualquier feature nueva con `implementador-swl` o `backend-*-swl`
@@ -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.
@@ -58,7 +58,7 @@ node scripts/auditar-claudemd.js --json # para parsing
58
58
  node scripts/auditar-claudemd.js --strict # exit 1 si WARN
59
59
  ```
60
60
 
61
- El auditor verifica seis dimensiones:
61
+ El auditor verifica siete dimensiones:
62
62
 
63
63
  | Dimensión | Regla | Severidad |
64
64
  |---|---|---|
@@ -68,6 +68,7 @@ El auditor verifica seis dimensiones:
68
68
  | Secciones canónicas | Stack, Comandos, Code style, Conventions presentes | WARN |
69
69
  | @references | Archivos >80 líneas usan al menos un `@docs/...md` | WARN |
70
70
  | Placeholders | `[TBD]`, `[TODO]`, `[COMPLETAR]` | ERROR |
71
+ | Karpathy reference | Project-level >50 LOC menciona "Karpathy", los 4 principios, o `prevencion-sobreingenieria` | WARN |
71
72
 
72
73
  Veredicto final: ERROR → WARN → OK (el más severo gana).
73
74
 
@@ -168,6 +169,9 @@ Genera `./CLAUDE.md` raíz del proyecto detectando stack actual.
168
169
  2. Ejecutar `detectarStackDetallado(process.cwd())` (de
169
170
  `scripts/lib/detectar-stack-detallado.js`).
170
171
  3. Generar archivo con secciones pobladas:
172
+ - **Reglas obligatorias**: `@reglas/usar-sistema-swl.md`
173
+ - **Reglas de máxima prioridad** con sub-sección **Cuatro principios de
174
+ implementación (Karpathy)** — bloque exacto definido abajo
171
175
  - **Stack**: lenguaje + framework + ORM + package manager detectados
172
176
  - **Comandos**: npm scripts detectados o comandos típicos por lenguaje
173
177
  - **Code style**: placeholders explícitos (`<!-- pendiente: definir convención de X -->`)
@@ -177,6 +181,42 @@ Genera `./CLAUDE.md` raíz del proyecto detectando stack actual.
177
181
  4. Imprimir mensaje con próximo paso: "ejecuta `/swl:claudemd audit`
178
182
  para verificar".
179
183
 
184
+ ### Bloque obligatorio Karpathy en init-project
185
+
186
+ Toda generación nueva incluye esta sub-sección literal en "Reglas de
187
+ máxima prioridad":
188
+
189
+ ```markdown
190
+ ### Cuatro principios de implementación (Karpathy)
191
+ Antes de implementar, refactorizar o corregir bugs: (1) **pensar antes de codificar** (no asumir en silencio), (2) **simplicidad primero** (sin abstracciones especulativas), (3) **cambios quirúrgicos** (leer archivo completo antes de editar, no refactor de oportunidad), (4) **ejecución orientada a metas** (criterios verificables, test que reproduce bugs antes del fix). Detalle + 9 ejemplos MAL→BIEN: `Skill("prevencion-sobreingenieria")` + `recursos/EXAMPLES.md`.
192
+ ```
193
+
194
+ Justificación: el skill `prevencion-sobreingenieria` y su `recursos/EXAMPLES.md`
195
+ se distribuyen con cada instalación SWL, así que la referencia siempre resuelve.
196
+ La sub-sección compacta (5 líneas) ataca los gaps cognitivos más frecuentes
197
+ de cualquier proyecto sin agregar opinión específica.
198
+
199
+ ### Check Karpathy en audit
200
+
201
+ `scripts/auditar-claudemd.js` debe verificar (search regex case-insensitive):
202
+
203
+ - `/karpathy/i`
204
+ - `/cuatro principios/i` o `/4 principios/i`
205
+ - `/prevencion-sobreingenieria/`
206
+ - `/Skill\(["'`]prevencion-sobreingenieria["'`]\)/`
207
+
208
+ Si NO encuentra ninguna AND el archivo tiene >50 LOC AND es project-level
209
+ (no `~/.claude/CLAUDE.md`): emitir hallazgo WARN con mensaje:
210
+
211
+ ```
212
+ CLAUDE.md no menciona los cuatro principios Karpathy ni el skill
213
+ prevencion-sobreingenieria. Considerar agregar la sub-sección compacta
214
+ (ver /swl:claudemd init-project § Bloque obligatorio Karpathy).
215
+ ```
216
+
217
+ Esta dimensión es WARN, no ERROR — guía sin imponer. CLAUDE.md user-level
218
+ (`~/.claude/`) NO se evalúa contra este check.
219
+
180
220
  ---
181
221
 
182
222
  ## Gotchas / Errores comunes no obvios
@@ -1,8 +1,8 @@
1
1
  {
2
2
  "lockfileVersion": 1,
3
- "generatedAt": "2026-05-21T20:46:43.799Z",
3
+ "generatedAt": "2026-05-22T17:02:33.598Z",
4
4
  "skillsCount": 177,
5
- "lockHash": "sha256:f87e22a97a86a4abe7f1da215dc894433c3c8c9dfe9434e53d5a70e5b4912406",
5
+ "lockHash": "sha256:5d644809f2e8c7741a4553f1f6f493d309d4913782c9560bf06bd735eed1701e",
6
6
  "skills": [
7
7
  {
8
8
  "nombre": "accesibilidad-a11y",
@@ -889,9 +889,9 @@
889
889
  {
890
890
  "nombre": "prevencion-sobreingenieria",
891
891
  "path": "habilidades/prevencion-sobreingenieria/SKILL.md",
892
- "hash": "sha256:7396e1d8e7c87dd0154e9b34d1eff4124997566ca9a9da63b8a0514680af131e",
893
- "bytes": 16468,
894
- "version": "\"1.1.0\""
892
+ "hash": "sha256:9f6d0d4f5c54e29aff3e5fa66d89c42a0dabfd0a59eefcfbf6715ca085984f3e",
893
+ "bytes": 16812,
894
+ "version": "\"1.2.0\""
895
895
  },
896
896
  {
897
897
  "nombre": "privacy-memoria",
@@ -1099,8 +1099,8 @@
1099
1099
  {
1100
1100
  "nombre": "swl-claudemd",
1101
1101
  "path": "habilidades/swl-claudemd/SKILL.md",
1102
- "hash": "sha256:18cef250bd32a7841db3b6362b9303679bf2851e5e1efe7f3d81f0dd514ec5eb",
1103
- "bytes": 11585,
1102
+ "hash": "sha256:dd8ef172d38668e261fc88ff22e614559a05a4f13b2c612c2da3ea2388494377",
1103
+ "bytes": 13666,
1104
1104
  "version": "\"1.0.2\""
1105
1105
  },
1106
1106
  {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@saulwade/swl-ses",
3
- "version": "1.6.7",
3
+ "version": "1.6.8",
4
4
  "description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot con 61 agentes, 177 habilidades, 44 comandos, 69 reglas y 42 hooks. Soporta 11 lenguajes y 7 runtimes: Claude Code, OpenClaude, OpenCode, Gemini CLI, Cursor, Codex CLI (soporte completo); GitHub Copilot (soporte parcial). 100% en espanol (Mexico). Multi-target install (--target CSV / --all-runtimes), autoconfig MCP en Cursor/Codex con --with-mcp, agentes Codex en TOML, hooks Cursor (17 eventos) y Codex (6 eventos). Gateway bidireccional con relay Telegram y auditoria profunda Nemesis con loop evaluator-optimizer opt-in (ADR-0021) y 8 tools ejecutables. v1.6.5 integra 3 patrones de awesome-codex-skills (ComposioHQ, MIT) — agent-deep-links + changelog-generator + gh-fix-ci-swl — ADR-0029, y promueve 3 evoluciones SIGAF al sistema global (D1 nemesis SendMessage + D2 verificar smoke frontend + L2 alineacion veredicto).",
5
5
  "bin": {
6
6
  "swl-ses": "bin/swl-ses.js",
package/plugin.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "swl-ses",
3
- "version": "1.6.7",
3
+ "version": "1.6.8",
4
4
  "description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot. 61 agentes, 177 habilidades, 44 comandos, 69 reglas y 42 hooks. 62 librerias. 11 lenguajes. Soporta Claude Code, Copilot, OpenCode, Codex y Gemini CLI. Loop evaluator-optimizer en /swl:nemesis (ADR-0021). 3 patrones de awesome-codex-skills (ComposioHQ, MIT) adoptados (ADR-0029) — agent-deep-links + changelog-generator + gh-fix-ci-swl. Promueve 3 evoluciones SIGAF (D1 nemesis SendMessage, D2 verificar smoke frontend, L2 alineacion veredicto).",
5
5
  "author": "Saul Wade Leon",
6
6
  "license": "MIT",
@@ -13,6 +13,7 @@
13
13
  * - Presencia de secciones canónicas (Stack, Comandos, Code style, Conventions, @references)
14
14
  * - Uso de @references (al menos 1 si el archivo supera 80 líneas)
15
15
  * - Placeholders sin reemplazar ([TBD], [TODO], [COMPLETAR])
16
+ * - Referencia Karpathy (CLAUDE.md project-level >50 LOC menciona los 4 principios o el skill)
16
17
  *
17
18
  * Uso:
18
19
  * node scripts/auditar-claudemd.js [ruta] # Audita CLAUDE.md (default: ./)
@@ -26,11 +27,13 @@
26
27
 
27
28
  const fs = require('fs');
28
29
  const path = require('path');
30
+ const os = require('os');
29
31
 
30
32
  // ─── Config ───────────────────────────────────────────────────────────────
31
33
  const MAX_LINES = parseInt(process.env.SWL_CLAUDEMD_MAX_LINES, 10) || 200;
32
34
  const MAX_BULLET_CHARS =
33
35
  parseInt(process.env.SWL_CLAUDEMD_MAX_BULLET_CHARS, 10) || 1000;
36
+ const KARPATHY_MIN_LINES = 50;
34
37
 
35
38
  const SECCIONES_CANONICAS = [
36
39
  { nombre: 'Stack', regex: /^##\s+Stack/m },
@@ -39,6 +42,15 @@ const SECCIONES_CANONICAS = [
39
42
  { nombre: 'Conventions', regex: /^##\s+(Conventions|Convenciones)/im },
40
43
  ];
41
44
 
45
+ // Patrones que reconocen que el CLAUDE.md incorpora los 4 principios Karpathy
46
+ // o referencia explícita al skill prevencion-sobreingenieria.
47
+ const KARPATHY_PATTERNS = [
48
+ /karpathy/i,
49
+ /cuatro\s+principios/i,
50
+ /4\s+principios/i,
51
+ /prevencion-sobreingenieria/i,
52
+ ];
53
+
42
54
  const PLACEHOLDERS = /\[(TBD|TODO|COMPLETAR|PENDIENTE|XXX|FIXME)\]/g;
43
55
 
44
56
  // ─── Auditoría ────────────────────────────────────────────────────────────
@@ -129,6 +141,19 @@ function auditar(rutaClaudeMd) {
129
141
  });
130
142
  }
131
143
 
144
+ // 6. Referencia a los 4 principios Karpathy o al skill prevencion-sobreingenieria.
145
+ // Solo aplica a CLAUDE.md project-level con tamaño no trivial.
146
+ const tieneReferenciaKarpathy = detectarReferenciaKarpathy(contenido);
147
+ const esProjectLevel = !esRutaUserLevel(rutaClaudeMd);
148
+ if (!tieneReferenciaKarpathy && lineas.length > KARPATHY_MIN_LINES && esProjectLevel) {
149
+ hallazgos.push({
150
+ severidad: 'WARN',
151
+ regla: 'karpathy-reference',
152
+ mensaje: 'CLAUDE.md no menciona los cuatro principios Karpathy ni el skill prevencion-sobreingenieria',
153
+ sugerencia: 'Agregar la sub-sección compacta (ver `/swl:claudemd init-project` § Bloque obligatorio Karpathy)',
154
+ });
155
+ }
156
+
132
157
  // ─── Veredicto ───────────────────────────────────────────────────────────
133
158
  const tieneError = hallazgos.some(h => h.severidad === 'ERROR');
134
159
  const tieneWarn = hallazgos.some(h => h.severidad === 'WARN');
@@ -145,11 +170,33 @@ function auditar(rutaClaudeMd) {
145
170
  secciones_presentes: SECCIONES_CANONICAS.filter(s => s.regex.test(contenido)).map(s => s.nombre),
146
171
  secciones_ausentes: seccionesAusentes.map(s => s.nombre),
147
172
  tiene_at_references: /@[a-zA-Z][a-zA-Z0-9_\-./]+\.md/.test(contenido),
173
+ tiene_referencia_karpathy: tieneReferenciaKarpathy,
174
+ es_project_level: esProjectLevel,
148
175
  },
149
176
  hallazgos,
150
177
  };
151
178
  }
152
179
 
180
+ /**
181
+ * True si el contenido menciona los 4 principios Karpathy o el skill
182
+ * prevencion-sobreingenieria (cualquier coincidencia case-insensitive).
183
+ */
184
+ function detectarReferenciaKarpathy(contenido) {
185
+ return KARPATHY_PATTERNS.some(re => re.test(contenido));
186
+ }
187
+
188
+ /**
189
+ * True si la ruta apunta a un CLAUDE.md user-level (~/.claude/CLAUDE.md).
190
+ * El check Karpathy NO aplica a user-level — esos siguen otro contrato
191
+ * (preferencias personales, no implementación).
192
+ */
193
+ function esRutaUserLevel(rutaClaudeMd) {
194
+ if (!rutaClaudeMd) return false;
195
+ const homeClaudeDir = path.resolve(path.join(os.homedir(), '.claude'));
196
+ const rutaAbs = path.resolve(rutaClaudeMd);
197
+ return rutaAbs.startsWith(homeClaudeDir + path.sep) || rutaAbs === path.join(homeClaudeDir, 'CLAUDE.md');
198
+ }
199
+
153
200
  /**
154
201
  * Detecta bullets/párrafos cuyo contenido excede MAX_BULLET_CHARS.
155
202
  * Un "bullet" es una línea que empieza con `-` o `*` (ignorando indentación).
@@ -249,6 +296,9 @@ function imprimirReporte(resultado) {
249
296
  console.log(` - Secciones ausentes: ${m.secciones_ausentes.join(', ')}`);
250
297
  }
251
298
  console.log(` - @references: ${m.tiene_at_references ? 'sí' : 'no'}`);
299
+ if (m.es_project_level) {
300
+ console.log(` - Referencia Karpathy: ${m.tiene_referencia_karpathy ? 'sí' : 'no'}`);
301
+ }
252
302
  console.log('');
253
303
  }
254
304
 
@@ -294,4 +344,14 @@ if (require.main === module) {
294
344
  main();
295
345
  }
296
346
 
297
- module.exports = { auditar, ubicarClaudeMd, detectarBulletsGigantes, main, MAX_LINES, MAX_BULLET_CHARS };
347
+ module.exports = {
348
+ auditar,
349
+ ubicarClaudeMd,
350
+ detectarBulletsGigantes,
351
+ detectarReferenciaKarpathy,
352
+ esRutaUserLevel,
353
+ main,
354
+ MAX_LINES,
355
+ MAX_BULLET_CHARS,
356
+ KARPATHY_MIN_LINES,
357
+ };