@saulwade/swl-ses 1.6.3 → 1.6.5
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 +2 -2
- package/agentes/gh-fix-ci-swl.md +275 -0
- package/agentes/nemesis-auditor-swl.md +90 -1
- package/comandos/swl/exportar-vault.md +106 -14
- package/comandos/swl/nemesis.md +70 -3
- package/comandos/swl/release.md +62 -2
- package/comandos/swl/salud.md +32 -0
- package/comandos/swl/verificar.md +116 -2
- package/habilidades/agent-browser/SKILL.md +111 -4
- package/habilidades/agent-deep-links/SKILL.md +148 -0
- package/habilidades/backend-async-postgres-testing/SKILL.md +215 -0
- package/habilidades/backend-error-design/SKILL.md +221 -0
- package/habilidades/browser-interaction-patterns/SKILL.md +514 -0
- package/habilidades/browser-research-domains/SKILL.md +635 -0
- package/habilidades/changelog-generator/SKILL.md +172 -0
- package/habilidades/changelog-generator/scripts/parse-commits.js +354 -0
- package/habilidades/devsecops-pipeline-security/SKILL.md +3 -0
- package/habilidades/fastapi-experto/SKILL.md +49 -4
- package/habilidades/harness-claude-code/SKILL.md +4 -1
- package/habilidades/postgresql-experto/SKILL.md +80 -4
- package/habilidades/proceso-discovery-machote/SKILL.md +157 -0
- package/habilidades/proceso-modular-split/SKILL.md +256 -0
- package/habilidades/tdd-workflow/SKILL.md +12 -5
- package/hooks/extraccion-aprendizajes.js +8 -0
- package/hooks/lib/deep-links.js +185 -0
- package/hooks/lib/evolution-tracker.js +115 -18
- package/hooks/lib/gateway-notify.js +70 -7
- package/manifiestos/modulos.json +13 -3
- package/manifiestos/skills-lock.json +1247 -1191
- package/package.json +3 -3
- package/plugin.json +11 -2
- package/reglas/arquitectura.md +38 -0
- package/reglas/arreglar-al-detectar.md +93 -0
- package/reglas/auditorias-documentales-estructurales.md +38 -0
- package/reglas/registro-componentes-nuevos.md +14 -0
- package/reglas/tests-cleanup.md +220 -0
- package/scripts/lib/mcp_config.py +29 -14
- package/scripts/mcp-orchestrator.py +153 -131
- package/scripts/mcp-pool-manager.py +132 -107
- package/scripts/mcp-telemetry.py +139 -120
- package/scripts/verificar-release.js +199 -1
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: backend-error-design
|
|
3
|
+
description: >
|
|
4
|
+
Diseño de jerarquías de excepciones de dominio en backend Python: contrato
|
|
5
|
+
consistente (code, message, field opcional), kwargs explícitos obligatorios
|
|
6
|
+
para prevenir args invertidos silenciosamente, mapeo a códigos HTTP, testing
|
|
7
|
+
con exc_info.value.code (no "X" in str). Cargar cuando se diseñe o refactore
|
|
8
|
+
excepciones de dominio (BusinessLogicError, ValidationError, NotFoundError,
|
|
9
|
+
AuthorizationError, DuplicateError), cuando un test pase pero el contrato del
|
|
10
|
+
error esté invertido en producción, o cuando manejo-errores no cubra el caso
|
|
11
|
+
específico de excepciones de dominio con contrato code+message.
|
|
12
|
+
version: "1.0.0"
|
|
13
|
+
herramientasPermitidas: [Read, Grep]
|
|
14
|
+
exclusiones:
|
|
15
|
+
- "No cargar para manejo de errores HTTP genérico (try/except en endpoints, middleware de error handling) — para eso cargar `manejo-errores` que cubre el flujo HTTP completo."
|
|
16
|
+
- "No cargar para frameworks no-Python (Node.js error patterns, Java exception hierarchies, Go error wrapping) — este skill es específico de excepciones Python con dataclass-like signature."
|
|
17
|
+
- "No cargar para errores de capa externa (boto3 ClientError, httpx RequestError, BD driver) — esos NO son de dominio; usar el patrón genérico de mapear a HTTP 502 sin exponer el str(exc) interno."
|
|
18
|
+
- "No cargar para mensajes de validación de Pydantic — Pydantic genera sus propios ValidationError; cargar `fastapi-experto` para el patrón de respuesta."
|
|
19
|
+
evolvable: true
|
|
20
|
+
---
|
|
21
|
+
|
|
22
|
+
# Backend Error Design
|
|
23
|
+
|
|
24
|
+
Patrón para diseñar excepciones de dominio que NO se rompen silenciosamente al cliente.
|
|
25
|
+
|
|
26
|
+
## Cuándo NO cargar
|
|
27
|
+
|
|
28
|
+
- La tarea es manejo de errores genérico HTTP (try/except en endpoint, middleware global de error) — cargar `manejo-errores`.
|
|
29
|
+
- El framework es Node.js, Java, Go o Rust — los patrones de excepciones difieren; este skill es específico Python.
|
|
30
|
+
- El error es de capa externa (S3, HTTP client, BD driver) — esos NO son excepciones de dominio; el patrón genérico es mapear a HTTP 502 con detail genérico.
|
|
31
|
+
- La validación es de Pydantic — Pydantic ya genera `ValidationError`; cargar `fastapi-experto` para el patrón de respuesta.
|
|
32
|
+
|
|
33
|
+
## El bug recurrente: args posicionales se invierten silenciosamente
|
|
34
|
+
|
|
35
|
+
Caso real SIGAF 2026-05-15 Pasada 2 verificar (commit `4d2fb23`). Firma de excepción:
|
|
36
|
+
|
|
37
|
+
```python
|
|
38
|
+
class BusinessLogicError(Exception):
|
|
39
|
+
def __init__(self, message: str, code: str, field: str | None = None):
|
|
40
|
+
super().__init__(message)
|
|
41
|
+
self.message = message
|
|
42
|
+
self.code = code
|
|
43
|
+
self.field = field
|
|
44
|
+
```
|
|
45
|
+
|
|
46
|
+
3 sitios en `ordenes/service.py` invocaban la excepción así:
|
|
47
|
+
|
|
48
|
+
```python
|
|
49
|
+
# MAL — args invertidos: el primer string parece código, el segundo descripción.
|
|
50
|
+
# Pero la firma es (message, code), no (code, message).
|
|
51
|
+
raise BusinessLogicError(
|
|
52
|
+
"HALLAZGO_ACTO_MISMATCH",
|
|
53
|
+
f"El hallazgo {data.hallazgo_id} no pertenece al acto {data.acto_id}.",
|
|
54
|
+
)
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
Resultado: la excepción tiene `message="HALLAZGO_ACTO_MISMATCH"` y `code="El hallazgo X no pertenece al acto Y"`. El contrato HTTP devuelve al cliente:
|
|
58
|
+
|
|
59
|
+
```json
|
|
60
|
+
{
|
|
61
|
+
"error": {
|
|
62
|
+
"code": "El hallazgo 1234 no pertenece al acto 5678.",
|
|
63
|
+
"message": "HALLAZGO_ACTO_MISMATCH"
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
```
|
|
67
|
+
|
|
68
|
+
Inversión total. El cliente trata el texto descriptivo como código de error y el código como mensaje user-facing. El frontend no puede ramificar lógica por `code` porque cada error es único (incluye UUIDs).
|
|
69
|
+
|
|
70
|
+
### Por qué los tests NO detectaron el bug
|
|
71
|
+
|
|
72
|
+
Los tests asociados usaban:
|
|
73
|
+
|
|
74
|
+
```python
|
|
75
|
+
# MAL — la aserción pasa aunque el contrato esté invertido
|
|
76
|
+
with pytest.raises(BusinessLogicError) as exc_info:
|
|
77
|
+
await service.crear_hallazgo_acto(data)
|
|
78
|
+
assert "HALLAZGO_ACTO_MISMATCH" in str(exc_info.value)
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
`str(BusinessLogicError(...))` retorna `self.message` (porque `super().__init__(message)` configura `__str__`). El test encuentra "HALLAZGO_ACTO_MISMATCH" en el `message` aunque ese campo debería contener la descripción, no el código. La aserción es estructuralmente débil.
|
|
82
|
+
|
|
83
|
+
## Patrón obligatorio: kwargs explícitos
|
|
84
|
+
|
|
85
|
+
### Regla 1: el llamador SIEMPRE usa kwargs
|
|
86
|
+
|
|
87
|
+
```python
|
|
88
|
+
# BIEN — kwargs explícitos no se pueden invertir
|
|
89
|
+
raise BusinessLogicError(
|
|
90
|
+
message=f"El hallazgo {data.hallazgo_id} no pertenece al acto {data.acto_id}.",
|
|
91
|
+
code="HALLAZGO_ACTO_MISMATCH",
|
|
92
|
+
)
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
Si el llamador escribe args posicionales, el linter debe alertar. Configurar Ruff con regla `B026` (estrella en kwargs) o crear un pre-commit hook que detecte el patrón:
|
|
96
|
+
|
|
97
|
+
```bash
|
|
98
|
+
# Detectar excepciones de dominio invocadas con primer arg posicional string corto en SCREAMING_CASE
|
|
99
|
+
grep -rnE "raise (BusinessLogicError|AuthorizationError|NotFoundError|ValidationV3Error|BusinessRuleError|DuplicateError)\(\s*\"[A-Z_]+\"" backend/ --include="*.py"
|
|
100
|
+
```
|
|
101
|
+
|
|
102
|
+
Si hay match: 99% es un bug latente de args invertidos.
|
|
103
|
+
|
|
104
|
+
### Regla 2: tests verifican `exc_info.value.code`, NO `str()`
|
|
105
|
+
|
|
106
|
+
```python
|
|
107
|
+
# BIEN — verifica el campo correcto del contrato
|
|
108
|
+
with pytest.raises(BusinessLogicError) as exc_info:
|
|
109
|
+
await service.crear_hallazgo_acto(data)
|
|
110
|
+
assert exc_info.value.code == "HALLAZGO_ACTO_MISMATCH"
|
|
111
|
+
assert "no pertenece al acto" in exc_info.value.message
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
Esto fuerza al llamador a poner `code` en el campo `code`. Si está invertido, el test falla con `assert "El hallazgo X no pertenece..." == "HALLAZGO_ACTO_MISMATCH"`.
|
|
115
|
+
|
|
116
|
+
### Regla 3: la firma de la excepción enfuerza kwargs-only (Python 3.8+)
|
|
117
|
+
|
|
118
|
+
Para excepciones nuevas, usar el separador `*` para forzar kwargs:
|
|
119
|
+
|
|
120
|
+
```python
|
|
121
|
+
class BusinessLogicError(Exception):
|
|
122
|
+
def __init__(
|
|
123
|
+
self,
|
|
124
|
+
*, # ← obliga kwargs en TODOS los args posteriores
|
|
125
|
+
message: str,
|
|
126
|
+
code: str,
|
|
127
|
+
field: str | None = None,
|
|
128
|
+
):
|
|
129
|
+
super().__init__(message)
|
|
130
|
+
self.message = message
|
|
131
|
+
self.code = code
|
|
132
|
+
self.field = field
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
Con esta firma, `BusinessLogicError("CODE", "msg")` falla en compile-time con `TypeError: __init__() takes 1 positional argument but 3 were given`. El linter detecta el bug ANTES de que llegue a runtime.
|
|
136
|
+
|
|
137
|
+
**Cuándo NO retrofitear `*` a una excepción existente**: si la excepción ya tiene 100+ sitios de invocación legacy con args posicionales, el retrofit es un breaking change. Estrategia: documentar la regla en CLAUDE.md, agregar el linter rule a CI, migrar incrementalmente, y aplicar `*` solo cuando todos los sitios estén migrados.
|
|
138
|
+
|
|
139
|
+
## Jerarquía de excepciones de dominio recomendada
|
|
140
|
+
|
|
141
|
+
```python
|
|
142
|
+
# core/exceptions.py
|
|
143
|
+
class DomainError(Exception):
|
|
144
|
+
"""Base de toda excepción de dominio. NO usar directamente, usar subclases."""
|
|
145
|
+
|
|
146
|
+
def __init__(
|
|
147
|
+
self,
|
|
148
|
+
*,
|
|
149
|
+
message: str,
|
|
150
|
+
code: str,
|
|
151
|
+
field: str | None = None,
|
|
152
|
+
http_status: int = 400,
|
|
153
|
+
):
|
|
154
|
+
super().__init__(message)
|
|
155
|
+
self.message = message
|
|
156
|
+
self.code = code
|
|
157
|
+
self.field = field
|
|
158
|
+
self.http_status = http_status
|
|
159
|
+
|
|
160
|
+
class ValidationV3Error(DomainError):
|
|
161
|
+
"""Input inválido — HTTP 422."""
|
|
162
|
+
|
|
163
|
+
def __init__(self, *, message: str, code: str, field: str | None = None):
|
|
164
|
+
super().__init__(message=message, code=code, field=field, http_status=422)
|
|
165
|
+
|
|
166
|
+
class NotFoundError(DomainError):
|
|
167
|
+
"""Recurso no encontrado — HTTP 404."""
|
|
168
|
+
|
|
169
|
+
def __init__(self, *, message: str, code: str):
|
|
170
|
+
super().__init__(message=message, code=code, http_status=404)
|
|
171
|
+
|
|
172
|
+
class AuthorizationError(DomainError):
|
|
173
|
+
"""Sin permiso para la acción — HTTP 403."""
|
|
174
|
+
|
|
175
|
+
def __init__(self, *, message: str, code: str):
|
|
176
|
+
super().__init__(message=message, code=code, http_status=403)
|
|
177
|
+
|
|
178
|
+
class BusinessLogicError(DomainError):
|
|
179
|
+
"""Reglas de negocio violadas — HTTP 422."""
|
|
180
|
+
|
|
181
|
+
def __init__(self, *, message: str, code: str, field: str | None = None):
|
|
182
|
+
super().__init__(message=message, code=code, field=field, http_status=422)
|
|
183
|
+
|
|
184
|
+
class DuplicateError(DomainError):
|
|
185
|
+
"""Recurso ya existe — HTTP 409."""
|
|
186
|
+
|
|
187
|
+
def __init__(self, *, message: str, code: str, field: str | None = None):
|
|
188
|
+
super().__init__(message=message, code=code, field=field, http_status=409)
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
El handler global de FastAPI mapea cada excepción al código HTTP:
|
|
192
|
+
|
|
193
|
+
```python
|
|
194
|
+
@app.exception_handler(DomainError)
|
|
195
|
+
async def domain_error_handler(request: Request, exc: DomainError):
|
|
196
|
+
return JSONResponse(
|
|
197
|
+
status_code=exc.http_status,
|
|
198
|
+
content={
|
|
199
|
+
"error": {
|
|
200
|
+
"code": exc.code,
|
|
201
|
+
"message": exc.message,
|
|
202
|
+
"field": exc.field,
|
|
203
|
+
}
|
|
204
|
+
},
|
|
205
|
+
)
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
## Anti-patrones
|
|
209
|
+
|
|
210
|
+
- **`raise DomainError("CODE", "msg")` con args posicionales** — invierte el contrato si firma es `(message, code)`. Fix: kwargs explícitos.
|
|
211
|
+
- **`assert "CODE" in str(exc_info.value)` en tests** — pasa aunque el código esté en el campo equivocado. Fix: `exc_info.value.code == "CODE"`.
|
|
212
|
+
- **Heredar de `Exception` sin definir `code`** — el handler global no puede mapear a HTTP estructurado. Fix: heredar de `DomainError` con código obligatorio.
|
|
213
|
+
- **`detail=str(exc)` en HTTPException raised a partir de capa externa** — fuga arquitectura (paths, hosts, credenciales parciales). Fix: log internamente con detalle, devolver mensaje genérico al cliente.
|
|
214
|
+
- **Inventar códigos sin convención** — `code="error_1"`, `code="oops"`. Fix: `code` en SCREAMING_SNAKE_CASE con prefijo de dominio (`HALLAZGO_ACTO_MISMATCH`, `CEDULA_PRELIMINAR_INMUTABLE`).
|
|
215
|
+
|
|
216
|
+
## Gotchas / Errores comunes no obvios
|
|
217
|
+
|
|
218
|
+
- **`str(excepción)` retorna `args[0]` aunque tengas `self.message`**: `super().__init__(message)` pasa `message` como `args[0]`, y `Exception.__str__` retorna `args[0]`. Si cambias la firma sin actualizar `super().__init__(...)`, `str(exc)` muestra el valor equivocado pero `exc.message` muestra el correcto. Las dos representaciones divergen y bugs distintos se manifiestan según qué tests usen `str()` vs `.message`. Fix: documentar que `str()` SIEMPRE refleja `message` y nunca cambiar `super().__init__(...)` sin actualizar la convención.
|
|
219
|
+
- **Excepciones de dominio que envuelven excepciones de capa externa pierden el traceback original**: `raise BusinessLogicError(...)` dentro de un `except BotoCoreError as exc` sin `raise BusinessLogicError(...) from exc` pierde la cadena `__cause__`. Fix: SIEMPRE `raise DomainError(...) from exc` cuando se envuelve excepción externa.
|
|
220
|
+
- **`field` se usa para validación pero no se valida en runtime**: nada impide pasar `field="invalid_field_name"` con un nombre que NO existe en el schema. Si el frontend usa `field` para highlightear el campo del formulario, recibe nombres muertos. Fix: validar en tests que cada `field=` referencia un campo real del schema asociado. Considera generar el conjunto de fields válidos desde el schema Pydantic con `Schema.model_fields.keys()`.
|
|
221
|
+
- **Tests que importan la excepción del módulo equivocado**: dos módulos pueden definir `BusinessLogicError` distintos. El test importa de `auth.exceptions`, el service raise desde `ordenes.exceptions`. `pytest.raises(BusinessLogicError)` NO matchea. Fix: una sola jerarquía en `core/exceptions.py`, prohibir definir excepciones duplicadas en módulos hijos.
|