@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
|
@@ -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.
|
|
4
|
+
version: "1.2.0"
|
|
5
5
|
evolved: true
|
|
6
|
-
evolved-from: "1.
|
|
7
|
-
evolved-at: "2026-05-
|
|
6
|
+
evolved-from: "1.1.0"
|
|
7
|
+
evolved-at: "2026-05-20"
|
|
8
8
|
evolved-by: "aprender"
|
|
9
|
-
evolved-note: "
|
|
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.
|
|
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
|
|
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
|
|
405
|
-
|
|
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,
|