@saulwade/swl-ses 1.6.3 → 1.6.6

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.
Files changed (46) hide show
  1. package/CLAUDE.md +3 -3
  2. package/README.md +2 -2
  3. package/agentes/gh-fix-ci-swl.md +275 -0
  4. package/agentes/nemesis-auditor-swl.md +90 -1
  5. package/comandos/swl/exportar-vault.md +106 -14
  6. package/comandos/swl/nemesis.md +70 -3
  7. package/comandos/swl/release.md +62 -2
  8. package/comandos/swl/salud.md +32 -0
  9. package/comandos/swl/verificar.md +116 -2
  10. package/habilidades/agent-browser/SKILL.md +111 -4
  11. package/habilidades/agent-deep-links/SKILL.md +148 -0
  12. package/habilidades/backend-async-postgres-testing/SKILL.md +215 -0
  13. package/habilidades/backend-error-design/SKILL.md +221 -0
  14. package/habilidades/browser-interaction-patterns/SKILL.md +514 -0
  15. package/habilidades/browser-research-domains/SKILL.md +635 -0
  16. package/habilidades/changelog-generator/SKILL.md +172 -0
  17. package/habilidades/changelog-generator/scripts/parse-commits.js +354 -0
  18. package/habilidades/devsecops-pipeline-security/SKILL.md +3 -0
  19. package/habilidades/fastapi-experto/SKILL.md +49 -4
  20. package/habilidades/harness-claude-code/SKILL.md +4 -1
  21. package/habilidades/postgresql-experto/SKILL.md +80 -4
  22. package/habilidades/proceso-discovery-machote/SKILL.md +157 -0
  23. package/habilidades/proceso-modular-split/SKILL.md +256 -0
  24. package/habilidades/tdd-workflow/SKILL.md +12 -5
  25. package/hooks/extraccion-aprendizajes.js +8 -0
  26. package/hooks/lib/deep-links.js +185 -0
  27. package/hooks/lib/evolution-tracker.js +148 -20
  28. package/hooks/lib/gateway-notify.js +70 -7
  29. package/manifiestos/modulos.json +13 -3
  30. package/manifiestos/skills-lock.json +1247 -1191
  31. package/package.json +92 -92
  32. package/plugin.json +371 -362
  33. package/reglas/arquitectura.md +38 -0
  34. package/reglas/arreglar-al-detectar.md +93 -0
  35. package/reglas/auditorias-documentales-estructurales.md +38 -0
  36. package/reglas/registro-componentes-nuevos.md +14 -0
  37. package/reglas/tests-cleanup.md +220 -0
  38. package/scripts/instalador.js +72 -4
  39. package/scripts/lib/mcp_config.py +29 -14
  40. package/scripts/lib/notificaciones-telegram.js +14 -0
  41. package/scripts/lib/transformadores/codex.js +4 -0
  42. package/scripts/lib/transformadores/cursor.js +5 -0
  43. package/scripts/mcp-orchestrator.py +153 -131
  44. package/scripts/mcp-pool-manager.py +132 -107
  45. package/scripts/mcp-telemetry.py +139 -120
  46. 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.