milens 0.6.2 → 0.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.
Files changed (96) hide show
  1. package/README.md +157 -14
  2. package/dist/analyzer/engine.d.ts +1 -0
  3. package/dist/analyzer/engine.d.ts.map +1 -1
  4. package/dist/analyzer/engine.js +27 -8
  5. package/dist/analyzer/engine.js.map +1 -1
  6. package/dist/analyzer/review.d.ts +23 -0
  7. package/dist/analyzer/review.d.ts.map +1 -0
  8. package/dist/analyzer/review.js +143 -0
  9. package/dist/analyzer/review.js.map +1 -0
  10. package/dist/analyzer/testplan.d.ts +59 -0
  11. package/dist/analyzer/testplan.d.ts.map +1 -0
  12. package/dist/analyzer/testplan.js +218 -0
  13. package/dist/analyzer/testplan.js.map +1 -0
  14. package/dist/cli.js +2 -0
  15. package/dist/cli.js.map +1 -1
  16. package/dist/gateway/analyzer.d.ts +6 -0
  17. package/dist/gateway/analyzer.d.ts.map +1 -0
  18. package/dist/gateway/analyzer.js +218 -0
  19. package/dist/gateway/analyzer.js.map +1 -0
  20. package/dist/gateway/cache.d.ts +35 -0
  21. package/dist/gateway/cache.d.ts.map +1 -0
  22. package/dist/gateway/cache.js +175 -0
  23. package/dist/gateway/cache.js.map +1 -0
  24. package/dist/gateway/config.d.ts +10 -0
  25. package/dist/gateway/config.d.ts.map +1 -0
  26. package/dist/gateway/config.js +167 -0
  27. package/dist/gateway/config.js.map +1 -0
  28. package/dist/gateway/context-memory.d.ts +68 -0
  29. package/dist/gateway/context-memory.d.ts.map +1 -0
  30. package/dist/gateway/context-memory.js +157 -0
  31. package/dist/gateway/context-memory.js.map +1 -0
  32. package/dist/gateway/observability.d.ts +83 -0
  33. package/dist/gateway/observability.d.ts.map +1 -0
  34. package/dist/gateway/observability.js +152 -0
  35. package/dist/gateway/observability.js.map +1 -0
  36. package/dist/gateway/privacy.d.ts +27 -0
  37. package/dist/gateway/privacy.d.ts.map +1 -0
  38. package/dist/gateway/privacy.js +139 -0
  39. package/dist/gateway/privacy.js.map +1 -0
  40. package/dist/gateway/providers.d.ts +66 -0
  41. package/dist/gateway/providers.d.ts.map +1 -0
  42. package/dist/gateway/providers.js +377 -0
  43. package/dist/gateway/providers.js.map +1 -0
  44. package/dist/gateway/router.d.ts +18 -0
  45. package/dist/gateway/router.d.ts.map +1 -0
  46. package/dist/gateway/router.js +102 -0
  47. package/dist/gateway/router.js.map +1 -0
  48. package/dist/gateway/server.d.ts +20 -0
  49. package/dist/gateway/server.d.ts.map +1 -0
  50. package/dist/gateway/server.js +387 -0
  51. package/dist/gateway/server.js.map +1 -0
  52. package/dist/gateway/translator.d.ts +19 -0
  53. package/dist/gateway/translator.d.ts.map +1 -0
  54. package/dist/gateway/translator.js +340 -0
  55. package/dist/gateway/translator.js.map +1 -0
  56. package/dist/gateway/types.d.ts +215 -0
  57. package/dist/gateway/types.d.ts.map +1 -0
  58. package/dist/gateway/types.js +3 -0
  59. package/dist/gateway/types.js.map +1 -0
  60. package/dist/parser/extract.d.ts +1 -0
  61. package/dist/parser/extract.d.ts.map +1 -1
  62. package/dist/parser/extract.js +8 -0
  63. package/dist/parser/extract.js.map +1 -1
  64. package/dist/parser/lang-go.d.ts.map +1 -1
  65. package/dist/parser/lang-go.js +41 -5
  66. package/dist/parser/lang-go.js.map +1 -1
  67. package/dist/parser/lang-java.d.ts.map +1 -1
  68. package/dist/parser/lang-java.js +1 -0
  69. package/dist/parser/lang-java.js.map +1 -1
  70. package/dist/parser/lang-py.d.ts.map +1 -1
  71. package/dist/parser/lang-py.js +22 -0
  72. package/dist/parser/lang-py.js.map +1 -1
  73. package/dist/parser/lang-ruby.d.ts.map +1 -1
  74. package/dist/parser/lang-ruby.js +1 -0
  75. package/dist/parser/lang-ruby.js.map +1 -1
  76. package/dist/server/mcp.d.ts.map +1 -1
  77. package/dist/server/mcp.js +615 -106
  78. package/dist/server/mcp.js.map +1 -1
  79. package/dist/skills.js +32 -0
  80. package/dist/skills.js.map +1 -1
  81. package/dist/store/db.d.ts +44 -0
  82. package/dist/store/db.d.ts.map +1 -1
  83. package/dist/store/db.js +142 -25
  84. package/dist/store/db.js.map +1 -1
  85. package/dist/store/gateway-schema.sql +53 -0
  86. package/dist/store/schema.sql +33 -0
  87. package/dist/store/vectors.d.ts +65 -0
  88. package/dist/store/vectors.d.ts.map +1 -0
  89. package/dist/store/vectors.js +212 -0
  90. package/dist/store/vectors.js.map +1 -0
  91. package/dist/utils.d.ts +3 -0
  92. package/dist/utils.d.ts.map +1 -0
  93. package/dist/utils.js +9 -0
  94. package/dist/utils.js.map +1 -0
  95. package/docs/diagram2.svg +1 -1
  96. package/package.json +2 -1
@@ -0,0 +1,377 @@
1
+ // ── Provider registry + API adapters ──
2
+ // Native fetch + ReadableStream SSE — no SDK dependencies
3
+ import { toOpenAIRequest, toAnthropicRequest, fromAnthropicResponse, fromAnthropicStreamEvent, toGeminiRequest, fromGeminiResponse, fromGeminiStreamChunk, } from './translator.js';
4
+ import { randomUUID } from 'node:crypto';
5
+ export class BaseAdapter {
6
+ config;
7
+ constructor(config) {
8
+ this.config = config;
9
+ }
10
+ async fetchWithTimeout(url, init) {
11
+ const timeout = this.config.timeoutMs ?? 120_000;
12
+ const controller = new AbortController();
13
+ const timer = setTimeout(() => controller.abort(), timeout);
14
+ try {
15
+ const res = await fetch(url, { ...init, signal: controller.signal });
16
+ return res;
17
+ }
18
+ finally {
19
+ clearTimeout(timer);
20
+ }
21
+ }
22
+ makeSSEChunk(delta, model) {
23
+ return {
24
+ id: `chatcmpl-${randomUUID().slice(0, 8)}`,
25
+ object: 'chat.completion.chunk',
26
+ created: Math.floor(Date.now() / 1000),
27
+ model,
28
+ choices: [delta],
29
+ };
30
+ }
31
+ }
32
+ // ── OpenAI-compatible adapter ──
33
+ // Works for: OpenAI, Groq, Together, GLM, MiniMax, DeepSeek, Ollama, vLLM, etc.
34
+ export class OpenAIAdapter extends BaseAdapter {
35
+ async chat(req) {
36
+ const start = Date.now();
37
+ const body = toOpenAIRequest({ ...req, stream: false });
38
+ const url = `${this.config.endpoint}/chat/completions`;
39
+ const res = await this.fetchWithTimeout(url, {
40
+ method: 'POST',
41
+ headers: this.headers(),
42
+ body: JSON.stringify(body),
43
+ });
44
+ if (!res.ok) {
45
+ const text = await res.text().catch(() => '');
46
+ throw new Error(`OpenAI API error ${res.status}: ${text.slice(0, 300)}`);
47
+ }
48
+ const data = await res.json();
49
+ return { response: data, durationMs: Date.now() - start };
50
+ }
51
+ async *chatStream(req) {
52
+ const body = toOpenAIRequest({ ...req, stream: true });
53
+ const url = `${this.config.endpoint}/chat/completions`;
54
+ const res = await this.fetchWithTimeout(url, {
55
+ method: 'POST',
56
+ headers: this.headers(),
57
+ body: JSON.stringify(body),
58
+ });
59
+ if (!res.ok) {
60
+ const text = await res.text().catch(() => '');
61
+ throw new Error(`OpenAI API error ${res.status}: ${text.slice(0, 300)}`);
62
+ }
63
+ yield* this.parseSSEStream(res, req.model);
64
+ }
65
+ headers() {
66
+ return {
67
+ 'Content-Type': 'application/json',
68
+ 'Authorization': `Bearer ${this.config.apiKey}`,
69
+ ...this.config.headers,
70
+ };
71
+ }
72
+ async *parseSSEStream(res, model) {
73
+ const reader = res.body?.getReader();
74
+ if (!reader)
75
+ return;
76
+ const decoder = new TextDecoder();
77
+ let buffer = '';
78
+ try {
79
+ while (true) {
80
+ const { done, value } = await reader.read();
81
+ if (done)
82
+ break;
83
+ buffer += decoder.decode(value, { stream: true });
84
+ const lines = buffer.split('\n');
85
+ buffer = lines.pop() ?? '';
86
+ for (const line of lines) {
87
+ const trimmed = line.trim();
88
+ if (!trimmed || trimmed === 'data: [DONE]')
89
+ continue;
90
+ if (!trimmed.startsWith('data: '))
91
+ continue;
92
+ try {
93
+ const chunk = JSON.parse(trimmed.slice(6));
94
+ yield { chunk, raw: trimmed };
95
+ }
96
+ catch { /* skip malformed chunks */ }
97
+ }
98
+ }
99
+ }
100
+ finally {
101
+ reader.releaseLock();
102
+ }
103
+ }
104
+ }
105
+ // ── Anthropic adapter ──
106
+ export class AnthropicAdapter extends BaseAdapter {
107
+ async chat(req) {
108
+ const start = Date.now();
109
+ const body = toAnthropicRequest({ ...req, stream: false });
110
+ const url = `${this.config.endpoint}/messages`;
111
+ const res = await this.fetchWithTimeout(url, {
112
+ method: 'POST',
113
+ headers: this.headers(),
114
+ body: JSON.stringify(body),
115
+ });
116
+ if (!res.ok) {
117
+ const text = await res.text().catch(() => '');
118
+ throw new Error(`Anthropic API error ${res.status}: ${text.slice(0, 300)}`);
119
+ }
120
+ const data = await res.json();
121
+ const parsed = fromAnthropicResponse(data);
122
+ const response = {
123
+ id: data.id ?? `msg_${randomUUID().slice(0, 8)}`,
124
+ object: 'chat.completion',
125
+ created: Math.floor(Date.now() / 1000),
126
+ model: parsed.model || req.model,
127
+ choices: [{ index: 0, message: parsed.message, finish_reason: parsed.finishReason }],
128
+ usage: parsed.usage,
129
+ };
130
+ return { response, durationMs: Date.now() - start };
131
+ }
132
+ async *chatStream(req) {
133
+ const body = toAnthropicRequest({ ...req, stream: true });
134
+ const url = `${this.config.endpoint}/messages`;
135
+ const res = await this.fetchWithTimeout(url, {
136
+ method: 'POST',
137
+ headers: this.headers(),
138
+ body: JSON.stringify(body),
139
+ });
140
+ if (!res.ok) {
141
+ const text = await res.text().catch(() => '');
142
+ throw new Error(`Anthropic API error ${res.status}: ${text.slice(0, 300)}`);
143
+ }
144
+ const reader = res.body?.getReader();
145
+ if (!reader)
146
+ return;
147
+ const decoder = new TextDecoder();
148
+ let buffer = '';
149
+ try {
150
+ while (true) {
151
+ const { done, value } = await reader.read();
152
+ if (done)
153
+ break;
154
+ buffer += decoder.decode(value, { stream: true });
155
+ const lines = buffer.split('\n');
156
+ buffer = lines.pop() ?? '';
157
+ for (const line of lines) {
158
+ const trimmed = line.trim();
159
+ if (!trimmed.startsWith('data: '))
160
+ continue;
161
+ try {
162
+ const event = JSON.parse(trimmed.slice(6));
163
+ const delta = fromAnthropicStreamEvent(event);
164
+ if (delta) {
165
+ yield { chunk: this.makeSSEChunk(delta, req.model) };
166
+ }
167
+ }
168
+ catch { /* skip */ }
169
+ }
170
+ }
171
+ }
172
+ finally {
173
+ reader.releaseLock();
174
+ }
175
+ }
176
+ headers() {
177
+ return {
178
+ 'Content-Type': 'application/json',
179
+ 'x-api-key': this.config.apiKey,
180
+ 'anthropic-version': '2023-06-01',
181
+ ...this.config.headers,
182
+ };
183
+ }
184
+ }
185
+ // ── Gemini adapter ──
186
+ export class GeminiAdapter extends BaseAdapter {
187
+ async chat(req) {
188
+ const start = Date.now();
189
+ const body = toGeminiRequest({ ...req, stream: false });
190
+ const url = this.buildUrl(req.model, false);
191
+ const res = await this.fetchWithTimeout(url, {
192
+ method: 'POST',
193
+ headers: { 'Content-Type': 'application/json' },
194
+ body: JSON.stringify(body),
195
+ });
196
+ if (!res.ok) {
197
+ const text = await res.text().catch(() => '');
198
+ throw new Error(`Gemini API error ${res.status}: ${text.slice(0, 300)}`);
199
+ }
200
+ const data = await res.json();
201
+ const parsed = fromGeminiResponse(data);
202
+ const response = {
203
+ id: `gemini-${randomUUID().slice(0, 8)}`,
204
+ object: 'chat.completion',
205
+ created: Math.floor(Date.now() / 1000),
206
+ model: parsed.model || req.model,
207
+ choices: [{ index: 0, message: parsed.message, finish_reason: parsed.finishReason }],
208
+ usage: parsed.usage,
209
+ };
210
+ return { response, durationMs: Date.now() - start };
211
+ }
212
+ async *chatStream(req) {
213
+ const body = toGeminiRequest({ ...req, stream: true });
214
+ const url = this.buildUrl(req.model, true);
215
+ const res = await this.fetchWithTimeout(url, {
216
+ method: 'POST',
217
+ headers: { 'Content-Type': 'application/json' },
218
+ body: JSON.stringify(body),
219
+ });
220
+ if (!res.ok) {
221
+ const text = await res.text().catch(() => '');
222
+ throw new Error(`Gemini API error ${res.status}: ${text.slice(0, 300)}`);
223
+ }
224
+ const reader = res.body?.getReader();
225
+ if (!reader)
226
+ return;
227
+ const decoder = new TextDecoder();
228
+ let buffer = '';
229
+ try {
230
+ while (true) {
231
+ const { done, value } = await reader.read();
232
+ if (done)
233
+ break;
234
+ buffer += decoder.decode(value, { stream: true });
235
+ // Gemini streams as JSON array chunks or newline-separated JSON
236
+ const lines = buffer.split('\n');
237
+ buffer = lines.pop() ?? '';
238
+ for (const line of lines) {
239
+ const trimmed = line.trim().replace(/^,/, '').replace(/^\[/, '').replace(/\]$/, '');
240
+ if (!trimmed || trimmed === '[' || trimmed === ']')
241
+ continue;
242
+ try {
243
+ const data = JSON.parse(trimmed);
244
+ const delta = fromGeminiStreamChunk(data);
245
+ if (delta) {
246
+ yield { chunk: this.makeSSEChunk(delta, req.model) };
247
+ }
248
+ }
249
+ catch { /* skip */ }
250
+ }
251
+ }
252
+ }
253
+ finally {
254
+ reader.releaseLock();
255
+ }
256
+ }
257
+ buildUrl(model, stream) {
258
+ const base = this.config.endpoint || 'https://generativelanguage.googleapis.com/v1beta';
259
+ const action = stream ? 'streamGenerateContent?alt=sse' : 'generateContent';
260
+ return `${base}/models/${model}:${action}&key=${this.config.apiKey}`;
261
+ }
262
+ }
263
+ // ── Provider Registry ──
264
+ export class ProviderRegistry {
265
+ providers = new Map();
266
+ adapters = new Map();
267
+ roundRobinIndex = new Map();
268
+ addProvider(config) {
269
+ this.providers.set(config.id, {
270
+ config,
271
+ status: 'healthy',
272
+ consecutiveFailures: 0,
273
+ lastUsed: 0,
274
+ totalRequests: 0,
275
+ totalTokens: 0,
276
+ });
277
+ this.adapters.set(config.id, createAdapter(config));
278
+ }
279
+ removeProvider(id) {
280
+ this.providers.delete(id);
281
+ this.adapters.delete(id);
282
+ }
283
+ getAdapter(id) {
284
+ return this.adapters.get(id);
285
+ }
286
+ getState(id) {
287
+ return this.providers.get(id);
288
+ }
289
+ /** Get all providers for a tier, ordered by round-robin */
290
+ getByTier(tier) {
291
+ const matching = Array.from(this.providers.values())
292
+ .filter(p => p.config.tier === tier && p.status !== 'down')
293
+ .sort((a, b) => a.lastUsed - b.lastUsed)
294
+ .map(p => p.config);
295
+ return matching;
296
+ }
297
+ /** Build a fallback chain: requested tier → lower tiers */
298
+ buildFallbackChain(startTier) {
299
+ const tierOrder = ['premium', 'standard', 'economy', 'local'];
300
+ const startIdx = tierOrder.indexOf(startTier);
301
+ const chain = [];
302
+ for (let i = startIdx; i < tierOrder.length; i++) {
303
+ chain.push(...this.getByTier(tierOrder[i]));
304
+ }
305
+ return chain;
306
+ }
307
+ /** Mark a provider as used, update stats */
308
+ recordUse(id, tokens) {
309
+ const state = this.providers.get(id);
310
+ if (!state)
311
+ return;
312
+ state.lastUsed = Date.now();
313
+ state.totalRequests++;
314
+ state.totalTokens += tokens;
315
+ state.consecutiveFailures = 0;
316
+ state.status = 'healthy';
317
+ }
318
+ /** Mark a provider failure */
319
+ recordFailure(id, error) {
320
+ const state = this.providers.get(id);
321
+ if (!state)
322
+ return;
323
+ state.consecutiveFailures++;
324
+ state.lastError = error;
325
+ if (state.consecutiveFailures >= (state.config.maxRetries ?? 3)) {
326
+ state.status = 'down';
327
+ // Auto-recover after 5 minutes
328
+ setTimeout(() => {
329
+ const s = this.providers.get(id);
330
+ if (s && s.status === 'down') {
331
+ s.status = 'degraded';
332
+ s.consecutiveFailures = 0;
333
+ }
334
+ }, 5 * 60_000);
335
+ }
336
+ else {
337
+ state.status = 'degraded';
338
+ }
339
+ }
340
+ /** List all model IDs across all providers */
341
+ listModels() {
342
+ const models = [];
343
+ for (const state of this.providers.values()) {
344
+ for (const model of state.config.models) {
345
+ models.push({ id: model, provider: state.config.id, tier: state.config.tier });
346
+ }
347
+ }
348
+ return models;
349
+ }
350
+ /** Find which provider has a specific model */
351
+ findByModel(model) {
352
+ for (const state of this.providers.values()) {
353
+ if (state.config.models.includes(model) && state.status !== 'down') {
354
+ return state.config;
355
+ }
356
+ }
357
+ return undefined;
358
+ }
359
+ /** Get health status for all providers */
360
+ getHealthStatus() {
361
+ return Array.from(this.providers.values()).map(p => ({
362
+ id: p.config.id,
363
+ status: p.status,
364
+ tier: p.config.tier,
365
+ requests: p.totalRequests,
366
+ }));
367
+ }
368
+ }
369
+ function createAdapter(config) {
370
+ switch (config.format) {
371
+ case 'anthropic': return new AnthropicAdapter(config);
372
+ case 'gemini': return new GeminiAdapter(config);
373
+ case 'openai':
374
+ default: return new OpenAIAdapter(config);
375
+ }
376
+ }
377
+ //# sourceMappingURL=providers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"providers.js","sourceRoot":"","sources":["../../src/gateway/providers.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { ChatRequest, RoutingDecision, RoutingConfig, MessageAnalysis, ContextInjectionConfig } from './types.js';
2
+ import type { Database } from '../store/db.js';
3
+ import { ProviderRegistry } from './providers.js';
4
+ export declare class CodeAwareRouter {
5
+ private db;
6
+ private providers;
7
+ private routingConfig;
8
+ private contextConfig;
9
+ constructor(db: Database | null, providers: ProviderRegistry, routingConfig: RoutingConfig, contextConfig: ContextInjectionConfig);
10
+ /** Analyze request and produce a routing decision */
11
+ route(req: ChatRequest): RoutingDecision & {
12
+ analysis: MessageAnalysis;
13
+ };
14
+ private complexityToTier;
15
+ private parseTierOverride;
16
+ private buildReason;
17
+ }
18
+ //# sourceMappingURL=router.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.d.ts","sourceRoot":"","sources":["../../src/gateway/router.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EACV,WAAW,EAAE,eAAe,EAAE,aAAa,EAC3C,eAAe,EAAiB,sBAAsB,EACvD,MAAM,YAAY,CAAC;AACpB,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAC/C,OAAO,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAGlD,qBAAa,eAAe;IAExB,OAAO,CAAC,EAAE;IACV,OAAO,CAAC,SAAS;IACjB,OAAO,CAAC,aAAa;IACrB,OAAO,CAAC,aAAa;gBAHb,EAAE,EAAE,QAAQ,GAAG,IAAI,EACnB,SAAS,EAAE,gBAAgB,EAC3B,aAAa,EAAE,aAAa,EAC5B,aAAa,EAAE,sBAAsB;IAG/C,qDAAqD;IACrD,KAAK,CAAC,GAAG,EAAE,WAAW,GAAG,eAAe,GAAG;QAAE,QAAQ,EAAE,eAAe,CAAA;KAAE;IAqDxE,OAAO,CAAC,gBAAgB;IAOxB,OAAO,CAAC,iBAAiB;IAgBzB,OAAO,CAAC,WAAW;CAwBpB"}
@@ -0,0 +1,102 @@
1
+ // ── Code-aware router — complexity-based tier selection with fallback ──
2
+ import { analyzeMessages, lookupSymbols, buildContextInjection } from './analyzer.js';
3
+ export class CodeAwareRouter {
4
+ db;
5
+ providers;
6
+ routingConfig;
7
+ contextConfig;
8
+ constructor(db, providers, routingConfig, contextConfig) {
9
+ this.db = db;
10
+ this.providers = providers;
11
+ this.routingConfig = routingConfig;
12
+ this.contextConfig = contextConfig;
13
+ }
14
+ /** Analyze request and produce a routing decision */
15
+ route(req) {
16
+ // 1. Analyze messages for code context
17
+ let analysis = analyzeMessages(req.messages);
18
+ // 2. Enrich with milens data (if DB available)
19
+ if (this.db && (analysis.symbolNames.length > 0 || analysis.files.length > 0)) {
20
+ analysis = lookupSymbols(analysis, this.db);
21
+ }
22
+ // 3. Check for user tier override via model prefix (@premium/, @local/, etc.)
23
+ const tierOverride = this.parseTierOverride(req);
24
+ // 4. Compute target tier
25
+ const tier = tierOverride ?? this.complexityToTier(analysis.complexity);
26
+ // 5. Select provider and build fallback chain
27
+ const chain = this.providers.buildFallbackChain(tier);
28
+ const primary = chain[0];
29
+ if (!primary) {
30
+ // No providers available — return a stub decision
31
+ return {
32
+ provider: { id: 'none', name: 'none', tier: 'economy', format: 'openai', endpoint: '', apiKey: '', models: [] },
33
+ tier,
34
+ reason: 'No providers available',
35
+ complexity: analysis.complexity,
36
+ intent: analysis.intent,
37
+ fallbackChain: [],
38
+ analysis,
39
+ };
40
+ }
41
+ // 6. Build context injection
42
+ let contextInjection;
43
+ if (this.contextConfig.enabled && this.db && analysis.symbols.length > 0) {
44
+ contextInjection = buildContextInjection(analysis, this.db, this.contextConfig.maxTokens);
45
+ }
46
+ // 7. Build reason string
47
+ const reason = this.buildReason(analysis, tier, tierOverride !== undefined);
48
+ return {
49
+ provider: primary,
50
+ tier,
51
+ reason,
52
+ complexity: analysis.complexity,
53
+ intent: analysis.intent,
54
+ fallbackChain: chain.slice(1),
55
+ contextInjection,
56
+ analysis,
57
+ };
58
+ }
59
+ complexityToTier(complexity) {
60
+ const { economy, standard } = this.routingConfig.thresholds;
61
+ if (complexity <= economy)
62
+ return 'economy';
63
+ if (complexity <= standard)
64
+ return 'standard';
65
+ return 'premium';
66
+ }
67
+ parseTierOverride(req) {
68
+ // Check _gateway metadata
69
+ if (req._gateway?.tier)
70
+ return req._gateway.tier;
71
+ // Check model prefix: @premium/gpt-4o, @local/qwen, etc.
72
+ const model = req.model;
73
+ const prefixMatch = model.match(/^@(premium|standard|economy|local)\//);
74
+ if (prefixMatch) {
75
+ // Strip prefix from model name
76
+ req.model = model.slice(prefixMatch[0].length);
77
+ return prefixMatch[1];
78
+ }
79
+ return undefined;
80
+ }
81
+ buildReason(analysis, tier, overridden) {
82
+ const parts = [];
83
+ if (overridden) {
84
+ parts.push('tier override');
85
+ }
86
+ else {
87
+ parts.push(`complexity=${analysis.complexity}`);
88
+ }
89
+ if (analysis.intent !== 'unknown') {
90
+ parts.push(`intent=${analysis.intent}`);
91
+ }
92
+ if (analysis.symbols.length > 0) {
93
+ const topSymbols = analysis.symbols.slice(0, 3).map(s => s.name);
94
+ parts.push(`symbols=[${topSymbols.join(', ')}]`);
95
+ }
96
+ if (analysis.files.length > 0) {
97
+ parts.push(`files=${analysis.files.length}`);
98
+ }
99
+ return `→ ${tier} (${parts.join(', ')})`;
100
+ }
101
+ }
102
+ //# sourceMappingURL=router.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"router.js","sourceRoot":"","sources":["../../src/gateway/router.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,20 @@
1
+ import { IncomingMessage, ServerResponse } from 'node:http';
2
+ import type { GatewayConfig } from './types.js';
3
+ import { ProviderRegistry } from './providers.js';
4
+ import type { CodeAwareRouter } from './router.js';
5
+ import type { SemanticCache } from './cache.js';
6
+ import type { PrivacyFilter } from './privacy.js';
7
+ import type { CostTracker } from './observability.js';
8
+ import type { ContextMemory } from './context-memory.js';
9
+ export interface GatewayComponents {
10
+ config: GatewayConfig;
11
+ providers: ProviderRegistry;
12
+ router?: CodeAwareRouter;
13
+ cache?: SemanticCache;
14
+ privacy?: PrivacyFilter;
15
+ costTracker?: CostTracker;
16
+ contextMemory?: ContextMemory;
17
+ }
18
+ export declare function createGatewayServer(components: GatewayComponents): import("http").Server<typeof IncomingMessage, typeof ServerResponse>;
19
+ export declare function startGateway(config: GatewayConfig, dbPath?: string): Promise<void>;
20
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/gateway/server.ts"],"names":[],"mappings":"AAEA,OAAO,EAAgB,eAAe,EAAE,cAAc,EAAE,MAAM,WAAW,CAAC;AAE1E,OAAO,KAAK,EACV,aAAa,EACd,MAAM,YAAY,CAAC;AACpB,OAAO,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAClD,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AACnD,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAChD,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAClD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACtD,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAwBzD,MAAM,WAAW,iBAAiB;IAChC,MAAM,EAAE,aAAa,CAAC;IACtB,SAAS,EAAE,gBAAgB,CAAC;IAC5B,MAAM,CAAC,EAAE,eAAe,CAAC;IACzB,KAAK,CAAC,EAAE,aAAa,CAAC;IACtB,OAAO,CAAC,EAAE,aAAa,CAAC;IACxB,WAAW,CAAC,EAAE,WAAW,CAAC;IAC1B,aAAa,CAAC,EAAE,aAAa,CAAC;CAC/B;AAED,wBAAgB,mBAAmB,CAAC,UAAU,EAAE,iBAAiB,wEAoDhE;AA4QD,wBAAsB,YAAY,CAAC,MAAM,EAAE,aAAa,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAuCxF"}