@saulwade/swl-ses 1.6.1 → 1.6.3
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 +2 -2
- package/README.md +4 -4
- package/agentes/_intent-spec.md +73 -0
- package/agentes/auto-evolucion-swl.md +24 -0
- package/agentes/cloud-infra-swl.md +25 -0
- package/agentes/datos-swl.md +23 -0
- package/agentes/devops-ci-swl.md +24 -0
- package/agentes/migrador-swl.md +22 -0
- package/agentes/pagos-swl.md +25 -0
- package/agentes/release-manager-swl.md +24 -0
- package/agentes/sre-swl.md +24 -0
- package/comandos/swl/planear-fase.md +16 -0
- package/habilidades/aprender-de-git-diff/SKILL.md +288 -0
- package/habilidades/diseno-herramientas-agente/SKILL.md +17 -1
- package/habilidades/meta-skills-estandar/SKILL.md +6 -0
- package/habilidades/meta-skills-estandar/recursos/skill-judge-rubrica.md +281 -0
- package/habilidades/proceso-autoverificacion-evidencias/SKILL.md +258 -0
- package/habilidades/proceso-confianza-pre-implementacion/SKILL.md +246 -0
- package/habilidades/proceso-ddia-fundamentos/SKILL.md +255 -0
- package/habilidades/proceso-ddia-streaming/SKILL.md +231 -0
- package/habilidades/proceso-intent-engineering/SKILL.md +269 -0
- package/habilidades/reducir-entropia/SKILL.md +219 -0
- package/hooks/lib/task-budget.js +218 -0
- package/hooks/validar-intent-spec.js +222 -0
- package/manifiestos/hooks-config.json +9 -0
- package/manifiestos/modulos.json +11 -2
- package/manifiestos/skills-lock.json +90 -41
- package/package.json +2 -2
- package/plugin.json +9 -2
- package/reglas/fragmentos-compartidos.md +26 -0
- package/reglas/intent-engineering.md +214 -0
- package/reglas/registro-componentes-nuevos.md +38 -0
- package/schemas/agent-frontmatter.schema.json +294 -167
- package/schemas/agent-message.schema.json +73 -53
- package/schemas/agent-output-implementacion.schema.json +114 -85
- package/schemas/agent-output-planificacion.schema.json +150 -113
- package/schemas/agent-output-review.schema.json +98 -78
- package/schemas/diary-entry.schema.json +42 -10
- package/schemas/hook-profiles.schema.json +54 -39
- package/schemas/hooks-config.schema.json +89 -74
- package/schemas/instinct.schema.json +152 -115
- package/schemas/modulos.schema.json +38 -29
- package/schemas/perfiles.schema.json +36 -28
- package/schemas/plugin.schema.json +77 -64
- package/schemas/skill-evals.schema.json +119 -95
- package/schemas/skill-frontmatter.schema.json +245 -170
- package/scripts/generar-inventario.js +3 -1
- package/scripts/lib/schema-version.js +164 -0
- package/scripts/validar-manifest.js +1 -1
- package/scripts/validar.js +3 -2
|
@@ -0,0 +1,255 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: proceso-ddia-fundamentos
|
|
3
|
+
description: >
|
|
4
|
+
Conceptos fundamentales del libro "Designing Data-Intensive Applications" de
|
|
5
|
+
Martin Kleppmann aplicables a sistemas no-distribuidos (como SWL): Cap 1
|
|
6
|
+
Reliability/Scalability/Maintainability como framework de evaluación, Cap 4
|
|
7
|
+
Schema Evolution con backward/forward compatibility, Cap 12 End-to-End
|
|
8
|
+
Argument y Trust-but-Verify. Cargar al diseñar manifiestos versionados,
|
|
9
|
+
evaluar arquitectura de un módulo, decidir dónde validar correctitud.
|
|
10
|
+
when_to_use: >
|
|
11
|
+
Usar cuando el usuario menciona schema versioning, backward/forward compat,
|
|
12
|
+
end-to-end verification, trust but verify, reliability scalability
|
|
13
|
+
maintainability, RSM framework, Operability Simplicity Evolvability,
|
|
14
|
+
schema-evolution, evaluar arquitectura, decidir dónde validar.
|
|
15
|
+
---
|
|
16
|
+
|
|
17
|
+
# DDIA Fundamentos — RSM, Schema Evolution, End-to-End
|
|
18
|
+
|
|
19
|
+
Adaptado de Martin Kleppmann, *"Designing Data-Intensive Applications"*
|
|
20
|
+
(O'Reilly, 2017). Skill cubre los tres ejes del libro que aplican a SWL:
|
|
21
|
+
|
|
22
|
+
- **Capítulo 1** — Reliability, Scalability, Maintainability como framework.
|
|
23
|
+
- **Capítulo 4** — Schema Evolution (back/forward compatibility).
|
|
24
|
+
- **Capítulo 12** — End-to-End Argument + Trust-but-Verify.
|
|
25
|
+
|
|
26
|
+
Los demás capítulos del libro (2-3 data models / storage, 5-10 distributed
|
|
27
|
+
systems) tratan dominios que SWL no es (no es BD, no es sistema distribuido).
|
|
28
|
+
Se descartan por filtro de dominio (regla `arquitectura.md`).
|
|
29
|
+
|
|
30
|
+
## Cuándo cargar
|
|
31
|
+
|
|
32
|
+
- Diseñar un schema nuevo (manifest, frontmatter, JSON Schema) que
|
|
33
|
+
evolucionará en el tiempo.
|
|
34
|
+
- Auditar un módulo SWL para evaluar Reliability/Scalability/Maintainability.
|
|
35
|
+
- Decidir dónde poner una validación: en cada capa o solo end-to-end.
|
|
36
|
+
- Resolver una duda sobre "¿esto se rompe si actualizo el sistema?" — es
|
|
37
|
+
pregunta de schema compatibility.
|
|
38
|
+
|
|
39
|
+
## Cuándo NO cargar
|
|
40
|
+
|
|
41
|
+
- Discusiones sobre replication, partitioning, consenso (Paxos/Raft) — esos
|
|
42
|
+
capítulos DDIA no aplican a SWL.
|
|
43
|
+
- Streaming/event sourcing avanzado — cargar `Skill("proceso-ddia-streaming")`.
|
|
44
|
+
- Optimización puntual de query SQL — cargar `Skill("performance-baseline")`.
|
|
45
|
+
|
|
46
|
+
---
|
|
47
|
+
|
|
48
|
+
## Parte 1 — Reliability, Scalability, Maintainability (RSM)
|
|
49
|
+
|
|
50
|
+
Cita textual del libro (DDIA p.18, línea 1411-1453):
|
|
51
|
+
|
|
52
|
+
> "It is well known that the majority of the cost of software is not in its
|
|
53
|
+
> initial development, but in its ongoing maintenance — fixing bugs, keeping
|
|
54
|
+
> its systems operational, investigating failures, adapting it to new
|
|
55
|
+
> platforms, modifying it for new use cases, repaying technical debt, and
|
|
56
|
+
> adding new features."
|
|
57
|
+
|
|
58
|
+
### Las 3 dimensiones del RSM
|
|
59
|
+
|
|
60
|
+
**Reliability** — el sistema sigue funcionando correctamente bajo:
|
|
61
|
+
- Hardware faults
|
|
62
|
+
- Software errors
|
|
63
|
+
- Human errors
|
|
64
|
+
|
|
65
|
+
Tipos de respuesta:
|
|
66
|
+
- **Fault-tolerant**: el sistema absorbe la falla sin propagarla.
|
|
67
|
+
- **Resilient**: degrada con gracia, no en cascada.
|
|
68
|
+
|
|
69
|
+
**Aplicación a SWL**: hooks bloqueantes (`calidad-pre-commit`,
|
|
70
|
+
`escaneo-secretos`) son fault-tolerance. Las escrituras atómicas
|
|
71
|
+
(`hooks/lib/atomic-write.js`) son resilience contra crashes durante write.
|
|
72
|
+
|
|
73
|
+
**Scalability** — cómo cambia el rendimiento cuando crece la carga:
|
|
74
|
+
- Describir **carga** (parámetros: throughput, concurrent users, etc.)
|
|
75
|
+
- Describir **rendimiento** (latencia: p50, p95, p99)
|
|
76
|
+
|
|
77
|
+
**Aplicación a SWL**: las métricas `Skill("performance-baseline")` siguen
|
|
78
|
+
este modelo. Carga = sesiones concurrentes, archivos por proyecto, agentes
|
|
79
|
+
en cadena. Rendimiento = latencia de hooks, tiempo de respuesta de
|
|
80
|
+
validadores.
|
|
81
|
+
|
|
82
|
+
**Maintainability** — tres principios de diseño del libro:
|
|
83
|
+
|
|
84
|
+
> "Operability: Make it easy for operations teams to keep the system running
|
|
85
|
+
> smoothly.
|
|
86
|
+
> Simplicity: Make it easy for new engineers to understand the system, by
|
|
87
|
+
> removing as much complexity as possible from the system. (Note this is
|
|
88
|
+
> not the same as simplicity of the user interface.)
|
|
89
|
+
> Evolvability: Make it easy for engineers to make changes to the system in
|
|
90
|
+
> the future, adapting it for unanticipated use cases as requirements
|
|
91
|
+
> change. Also known as extensibility, modifiability, or plasticity."
|
|
92
|
+
> (DDIA p.18-19, líneas 1435-1449)
|
|
93
|
+
|
|
94
|
+
**Aplicación a SWL**:
|
|
95
|
+
|
|
96
|
+
| Principio DDIA | Implementación SWL |
|
|
97
|
+
|---|---|
|
|
98
|
+
| Operability | `/swl:salud`, `/swl:dashboard`, `/swl:doctor`, runbooks |
|
|
99
|
+
| Simplicity | Skills ≤300 líneas, módulos profundos (Ousterhout), zero-deps en `hooks/lib/` |
|
|
100
|
+
| Evolvability | ADRs versionados, schemas opcionales aditivos, fragments compartidos |
|
|
101
|
+
|
|
102
|
+
### Cómo usar RSM como rúbrica de evaluación
|
|
103
|
+
|
|
104
|
+
Antes de declarar un módulo "listo", pregúntate:
|
|
105
|
+
|
|
106
|
+
1. **Reliability**: ¿qué pasa si una entrada inesperada llega? ¿Si un hook
|
|
107
|
+
externo falla? ¿Si el filesystem se llena?
|
|
108
|
+
2. **Scalability**: ¿el módulo escala con el tamaño del proyecto del
|
|
109
|
+
usuario? Si tiene 1000 archivos vs 10, ¿degrada linealmente o se
|
|
110
|
+
cuadratiza?
|
|
111
|
+
3. **Maintainability** (los 3 sub-principios):
|
|
112
|
+
- ¿Un operario nuevo puede operarlo sin leer todo el código?
|
|
113
|
+
- ¿Un ingeniero nuevo puede entenderlo sin un curso?
|
|
114
|
+
- ¿Un cambio de requisito se acomoda sin reescribir?
|
|
115
|
+
|
|
116
|
+
Si la respuesta es "no" a más de uno, el módulo necesita refactor antes de
|
|
117
|
+
release.
|
|
118
|
+
|
|
119
|
+
---
|
|
120
|
+
|
|
121
|
+
## Parte 2 — Schema Evolution (Cap 4)
|
|
122
|
+
|
|
123
|
+
Cita textual del libro (DDIA p.112, línea 5706-5719):
|
|
124
|
+
|
|
125
|
+
> "Backward compatibility — Newer code can read data that was written by
|
|
126
|
+
> older code.
|
|
127
|
+
>
|
|
128
|
+
> Forward compatibility — Older code can read data that was written by
|
|
129
|
+
> newer code.
|
|
130
|
+
>
|
|
131
|
+
> Backward compatibility is normally not hard to achieve: as author of the
|
|
132
|
+
> newer code, you know the format of data written by older code, and so
|
|
133
|
+
> you can explicitly handle it (if necessary by simply keeping the old code
|
|
134
|
+
> to read the old data). Forward compatibility can be trickier, because it
|
|
135
|
+
> requires older code to ignore additions made by a newer version of the
|
|
136
|
+
> code."
|
|
137
|
+
|
|
138
|
+
### Las 2 direcciones de compatibilidad
|
|
139
|
+
|
|
140
|
+
Cuando un schema evoluciona, hay dos preguntas distintas:
|
|
141
|
+
|
|
142
|
+
| Pregunta | Tipo | Difficulty |
|
|
143
|
+
|---|---|---|
|
|
144
|
+
| ¿Código nuevo lee datos viejos? | Backward | Fácil |
|
|
145
|
+
| ¿Código viejo lee datos nuevos? | Forward | Más difícil |
|
|
146
|
+
|
|
147
|
+
**Backward compat es fácil**: el código nuevo conoce ambas versiones del
|
|
148
|
+
schema, puede manejar los dos formatos.
|
|
149
|
+
|
|
150
|
+
**Forward compat es difícil**: el código viejo NO sabe que existe el campo
|
|
151
|
+
nuevo. Para que funcione, el código viejo debe **ignorar campos
|
|
152
|
+
desconocidos** sin romperse.
|
|
153
|
+
|
|
154
|
+
### Aplicación a schemas SWL
|
|
155
|
+
|
|
156
|
+
SWL tiene 14 archivos `schemas/*.schema.json`. Reglas operativas:
|
|
157
|
+
|
|
158
|
+
1. **Campos nuevos siempre opcionales**: agregar `myField: { type: "..." }`
|
|
159
|
+
sin marcarlo `required`. Código viejo lo ignora (forward-compat OK).
|
|
160
|
+
Código nuevo lo usa si está presente (backward-compat OK).
|
|
161
|
+
|
|
162
|
+
2. **NUNCA renombrar campos en MINOR/PATCH**: rename = breaking. Si hay
|
|
163
|
+
que renombrar, mantener ambos durante 1 release (deprecación) y
|
|
164
|
+
eliminar en el MAJOR siguiente.
|
|
165
|
+
|
|
166
|
+
3. **NUNCA cambiar tipo de un campo en MINOR/PATCH**: cambiar `string` →
|
|
167
|
+
`array` rompe ambas direcciones. Es MAJOR.
|
|
168
|
+
|
|
169
|
+
4. **Agregar enum value es compatible**: si `nivelRiesgo` acepta
|
|
170
|
+
`[BAJO, MEDIO, ALTO]`, agregar `CRÍTICO` es backward-compat (código
|
|
171
|
+
nuevo lo entiende, código viejo lo rechaza pero no se rompe).
|
|
172
|
+
|
|
173
|
+
5. **Eliminar enum value es breaking**: código nuevo rechaza algo que
|
|
174
|
+
código viejo escribía.
|
|
175
|
+
|
|
176
|
+
### Detección de breaking changes
|
|
177
|
+
|
|
178
|
+
Helper `scripts/lib/schema-version.js` (creado en F4) implementa estas
|
|
179
|
+
reglas. Uso esperado:
|
|
180
|
+
|
|
181
|
+
```js
|
|
182
|
+
const { verificarCompatibilidad } = require('./scripts/lib/schema-version');
|
|
183
|
+
const r = verificarCompatibilidad(documento, schema);
|
|
184
|
+
// r = { compatible: bool, modo: "backward"|"forward"|"none", advertencias }
|
|
185
|
+
```
|
|
186
|
+
|
|
187
|
+
Tres modos según versiones:
|
|
188
|
+
- Doc 1.0.0 + Schema 1.0.0 → `modo: "none"` (idénticas).
|
|
189
|
+
- Doc 1.0.0 + Schema 1.1.0 → `modo: "forward"` (dato viejo, código nuevo).
|
|
190
|
+
- Doc 1.1.0 + Schema 1.0.0 → `modo: "backward"` (dato nuevo, código viejo).
|
|
191
|
+
- Doc 2.0.0 + Schema 1.0.0 → `compatible: false` (MAJOR rompe).
|
|
192
|
+
|
|
193
|
+
---
|
|
194
|
+
|
|
195
|
+
## Parte 3 — End-to-End Argument (Cap 12)
|
|
196
|
+
|
|
197
|
+
Concepto del libro: la verificación de correctitud debe vivir en la
|
|
198
|
+
frontera del sistema (end-to-end), no en cada capa intermedia.
|
|
199
|
+
|
|
200
|
+
> "The End-to-End Argument for Databases" (DDIA p.516, sección referenciada)
|
|
201
|
+
|
|
202
|
+
### El principio aplicado
|
|
203
|
+
|
|
204
|
+
Si validas en cada capa intermedia:
|
|
205
|
+
- Capa 1: valida → pasa
|
|
206
|
+
- Capa 2: valida → pasa
|
|
207
|
+
- Capa 3: valida → pasa
|
|
208
|
+
- Resultado: dato corrupto entregado al usuario porque ninguna capa
|
|
209
|
+
verificó la **invariante de negocio** completa.
|
|
210
|
+
|
|
211
|
+
Si validas end-to-end:
|
|
212
|
+
- Capa 1, 2, 3: hacen su trabajo sin validar
|
|
213
|
+
- Frontera del sistema: verifica la invariante completa
|
|
214
|
+
- Resultado: si falla, se detecta antes de entregar al usuario.
|
|
215
|
+
|
|
216
|
+
### Aplicación a SWL
|
|
217
|
+
|
|
218
|
+
- `/swl:verificar` con goal-backward 4 niveles es end-to-end.
|
|
219
|
+
- Hooks de validación intermedios (`calidad-pre-commit`,
|
|
220
|
+
`escaneo-secretos`) son fault-tolerance, no end-to-end. Detectan
|
|
221
|
+
problemas obvios temprano para reducir blast radius — no reemplazan la
|
|
222
|
+
verificación end-to-end.
|
|
223
|
+
- Las dos capas son complementarias: hooks rápidos en cada step, verifier
|
|
224
|
+
end-to-end al cierre de fase.
|
|
225
|
+
|
|
226
|
+
### Trust-but-Verify (Cap 12, p.528)
|
|
227
|
+
|
|
228
|
+
El sistema debe **confiar provisionalmente** en sus componentes
|
|
229
|
+
(hooks pasan → asumimos correcto) pero **verificar** la invariante final
|
|
230
|
+
con evidencia, no con auto-reporte de los componentes.
|
|
231
|
+
|
|
232
|
+
Aplicación SWL: regla `verificar-citas-normativas.md` aplica este
|
|
233
|
+
principio a citas archivo:línea — un sub-agente reportó "está
|
|
234
|
+
implementado en X:42"; el padre verifica con `Read`+`grep` antes de
|
|
235
|
+
aceptar.
|
|
236
|
+
|
|
237
|
+
---
|
|
238
|
+
|
|
239
|
+
## Resumen — qué se llevó SWL del libro
|
|
240
|
+
|
|
241
|
+
| Concepto DDIA | Dónde vive en SWL |
|
|
242
|
+
|---|---|
|
|
243
|
+
| Reliability/Scalability/Maintainability como framework | Este skill como rúbrica de evaluación |
|
|
244
|
+
| Operability / Simplicity / Evolvability | Skills ≤300 líneas + observabilidad-swl + ADRs |
|
|
245
|
+
| Backward / Forward compat | `$schemaVersion` + `scripts/lib/schema-version.js` (F4) |
|
|
246
|
+
| End-to-End Argument | `/swl:verificar` goal-backward |
|
|
247
|
+
| Trust-but-Verify | regla `verificar-citas-normativas.md` |
|
|
248
|
+
|
|
249
|
+
## Origen
|
|
250
|
+
|
|
251
|
+
Skill creado el 2026-05-18 como parte de Opción B de integración
|
|
252
|
+
`temp/Designing Data Intensive Applications by Martin Kleppmann.md`. Ver
|
|
253
|
+
ADR-0025 para la decisión de scope. Skill hermano:
|
|
254
|
+
`proceso-ddia-streaming` (Cap 11 event sourcing aplicado a
|
|
255
|
+
`evolucion/*.jsonl`).
|
|
@@ -0,0 +1,231 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: proceso-ddia-streaming
|
|
3
|
+
description: >
|
|
4
|
+
Patrones de Stream Processing del libro DDIA (Cap 11, Martin Kleppmann)
|
|
5
|
+
aplicados a archivos JSONL append-only de SWL: event sourcing, idempotencia,
|
|
6
|
+
exactly-once vs at-least-once, replay desde offset. Cargar al diseñar hooks
|
|
7
|
+
que persistan eventos, modificar evolucion/*.jsonl, audit-trail.jsonl o
|
|
8
|
+
nudges.jsonl, o cuando un consumidor de JSONL deba ser idempotente.
|
|
9
|
+
when_to_use: >
|
|
10
|
+
Usar cuando el usuario menciona JSONL append-only, event sourcing, audit
|
|
11
|
+
trail, idempotencia, exactly-once, replay, offset, consumidor de stream,
|
|
12
|
+
evolucion/, nudges, telemetria, hooks que escriben eventos.
|
|
13
|
+
---
|
|
14
|
+
|
|
15
|
+
# DDIA Streaming — event sourcing aplicado a JSONL de SWL
|
|
16
|
+
|
|
17
|
+
Adaptado de Martin Kleppmann, *"Designing Data-Intensive Applications"*
|
|
18
|
+
Capítulo 11 — Stream Processing. SWL no tiene Kafka ni stream processor
|
|
19
|
+
real, pero opera con archivos JSONL append-only que se comportan como
|
|
20
|
+
streams locales. Los patrones del Cap 11 aplican directamente.
|
|
21
|
+
|
|
22
|
+
## Cuándo cargar
|
|
23
|
+
|
|
24
|
+
- Crear un hook que persiste eventos en JSONL (telemetría, audit, evolución).
|
|
25
|
+
- Diseñar un consumidor de `.planning/evolucion/*.jsonl`,
|
|
26
|
+
`.planning/audit.jsonl`, `.planning/comms/nudges.jsonl`.
|
|
27
|
+
- Diagnosticar duplicación de eventos en JSONL.
|
|
28
|
+
- Decidir si un hook debe ser idempotente o si basta con append.
|
|
29
|
+
- Implementar replay de eventos desde un offset.
|
|
30
|
+
|
|
31
|
+
## Cuándo NO cargar
|
|
32
|
+
|
|
33
|
+
- Trabajo síncrono request/response sin event log → no aplica.
|
|
34
|
+
- Sistema sin JSONL (sin streams locales) → cargar `proceso-ddia-fundamentos`.
|
|
35
|
+
- Discusiones sobre Kafka/Flink/Spark reales → SWL no opera con esos.
|
|
36
|
+
|
|
37
|
+
## Cita textual base (DDIA Cap 11, p.439-479)
|
|
38
|
+
|
|
39
|
+
### Event Sourcing (p.457, línea 21753-21798)
|
|
40
|
+
|
|
41
|
+
> "Event sourcing is a powerful technique for data modeling: from an
|
|
42
|
+
> application point of view [...]"
|
|
43
|
+
|
|
44
|
+
El libro distingue event sourcing de change data capture (CDC):
|
|
45
|
+
- **CDC**: captura cambios de una base de datos relacional como stream.
|
|
46
|
+
- **Event Sourcing**: la aplicación misma emite eventos como log primario;
|
|
47
|
+
el estado se deriva replicando los eventos.
|
|
48
|
+
|
|
49
|
+
**SWL aplica event sourcing** (no CDC): no hay BD relacional intermedia.
|
|
50
|
+
Los archivos JSONL son el log primario; cualquier proyección (instintos,
|
|
51
|
+
métricas, dashboard) se deriva replicándolos.
|
|
52
|
+
|
|
53
|
+
### Exactly-Once vs Effectively-Once (p.476, línea 22755)
|
|
54
|
+
|
|
55
|
+
> "This principle is known as exactly-once semantics, although
|
|
56
|
+
> effectively-once would be a more descriptive term."
|
|
57
|
+
|
|
58
|
+
Distinción clave del libro:
|
|
59
|
+
- **Exactly-once teórico**: el sistema garantiza que cada evento se
|
|
60
|
+
procesa una y solo una vez.
|
|
61
|
+
- **Effectively-once práctico**: el evento puede llegar N veces, pero
|
|
62
|
+
el efecto visible es el de procesarlo 1 vez. Se logra con idempotencia.
|
|
63
|
+
|
|
64
|
+
### Idempotencia (p.478, línea 22828-22845)
|
|
65
|
+
|
|
66
|
+
> "An idempotent operation is one that you can perform multiple times,
|
|
67
|
+
> and it has the same effect as if you performed it only once. For
|
|
68
|
+
> example, setting a key in a key-value store to some fixed value is
|
|
69
|
+
> idempotent (writing the value again simply overwrites the value with
|
|
70
|
+
> an identical value), whereas incrementing a counter is not idempotent
|
|
71
|
+
> (performing the increment again means the value is incremented twice)."
|
|
72
|
+
|
|
73
|
+
Operación clave del libro: incluso operaciones no-idempotentes pueden
|
|
74
|
+
hacerse idempotentes con metadata adicional (el offset del mensaje
|
|
75
|
+
fuente).
|
|
76
|
+
|
|
77
|
+
---
|
|
78
|
+
|
|
79
|
+
## Aplicación a SWL — archivos JSONL del sistema
|
|
80
|
+
|
|
81
|
+
| Archivo | Productor | Consumidores | ¿Es event source? |
|
|
82
|
+
|---|---|---|---|
|
|
83
|
+
| `.planning/evolucion/evoluciones.jsonl` | `/swl:evolucionar`, hook `evolucion-detector` | `/swl:evolucion-estado`, dashboard | Sí |
|
|
84
|
+
| `.planning/evolucion/nudges.jsonl` | hooks varios | `/swl:salud`, `red-team-swl` | Sí |
|
|
85
|
+
| `.planning/evolucion/agentes.jsonl` | hook `telemetria-agentes` | `/swl:metricas` | Sí |
|
|
86
|
+
| `.planning/audit.jsonl` | hook `audit-trail` | auditorías, post-mortems | Sí (con Merkle) |
|
|
87
|
+
| `.planning/comms/*.jsonl` | `notificador-swl` | inbox, gateway | Sí |
|
|
88
|
+
|
|
89
|
+
Todos son **append-only logs** — escritura por `fs.appendFileSync` (regla
|
|
90
|
+
CLAUDE.md). Nadie modifica ni borra entradas existentes.
|
|
91
|
+
|
|
92
|
+
## Patrón 1 — Idempotencia con event ID
|
|
93
|
+
|
|
94
|
+
**Problema**: un hook se ejecuta dos veces (re-trigger, retry) y emite el
|
|
95
|
+
mismo evento. El consumidor lo procesa 2 veces, produciendo doble efecto.
|
|
96
|
+
|
|
97
|
+
**Solución aplicada en SWL**: cada evento JSONL incluye un campo `id` único
|
|
98
|
+
(UUID v4 o `timestamp + hash de payload`). El consumidor mantiene un
|
|
99
|
+
set de IDs procesados (en `.planning/.processed-ids/`) y descarta
|
|
100
|
+
duplicados.
|
|
101
|
+
|
|
102
|
+
Ejemplo en hooks/lib (idiomático SWL):
|
|
103
|
+
|
|
104
|
+
```js
|
|
105
|
+
// Productor (hook)
|
|
106
|
+
const crypto = require('crypto');
|
|
107
|
+
const evento = {
|
|
108
|
+
id: crypto.randomUUID(),
|
|
109
|
+
ts: new Date().toISOString(),
|
|
110
|
+
type: 'agent.completed',
|
|
111
|
+
payload: { ... }
|
|
112
|
+
};
|
|
113
|
+
fs.appendFileSync(ruta, JSON.stringify(evento) + '\n');
|
|
114
|
+
|
|
115
|
+
// Consumidor
|
|
116
|
+
const procesados = cargarSetIds(); // del filesystem
|
|
117
|
+
for (const linea of leerJSONL(ruta)) {
|
|
118
|
+
const e = JSON.parse(linea);
|
|
119
|
+
if (procesados.has(e.id)) continue; // idempotencia
|
|
120
|
+
aplicar(e);
|
|
121
|
+
procesados.add(e.id);
|
|
122
|
+
guardarSetIds(procesados);
|
|
123
|
+
}
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
**Cuándo aplicar**: consumidores que ejecutan side-effects no-idempotentes
|
|
127
|
+
(notificaciones, mutaciones de estado, incrementos de contador).
|
|
128
|
+
|
|
129
|
+
**Cuándo NO aplicar**: consumidores que solo proyectan estado (instintos,
|
|
130
|
+
métricas) — la idempotencia natural (re-procesar produce mismo estado)
|
|
131
|
+
es suficiente.
|
|
132
|
+
|
|
133
|
+
## Patrón 2 — Replay desde offset
|
|
134
|
+
|
|
135
|
+
**Problema**: un consumidor crashea a mitad del JSONL. Al reiniciar,
|
|
136
|
+
¿desde dónde continúa?
|
|
137
|
+
|
|
138
|
+
**Solución aplicada en SWL**: persistir el byte-offset del último evento
|
|
139
|
+
procesado en archivo separado (`.planning/.offsets/consumidor-X.offset`).
|
|
140
|
+
Al iniciar, leer desde ese offset.
|
|
141
|
+
|
|
142
|
+
```js
|
|
143
|
+
const offsetFile = `.planning/.offsets/${consumidor}.offset`;
|
|
144
|
+
const offset = fs.existsSync(offsetFile)
|
|
145
|
+
? parseInt(fs.readFileSync(offsetFile, 'utf-8'), 10)
|
|
146
|
+
: 0;
|
|
147
|
+
|
|
148
|
+
const fd = fs.openSync(ruta, 'r');
|
|
149
|
+
// leer desde offset, procesar evento por evento
|
|
150
|
+
// tras cada evento procesado: fs.writeFileSync(offsetFile, nuevoOffset)
|
|
151
|
+
```
|
|
152
|
+
|
|
153
|
+
**Garantía**: at-least-once. Si crashea entre procesar y guardar offset,
|
|
154
|
+
el evento se reprocesa. Combinado con idempotencia (Patrón 1), se logra
|
|
155
|
+
**effectively-once** (DDIA p.476).
|
|
156
|
+
|
|
157
|
+
## Patrón 3 — Append-only sin borrado
|
|
158
|
+
|
|
159
|
+
**Regla operacional SWL**: los archivos JSONL NUNCA se reescriben enteros
|
|
160
|
+
ni se editan entradas pasadas. Compactación (eliminar duplicados, agrupar)
|
|
161
|
+
se hace creando un archivo `.jsonl.compacted` separado, dejando el
|
|
162
|
+
original como audit trail.
|
|
163
|
+
|
|
164
|
+
**Por qué importa**: si un hook reescribiera `audit.jsonl` con un nuevo
|
|
165
|
+
contenido (incluso con razón válida), perderíamos la garantía de
|
|
166
|
+
inmutabilidad que `audit-trail.js` con Merkle hash depende. El audit
|
|
167
|
+
trail roto deja de ser auditable.
|
|
168
|
+
|
|
169
|
+
## Patrón 4 — Schema evolution en eventos JSONL
|
|
170
|
+
|
|
171
|
+
Los eventos JSONL evolucionan: campos nuevos, payloads modificados. Los
|
|
172
|
+
consumidores deben tolerar eventos viejos sin romperse.
|
|
173
|
+
|
|
174
|
+
Aplicar las reglas de `Skill("proceso-ddia-fundamentos")` § Schema
|
|
175
|
+
Evolution:
|
|
176
|
+
- Campos nuevos siempre opcionales.
|
|
177
|
+
- `schemaVersion: "1.0.0"` por evento opcional pero recomendado.
|
|
178
|
+
- Consumidores ignoran campos desconocidos (forward-compat).
|
|
179
|
+
|
|
180
|
+
## Patrón 5 — Backpressure (cuándo NO aplica a SWL)
|
|
181
|
+
|
|
182
|
+
DDIA trata extensamente backpressure: cuando el productor escribe más
|
|
183
|
+
rápido que el consumidor. **SWL NO sufre este problema** porque:
|
|
184
|
+
- Volumen de eventos por sesión: ~10²-10³ líneas, no millones.
|
|
185
|
+
- Consumidores son sincrónicos bajo demanda (`/swl:salud`, dashboard).
|
|
186
|
+
- No hay productor continuo de alta frecuencia.
|
|
187
|
+
|
|
188
|
+
NO implementar backpressure / rate-limiting en JSONL writers. Es
|
|
189
|
+
sobre-ingeniería.
|
|
190
|
+
|
|
191
|
+
## Anti-patrones específicos a evitar
|
|
192
|
+
|
|
193
|
+
- **Compactar JSONL en mismo archivo**: rompe inmutabilidad. Crear
|
|
194
|
+
`.compacted` separado.
|
|
195
|
+
- **Confiar en orden de líneas para totalidad**: dos hooks concurrentes
|
|
196
|
+
pueden escribir entrelazado. Usar IDs únicos + timestamps, NO orden.
|
|
197
|
+
- **Indexar JSONL en disco como BD**: el archivo es secuencial por
|
|
198
|
+
diseño. Si necesitas índice, proyecta a otra estructura (instintos,
|
|
199
|
+
métricas agregadas).
|
|
200
|
+
- **Bloquear el productor durante write**: `fs.appendFileSync` es atómico
|
|
201
|
+
para writes ≤4096 bytes en sistemas POSIX (no aplica Windows, donde
|
|
202
|
+
hay carrera de race). Para Windows usar `atomicAppend` de
|
|
203
|
+
`hooks/lib/atomic-write.js` cuando el evento supera 4KB.
|
|
204
|
+
|
|
205
|
+
## Gotchas no obvios
|
|
206
|
+
|
|
207
|
+
- **`fs.appendFileSync` no es atómico en Windows con escrituras grandes**:
|
|
208
|
+
Node usa `WriteFile` que no garantiza atomicity para >4KB. Si el
|
|
209
|
+
evento JSON serializado supera 4KB, usar buffer + `atomicAppend`.
|
|
210
|
+
- **Lectura de JSONL con `fs.readFileSync` carga todo a memoria**: si
|
|
211
|
+
un archivo crece a 100MB+, usar `readline` con stream. Pero para
|
|
212
|
+
swl-ses (volumen bajo), `readFileSync` es aceptable.
|
|
213
|
+
- **`JSON.parse` en línea malformada lanza**: envolver en try/catch +
|
|
214
|
+
log de línea problemática. NUNCA dejar que un evento corrupto rompa
|
|
215
|
+
todo el replay.
|
|
216
|
+
|
|
217
|
+
## Resumen — del libro a SWL
|
|
218
|
+
|
|
219
|
+
| Concepto DDIA Cap 11 | Aplicación SWL |
|
|
220
|
+
|---|---|
|
|
221
|
+
| Event sourcing | Todos los `.planning/*.jsonl` |
|
|
222
|
+
| Exactly-once → effectively-once | Idempotencia + at-least-once delivery |
|
|
223
|
+
| Idempotencia con offset | Patrón 1 de este skill |
|
|
224
|
+
| Replay desde offset | Patrón 2 de este skill |
|
|
225
|
+
| Append-only inmutable | Regla operacional SWL |
|
|
226
|
+
| Backpressure | NO aplica (volumen bajo) |
|
|
227
|
+
|
|
228
|
+
## Origen
|
|
229
|
+
|
|
230
|
+
Skill creado el 2026-05-18 como parte de Opción B (ver ADR-0025). Skill
|
|
231
|
+
hermano: `proceso-ddia-fundamentos` (Cap 1, 4, 12 del mismo libro).
|