@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 +3 -3
- package/README.md +1 -1
- package/comandos/swl/claudemd.md +23 -1
- package/habilidades/prevencion-sobreingenieria/SKILL.md +9 -5
- package/habilidades/prevencion-sobreingenieria/recursos/EXAMPLES.md +580 -0
- package/habilidades/swl-claudemd/SKILL.md +41 -1
- package/manifiestos/skills-lock.json +7 -7
- package/package.json +1 -1
- package/plugin.json +1 -1
- package/scripts/auditar-claudemd.js +61 -1
package/CLAUDE.md
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
# CLAUDE.md — @saulwade/swl-ses v1.6.
|
|
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
|
-
###
|
|
12
|
-
|
|
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
package/comandos/swl/claudemd.md
CHANGED
|
@@ -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
|
|
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.
|
|
10
|
+
version: "1.2.0"
|
|
11
11
|
evolved: true
|
|
12
|
-
evolved-from: "1.
|
|
13
|
-
evolved-at: "2026-
|
|
14
|
-
evolved-by: "
|
|
15
|
-
evolved-note: "
|
|
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
|
|
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-
|
|
3
|
+
"generatedAt": "2026-05-22T17:02:33.598Z",
|
|
4
4
|
"skillsCount": 177,
|
|
5
|
-
"lockHash": "sha256:
|
|
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:
|
|
893
|
-
"bytes":
|
|
894
|
-
"version": "\"1.
|
|
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:
|
|
1103
|
-
"bytes":
|
|
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.
|
|
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.
|
|
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 = {
|
|
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
|
+
};
|