@saulwade/swl-ses 1.3.8 → 1.4.0

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 (128) hide show
  1. package/CLAUDE.md +12 -4
  2. package/README.md +1 -1
  3. package/bin/swl-mcp-server.js +187 -187
  4. package/bin/swl-webhook-server.js +198 -0
  5. package/comandos/swl/.evolved.json +22 -22
  6. package/comandos/swl/adoptar-proyecto.md +21 -1
  7. package/comandos/swl/claudemd.md +14 -1
  8. package/comandos/swl/contribuir.md +233 -233
  9. package/comandos/swl/exportar-vault.md +108 -0
  10. package/comandos/swl/nuevo-proyecto.md +24 -2
  11. package/gateway/adapters/base.js +109 -0
  12. package/gateway/adapters/discord.js +167 -0
  13. package/gateway/adapters/email.js +221 -0
  14. package/gateway/adapters/slack.js +192 -0
  15. package/gateway/adapters/telegram.js +183 -0
  16. package/gateway/adapters/webhook.js +113 -0
  17. package/gateway/adapters/whatsapp.js +214 -0
  18. package/gateway/agent-executor.js +322 -0
  19. package/gateway/command-relay.js +271 -0
  20. package/gateway/cron/jobs.js +263 -0
  21. package/gateway/cron/scheduler.js +322 -0
  22. package/gateway/cron/store.js +335 -0
  23. package/gateway/index.js +320 -0
  24. package/gateway/lib/event-channel.js +191 -0
  25. package/gateway/session.js +131 -0
  26. package/gateway/webhook-server.js +324 -0
  27. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  28. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  29. package/habilidades/build-errors-nextjs/SKILL.md +55 -1
  30. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  31. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  32. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  33. package/habilidades/eval-framework/SKILL.md +212 -212
  34. package/habilidades/extractor-de-aprendizajes/SKILL.md +20 -10
  35. package/habilidades/harness-claude-code/SKILL.md +299 -299
  36. package/habilidades/infra-github-actions/SKILL.md +166 -166
  37. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  38. package/habilidades/manejo-errores/.evolved.json +8 -8
  39. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  40. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  41. package/habilidades/nextjs-testing/SKILL.md +89 -5
  42. package/habilidades/node-experto/SKILL.md +37 -1
  43. package/habilidades/patrones-python/SKILL.md +229 -229
  44. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  45. package/habilidades/planear-fase/SKILL.md +319 -319
  46. package/habilidades/react-experto/SKILL.md +45 -4
  47. package/habilidades/release-semver/.evolved.json +8 -8
  48. package/habilidades/tdd-workflow/SKILL.md +36 -4
  49. package/habilidades/testing-python/SKILL.md +340 -340
  50. package/hooks/claudemd-bloat-detector.js +161 -161
  51. package/hooks/inyeccion-contexto.js +8 -3
  52. package/hooks/lib/agent-routing.js +107 -107
  53. package/hooks/lib/auto-consolidator.js +335 -335
  54. package/hooks/lib/error-classifier.js +308 -308
  55. package/hooks/lib/merkle-audit.js +96 -96
  56. package/hooks/lib/provenance-tracker.js +191 -191
  57. package/hooks/lib/rate-limit-ip.js +177 -0
  58. package/hooks/lib/rate-limit-tracker.js +253 -253
  59. package/hooks/lib/resource-quota.js +122 -122
  60. package/hooks/lib/retry-jitter.js +165 -165
  61. package/hooks/lib/skill-auditor.js +588 -588
  62. package/hooks/lib/sync-status.js +228 -228
  63. package/hooks/lib/taint-tracker.js +107 -107
  64. package/hooks/lib/text-similarity.js +241 -241
  65. package/hooks/lib/toon-compressor.js +245 -245
  66. package/hooks/lib/webhook-dedup.js +184 -0
  67. package/hooks/lib/webhook-verify.js +123 -0
  68. package/hooks/proteccion-rutas.js +120 -15
  69. package/hooks/registro-turnos.js +209 -209
  70. package/hooks/sugerir-regenerar-inventario.js +170 -170
  71. package/hooks/validar-formato-post-subagente.js +140 -140
  72. package/hooks/validar-memoria-hook.js +218 -218
  73. package/instintos/prompt-appendices.yaml +57 -57
  74. package/manifiestos/agent-output-schemas.json +57 -57
  75. package/manifiestos/modulos.json +1 -0
  76. package/manifiestos/skills-lock.json +34 -34
  77. package/package.json +5 -3
  78. package/plantillas/auditor-veto-template.md +105 -105
  79. package/plantillas/github-workflows/README.md +47 -47
  80. package/plantillas/github-workflows/release-please.yml +44 -44
  81. package/plantillas/github-workflows/swl-ci.yml +107 -107
  82. package/plantillas/github-workflows/swl-security.yml +51 -51
  83. package/plugin.json +1 -1
  84. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  85. package/reglas/arreglar-al-detectar.md +147 -147
  86. package/reglas/fragmentos-compartidos.md +152 -152
  87. package/reglas/harness-claude-code.md +213 -213
  88. package/reglas/usar-context7.md +226 -226
  89. package/reglas/usar-sistema-swl.md +251 -0
  90. package/schemas/diary-entry.schema.json +80 -80
  91. package/scripts/benchmark-memoria.js +167 -167
  92. package/scripts/comandos/skills.js +251 -2
  93. package/scripts/configurar-branch-protection.js +418 -418
  94. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  95. package/scripts/field-report.js +199 -199
  96. package/scripts/generar-checklists-consolidados.js +273 -273
  97. package/scripts/generar-inventario.js +420 -420
  98. package/scripts/generar-matriz-lenguajes.js +271 -271
  99. package/scripts/lib/artefactos-python.js +43 -43
  100. package/scripts/lib/benchmark-metrics.js +160 -160
  101. package/scripts/lib/budget-enforcer.js +252 -252
  102. package/scripts/lib/configurar-ci.js +380 -380
  103. package/scripts/lib/contadores-inventario.js +217 -217
  104. package/scripts/lib/detectar-stack-detallado.js +307 -307
  105. package/scripts/lib/diary-entry.js +234 -234
  106. package/scripts/lib/eval-metrics-store.js +218 -218
  107. package/scripts/lib/eval-quality.js +171 -171
  108. package/scripts/lib/eval-schemas.js +144 -144
  109. package/scripts/lib/eval-self-correct.js +106 -106
  110. package/scripts/lib/eval-validator.js +185 -185
  111. package/scripts/lib/jaccard-similarity.js +98 -98
  112. package/scripts/lib/longmemeval-runner.js +125 -125
  113. package/scripts/lib/npm-version.js +261 -261
  114. package/scripts/lib/paquetes-conocidos.js +50 -50
  115. package/scripts/lib/prompt-builder.js +264 -264
  116. package/scripts/lib/rrf-fusion.js +175 -175
  117. package/scripts/lib/scoring-instintos.js +277 -277
  118. package/scripts/lib/semantic-search.js +252 -252
  119. package/scripts/limpiar-artefactos-python.js +131 -131
  120. package/scripts/mcp-server/README.md +128 -128
  121. package/scripts/mcp-server/handlers.js +206 -206
  122. package/scripts/migrar-csv-a-array.js +168 -168
  123. package/scripts/migrar-fase-dominio.js +201 -201
  124. package/scripts/publicar.js +511 -511
  125. package/scripts/run-eval.js +141 -141
  126. package/scripts/validar-manifest.js +195 -195
  127. package/scripts/validar-userland-vacio.js +110 -110
  128. package/scripts/verificar-release.js +110 -0
@@ -0,0 +1,177 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * rate-limit-ip.js — Token bucket per-IP para webhook entrante.
5
+ *
6
+ * Implementa rate limiting con algoritmo de token bucket por dirección IP:
7
+ *
8
+ * - Cada IP recibe un bucket de capacidad `capacidad` (default = rpm).
9
+ * - El bucket se rellena a razón de `rpm/60` tokens por segundo.
10
+ * - Cada request consume 1 token. Si no hay tokens, rechazo.
11
+ * - Burst handling: hasta `capacidad` requests instantáneos antes del
12
+ * refill, lo que permite picos legítimos (ej: GitHub push con muchos
13
+ * events en segundos).
14
+ *
15
+ * El reloj se inyecta vía parámetro `ahora` (ms epoch) para tests
16
+ * deterministas. Sin parámetro, usa `Date.now()`.
17
+ *
18
+ * Origen: port reducido de `temp/claude-code-telegram-main/src/security/rate_limiter.py`
19
+ * (294 LOC Python → ~150 LOC JS, eliminando cost-based limiting y locks
20
+ * asíncronos que no aplican a un servidor HTTP síncrono en Node).
21
+ *
22
+ * Zero-deps (sólo módulos nativos).
23
+ *
24
+ * @module hooks/lib/rate-limit-ip
25
+ */
26
+
27
+ const INTERVALO_LIMPIEZA_MS = 5 * 60 * 1000; // 5 minutos
28
+
29
+ /**
30
+ * Bucket individual de tokens. No es thread-safe (Node es single-threaded).
31
+ */
32
+ class RateLimitBucket {
33
+ /**
34
+ * @param {number} capacidad Tokens máximos (también valor inicial).
35
+ * @param {number} refillRate Tokens por segundo.
36
+ * @param {number} ahora Timestamp en ms (Date.now()).
37
+ */
38
+ constructor(capacidad, refillRate, ahora) {
39
+ if (!Number.isFinite(capacidad) || capacidad <= 0) {
40
+ throw new Error('rate-limit-ip: capacidad debe ser número positivo');
41
+ }
42
+ if (!Number.isFinite(refillRate) || refillRate <= 0) {
43
+ throw new Error('rate-limit-ip: refillRate debe ser número positivo');
44
+ }
45
+ this.capacidad = capacidad;
46
+ this.tokens = capacidad;
47
+ this.refillRate = refillRate;
48
+ this.ultimaActualizacion = ahora;
49
+ }
50
+
51
+ /**
52
+ * Intenta consumir tokens.
53
+ * @param {number} tokens Cantidad a consumir (default 1).
54
+ * @param {number} ahora Timestamp en ms.
55
+ * @returns {boolean} true si se consumieron; false si no había suficientes.
56
+ */
57
+ consumir(tokens, ahora) {
58
+ this._rellenar(ahora);
59
+ if (this.tokens >= tokens) {
60
+ this.tokens -= tokens;
61
+ return true;
62
+ }
63
+ return false;
64
+ }
65
+
66
+ /**
67
+ * Rellena tokens según el tiempo transcurrido desde la última actualización.
68
+ * @param {number} ahora Timestamp en ms.
69
+ */
70
+ _rellenar(ahora) {
71
+ const transcurridoSeg = Math.max(0, (ahora - this.ultimaActualizacion) / 1000);
72
+ this.tokens = Math.min(this.capacidad, this.tokens + transcurridoSeg * this.refillRate);
73
+ this.ultimaActualizacion = ahora;
74
+ }
75
+
76
+ /**
77
+ * Estado actual del bucket (para diagnostics).
78
+ * @param {number} ahora Timestamp en ms.
79
+ * @returns {{capacidad: number, tokens: number, refillRate: number}}
80
+ */
81
+ estado(ahora) {
82
+ this._rellenar(ahora);
83
+ return {
84
+ capacidad: this.capacidad,
85
+ tokens: this.tokens,
86
+ refillRate: this.refillRate,
87
+ };
88
+ }
89
+ }
90
+
91
+ /**
92
+ * Limitador per-IP con cleanup periódico de buckets inactivos.
93
+ */
94
+ class RateLimiterIP {
95
+ /**
96
+ * @param {object} opciones
97
+ * @param {number} [opciones.rpm=60] Requests permitidos por minuto.
98
+ * @param {number} [opciones.capacidad] Burst máximo (default = rpm).
99
+ * @param {number} [opciones.intervaloLimpiezaMs=300000] Cada cuánto eliminar buckets llenos.
100
+ */
101
+ constructor({ rpm = 60, capacidad = null, intervaloLimpiezaMs = INTERVALO_LIMPIEZA_MS } = {}) {
102
+ if (!Number.isFinite(rpm) || rpm <= 0) {
103
+ throw new Error('rate-limit-ip: rpm debe ser número positivo');
104
+ }
105
+ this.rpm = rpm;
106
+ this.capacidad = capacidad ?? rpm;
107
+ this.refillRate = rpm / 60; // tokens por segundo
108
+ this.intervaloLimpiezaMs = intervaloLimpiezaMs;
109
+ this.buckets = new Map();
110
+ this.ultimaLimpieza = 0;
111
+ }
112
+
113
+ /**
114
+ * Decide si una IP puede hacer la siguiente request.
115
+ * @param {string} ip Dirección IP del cliente.
116
+ * @param {number} [ahora=Date.now()] Timestamp en ms.
117
+ * @returns {boolean} true si la request está permitida; false si throttled.
118
+ */
119
+ permite(ip, ahora) {
120
+ if (typeof ahora !== 'number') ahora = Date.now();
121
+ if (!ip || typeof ip !== 'string') return false;
122
+
123
+ let bucket = this.buckets.get(ip);
124
+ if (!bucket) {
125
+ bucket = new RateLimitBucket(this.capacidad, this.refillRate, ahora);
126
+ this.buckets.set(ip, bucket);
127
+ }
128
+ const permitido = bucket.consumir(1, ahora);
129
+ this._limpiarSiCorresponde(ahora);
130
+ return permitido;
131
+ }
132
+
133
+ /**
134
+ * Estado de una IP específica (sin consumir token).
135
+ * @param {string} ip
136
+ * @param {number} [ahora=Date.now()]
137
+ * @returns {{capacidad: number, tokens: number, refillRate: number}}
138
+ */
139
+ estado(ip, ahora) {
140
+ if (typeof ahora !== 'number') ahora = Date.now();
141
+ const bucket = this.buckets.get(ip);
142
+ if (!bucket) {
143
+ return { capacidad: this.capacidad, tokens: this.capacidad, refillRate: this.refillRate };
144
+ }
145
+ return bucket.estado(ahora);
146
+ }
147
+
148
+ /**
149
+ * Elimina buckets que tienen tokens al máximo (IPs que no usaron su cuota).
150
+ * Previene crecimiento ilimitado del Map en servidores de larga duración.
151
+ * @param {number} ahora
152
+ */
153
+ _limpiarSiCorresponde(ahora) {
154
+ if (ahora - this.ultimaLimpieza < this.intervaloLimpiezaMs) return;
155
+ this.ultimaLimpieza = ahora;
156
+ for (const [ip, bucket] of this.buckets) {
157
+ bucket._rellenar(ahora);
158
+ if (bucket.tokens >= bucket.capacidad) {
159
+ this.buckets.delete(ip);
160
+ }
161
+ }
162
+ }
163
+
164
+ /**
165
+ * Tamaño actual del Map (para diagnostics y tests).
166
+ * @returns {number}
167
+ */
168
+ tamano() {
169
+ return this.buckets.size;
170
+ }
171
+ }
172
+
173
+ module.exports = {
174
+ RateLimitBucket,
175
+ RateLimiterIP,
176
+ INTERVALO_LIMPIEZA_MS,
177
+ };
@@ -1,253 +1,253 @@
1
- 'use strict';
2
-
3
- /**
4
- * Rate Limit Tracker — Awareness de límites de tasa per-provider.
5
- *
6
- * Patrón adoptado de Hermes Agent (agent/rate_limit_tracker.py).
7
- * Registra respuestas 429 y headers de rate limit para ajustar
8
- * timing de llamadas y prevenir cascadas de error.
9
- *
10
- * Zero dependencias externas.
11
- *
12
- * Uso:
13
- * const { tracker } = require('./lib/rate-limit-tracker');
14
- *
15
- * // Registrar un 429
16
- * tracker.registrar429('anthropic', { retryAfter: 30, modelo: 'claude-sonnet-4-6' });
17
- *
18
- * // Consultar antes de llamar
19
- * if (tracker.estaThrottled('anthropic')) {
20
- * const espera = tracker.tiempoEspera('anthropic');
21
- * console.log(`Provider throttled, esperar ${espera}s`);
22
- * }
23
- *
24
- * // Registrar éxito (resetea contadores)
25
- * tracker.registrarExito('anthropic');
26
- *
27
- * @module hooks/lib/rate-limit-tracker
28
- */
29
-
30
- // ---------------------------------------------------------------------------
31
- // Constantes
32
- // ---------------------------------------------------------------------------
33
-
34
- /** Máximo de 429s consecutivos antes de marcar provider como BLOQUEADO. */
35
- const MAX_429_CONSECUTIVOS = 5;
36
-
37
- /** Tiempo base de backoff en ms tras un 429 (si no hay Retry-After). */
38
- const BACKOFF_BASE_MS = 10_000;
39
-
40
- /** Multiplicador de backoff exponencial por 429 consecutivo. */
41
- const BACKOFF_MULTIPLICADOR = 2;
42
-
43
- /** Tiempo máximo de backoff en ms (5 minutos). */
44
- const BACKOFF_MAX_MS = 300_000;
45
-
46
- /** Tiempo de recuperación half-open en ms (1 minuto). */
47
- const HALF_OPEN_MS = 60_000;
48
-
49
- /** Tiempo para considerar un registro como stale y limpiar (30 minutos). */
50
- const STALE_MS = 1_800_000;
51
-
52
- // ---------------------------------------------------------------------------
53
- // Estado del tracker (en memoria, por sesión)
54
- // ---------------------------------------------------------------------------
55
-
56
- /**
57
- * @typedef {object} ProviderState
58
- * @property {number} consecutivos429 - Conteo de 429s consecutivos
59
- * @property {number} ultimoTimestamp - Timestamp del último 429
60
- * @property {number} esperaHastaMs - Timestamp hasta el cual esperar
61
- * @property {string} estado - 'ok' | 'throttled' | 'bloqueado' | 'half-open'
62
- * @property {string|null} modelo - Último modelo que causó 429
63
- * @property {number} totalHistorico - Total de 429s en la sesión
64
- */
65
-
66
- /** @type {Map<string, ProviderState>} */
67
- const _providers = new Map();
68
-
69
- // ---------------------------------------------------------------------------
70
- // Funciones internas
71
- // ---------------------------------------------------------------------------
72
-
73
- /**
74
- * Obtiene o crea el estado de un provider.
75
- * @param {string} provider
76
- * @returns {ProviderState}
77
- */
78
- function _obtenerEstado(provider) {
79
- if (!_providers.has(provider)) {
80
- _providers.set(provider, {
81
- consecutivos429: 0,
82
- ultimoTimestamp: 0,
83
- esperaHastaMs: 0,
84
- estado: 'ok',
85
- modelo: null,
86
- totalHistorico: 0,
87
- });
88
- }
89
- return _providers.get(provider);
90
- }
91
-
92
- /**
93
- * Calcula el delay de backoff para un número de 429s consecutivos.
94
- * @param {number} consecutivos
95
- * @param {number|null} retryAfterSeg - Valor de Retry-After del header (en segundos)
96
- * @returns {number} Delay en milisegundos
97
- */
98
- function _calcularBackoff(consecutivos, retryAfterSeg) {
99
- if (retryAfterSeg && retryAfterSeg > 0) {
100
- return retryAfterSeg * 1000;
101
- }
102
- const delay = BACKOFF_BASE_MS * Math.pow(BACKOFF_MULTIPLICADOR, consecutivos - 1);
103
- return Math.min(delay, BACKOFF_MAX_MS);
104
- }
105
-
106
- // ---------------------------------------------------------------------------
107
- // API pública
108
- // ---------------------------------------------------------------------------
109
-
110
- const tracker = {
111
- /**
112
- * Registra una respuesta 429 de un provider.
113
- *
114
- * @param {string} provider - Nombre del provider ('anthropic', 'openai', etc.)
115
- * @param {object} [info]
116
- * @param {number} [info.retryAfter] - Valor del header Retry-After en segundos
117
- * @param {string} [info.modelo] - Modelo que causó el 429
118
- */
119
- registrar429(provider, info = {}) {
120
- const estado = _obtenerEstado(provider);
121
- const ahora = Date.now();
122
-
123
- estado.consecutivos429 += 1;
124
- estado.ultimoTimestamp = ahora;
125
- estado.totalHistorico += 1;
126
- estado.modelo = info.modelo || estado.modelo;
127
-
128
- const backoff = _calcularBackoff(estado.consecutivos429, info.retryAfter);
129
- estado.esperaHastaMs = ahora + backoff;
130
-
131
- if (estado.consecutivos429 >= MAX_429_CONSECUTIVOS) {
132
- estado.estado = 'bloqueado';
133
- } else {
134
- estado.estado = 'throttled';
135
- }
136
- },
137
-
138
- /**
139
- * Registra una llamada exitosa — resetea contadores del provider.
140
- *
141
- * @param {string} provider
142
- */
143
- registrarExito(provider) {
144
- const estado = _obtenerEstado(provider);
145
- estado.consecutivos429 = 0;
146
- estado.esperaHastaMs = 0;
147
- estado.estado = 'ok';
148
- },
149
-
150
- /**
151
- * Verifica si un provider está throttled o bloqueado.
152
- *
153
- * @param {string} provider
154
- * @returns {boolean}
155
- */
156
- estaThrottled(provider) {
157
- const estado = _obtenerEstado(provider);
158
- const ahora = Date.now();
159
-
160
- if (estado.estado === 'ok') return false;
161
-
162
- // Verificar si ya pasó el tiempo de espera
163
- if (ahora >= estado.esperaHastaMs) {
164
- if (estado.estado === 'bloqueado') {
165
- // Transición a half-open para permitir un intento de prueba
166
- estado.estado = 'half-open';
167
- estado.esperaHastaMs = ahora + HALF_OPEN_MS;
168
- return false; // permitir un intento
169
- }
170
- // Throttled normal — ya pasó el backoff
171
- estado.estado = 'ok';
172
- estado.consecutivos429 = 0;
173
- return false;
174
- }
175
-
176
- return true;
177
- },
178
-
179
- /**
180
- * Retorna el tiempo de espera restante en segundos.
181
- *
182
- * @param {string} provider
183
- * @returns {number} Segundos restantes (0 si no está throttled)
184
- */
185
- tiempoEspera(provider) {
186
- const estado = _obtenerEstado(provider);
187
- const restante = estado.esperaHastaMs - Date.now();
188
- return restante > 0 ? Math.ceil(restante / 1000) : 0;
189
- },
190
-
191
- /**
192
- * Retorna el estado completo de un provider.
193
- *
194
- * @param {string} provider
195
- * @returns {ProviderState}
196
- */
197
- estado(provider) {
198
- return { ..._obtenerEstado(provider) };
199
- },
200
-
201
- /**
202
- * Retorna resumen de todos los providers trackeados.
203
- *
204
- * @returns {Array<{ provider: string, estado: string, consecutivos429: number, totalHistorico: number, tiempoEspera: number }>}
205
- */
206
- resumen() {
207
- const resultado = [];
208
- for (const [provider, estado] of _providers) {
209
- resultado.push({
210
- provider,
211
- estado: estado.estado,
212
- consecutivos429: estado.consecutivos429,
213
- totalHistorico: estado.totalHistorico,
214
- tiempoEspera: tracker.tiempoEspera(provider),
215
- modelo: estado.modelo,
216
- });
217
- }
218
- return resultado;
219
- },
220
-
221
- /**
222
- * Limpia registros stale (sin actividad reciente).
223
- * Llamar periódicamente para evitar memory leak en sesiones largas.
224
- */
225
- limpiarStale() {
226
- const ahora = Date.now();
227
- for (const [provider, estado] of _providers) {
228
- if (estado.estado === 'ok' && (ahora - estado.ultimoTimestamp) > STALE_MS) {
229
- _providers.delete(provider);
230
- }
231
- }
232
- },
233
-
234
- /**
235
- * Resetea todo el tracker (útil para tests).
236
- */
237
- reset() {
238
- _providers.clear();
239
- },
240
- };
241
-
242
- // ---------------------------------------------------------------------------
243
- // Exports
244
- // ---------------------------------------------------------------------------
245
-
246
- module.exports = {
247
- tracker,
248
- // Exponer constantes para tests
249
- MAX_429_CONSECUTIVOS,
250
- BACKOFF_BASE_MS,
251
- BACKOFF_MAX_MS,
252
- HALF_OPEN_MS,
253
- };
1
+ 'use strict';
2
+
3
+ /**
4
+ * Rate Limit Tracker — Awareness de límites de tasa per-provider.
5
+ *
6
+ * Patrón adoptado de Hermes Agent (agent/rate_limit_tracker.py).
7
+ * Registra respuestas 429 y headers de rate limit para ajustar
8
+ * timing de llamadas y prevenir cascadas de error.
9
+ *
10
+ * Zero dependencias externas.
11
+ *
12
+ * Uso:
13
+ * const { tracker } = require('./lib/rate-limit-tracker');
14
+ *
15
+ * // Registrar un 429
16
+ * tracker.registrar429('anthropic', { retryAfter: 30, modelo: 'claude-sonnet-4-6' });
17
+ *
18
+ * // Consultar antes de llamar
19
+ * if (tracker.estaThrottled('anthropic')) {
20
+ * const espera = tracker.tiempoEspera('anthropic');
21
+ * console.log(`Provider throttled, esperar ${espera}s`);
22
+ * }
23
+ *
24
+ * // Registrar éxito (resetea contadores)
25
+ * tracker.registrarExito('anthropic');
26
+ *
27
+ * @module hooks/lib/rate-limit-tracker
28
+ */
29
+
30
+ // ---------------------------------------------------------------------------
31
+ // Constantes
32
+ // ---------------------------------------------------------------------------
33
+
34
+ /** Máximo de 429s consecutivos antes de marcar provider como BLOQUEADO. */
35
+ const MAX_429_CONSECUTIVOS = 5;
36
+
37
+ /** Tiempo base de backoff en ms tras un 429 (si no hay Retry-After). */
38
+ const BACKOFF_BASE_MS = 10_000;
39
+
40
+ /** Multiplicador de backoff exponencial por 429 consecutivo. */
41
+ const BACKOFF_MULTIPLICADOR = 2;
42
+
43
+ /** Tiempo máximo de backoff en ms (5 minutos). */
44
+ const BACKOFF_MAX_MS = 300_000;
45
+
46
+ /** Tiempo de recuperación half-open en ms (1 minuto). */
47
+ const HALF_OPEN_MS = 60_000;
48
+
49
+ /** Tiempo para considerar un registro como stale y limpiar (30 minutos). */
50
+ const STALE_MS = 1_800_000;
51
+
52
+ // ---------------------------------------------------------------------------
53
+ // Estado del tracker (en memoria, por sesión)
54
+ // ---------------------------------------------------------------------------
55
+
56
+ /**
57
+ * @typedef {object} ProviderState
58
+ * @property {number} consecutivos429 - Conteo de 429s consecutivos
59
+ * @property {number} ultimoTimestamp - Timestamp del último 429
60
+ * @property {number} esperaHastaMs - Timestamp hasta el cual esperar
61
+ * @property {string} estado - 'ok' | 'throttled' | 'bloqueado' | 'half-open'
62
+ * @property {string|null} modelo - Último modelo que causó 429
63
+ * @property {number} totalHistorico - Total de 429s en la sesión
64
+ */
65
+
66
+ /** @type {Map<string, ProviderState>} */
67
+ const _providers = new Map();
68
+
69
+ // ---------------------------------------------------------------------------
70
+ // Funciones internas
71
+ // ---------------------------------------------------------------------------
72
+
73
+ /**
74
+ * Obtiene o crea el estado de un provider.
75
+ * @param {string} provider
76
+ * @returns {ProviderState}
77
+ */
78
+ function _obtenerEstado(provider) {
79
+ if (!_providers.has(provider)) {
80
+ _providers.set(provider, {
81
+ consecutivos429: 0,
82
+ ultimoTimestamp: 0,
83
+ esperaHastaMs: 0,
84
+ estado: 'ok',
85
+ modelo: null,
86
+ totalHistorico: 0,
87
+ });
88
+ }
89
+ return _providers.get(provider);
90
+ }
91
+
92
+ /**
93
+ * Calcula el delay de backoff para un número de 429s consecutivos.
94
+ * @param {number} consecutivos
95
+ * @param {number|null} retryAfterSeg - Valor de Retry-After del header (en segundos)
96
+ * @returns {number} Delay en milisegundos
97
+ */
98
+ function _calcularBackoff(consecutivos, retryAfterSeg) {
99
+ if (retryAfterSeg && retryAfterSeg > 0) {
100
+ return retryAfterSeg * 1000;
101
+ }
102
+ const delay = BACKOFF_BASE_MS * Math.pow(BACKOFF_MULTIPLICADOR, consecutivos - 1);
103
+ return Math.min(delay, BACKOFF_MAX_MS);
104
+ }
105
+
106
+ // ---------------------------------------------------------------------------
107
+ // API pública
108
+ // ---------------------------------------------------------------------------
109
+
110
+ const tracker = {
111
+ /**
112
+ * Registra una respuesta 429 de un provider.
113
+ *
114
+ * @param {string} provider - Nombre del provider ('anthropic', 'openai', etc.)
115
+ * @param {object} [info]
116
+ * @param {number} [info.retryAfter] - Valor del header Retry-After en segundos
117
+ * @param {string} [info.modelo] - Modelo que causó el 429
118
+ */
119
+ registrar429(provider, info = {}) {
120
+ const estado = _obtenerEstado(provider);
121
+ const ahora = Date.now();
122
+
123
+ estado.consecutivos429 += 1;
124
+ estado.ultimoTimestamp = ahora;
125
+ estado.totalHistorico += 1;
126
+ estado.modelo = info.modelo || estado.modelo;
127
+
128
+ const backoff = _calcularBackoff(estado.consecutivos429, info.retryAfter);
129
+ estado.esperaHastaMs = ahora + backoff;
130
+
131
+ if (estado.consecutivos429 >= MAX_429_CONSECUTIVOS) {
132
+ estado.estado = 'bloqueado';
133
+ } else {
134
+ estado.estado = 'throttled';
135
+ }
136
+ },
137
+
138
+ /**
139
+ * Registra una llamada exitosa — resetea contadores del provider.
140
+ *
141
+ * @param {string} provider
142
+ */
143
+ registrarExito(provider) {
144
+ const estado = _obtenerEstado(provider);
145
+ estado.consecutivos429 = 0;
146
+ estado.esperaHastaMs = 0;
147
+ estado.estado = 'ok';
148
+ },
149
+
150
+ /**
151
+ * Verifica si un provider está throttled o bloqueado.
152
+ *
153
+ * @param {string} provider
154
+ * @returns {boolean}
155
+ */
156
+ estaThrottled(provider) {
157
+ const estado = _obtenerEstado(provider);
158
+ const ahora = Date.now();
159
+
160
+ if (estado.estado === 'ok') return false;
161
+
162
+ // Verificar si ya pasó el tiempo de espera
163
+ if (ahora >= estado.esperaHastaMs) {
164
+ if (estado.estado === 'bloqueado') {
165
+ // Transición a half-open para permitir un intento de prueba
166
+ estado.estado = 'half-open';
167
+ estado.esperaHastaMs = ahora + HALF_OPEN_MS;
168
+ return false; // permitir un intento
169
+ }
170
+ // Throttled normal — ya pasó el backoff
171
+ estado.estado = 'ok';
172
+ estado.consecutivos429 = 0;
173
+ return false;
174
+ }
175
+
176
+ return true;
177
+ },
178
+
179
+ /**
180
+ * Retorna el tiempo de espera restante en segundos.
181
+ *
182
+ * @param {string} provider
183
+ * @returns {number} Segundos restantes (0 si no está throttled)
184
+ */
185
+ tiempoEspera(provider) {
186
+ const estado = _obtenerEstado(provider);
187
+ const restante = estado.esperaHastaMs - Date.now();
188
+ return restante > 0 ? Math.ceil(restante / 1000) : 0;
189
+ },
190
+
191
+ /**
192
+ * Retorna el estado completo de un provider.
193
+ *
194
+ * @param {string} provider
195
+ * @returns {ProviderState}
196
+ */
197
+ estado(provider) {
198
+ return { ..._obtenerEstado(provider) };
199
+ },
200
+
201
+ /**
202
+ * Retorna resumen de todos los providers trackeados.
203
+ *
204
+ * @returns {Array<{ provider: string, estado: string, consecutivos429: number, totalHistorico: number, tiempoEspera: number }>}
205
+ */
206
+ resumen() {
207
+ const resultado = [];
208
+ for (const [provider, estado] of _providers) {
209
+ resultado.push({
210
+ provider,
211
+ estado: estado.estado,
212
+ consecutivos429: estado.consecutivos429,
213
+ totalHistorico: estado.totalHistorico,
214
+ tiempoEspera: tracker.tiempoEspera(provider),
215
+ modelo: estado.modelo,
216
+ });
217
+ }
218
+ return resultado;
219
+ },
220
+
221
+ /**
222
+ * Limpia registros stale (sin actividad reciente).
223
+ * Llamar periódicamente para evitar memory leak en sesiones largas.
224
+ */
225
+ limpiarStale() {
226
+ const ahora = Date.now();
227
+ for (const [provider, estado] of _providers) {
228
+ if (estado.estado === 'ok' && (ahora - estado.ultimoTimestamp) > STALE_MS) {
229
+ _providers.delete(provider);
230
+ }
231
+ }
232
+ },
233
+
234
+ /**
235
+ * Resetea todo el tracker (útil para tests).
236
+ */
237
+ reset() {
238
+ _providers.clear();
239
+ },
240
+ };
241
+
242
+ // ---------------------------------------------------------------------------
243
+ // Exports
244
+ // ---------------------------------------------------------------------------
245
+
246
+ module.exports = {
247
+ tracker,
248
+ // Exponer constantes para tests
249
+ MAX_429_CONSECUTIVOS,
250
+ BACKOFF_BASE_MS,
251
+ BACKOFF_MAX_MS,
252
+ HALF_OPEN_MS,
253
+ };