@saulwade/swl-ses 1.3.7 → 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.
- package/CLAUDE.md +12 -4
- package/README.md +1 -1
- package/bin/swl-mcp-server.js +187 -187
- package/bin/swl-webhook-server.js +198 -0
- package/comandos/swl/.evolved.json +22 -22
- package/comandos/swl/adoptar-proyecto.md +21 -1
- package/comandos/swl/claudemd.md +14 -1
- package/comandos/swl/contribuir.md +233 -233
- package/comandos/swl/exportar-vault.md +207 -7
- package/comandos/swl/nuevo-proyecto.md +24 -2
- package/gateway/adapters/base.js +109 -0
- package/gateway/adapters/discord.js +167 -0
- package/gateway/adapters/email.js +221 -0
- package/gateway/adapters/slack.js +192 -0
- package/gateway/adapters/telegram.js +183 -0
- package/gateway/adapters/webhook.js +113 -0
- package/gateway/adapters/whatsapp.js +214 -0
- package/gateway/agent-executor.js +322 -0
- package/gateway/command-relay.js +271 -0
- package/gateway/cron/jobs.js +263 -0
- package/gateway/cron/scheduler.js +322 -0
- package/gateway/cron/store.js +335 -0
- package/gateway/index.js +320 -0
- package/gateway/lib/event-channel.js +191 -0
- package/gateway/session.js +131 -0
- package/gateway/webhook-server.js +324 -0
- package/habilidades/backend-production-resilience/SKILL.md +288 -288
- package/habilidades/benchmark-memoria/SKILL.md +186 -186
- package/habilidades/build-errors-nextjs/SKILL.md +55 -1
- package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
- package/habilidades/doubt-driven-review/SKILL.md +171 -171
- package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
- package/habilidades/eval-framework/SKILL.md +212 -212
- package/habilidades/extractor-de-aprendizajes/SKILL.md +24 -10
- package/habilidades/harness-claude-code/SKILL.md +299 -299
- package/habilidades/infra-github-actions/SKILL.md +166 -166
- package/habilidades/legacy-code-rescue/SKILL.md +267 -267
- package/habilidades/manejo-errores/.evolved.json +8 -8
- package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
- package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
- package/habilidades/nextjs-testing/SKILL.md +89 -5
- package/habilidades/node-experto/SKILL.md +37 -1
- package/habilidades/patrones-python/SKILL.md +229 -229
- package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
- package/habilidades/planear-fase/SKILL.md +319 -319
- package/habilidades/react-experto/SKILL.md +45 -4
- package/habilidades/release-semver/.evolved.json +8 -8
- package/habilidades/swl-claudemd/SKILL.md +15 -1
- package/habilidades/tdd-workflow/SKILL.md +36 -4
- package/habilidades/testing-python/SKILL.md +340 -340
- package/hooks/claudemd-bloat-detector.js +161 -161
- package/hooks/inyeccion-contexto.js +8 -3
- package/hooks/lib/agent-routing.js +107 -107
- package/hooks/lib/auto-consolidator.js +335 -335
- package/hooks/lib/error-classifier.js +308 -308
- package/hooks/lib/merkle-audit.js +96 -96
- package/hooks/lib/provenance-tracker.js +191 -191
- package/hooks/lib/rate-limit-ip.js +177 -0
- package/hooks/lib/rate-limit-tracker.js +253 -253
- package/hooks/lib/resource-quota.js +122 -122
- package/hooks/lib/retry-jitter.js +165 -165
- package/hooks/lib/skill-auditor.js +588 -588
- package/hooks/lib/sync-status.js +228 -228
- package/hooks/lib/taint-tracker.js +107 -107
- package/hooks/lib/text-similarity.js +241 -241
- package/hooks/lib/toon-compressor.js +245 -245
- package/hooks/lib/webhook-dedup.js +184 -0
- package/hooks/lib/webhook-verify.js +123 -0
- package/hooks/proteccion-rutas.js +120 -15
- package/hooks/registro-turnos.js +209 -209
- package/hooks/sugerir-regenerar-inventario.js +170 -170
- package/hooks/validar-formato-post-subagente.js +140 -140
- package/hooks/validar-memoria-hook.js +218 -218
- package/instintos/prompt-appendices.yaml +57 -57
- package/manifiestos/agent-output-schemas.json +57 -57
- package/manifiestos/modulos.json +1 -0
- package/manifiestos/skills-lock.json +37 -37
- package/package.json +5 -3
- package/plantillas/auditor-veto-template.md +105 -105
- package/plantillas/github-workflows/README.md +47 -47
- package/plantillas/github-workflows/release-please.yml +44 -44
- package/plantillas/github-workflows/swl-ci.yml +107 -107
- package/plantillas/github-workflows/swl-security.yml +51 -51
- package/plugin.json +1 -1
- package/reglas/analisis-previo-tareas-grandes.md +172 -172
- package/reglas/arreglar-al-detectar.md +147 -147
- package/reglas/fragmentos-compartidos.md +152 -152
- package/reglas/harness-claude-code.md +213 -213
- package/reglas/usar-context7.md +226 -226
- package/reglas/usar-sistema-swl.md +251 -0
- package/schemas/diary-entry.schema.json +80 -80
- package/scripts/benchmark-memoria.js +167 -167
- package/scripts/comandos/skills.js +251 -2
- package/scripts/configurar-branch-protection.js +418 -418
- package/scripts/detectar-aprendizajes-duplicados.js +151 -151
- package/scripts/field-report.js +199 -199
- package/scripts/generar-checklists-consolidados.js +273 -273
- package/scripts/generar-inventario.js +420 -420
- package/scripts/generar-matriz-lenguajes.js +271 -271
- package/scripts/lib/artefactos-python.js +43 -43
- package/scripts/lib/benchmark-metrics.js +160 -160
- package/scripts/lib/budget-enforcer.js +252 -252
- package/scripts/lib/configurar-ci.js +380 -380
- package/scripts/lib/contadores-inventario.js +217 -217
- package/scripts/lib/detectar-stack-detallado.js +307 -307
- package/scripts/lib/diary-entry.js +234 -234
- package/scripts/lib/eval-metrics-store.js +218 -218
- package/scripts/lib/eval-quality.js +171 -171
- package/scripts/lib/eval-schemas.js +144 -144
- package/scripts/lib/eval-self-correct.js +106 -106
- package/scripts/lib/eval-validator.js +185 -185
- package/scripts/lib/jaccard-similarity.js +98 -98
- package/scripts/lib/longmemeval-runner.js +125 -125
- package/scripts/lib/npm-version.js +261 -261
- package/scripts/lib/paquetes-conocidos.js +50 -50
- package/scripts/lib/prompt-builder.js +264 -264
- package/scripts/lib/rrf-fusion.js +175 -175
- package/scripts/lib/scoring-instintos.js +277 -277
- package/scripts/lib/semantic-search.js +252 -252
- package/scripts/limpiar-artefactos-python.js +131 -131
- package/scripts/mcp-server/README.md +128 -128
- package/scripts/mcp-server/handlers.js +206 -206
- package/scripts/migrar-csv-a-array.js +168 -168
- package/scripts/migrar-fase-dominio.js +201 -201
- package/scripts/publicar.js +511 -511
- package/scripts/run-eval.js +141 -141
- package/scripts/validar-manifest.js +195 -195
- package/scripts/validar-userland-vacio.js +110 -110
- 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
|
+
};
|