@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
@@ -1,12 +1,12 @@
1
1
  ---
2
2
  name: postgresql-experto
3
3
  description: PostgreSQL avanzado. JSONB, arrays, tipos personalizados, búsqueda de texto completo, window functions, CTEs recursivos, Row Level Security y funciones almacenadas.
4
- version: "1.1.0"
4
+ version: "1.2.0"
5
5
  evolved: true
6
- evolved-from: "1.0.0"
7
- evolved-at: "2026-05-05"
6
+ evolved-from: "1.1.0"
7
+ evolved-at: "2026-05-20"
8
8
  evolved-by: "aprender"
9
- evolved-note: "3 gotchas nuevos de la sesión SIGM 2026-05-05: RLS bypass por superusuarios, UUIDs hex en seeds, consultar enum_range antes de seedear (L2/L4/L8)"
9
+ evolved-note: "4 gotchas nuevos consolidados SIGM L-152/L-153/L-156 + SIGAF 2026-05-15: race condition SELECT-then-write fuera de transaction (CONFIRMADO x4), UPDATE WHERE col1 sin dimensiones de scoping, drift triple Literal↔validator↔ENUM, partial unique index con WHERE requiere predicado puro (no subqueries)"
10
10
  herramientasPermitidas: [Read, Grep]
11
11
  exclusiones:
12
12
  - "No cargar para optimización de queries SQL (EXPLAIN ANALYZE, índices, partitioning) — para optimización cargar `sql-optimizacion`."
@@ -172,3 +172,79 @@ Si el valor que necesitas no existe, agregar la variante con `ALTER TYPE ... ADD
172
172
  **`FORCE ROW LEVEL SECURITY` no protege al usuario dueño de la tabla (superuser/owner)**: el propietario de la tabla y los superusuarios de PostgreSQL bypasean RLS por defecto aunque esté `FORCE` habilitado. Causa: `FORCE` aplica a todos los usuarios excepto al dueño de la tabla y a superusuarios. Fix: si la aplicación conecta como el dueño de la tabla, cambiar el rol de conexión a un rol de aplicación sin privilegios de dueño (`GRANT CONNECT ON DATABASE ... TO app_user`), no usar el usuario `postgres` para conexiones de aplicación.
173
173
 
174
174
  **`tsvector` `GENERATED ALWAYS AS STORED` no actualiza cuando cambia la configuración de idioma**: si la columna `busqueda_ts` fue generada con `to_tsvector('spanish', ...)` y luego se cambia el contenido con texto en otro idioma, las palabras no se stemmizan correctamente — pero la columna no se regenera. Causa: la columna generada se recalcula solo cuando cambia la fila, no cuando cambia la lógica de la expresión. Fix: si se cambia la expresión de la columna generada, hacer `ALTER TABLE ... DROP COLUMN busqueda_ts; ALTER TABLE ... ADD COLUMN ...` para regenerar todos los valores.
175
+
176
+ **SELECT-then-INSERT/UPDATE FUERA de `conn.transaction()` produce race condition** (CONFIRMADO x4 en sesiones SIGM 2026-05-20): patrón `SELECT id FROM X WHERE clave=$1` seguido de `INSERT/UPDATE X` sin envolver en transacción permite a otra petición concurrente insertar entre el SELECT y el write — viola unicidad lógica aunque haya UNIQUE constraint (la 2da petición recibe error pero después de hacer trabajo). Casos: doble VIGENTE en valuación, doble documento/orden con mismo folio, subdivisión/fusión/traslado de predios concurrentes. Fix obligatorio:
177
+ ```python
178
+ async with self._conn.transaction():
179
+ # SELECT con lock explícito
180
+ row = await self._conn.fetchrow(
181
+ "SELECT id, version FROM tabla WHERE clave=$1 FOR UPDATE",
182
+ clave,
183
+ )
184
+ if row is None:
185
+ await self._conn.execute(
186
+ "INSERT INTO tabla (clave, ...) VALUES ($1, ...)",
187
+ clave, ...,
188
+ )
189
+ else:
190
+ # Si requiere CAS: WHERE clave=$1 AND version=$N en el UPDATE
191
+ await self._conn.execute(
192
+ "UPDATE tabla SET ... WHERE id=$1",
193
+ row["id"],
194
+ )
195
+ ```
196
+ Alternativas:
197
+ - `INSERT ... ON CONFLICT (clave) DO NOTHING/UPDATE` cuando la lógica es upsert puro.
198
+ - `SELECT FOR UPDATE` para bloquear filas existentes durante el chequeo.
199
+ - CAS en `WHERE version=$N` cuando el lock es muy costoso y se prefiere reintento.
200
+
201
+ Regla de auditoría: cualquier endpoint con `SELECT ... WHERE` seguido de `INSERT`/`UPDATE` sobre la misma tabla DEBE estar dentro de `async with self._conn.transaction():`. El UNIQUE constraint NO sustituye el lock — es defensa de último recurso.
202
+
203
+ **`UPDATE ... WHERE columna1 = $1` sin filtrar dimensiones adicionales des-actualiza filas hermanas**: SIGM NEM-VA-03 (2026-05-20). `UPDATE valuacion SET es_vigente=false WHERE predio_id=$1` cuando el predio tiene múltiples ejercicios fiscales des-vigentaba TODOS los ejercicios, no solo el actual. El bug pasó UNIQUE constraint (cada fila era válida individualmente) pero rompió integridad de dominio (debe haber 1 VIGENTE por (predio, ejercicio), no 1 VIGENTE por predio). Fix:
204
+ ```sql
205
+ -- MAL
206
+ UPDATE valuacion SET es_vigente=false WHERE predio_id=$1;
207
+
208
+ -- BIEN
209
+ UPDATE valuacion SET es_vigente=false
210
+ WHERE predio_id=$1 AND ejercicio=$2;
211
+ ```
212
+ Regla de auditoría: antes de aceptar un UPDATE, listar TODAS las dimensiones de dominio que definen "una fila lógica" (tenant_id, ejercicio, tipo, estatus, ...) y verificar que el WHERE incluye cada dimensión que aplica al scope del cambio. Las dimensiones implícitas (`es_vigente=true`, `activo=true`) también cuentan — un UPDATE que apaga `es_vigente` debe filtrar por la dimensión que define la unicidad de "vigente".
213
+
214
+ **Drift triple Pydantic `Literal` ↔ validator de schema ↔ ENUM PostgreSQL**: SIGM NEM-VA-11 (2026-05-20). Tres universos incompatibles del mismo concepto: `Literal["A", "B", ..., "F"]` (6 valores) en el schema Pydantic, validator personalizado aceptando 8 valores, ENUM en PostgreSQL declarado con 12 valores. Cast `$N::schema.tipo_enum` falla en BD cuando llega un valor permitido por el Literal pero ausente del ENUM. Fix: **PostgreSQL ENUM como única fuente de verdad** cuando existe. Validar al arrancar la app:
215
+ ```python
216
+ from sqlalchemy import text
217
+
218
+ async def validar_enum_alineado(session, schema: str, enum_name: str, pydantic_values: set[str]):
219
+ rows = await session.execute(
220
+ text(f"SELECT enum_range(NULL::{schema}.{enum_name})")
221
+ )
222
+ pg_values = set(rows.scalar().strip("{}").split(","))
223
+ if pydantic_values != pg_values:
224
+ raise RuntimeError(
225
+ f"Drift ENUM {enum_name}: pydantic={pydantic_values} pg={pg_values}"
226
+ )
227
+ ```
228
+ O generar el `Literal` Pydantic desde el ENUM PostgreSQL al startup (más estricto pero requiere conexión a BD para arrancar). Tests obligatorios: para cada ENUM, un test que compare `enum_range` contra el `Literal` del schema. La regla simple: si existe ENUM en BD, NO declarar Literal en código — derivarlo. Si NO existe ENUM en BD, NO declarar Literal en código — usar string libre con validator que consulta tabla de catálogo.
229
+
230
+ **Partial unique index requiere predicado puro sobre columnas de la propia tabla**: SIGAF DT-TRANSICION-UC 2026-05-15. Intento inicial: `CREATE UNIQUE INDEX ... WHERE estatus_destino_id != (SELECT id FROM cat_estatus_acto WHERE clave='CANCELADO')`. PostgreSQL rechaza el predicado en partial index porque no permite subqueries. Workaround común: resolver el UUID en `upgrade()` con `SELECT` y embeber el literal — frágil porque en BD limpia (CI) la tabla está vacía → query retorna None → el índice se crea SIN filtro WHERE → seeds posteriores fallan al insertar lo que el índice ahora rechaza. Fix correcto:
231
+ ```sql
232
+ -- 1. Agregar columna discriminadora en la tabla (boolean o enum)
233
+ ALTER TABLE transicion_estatus_acto
234
+ ADD COLUMN es_automatica BOOLEAN NOT NULL DEFAULT FALSE;
235
+
236
+ -- 2. Backfill condicional desde el join con la tabla externa
237
+ UPDATE transicion_estatus_acto t
238
+ SET es_automatica = TRUE
239
+ FROM cat_estatus_acto c
240
+ WHERE t.estatus_destino_id = c.id AND c.clave = 'CANCELADO';
241
+
242
+ -- 3. Partial index con predicado puro sobre columnas propias
243
+ CREATE UNIQUE INDEX uq_transicion_origen_automatica
244
+ ON transicion_estatus_acto (estatus_origen_id)
245
+ WHERE es_automatica = TRUE;
246
+ ```
247
+ Regla:
248
+ - El `WHERE` de un partial index DEBE ser sobre columnas de la propia tabla. NUNCA subquery a otra tabla.
249
+ - Si la lógica requiere distinguir filas por relación con otro catálogo, agregar columna discriminadora con backfill desde el JOIN.
250
+ - Toda migración que dependa de datos de catálogo DEBE ser idempotente para BD vacía. NO confiar en que el seed corra antes — en CI fresh corre después.
@@ -0,0 +1,157 @@
1
+ ---
2
+ name: proceso-discovery-machote
3
+ description: >
4
+ Patrón de discovery para modelar entidades regulatorias (papeles de trabajo,
5
+ cédulas, oficios, informes, expedientes, actas, dictámenes). Antes de modelar
6
+ desde el Artículo legal, pedir al usuario el machote oficial real (template
7
+ institucional). El Art. define el mínimo normativo; el machote revela el
8
+ contrato institucional real con prácticas no codificadas pero exigidas.
9
+ Cargar al iniciar la modelación de cualquier entidad nueva que represente
10
+ un documento normativo emitido por una institución regulatoria (OIC, INE,
11
+ SAT, etc.), durante /swl:discutir-fase de un módulo regulatorio, o cuando
12
+ el usuario rechaza un primer diseño por "faltan campos del documento real".
13
+ version: "1.0.0"
14
+ herramientasPermitidas: [Read, Grep, Glob]
15
+ exclusiones:
16
+ - "No cargar para entidades NO regulatorias (productos, usuarios, sesiones, eventos de log) — el machote no aplica a entidades de aplicación generales."
17
+ - "No cargar cuando el modelo de la entidad ya está estabilizado y solo se va a agregar 1-2 campos — el discovery completo es overhead innecesario para ajustes pequeños."
18
+ - "No cargar para refactor de entidad existente que ya tiene 6+ meses en producción — el machote ya se consolidó implícitamente en el modelo; cualquier diferencia es decisión de evolución, no de discovery."
19
+ - "No cargar para CRUD administrativo simple (catálogos, configuraciones) — esos no tienen contrato institucional rico; cargar `discutir-fase` regular."
20
+ evolvable: true
21
+ ---
22
+
23
+ # Proceso Discovery Machote
24
+
25
+ Patrón obligatorio antes de modelar entidades regulatorias.
26
+
27
+ ## Cuándo NO cargar
28
+
29
+ - La entidad NO es regulatoria (un producto de catálogo, una sesión de usuario, un evento de log).
30
+ - El modelo ya está estabilizado en producción y solo se agregan 1-2 campos.
31
+ - Es refactor de entidad consolidada (≥6 meses prod) — cargar reglas de refactor, no de discovery.
32
+ - Es CRUD administrativo simple sin contrato institucional rico.
33
+
34
+ ## El gap conceptual
35
+
36
+ **El Artículo legal define el mínimo normativo. El machote oficial revela el contrato institucional real.**
37
+
38
+ Caso real SIGAF ADR-081 IPHI (2026-05-15). El Informe de Presunta Hipótesis de Irregularidad (IPHI) se modeló inicialmente desde Art. 44 DBC_FISC_OIC, que enumera 10 fracciones (I-X) de contenido obligatorio:
39
+
40
+ > Art. 44. El informe contendrá:
41
+ > I. Antecedentes
42
+ > II. Marco normativo
43
+ > III. Hechos
44
+ > IV. Análisis
45
+ > V. Presunto daño patrimonial
46
+ > VI. Cuantificación del daño
47
+ > VII. Sujetos involucrados
48
+ > VIII. Pruebas documentales
49
+ > IX. Conclusiones
50
+ > X. Auditores que intervinieron
51
+
52
+ Modelo Pydantic + tabla SQL creados con esas 10 secciones. Commit inicial Fase 1.
53
+
54
+ Tras Fase 1, el usuario proporcionó el machote oficial real `OIC-DATIC-02-ESP-2025-IPHI-01.docx`. Reveló **7 extensiones institucionales NO presentes en Art. 44**:
55
+
56
+ | # | Extensión del machote | NO está en Art. 44 |
57
+ |---|---|---|
58
+ | 1 | Distinción `presunto_dano_patrimonial` (directo) vs `perjuicio_total_estimado` (agregado: daño + depreciación + costo oportunidad) | Art. 44 fracción V/VI no distinguen |
59
+ | 2 | Observaciones origen N:M con `hallazgo_identificador` (tabla puente `iphi_observacion_origen`) | Art. 44 no define la relación |
60
+ | 3 | `procedimientos_contratacion_relacionados` JSONB (lista de LP-INE asociadas) | No aparece en el Art. |
61
+ | 4 | `normas_infringidas` JSONB estructurado `[{norma, articulo, contenido}]` | Art. 44 dice "marco normativo" sin estructura |
62
+ | 5 | `desglose_dano_patrimonial` JSONB (auditabilidad del cálculo) | No exigido por Art. |
63
+ | 6 | Estructura específica de `antecedentes` JSONB con 6 sub-secciones | Art. 44 dice "antecedentes" sin desglose |
64
+ | 7 | Roles claros de `auditores_integrantes`: Elaboró/Revisó/Supervisó/Autorizó mapeando a flujo VBO de 3 niveles | Art. 44 dice "auditores que intervinieron" sin roles |
65
+
66
+ Refactor de Fase 1 = ~600 LOC de cambios + 1 tabla puente nueva + modelo Pydantic re-estructurado. Si el machote hubiera estado disponible al inicio, ese refactor no existiría.
67
+
68
+ ## El patrón
69
+
70
+ ### Paso 1: identificar si la entidad es regulatoria
71
+
72
+ Una entidad es regulatoria cuando representa un **documento normativo emitido por una institución**:
73
+
74
+ - Papel de trabajo (de auditoría).
75
+ - Cédula (preliminar, definitiva, observación derivada).
76
+ - Oficio (de investigación, de notificación, de respuesta).
77
+ - Informe (de hipótesis, de irregularidad, de auditoría).
78
+ - Acta (de inicio, de hechos, de cierre).
79
+ - Dictamen.
80
+ - Expediente (con composición fija de documentos).
81
+ - Resolución.
82
+
83
+ NO es regulatoria:
84
+ - Producto de catálogo, usuario, sesión, configuración.
85
+ - Evento de log o auditoría técnica.
86
+ - Entidad transaccional puramente operacional (pago, factura interna).
87
+
88
+ ### Paso 2: durante /swl:discutir-fase, preguntar explícitamente
89
+
90
+ Antes de modelar la entidad, plantear al usuario las 3 preguntas:
91
+
92
+ 1. **¿Existe un machote oficial real del documento?** (formato Word/PDF emitido por la institución, no template generado por el equipo).
93
+ 2. **¿Existe un Artículo legal o Reglamento que defina el contenido obligatorio?** (cita exacta del Art.).
94
+ 3. **Si solo hay Art., ¿hay práctica institucional documentada (manuales operativos, guías, ejemplos previos)?**
95
+
96
+ Estados posibles del discovery:
97
+
98
+ | Machote | Art. legal | Acción |
99
+ |---|---|---|
100
+ | ✅ Disponible | ✅ Disponible | Modelar desde machote + verificar que cubre todos los puntos del Art. (el Art. es el mínimo, el machote es el contrato real). |
101
+ | ✅ Disponible | ❌ No disponible | Modelar desde machote. Marcar campos como "exigidos por práctica institucional, sin base legal explícita". |
102
+ | ❌ No disponible | ✅ Disponible | Modelar desde Art. con marca explícita "modelo derivado solo del Art. — pendiente de validar contra machote oficial cuando el usuario lo proporcione". Crear DT formal `DT-MODELO-{ENTIDAD}-MACHOTE-PENDIENTE` con criterio de disparo: "modelo se considera estable cuando el machote oficial confirme o requiera ajustes". |
103
+ | ❌ No disponible | ❌ No disponible | Escalar al usuario: "No hay fuente oficial ni machote — ¿la entidad existe como tal? ¿O debería modelarse como entidad de aplicación sin contrato institucional?". |
104
+
105
+ ### Paso 3: extraer del machote, no del modelo mental
106
+
107
+ Cuando el machote esté disponible, leerlo COMPLETO con `markitdown` (regla global `markitdown.md` para .docx/.pdf):
108
+
109
+ ```bash
110
+ markitdown OIC-DATIC-02-ESP-2025-IPHI-01.docx > /tmp/machote-iphi.md
111
+ ```
112
+
113
+ Auditar cada sección visible:
114
+
115
+ - **Tablas estructuradas**: cada fila/columna → posible columna de la entidad o tabla puente.
116
+ - **Campos repetidos con formato uniforme**: posible tipo enum o catálogo.
117
+ - **Secciones con sub-secciones**: posible JSONB estructurado.
118
+ - **Espacios para firma/aprobación**: posible flujo de estados (VBO, aprobación multi-nivel).
119
+ - **Numeración de párrafos**: posible array o relación 1:N.
120
+ - **Referencias a otros documentos**: posibles FKs a otras entidades.
121
+
122
+ NO inferir campos del modelo mental — extraer literal del machote.
123
+
124
+ ### Paso 4: registrar la divergencia Art. ↔ machote
125
+
126
+ En el ADR de la entidad (o sección `### Modelo derivado` del schema):
127
+
128
+ ```markdown
129
+ ## Origen del modelo
130
+
131
+ **Artículo legal**: DBC_FISC_OIC Art. 44 (10 fracciones I-X).
132
+ **Machote oficial**: OIC-DATIC-02-ESP-2025-IPHI-01.docx (~25 páginas).
133
+
134
+ **Extensiones del machote sobre el Art.**:
135
+ 1. `perjuicio_total_estimado` distinto de `presunto_dano_patrimonial`.
136
+ 2. Observaciones origen N:M con `hallazgo_identificador`.
137
+ 3. `procedimientos_contratacion_relacionados` JSONB.
138
+ 4. ...
139
+
140
+ **Cobertura**: el modelo cubre 100% del Art. 44 + 7 extensiones del machote.
141
+ **Divergencia**: cuando el Art. y el machote contradicen, prevalece el machote (es el contrato real de la institución).
142
+ ```
143
+
144
+ ## Anti-patrones
145
+
146
+ - **Modelar desde solo el Art. legal sin pedir el machote** — el modelo cubre el mínimo legal pero pierde 30-50% del contenido institucional real. Refactor garantizado cuando aparezca el machote.
147
+ - **Modelar desde un machote sin verificar contra el Art.** — riesgo de campos derivados de versiones obsoletas del documento. El Art. es la fuente de verdad legal; el machote puede tener campos "heredados" de versiones previas.
148
+ - **Asumir que el machote está internalizado en el modelo del equipo** — el equipo del agente NO tiene el machote en su contexto entrenado. Pedirlo explícitamente al usuario; no inferir.
149
+ - **Diferir el machote a "después del MVP"** — el refactor para incorporar el machote en una entidad regulatoria es 80% reescritura del modelo + cascada a schemas Pydantic + frontend forms. NO es post-MVP barato.
150
+ - **Aceptar un machote en formato propietario sin convertirlo a markdown legible** — abrir el Word/PDF directamente lleva al modelo a olvidar campos. Convertir SIEMPRE con `markitdown` y auditar el markdown resultante.
151
+
152
+ ## Gotchas / Errores comunes no obvios
153
+
154
+ - **El machote tiene campos "sólo para impresión" que NO son datos de la entidad**: encabezados con logos, secciones de "Para uso oficial", footers. Esos NO van al modelo. Discriminar mientras se audita el markdown extraído.
155
+ - **El machote tiene campos calculados o derivados que SE ALMACENAN para auditoría**: ej. `perjuicio_total_estimado` se calcula desde 3 sumandos pero se persiste explícitamente porque el documento legal lo cita como cifra. Almacenar como columna calculada con backfill o como dato denormalizado con trigger de recálculo.
156
+ - **El machote evoluciona entre años fiscales** (`OIC-DATIC-02-ESP-2024-...` vs `OIC-DATIC-02-ESP-2025-...`): el modelo debe versionar el contrato. Agregar `machote_version: str` a la entidad para registrar contra qué versión se emitió cada documento. Migrar es costoso si no se versiona desde el inicio.
157
+ - **El usuario puede proporcionar un machote "informal" que es solo un Word con texto libre**: discriminar entre machote oficial (emitido por la institución con código de documento, fecha de actualización, autoridad emisora) vs borrador interno del equipo. El borrador NO es contrato — usar Art. como base con DT formal.
@@ -0,0 +1,256 @@
1
+ ---
2
+ name: proceso-modular-split
3
+ description: >
4
+ Playbook validado a escala (~20,000 LOC, 2 módulos refactorizados en SIGM:
5
+ recaudacion ADR-0016 + catastro ADR-0017) para dividir un módulo monolítico
6
+ en sub-paquetes coherentes preservando la API pública. Usa compositor por
7
+ herencia múltiple (NO __getattr__), helpers transversales en clase base,
8
+ router compositor zero-LOC con APP_ROUTER raíz. Cubre las 8 sesiones del
9
+ playbook (S1-S8), criterios para decidir cuándo aplicar el split, y los
10
+ anti-patrones detectados por auditoría humana que las auditorías técnicas
11
+ NO detectan. Cargar cuando el módulo backend supera ~1500 LOC en un solo
12
+ archivo (router.py, service.py o repository.py), cuando hay solicitud
13
+ explícita de "refactorizar X aplicando ADR-0016/0017", o cuando el módulo
14
+ monolítico se vuelve unmaintainable por acoplamiento entre sub-dominios.
15
+ version: "1.0.0"
16
+ herramientasPermitidas: [Read, Grep, Glob]
17
+ exclusiones:
18
+ - "No cargar para módulos pequeños (<800 LOC en su archivo más grande) — el split agrega overhead arquitectural sin beneficio."
19
+ - "No cargar para frameworks no-Python — el patrón compositor por herencia múltiple es específico de la flexibilidad de MRO de Python; en Java/C# se hace con interfaces y composition diferente."
20
+ - "No cargar para refactor de capa de datos (BD schema split, sharding) — este playbook es para split de código aplicación; BD split requiere migración con expand-contract distinta."
21
+ - "No cargar para split de microservicios (separar en servicios HTTP distintos) — este playbook mantiene un solo proceso/deployment; para microservicios cargar `microservicios`."
22
+ evolvable: true
23
+ ---
24
+
25
+ # Proceso Modular Split (Playbook ADR-0016/0017)
26
+
27
+ Split de módulo monolítico en sub-paquetes preservando API pública. Validado a escala 20k LOC.
28
+
29
+ ## Cuándo NO cargar
30
+
31
+ - Módulo pequeño (<800 LOC en su archivo más grande) — overhead arquitectural sin beneficio.
32
+ - Framework no-Python (Java, Go, Rust) — el patrón compositor por herencia múltiple aprovecha MRO de Python.
33
+ - Refactor de capa de datos (BD schema, sharding) — requiere expand-contract distinto.
34
+ - Split de microservicios HTTP — cargar `microservicios`; este playbook mantiene un solo proceso.
35
+
36
+ ## Criterio para aplicar
37
+
38
+ Aplicar el playbook cuando AL MENOS DOS se cumplen:
39
+
40
+ 1. `router.py` (FastAPI) o equivalente supera **~1500 LOC** con ~30+ endpoints heterogéneos.
41
+ 2. `service.py` supera **~1700 LOC** con sub-dominios identificables (e.g., en `catastro/`: predios, valuación, vialidades, manzanas, construcciones, inspecciones, operaciones, factores, geo).
42
+ 3. `repository.py` supera **~2400 LOC** o tiene métodos de tablas que NO se cruzan entre sí.
43
+ 4. Tests del módulo tardan >60s o requieren imports cross-dominio frágiles.
44
+ 5. Cambio en una "sección" del módulo rompe tests de otra sección que no debería estar relacionada (signal de acoplamiento implícito).
45
+
46
+ Si solo se cumple 1 criterio, probablemente lo correcto es extraer una sub-función o helper, NO aplicar el split completo.
47
+
48
+ ## El patrón: compositor por herencia múltiple (NO `__getattr__`)
49
+
50
+ ### Lo que NO hacer
51
+
52
+ ```python
53
+ # MAL — compositor por __getattr__: invisible al type checker, frágil
54
+ class CatastroService:
55
+ def __init__(self, conn):
56
+ self._predios = CatastroPrediosService(conn)
57
+ self._valuacion = CatastroValuacionService(conn)
58
+ ...
59
+
60
+ def __getattr__(self, name):
61
+ # Mágico, pero el type checker no ve los métodos delegados.
62
+ for sub in (self._predios, self._valuacion, ...):
63
+ if hasattr(sub, name):
64
+ return getattr(sub, name)
65
+ raise AttributeError(name)
66
+ ```
67
+
68
+ Problemas:
69
+ - IDE no autocompleta los métodos delegados.
70
+ - `mypy`/`pyright` no detecta llamadas a métodos inexistentes.
71
+ - Stack traces opacos (la línea de error apunta al `__getattr__`, no al sitio real).
72
+ - Imposible inspeccionar el MRO efectivo.
73
+
74
+ ### Lo que SÍ hacer
75
+
76
+ ```python
77
+ # BIEN — herencia múltiple: MRO inspeccionable, type checker feliz
78
+ class CatastroPrediosService(CatastroServiceBase):
79
+ async def crear_predio(self, ...): ...
80
+ async def obtener_predio(self, ...): ...
81
+
82
+ class CatastroValuacionService(CatastroServiceBase):
83
+ async def crear_valuacion(self, ...): ...
84
+ async def listar_vigentes(self, ...): ...
85
+
86
+ class CatastroVialidadesService(CatastroServiceBase):
87
+ async def listar_vialidades(self, ...): ...
88
+
89
+ # El compositor hereda de TODOS los sub-services.
90
+ class CatastroService(
91
+ CatastroPrediosService,
92
+ CatastroValuacionService,
93
+ CatastroVialidadesService,
94
+ # ... 9 sub-services
95
+ ):
96
+ """Compositor monolítico transitorio.
97
+
98
+ Mantiene la API pública del módulo monolítico previo. Los routers de la
99
+ fase posterior consumen sub-services específicos directamente — el
100
+ compositor queda como fallback de retrocompatibilidad.
101
+ """
102
+ pass
103
+ ```
104
+
105
+ Verificación inspeccionable:
106
+ ```python
107
+ >>> CatastroService.__mro__
108
+ (CatastroService, CatastroPrediosService, CatastroValuacionService,
109
+ CatastroVialidadesService, ..., CatastroServiceBase, object)
110
+ ```
111
+
112
+ El type checker ve todos los métodos. El IDE autocompleta. El stack trace apunta al método real, no al compositor.
113
+
114
+ ### Helpers transversales en clase base
115
+
116
+ Cuando varios sub-services necesitan el mismo helper (ej. `_validar_predio_existe`), va en la clase base (NO se duplica entre sub-services):
117
+
118
+ ```python
119
+ class CatastroServiceBase:
120
+ """Helpers compartidos por todos los sub-services de catastro."""
121
+
122
+ def __init__(self, conn):
123
+ self._conn = conn
124
+
125
+ async def _validar_predio_existe(self, predio_id: uuid.UUID) -> None:
126
+ row = await self._conn.fetchrow(
127
+ "SELECT 1 FROM catastro.predio WHERE id=$1",
128
+ predio_id,
129
+ )
130
+ if row is None:
131
+ raise NotFoundError(
132
+ message=f"Predio {predio_id} no encontrado",
133
+ code="PREDIO_NO_ENCONTRADO",
134
+ )
135
+
136
+ async def _tenant_actual(self) -> uuid.UUID:
137
+ # Lee del contexto async (contextvars), no del estado de instancia.
138
+ ...
139
+ ```
140
+
141
+ Regla: helper que ≥80% de sub-services necesitan → clase base. Si solo lo necesitan 1-2 sub-services, vive en uno de ellos y los otros lo importan via composición (no herencia).
142
+
143
+ ## El playbook S1-S8
144
+
145
+ ### S1 — ADR + PLAN.md + consolidación previa
146
+
147
+ - Crear `.planning/adrs/NNNN-split-modulo-X.md` con:
148
+ - Motivación cuantitativa (LOC actuales, problemas concretos detectados).
149
+ - Lista de sub-dominios identificados con criterio de separación.
150
+ - Plan de migración (no hay big-bang — son 8 sesiones secuenciales).
151
+ - Consolidar primero archivos relacionados que ya están separados pero deberían convivir (ej. en catastro: `repository_geo.py` + `router_geo.py` → `geo/`).
152
+
153
+ ### S2 — Split `repository.py` en sub-repositorios
154
+
155
+ - Crear `<modulo>/<sub>/repository.py` por cada sub-dominio.
156
+ - Mover métodos del repository monolítico a su sub-repositorio.
157
+ - El `<modulo>/repository.py` original queda como compositor por herencia múltiple (mismo patrón que el service).
158
+
159
+ ### S3 — Split `router.py` en sub-routers + APP_ROUTER raíz
160
+
161
+ - Crear `<modulo>/<sub>/router.py` con sub-APIRouter.
162
+ - Crear `<modulo>/router.py` reducido (~38 LOC) que solo agrega APP_ROUTER raíz e incluye los sub-routers:
163
+ ```python
164
+ from fastapi import APIRouter
165
+ from .predios.router import router as predios_router
166
+ from .valuacion.router import router as valuacion_router
167
+ # ...
168
+
169
+ app_router = APIRouter(prefix="/catastro", tags=["catastro"])
170
+ app_router.include_router(predios_router, prefix="/predios")
171
+ app_router.include_router(valuacion_router, prefix="/valuacion")
172
+ # ...
173
+ ```
174
+
175
+ ### S4 — Split `service.py` en sub-services + `service_base.py`
176
+
177
+ - Aplicar el patrón "compositor por herencia múltiple" descrito arriba.
178
+ - Mover métodos del service monolítico a su sub-service.
179
+ - Helpers transversales → `service_base.py`.
180
+
181
+ ### S5 — Schemas: re-export por sub-paquete
182
+
183
+ - Schemas Pydantic NO se parten — siguen viviendo en `<modulo>/schemas.py` (un solo archivo) O se mueven a `<modulo>/<sub>/schemas.py` pero NO se duplica definición.
184
+ - Si se mueve, hacer re-export en `<modulo>/schemas.py` para retrocompatibilidad de imports:
185
+ ```python
186
+ from .predios.schemas import PredioCreate, PredioResponse # noqa: F401
187
+ from .valuacion.schemas import ValuacionCreate, ValuacionResponse # noqa: F401
188
+ ```
189
+
190
+ ### S6 — Reorganización de tests
191
+
192
+ - Mover `tests/<modulo>/test_*.py` a `tests/<modulo>/<sub>/test_*.py` con `git mv` (preserva historial).
193
+ - Conftest.py por sub-dominio si los fixtures divergen.
194
+
195
+ ### S7 — Auditoría nemesis por sub-router + arch-cleanup
196
+
197
+ - `/swl:nemesis --modulo <modulo>/<sub>/router.py --remediar` por cada sub-router.
198
+ - Documentar hallazgos por sub-dominio.
199
+ - **S7-arch-cleanup**: migrar routers que aún consumen el compositor a su sub-service específico:
200
+ ```python
201
+ # MAL — router consume compositor cuando ya existe sub-service
202
+ from app.catastro.service import CatastroService
203
+ servicio = CatastroService(conn)
204
+ predio = await servicio.obtener_predio(id)
205
+
206
+ # BIEN — router consume el sub-service directamente
207
+ from app.catastro.predios.service import CatastroPrediosService
208
+ servicio = CatastroPrediosService(conn)
209
+ predio = await servicio.obtener_predio(id)
210
+ ```
211
+ El compositor queda como fallback para código legacy, no como ruta normal.
212
+
213
+ ### S8 — Cierre formal + actualización docs
214
+
215
+ - Actualizar `CLAUDE.md` del proyecto con los anti-patrones detectados en S7-arch-cleanup.
216
+ - Actualizar `AGENTS.md` con la nueva estructura del módulo.
217
+ - Marcar la fase como cerrada en `.planning/HOJA-RUTA.md`.
218
+
219
+ ## Anti-patrones detectados que las auditorías NO detectan (S7-arch-cleanup)
220
+
221
+ La auditoría técnica (`revisor-codigo-swl`, `nemesis-auditor-swl`) NO detectó estos en SIGM 2026-05-20 — los detectó el usuario al revisar el código post-refactor:
222
+
223
+ 1. **Router importa el compositor cuando existe sub-service específico**: defeating-purpose del refactor. Detectable con `grep -rn "from app.<modulo>.service import <Modulo>Service" app/<modulo>/<sub>/router.py` — debería retornar vacío después del split.
224
+
225
+ 2. **Router accede a `servicio._repo` o métodos privados**: rompe encapsulación. Detectable con `grep -rn "servicio\._repo\|servicio\._mapear_" app/<modulo>/<sub>/router.py`. Fix: encapsular en método público del sub-service.
226
+
227
+ 3. **Default `None` pasado explícitamente bypasea el default del callee**: `await servicio.listar_vialidades(activa=None)` cuando el contrato dice "omitir devuelve solo activas". Fix: en el router, `activa=True if activa is None else activa` o cambiar firma del callee a `activa: bool = True` (no `bool | None`).
228
+
229
+ 4. **CLAUDE.md y AGENTS.md desincronizados**: tras el split, ambos archivos describen el módulo. Si solo se actualiza uno, los agentes que cargan AGENTS.md desconocen el cambio. Fix: gate en CI que compara secciones del módulo entre los dos archivos.
230
+
231
+ ## Reglas no-negociables
232
+
233
+ - **NUNCA `__getattr__`** para delegación entre sub-services. El compositor es herencia múltiple inspeccionable.
234
+ - **NUNCA partir schemas Pydantic sin re-export** — rompe imports existentes y obliga a refactor cascada.
235
+ - **NUNCA hacer big-bang** — el playbook ES 8 sesiones secuenciales. Saltarse S5 o S6 deja el módulo en estado inconsistente.
236
+ - **SIEMPRE `git mv` para mover archivos** — preserva blame y permite ver el historial post-split.
237
+ - **SIEMPRE actualizar CLAUDE.md + AGENTS.md en S8** — desincronizados generan deuda invisible que aflora en la siguiente fase.
238
+ - **SIEMPRE S7-arch-cleanup después de los fixes nemesis** — sin esa pasada, los routers mantienen acoplamiento al compositor que el split intentaba romper.
239
+
240
+ ## Gotchas / Errores comunes no obvios
241
+
242
+ - **MRO con conflict en herencia múltiple**: si dos sub-services definen método con el mismo nombre, Python resuelve por MRO (orden de declaración en la clase compositor). Si el conflict es accidental (dos sub-dominios distintos con método homónimo), el MRO oculta uno. Fix: tests de regresión sobre cada método público del compositor, no solo de los sub-services.
243
+ - **Sub-service que requiere estado de otro sub-service en su `__init__`**: ej. `CatastroValuacionService` necesita `CatastroPrediosService` para validar el predio antes de crear valuación. Si el constructor lo recibe, el compositor por herencia múltiple no puede instanciar (`__init__` debe ser igual en todos). Fix: los sub-services NO se referencian mutuamente; comparten helpers en la clase base, no instancias entre sí.
244
+ - **Tests de integración que importan el compositor pasan, tests unitarios del sub-service fallan**: el sub-service requiere fixtures que el compositor ya configura. Fix: cada sub-service tiene su `conftest.py` con fixtures locales (mock de `_conn`, helpers de creación de datos del sub-dominio).
245
+ - **Migración del git history con `git mv` no preserva blame si hay cambios en el mismo commit**: si haces `git mv router.py predios/router.py` y modificas el contenido en el mismo commit, `git blame --follow` puede fallar en versiones antiguas de git. Fix: hacer `git mv` puro en un commit, modificar contenido en commit separado.
246
+ - **El compositor monolítico es "transitorio" pero queda 6+ meses**: si el equipo no migra los consumers al sub-service específico, el compositor se vuelve permanente. Fix: agregar deprecation warning al `__init__` del compositor: `warnings.warn("Use <sub>Service directly", DeprecationWarning, stacklevel=2)`.
247
+
248
+ ## Referencias
249
+
250
+ | Tema | Recurso |
251
+ |------|---------|
252
+ | ADR-0016 (recaudación split) | SIGM `.planning/adrs/0016-split-modulo-recaudacion.md` |
253
+ | ADR-0017 (catastro split) | SIGM `.planning/adrs/0017-split-modulo-catastro.md` |
254
+ | Patrón de error design (kwargs explícitos) | `Skill("backend-error-design")` |
255
+ | Testing con transaction mock | `Skill("backend-async-postgres-testing")` |
256
+ | Arquitectura general (módulos profundos, DRY) | `~/.claude/rules/arquitectura.md` |
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  name: tdd-workflow
3
3
  description: Flujo completo de Test-Driven Development. Ciclo RED (el test falla) → GREEN (implementación mínima) → REFACTOR (limpieza). Incluye cobertura mínima obligatoria, tests de frontera, factories, fixtures y estrategias para diferentes tipos de código (APIs, services, componentes Angular).
4
- version: "1.0.5"
4
+ version: "1.0.6"
5
5
  evolved: true
6
6
  evolved-from: "1.0.4"
7
7
  evolved-at: "2026-05-16"
@@ -350,14 +350,18 @@ function enriquecerDesdeFases(fases, opciones = {}) {
350
350
  // ...
351
351
  }
352
352
 
353
- // En el test:
354
- const sandbox = fs.mkdtempSync(path.join(os.tmpdir(), 'test-'));
353
+ // En el test (usa setupSandboxes — regla tests-cleanup.md):
354
+ const { setupSandboxes } = require('../_helpers/sandbox');
355
+ const sandboxes = setupSandboxes('swl-test-');
356
+
357
+ const sandbox = sandboxes.create();
355
358
  fs.mkdirSync(path.join(sandbox, '.planning', 'fases'), { recursive: true });
356
359
  // Opción A: pasar cwd explícito (recomendado)
357
360
  const r = enriquecerDesdeFases([], { cwd: sandbox });
358
361
  // Opción B: process.chdir() — solo funciona con cwd dinámico
359
362
  process.chdir(sandbox);
360
363
  const r2 = enriquecerDesdeFases([]);
364
+ // Cleanup automático al final del archivo vía after() registrado por setupSandboxes.
361
365
  ```
362
366
 
363
367
  ---
@@ -399,10 +403,13 @@ deterministamente. Patrones típicos:
399
403
  **Patrón 1 — Aislamiento por path único** (recomendado):
400
404
 
401
405
  ```javascript
406
+ const { setupSandboxes } = require('../_helpers/sandbox');
407
+ const sandboxes = setupSandboxes('swl-mi-app-test-');
402
408
  const env = { ...process.env };
403
409
 
404
- // Path único por test usando mkdtempSync sin contención
405
- const dir = fs.mkdtempSync(path.join(os.tmpdir(), 'mi-app-test-'));
410
+ // Path único por test usando el helper canónico (regla tests-cleanup.md).
411
+ // El cleanup es automático al final del archivo vía after() registrado.
412
+ const dir = sandboxes.create();
406
413
  env.MI_APP_FLAG_PATH = path.join(dir, 'flag.json');
407
414
 
408
415
  const res = spawnSync('node', [BIN], { env, ... });
@@ -632,6 +632,14 @@ const PATRONES_ARCHIVO_SWL_EXCLUIDO = [
632
632
  // Todo .planning/ salvo wiki/ (que puede contener conocimiento del proyecto usuario).
633
633
  // En swl-ses .planning/ es meta del sistema; los aprendizajes se gestionan manualmente.
634
634
  /(?:^|[\\/])\.planning[\\/](?!wiki[\\/])/,
635
+ // Todo _userland/ — zona de trabajo del agente (staging para exports al vault,
636
+ // userland-agentes, userland-habilidades). Su contenido es meta del agente:
637
+ // promociones temporales, borradores, exports estructurados con keywords
638
+ // narrativas ("patrón", "decisión", "anti-patrón", "fix") que el hook
639
+ // confunde con aprendizajes genuinos. Detectado 2026-05-18 cuando staging
640
+ // con dec1-*.md y res1-*.md (escritos para promover a vault) generaron
641
+ // 2 entradas espurias en APRENDIZAJES.md.
642
+ /(?:^|[\\/])_userland[\\/]/,
635
643
  /[\\/]CHANGELOG\.md$/i,
636
644
  /[\\/]CLAUDE\.md$/i,
637
645
  /[\\/]AGENTS\.md$/i,