context-lens 0.2.0 → 0.3.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/README.md +120 -25
- package/dist/cli-utils.d.ts +2 -1
- package/dist/cli-utils.d.ts.map +1 -1
- package/dist/cli-utils.js +40 -13
- package/dist/cli-utils.js.map +1 -1
- package/dist/cli.js +184 -68
- package/dist/cli.js.map +1 -1
- package/dist/core/conversation.d.ts +63 -0
- package/dist/core/conversation.d.ts.map +1 -0
- package/dist/core/conversation.js +305 -0
- package/dist/core/conversation.js.map +1 -0
- package/dist/core/health.d.ts +7 -0
- package/dist/core/health.d.ts.map +1 -0
- package/dist/core/health.js +311 -0
- package/dist/core/health.js.map +1 -0
- package/dist/core/models.d.ts +36 -0
- package/dist/core/models.d.ts.map +1 -0
- package/dist/core/models.js +111 -0
- package/dist/core/models.js.map +1 -0
- package/dist/core/parse.d.ts +17 -0
- package/dist/core/parse.d.ts.map +1 -0
- package/dist/core/parse.js +349 -0
- package/dist/core/parse.js.map +1 -0
- package/dist/core/routing.d.ts +47 -0
- package/dist/core/routing.d.ts.map +1 -0
- package/dist/core/routing.js +132 -0
- package/dist/core/routing.js.map +1 -0
- package/dist/core/security.d.ts +8 -0
- package/dist/core/security.d.ts.map +1 -0
- package/dist/core/security.js +222 -0
- package/dist/core/security.js.map +1 -0
- package/dist/core/source.d.ts +22 -0
- package/dist/core/source.d.ts.map +1 -0
- package/dist/core/source.js +56 -0
- package/dist/core/source.js.map +1 -0
- package/dist/core/tokens.d.ts +29 -0
- package/dist/core/tokens.d.ts.map +1 -0
- package/dist/core/tokens.js +163 -0
- package/dist/core/tokens.js.map +1 -0
- package/dist/core.d.ts +14 -22
- package/dist/core.d.ts.map +1 -1
- package/dist/core.js +14 -471
- package/dist/core.js.map +1 -1
- package/dist/http/headers.d.ts +25 -0
- package/dist/http/headers.d.ts.map +1 -0
- package/dist/http/headers.js +54 -0
- package/dist/http/headers.js.map +1 -0
- package/dist/lhar/composition.d.ts +12 -0
- package/dist/lhar/composition.d.ts.map +1 -0
- package/dist/lhar/composition.js +258 -0
- package/dist/lhar/composition.js.map +1 -0
- package/dist/lhar/export.d.ts +5 -0
- package/dist/lhar/export.d.ts.map +1 -0
- package/dist/lhar/export.js +59 -0
- package/dist/lhar/export.js.map +1 -0
- package/dist/lhar/record.d.ts +6 -0
- package/dist/lhar/record.d.ts.map +1 -0
- package/dist/lhar/record.js +216 -0
- package/dist/lhar/record.js.map +1 -0
- package/dist/lhar/response.d.ts +11 -0
- package/dist/lhar/response.d.ts.map +1 -0
- package/dist/lhar/response.js +132 -0
- package/dist/lhar/response.js.map +1 -0
- package/dist/lhar-types.generated.d.ts +24 -3
- package/dist/lhar-types.generated.d.ts.map +1 -1
- package/dist/lhar.d.ts +12 -19
- package/dist/lhar.d.ts.map +1 -1
- package/dist/lhar.js +16 -473
- package/dist/lhar.js.map +1 -1
- package/dist/server/api.d.ts +8 -0
- package/dist/server/api.d.ts.map +1 -0
- package/dist/server/api.js +292 -0
- package/dist/server/api.js.map +1 -0
- package/dist/server/config.d.ts +13 -0
- package/dist/server/config.d.ts.map +1 -0
- package/dist/server/config.js +36 -0
- package/dist/server/config.js.map +1 -0
- package/dist/server/projection.d.ts +9 -0
- package/dist/server/projection.d.ts.map +1 -0
- package/dist/server/projection.js +47 -0
- package/dist/server/projection.js.map +1 -0
- package/dist/server/proxy.d.ts +13 -0
- package/dist/server/proxy.d.ts.map +1 -0
- package/dist/server/proxy.js +218 -0
- package/dist/server/proxy.js.map +1 -0
- package/dist/server/static.d.ts +9 -0
- package/dist/server/static.d.ts.map +1 -0
- package/dist/server/static.js +78 -0
- package/dist/server/static.js.map +1 -0
- package/dist/server/store.d.ts +81 -0
- package/dist/server/store.d.ts.map +1 -0
- package/dist/server/store.js +632 -0
- package/dist/server/store.js.map +1 -0
- package/dist/server/webui.d.ts +5 -0
- package/dist/server/webui.d.ts.map +1 -0
- package/dist/server/webui.js +42 -0
- package/dist/server/webui.js.map +1 -0
- package/dist/server-utils.d.ts +2 -2
- package/dist/server-utils.d.ts.map +1 -1
- package/dist/server-utils.js +12 -21
- package/dist/server-utils.js.map +1 -1
- package/dist/server.js +31 -697
- package/dist/server.js.map +1 -1
- package/dist/types.d.ts +94 -10
- package/dist/types.d.ts.map +1 -1
- package/dist/version.generated.d.ts +2 -0
- package/dist/version.generated.d.ts.map +1 -0
- package/dist/version.generated.js +2 -0
- package/dist/version.generated.js.map +1 -0
- package/findings-screenshot.png +0 -0
- package/messages-screenshot.png +0 -0
- package/package.json +23 -10
- package/schema/lhar.schema.json +58 -4
- package/screenshot-overview.png +0 -0
- package/sessions-screenshot.png +0 -0
- package/timeline-screenshot.png +0 -0
- package/diff.png +0 -0
- package/overview-sidebar.png +0 -0
- package/public/index.html +0 -2804
|
@@ -0,0 +1,292 @@
|
|
|
1
|
+
import url from "node:url";
|
|
2
|
+
import { parseContextInfo } from "../core.js";
|
|
3
|
+
import { toLharJson, toLharJsonl } from "../lhar.js";
|
|
4
|
+
import { projectEntry } from "./projection.js";
|
|
5
|
+
function projectEntryForApi(e) {
|
|
6
|
+
return projectEntry(e, e.contextInfo);
|
|
7
|
+
}
|
|
8
|
+
function getExportEntries(store, parsedUrl) {
|
|
9
|
+
const convoFilter = parsedUrl.query.conversation;
|
|
10
|
+
if (!convoFilter)
|
|
11
|
+
return store.getCapturedRequests();
|
|
12
|
+
return store
|
|
13
|
+
.getCapturedRequests()
|
|
14
|
+
.filter((e) => e.conversationId === convoFilter);
|
|
15
|
+
}
|
|
16
|
+
function handleIngest(store, req, res) {
|
|
17
|
+
if (req.method !== "POST") {
|
|
18
|
+
res.writeHead(405);
|
|
19
|
+
res.end();
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
const bodyChunks = [];
|
|
23
|
+
req.on("data", (chunk) => {
|
|
24
|
+
bodyChunks.push(chunk);
|
|
25
|
+
});
|
|
26
|
+
req.on("end", () => {
|
|
27
|
+
try {
|
|
28
|
+
const data = JSON.parse(Buffer.concat(bodyChunks).toString("utf8"));
|
|
29
|
+
const provider = data.provider || "unknown";
|
|
30
|
+
const apiFormat = data.apiFormat || "unknown";
|
|
31
|
+
const source = data.source || "unknown";
|
|
32
|
+
const contextInfo = parseContextInfo(provider, data.body || {}, apiFormat);
|
|
33
|
+
store.storeRequest(contextInfo, data.response || {}, source, data.body || {});
|
|
34
|
+
console.log(` 📥 Ingested: [${provider}] ${contextInfo.model} from ${source}`);
|
|
35
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
36
|
+
res.end(JSON.stringify({ ok: true }));
|
|
37
|
+
}
|
|
38
|
+
catch (e) {
|
|
39
|
+
const message = e instanceof Error ? e.message : String(e);
|
|
40
|
+
console.error("Ingest error:", message);
|
|
41
|
+
res.writeHead(400, { "Content-Type": "application/json" });
|
|
42
|
+
res.end(JSON.stringify({ error: message }));
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Group captured requests into conversations with agents.
|
|
48
|
+
* Shared by both full and summary endpoints.
|
|
49
|
+
*/
|
|
50
|
+
function buildConversationGroups(store) {
|
|
51
|
+
const capturedRequests = store.getCapturedRequests();
|
|
52
|
+
const grouped = new Map();
|
|
53
|
+
const ungrouped = [];
|
|
54
|
+
for (const entry of capturedRequests) {
|
|
55
|
+
if (entry.conversationId) {
|
|
56
|
+
if (!grouped.has(entry.conversationId))
|
|
57
|
+
grouped.set(entry.conversationId, []);
|
|
58
|
+
grouped.get(entry.conversationId)?.push(entry);
|
|
59
|
+
}
|
|
60
|
+
else {
|
|
61
|
+
ungrouped.push(entry);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
return { grouped, ungrouped };
|
|
65
|
+
}
|
|
66
|
+
function buildFullConversation(id, entries, conversations) {
|
|
67
|
+
const meta = conversations.get(id) || {
|
|
68
|
+
id,
|
|
69
|
+
label: "Unknown",
|
|
70
|
+
source: "unknown",
|
|
71
|
+
workingDirectory: null,
|
|
72
|
+
firstSeen: entries[entries.length - 1].timestamp,
|
|
73
|
+
};
|
|
74
|
+
const agentMap = new Map();
|
|
75
|
+
for (const e of entries) {
|
|
76
|
+
const ak = e.agentKey || "_default";
|
|
77
|
+
if (!agentMap.has(ak))
|
|
78
|
+
agentMap.set(ak, []);
|
|
79
|
+
agentMap.get(ak)?.push(e);
|
|
80
|
+
}
|
|
81
|
+
const agents = [];
|
|
82
|
+
for (const [_ak, agentEntries] of agentMap) {
|
|
83
|
+
agents.push({
|
|
84
|
+
key: _ak,
|
|
85
|
+
label: agentEntries[agentEntries.length - 1].agentLabel || "Unnamed",
|
|
86
|
+
model: agentEntries[0].contextInfo.model,
|
|
87
|
+
entries: agentEntries.map(projectEntryForApi),
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
agents.sort((a, b) => new Date(b.entries[0].timestamp).getTime() -
|
|
91
|
+
new Date(a.entries[0].timestamp).getTime());
|
|
92
|
+
return {
|
|
93
|
+
...meta,
|
|
94
|
+
agents,
|
|
95
|
+
entries: entries.map(projectEntryForApi),
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
function handleRequests(store, _req, res, parsedUrl) {
|
|
99
|
+
const isSummary = parsedUrl.query.summary === "true";
|
|
100
|
+
const { grouped, ungrouped } = buildConversationGroups(store);
|
|
101
|
+
const conversations = store.getConversations();
|
|
102
|
+
if (isSummary) {
|
|
103
|
+
// Lightweight: conversation metadata + summary stats, no entries
|
|
104
|
+
const summaries = [];
|
|
105
|
+
for (const [id, entries] of grouped) {
|
|
106
|
+
const meta = conversations.get(id) || {
|
|
107
|
+
id,
|
|
108
|
+
label: "Unknown",
|
|
109
|
+
source: "unknown",
|
|
110
|
+
workingDirectory: null,
|
|
111
|
+
firstSeen: entries[entries.length - 1].timestamp,
|
|
112
|
+
};
|
|
113
|
+
const latest = entries[0];
|
|
114
|
+
const totalCost = entries.reduce((sum, e) => sum + (e.costUsd ?? 0), 0);
|
|
115
|
+
// Token history for sparkline: main agent only, chronological (oldest → newest).
|
|
116
|
+
// Main agent = most frequent agentKey in the conversation.
|
|
117
|
+
const keyCounts = new Map();
|
|
118
|
+
for (const e of entries) {
|
|
119
|
+
const k = e.agentKey || "_default";
|
|
120
|
+
keyCounts.set(k, (keyCounts.get(k) || 0) + 1);
|
|
121
|
+
}
|
|
122
|
+
let mainKey = "_default";
|
|
123
|
+
let maxCount = 0;
|
|
124
|
+
for (const [k, count] of keyCounts) {
|
|
125
|
+
if (count > maxCount) {
|
|
126
|
+
mainKey = k;
|
|
127
|
+
maxCount = count;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
const tokenHistory = [];
|
|
131
|
+
for (let i = entries.length - 1; i >= 0; i--) {
|
|
132
|
+
const k = entries[i].agentKey || "_default";
|
|
133
|
+
if (k === mainKey) {
|
|
134
|
+
tokenHistory.push(entries[i].contextInfo.totalTokens);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
summaries.push({
|
|
138
|
+
...meta,
|
|
139
|
+
entryCount: entries.length,
|
|
140
|
+
latestTimestamp: latest.timestamp,
|
|
141
|
+
latestModel: latest.contextInfo.model,
|
|
142
|
+
latestTotalTokens: latest.contextInfo.totalTokens,
|
|
143
|
+
contextLimit: latest.contextLimit,
|
|
144
|
+
totalCost,
|
|
145
|
+
healthScore: latest.healthScore,
|
|
146
|
+
tokenHistory,
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
summaries.sort((a, b) => new Date(b.latestTimestamp).getTime() -
|
|
150
|
+
new Date(a.latestTimestamp).getTime());
|
|
151
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
152
|
+
res.end(JSON.stringify({
|
|
153
|
+
revision: store.getRevision(),
|
|
154
|
+
conversations: summaries,
|
|
155
|
+
ungroupedCount: ungrouped.length,
|
|
156
|
+
}));
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
// Full response (backwards compatible)
|
|
160
|
+
const convos = [];
|
|
161
|
+
for (const [id, entries] of grouped) {
|
|
162
|
+
convos.push(buildFullConversation(id, entries, conversations));
|
|
163
|
+
}
|
|
164
|
+
convos.sort((a, b) => new Date(b.entries[0].timestamp).getTime() -
|
|
165
|
+
new Date(a.entries[0].timestamp).getTime());
|
|
166
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
167
|
+
res.end(JSON.stringify({
|
|
168
|
+
revision: store.getRevision(),
|
|
169
|
+
conversations: convos,
|
|
170
|
+
ungrouped: ungrouped.map(projectEntryForApi),
|
|
171
|
+
}));
|
|
172
|
+
}
|
|
173
|
+
function handleConversationDetail(store, convoId, res) {
|
|
174
|
+
const { grouped } = buildConversationGroups(store);
|
|
175
|
+
const conversations = store.getConversations();
|
|
176
|
+
const entries = grouped.get(convoId);
|
|
177
|
+
if (!entries || entries.length === 0) {
|
|
178
|
+
res.writeHead(404, { "Content-Type": "application/json" });
|
|
179
|
+
res.end(JSON.stringify({ error: "Conversation not found" }));
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
const convo = buildFullConversation(convoId, entries, conversations);
|
|
183
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
184
|
+
res.end(JSON.stringify(convo));
|
|
185
|
+
}
|
|
186
|
+
function handleEvents(store, _req, res) {
|
|
187
|
+
res.writeHead(200, {
|
|
188
|
+
"Content-Type": "text/event-stream",
|
|
189
|
+
"Cache-Control": "no-cache",
|
|
190
|
+
Connection: "keep-alive",
|
|
191
|
+
"Access-Control-Allow-Origin": "*",
|
|
192
|
+
});
|
|
193
|
+
// Send initial revision so client knows current state
|
|
194
|
+
const initial = JSON.stringify({
|
|
195
|
+
revision: store.getRevision(),
|
|
196
|
+
type: "connected",
|
|
197
|
+
});
|
|
198
|
+
res.write(`data: ${initial}\n\n`);
|
|
199
|
+
const listener = (event) => {
|
|
200
|
+
const data = JSON.stringify(event);
|
|
201
|
+
res.write(`data: ${data}\n\n`);
|
|
202
|
+
};
|
|
203
|
+
store.on("change", listener);
|
|
204
|
+
// Clean up when client disconnects
|
|
205
|
+
_req.on("close", () => {
|
|
206
|
+
store.off("change", listener);
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
function handleExportLhar(store, parsedUrl, _req, res) {
|
|
210
|
+
const entries = getExportEntries(store, parsedUrl);
|
|
211
|
+
const jsonl = toLharJsonl(entries, store.getConversations());
|
|
212
|
+
res.writeHead(200, {
|
|
213
|
+
"Content-Type": "application/x-ndjson",
|
|
214
|
+
"Content-Disposition": 'attachment; filename="context-lens-export.lhar"',
|
|
215
|
+
});
|
|
216
|
+
res.end(jsonl);
|
|
217
|
+
}
|
|
218
|
+
function handleExportLharJson(store, parsedUrl, _req, res) {
|
|
219
|
+
const entries = getExportEntries(store, parsedUrl);
|
|
220
|
+
const wrapped = toLharJson(entries, store.getConversations());
|
|
221
|
+
res.writeHead(200, {
|
|
222
|
+
"Content-Type": "application/json",
|
|
223
|
+
"Content-Disposition": 'attachment; filename="context-lens-export.lhar.json"',
|
|
224
|
+
});
|
|
225
|
+
res.end(JSON.stringify(wrapped, null, 2));
|
|
226
|
+
}
|
|
227
|
+
/**
|
|
228
|
+
* Create an API-only request handler. Returns true if the request was handled,
|
|
229
|
+
* false if it should fall through to static file serving.
|
|
230
|
+
*/
|
|
231
|
+
export function createApiHandler(store) {
|
|
232
|
+
return function handleApi(req, res) {
|
|
233
|
+
const parsedUrl = url.parse(req.url, true);
|
|
234
|
+
const pathname = parsedUrl.pathname;
|
|
235
|
+
if (pathname === "/api/ingest" && req.method === "POST") {
|
|
236
|
+
handleIngest(store, req, res);
|
|
237
|
+
return true;
|
|
238
|
+
}
|
|
239
|
+
// Entry detail: full uncompacted contextInfo for a specific entry
|
|
240
|
+
const entryDetailMatch = pathname?.match(/^\/api\/entries\/(\d+)\/detail$/);
|
|
241
|
+
if (entryDetailMatch && req.method === "GET") {
|
|
242
|
+
const entryId = parseInt(entryDetailMatch[1], 10);
|
|
243
|
+
const contextInfo = store.getEntryDetail(entryId);
|
|
244
|
+
if (contextInfo) {
|
|
245
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
246
|
+
res.end(JSON.stringify({ contextInfo }));
|
|
247
|
+
}
|
|
248
|
+
else {
|
|
249
|
+
res.writeHead(404, { "Content-Type": "application/json" });
|
|
250
|
+
res.end(JSON.stringify({ error: "Detail not found for this entry" }));
|
|
251
|
+
}
|
|
252
|
+
return true;
|
|
253
|
+
}
|
|
254
|
+
const convoMatch = pathname?.match(/^\/api\/conversations\/(.+)$/);
|
|
255
|
+
if (convoMatch && req.method === "DELETE") {
|
|
256
|
+
const convoId = decodeURIComponent(convoMatch[1]);
|
|
257
|
+
store.deleteConversation(convoId);
|
|
258
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
259
|
+
res.end(JSON.stringify({ ok: true }));
|
|
260
|
+
return true;
|
|
261
|
+
}
|
|
262
|
+
if (convoMatch && req.method === "GET") {
|
|
263
|
+
const convoId = decodeURIComponent(convoMatch[1]);
|
|
264
|
+
handleConversationDetail(store, convoId, res);
|
|
265
|
+
return true;
|
|
266
|
+
}
|
|
267
|
+
if (pathname === "/api/reset" && req.method === "POST") {
|
|
268
|
+
store.resetAll();
|
|
269
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
270
|
+
res.end(JSON.stringify({ ok: true }));
|
|
271
|
+
return true;
|
|
272
|
+
}
|
|
273
|
+
if (pathname === "/api/requests") {
|
|
274
|
+
handleRequests(store, req, res, parsedUrl);
|
|
275
|
+
return true;
|
|
276
|
+
}
|
|
277
|
+
if (pathname === "/api/events") {
|
|
278
|
+
handleEvents(store, req, res);
|
|
279
|
+
return true;
|
|
280
|
+
}
|
|
281
|
+
if (pathname === "/api/export/lhar") {
|
|
282
|
+
handleExportLhar(store, parsedUrl, req, res);
|
|
283
|
+
return true;
|
|
284
|
+
}
|
|
285
|
+
if (pathname === "/api/export/lhar.json") {
|
|
286
|
+
handleExportLharJson(store, parsedUrl, req, res);
|
|
287
|
+
return true;
|
|
288
|
+
}
|
|
289
|
+
return false;
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
//# sourceMappingURL=api.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api.js","sourceRoot":"","sources":["../../src/server/api.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import type { PrivacyLevel, Upstreams } from "../types.js";
|
|
2
|
+
export interface ServerConfig {
|
|
3
|
+
upstreams: Upstreams;
|
|
4
|
+
bindHost: string;
|
|
5
|
+
allowTargetOverride: boolean;
|
|
6
|
+
dataDir: string;
|
|
7
|
+
stateFile: string;
|
|
8
|
+
maxSessions: number;
|
|
9
|
+
maxCompactMessages: number;
|
|
10
|
+
privacy: PrivacyLevel;
|
|
11
|
+
}
|
|
12
|
+
export declare function loadServerConfig(baseDir: string): ServerConfig;
|
|
13
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../src/server/config.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAE3D,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,SAAS,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,mBAAmB,EAAE,OAAO,CAAC;IAC7B,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,OAAO,EAAE,YAAY,CAAC;CACvB;AAED,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,MAAM,GAAG,YAAY,CA8C9D"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import path from "node:path";
|
|
2
|
+
export function loadServerConfig(baseDir) {
|
|
3
|
+
// Upstream targets, configurable via env vars
|
|
4
|
+
const UPSTREAM_OPENAI_URL = process.env.UPSTREAM_OPENAI_URL || "https://api.openai.com/v1";
|
|
5
|
+
const UPSTREAM_ANTHROPIC_URL = process.env.UPSTREAM_ANTHROPIC_URL || "https://api.anthropic.com";
|
|
6
|
+
const UPSTREAM_CHATGPT_URL = process.env.UPSTREAM_CHATGPT_URL || "https://chatgpt.com";
|
|
7
|
+
const UPSTREAM_GEMINI_URL = process.env.UPSTREAM_GEMINI_URL ||
|
|
8
|
+
"https://generativelanguage.googleapis.com";
|
|
9
|
+
const UPSTREAM_GEMINI_CODE_ASSIST_URL = process.env.UPSTREAM_GEMINI_CODE_ASSIST_URL ||
|
|
10
|
+
"https://cloudcode-pa.googleapis.com";
|
|
11
|
+
// Safety defaults:
|
|
12
|
+
// - Bind only to localhost unless explicitly overridden.
|
|
13
|
+
// - Do not honor `x-target-url` unless explicitly enabled (prevents accidental open-proxy/SSRF).
|
|
14
|
+
const bindHost = process.env.CONTEXT_LENS_BIND_HOST || "127.0.0.1";
|
|
15
|
+
const allowTargetOverride = process.env.CONTEXT_LENS_ALLOW_TARGET_OVERRIDE === "1";
|
|
16
|
+
const privacyEnv = (process.env.CONTEXT_LENS_PRIVACY || "standard").toLowerCase();
|
|
17
|
+
const privacy = privacyEnv === "minimal" || privacyEnv === "full" ? privacyEnv : "standard";
|
|
18
|
+
const dataDir = path.join(baseDir, "..", "data");
|
|
19
|
+
return {
|
|
20
|
+
upstreams: {
|
|
21
|
+
openai: UPSTREAM_OPENAI_URL,
|
|
22
|
+
anthropic: UPSTREAM_ANTHROPIC_URL,
|
|
23
|
+
chatgpt: UPSTREAM_CHATGPT_URL,
|
|
24
|
+
gemini: UPSTREAM_GEMINI_URL,
|
|
25
|
+
geminiCodeAssist: UPSTREAM_GEMINI_CODE_ASSIST_URL,
|
|
26
|
+
},
|
|
27
|
+
bindHost,
|
|
28
|
+
allowTargetOverride,
|
|
29
|
+
dataDir,
|
|
30
|
+
stateFile: path.join(dataDir, "state.jsonl"),
|
|
31
|
+
maxSessions: 100,
|
|
32
|
+
maxCompactMessages: 60,
|
|
33
|
+
privacy,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/server/config.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAe7B,MAAM,UAAU,gBAAgB,CAAC,OAAe;IAC9C,8CAA8C;IAC9C,MAAM,mBAAmB,GACvB,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,2BAA2B,CAAC;IACjE,MAAM,sBAAsB,GAC1B,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,2BAA2B,CAAC;IACpE,MAAM,oBAAoB,GACxB,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,qBAAqB,CAAC;IAC5D,MAAM,mBAAmB,GACvB,OAAO,CAAC,GAAG,CAAC,mBAAmB;QAC/B,2CAA2C,CAAC;IAC9C,MAAM,+BAA+B,GACnC,OAAO,CAAC,GAAG,CAAC,+BAA+B;QAC3C,qCAAqC,CAAC;IAExC,mBAAmB;IACnB,yDAAyD;IACzD,iGAAiG;IACjG,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,WAAW,CAAC;IACnE,MAAM,mBAAmB,GACvB,OAAO,CAAC,GAAG,CAAC,kCAAkC,KAAK,GAAG,CAAC;IAEzD,MAAM,UAAU,GAAG,CACjB,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,UAAU,CAC/C,CAAC,WAAW,EAAE,CAAC;IAChB,MAAM,OAAO,GACX,UAAU,KAAK,SAAS,IAAI,UAAU,KAAK,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC;IAE9E,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,MAAM,CAAC,CAAC;IAEjD,OAAO;QACL,SAAS,EAAE;YACT,MAAM,EAAE,mBAAmB;YAC3B,SAAS,EAAE,sBAAsB;YACjC,OAAO,EAAE,oBAAoB;YAC7B,MAAM,EAAE,mBAAmB;YAC3B,gBAAgB,EAAE,+BAA+B;SAClD;QACD,QAAQ;QACR,mBAAmB;QACnB,OAAO;QACP,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,CAAC;QAC5C,WAAW,EAAE,GAAG;QAChB,kBAAkB,EAAE,EAAE;QACtB,OAAO;KACR,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import type { CapturedEntry, ContextInfo, ProjectedEntry } from "../types.js";
|
|
2
|
+
/**
|
|
3
|
+
* Create a lightweight entry projection for APIs / persistence.
|
|
4
|
+
*
|
|
5
|
+
* `contextInfo` can be either the raw `ContextInfo` or a compacted projection.
|
|
6
|
+
* This avoids duplicating the same mapping logic across store + web UI.
|
|
7
|
+
*/
|
|
8
|
+
export declare function projectEntry(e: CapturedEntry, contextInfo: ContextInfo): ProjectedEntry;
|
|
9
|
+
//# sourceMappingURL=projection.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projection.d.ts","sourceRoot":"","sources":["../../src/server/projection.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,aAAa,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAE9E;;;;;GAKG;AACH,wBAAgB,YAAY,CAC1B,CAAC,EAAE,aAAa,EAChB,WAAW,EAAE,WAAW,GACvB,cAAc,CAyChB"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { parseResponseUsage } from "../lhar.js";
|
|
2
|
+
/**
|
|
3
|
+
* Create a lightweight entry projection for APIs / persistence.
|
|
4
|
+
*
|
|
5
|
+
* `contextInfo` can be either the raw `ContextInfo` or a compacted projection.
|
|
6
|
+
* This avoids duplicating the same mapping logic across store + web UI.
|
|
7
|
+
*/
|
|
8
|
+
export function projectEntry(e, contextInfo) {
|
|
9
|
+
// Use the canonical response parser instead of ad-hoc field access
|
|
10
|
+
const usage = parseResponseUsage(e.response);
|
|
11
|
+
// Return null for usage if no actual data is present (all zeros)
|
|
12
|
+
const hasUsageData = usage.inputTokens > 0 ||
|
|
13
|
+
usage.outputTokens > 0 ||
|
|
14
|
+
usage.cacheReadTokens > 0 ||
|
|
15
|
+
usage.cacheWriteTokens > 0;
|
|
16
|
+
return {
|
|
17
|
+
id: e.id,
|
|
18
|
+
timestamp: e.timestamp,
|
|
19
|
+
contextInfo,
|
|
20
|
+
response: e.response,
|
|
21
|
+
contextLimit: e.contextLimit,
|
|
22
|
+
source: e.source,
|
|
23
|
+
conversationId: e.conversationId,
|
|
24
|
+
agentKey: e.agentKey,
|
|
25
|
+
agentLabel: e.agentLabel,
|
|
26
|
+
httpStatus: e.httpStatus,
|
|
27
|
+
timings: e.timings,
|
|
28
|
+
requestBytes: e.requestBytes,
|
|
29
|
+
responseBytes: e.responseBytes,
|
|
30
|
+
targetUrl: e.targetUrl,
|
|
31
|
+
composition: e.composition,
|
|
32
|
+
costUsd: e.costUsd,
|
|
33
|
+
healthScore: e.healthScore,
|
|
34
|
+
securityAlerts: e.securityAlerts || [],
|
|
35
|
+
usage: hasUsageData
|
|
36
|
+
? {
|
|
37
|
+
inputTokens: usage.inputTokens,
|
|
38
|
+
outputTokens: usage.outputTokens,
|
|
39
|
+
cacheReadTokens: usage.cacheReadTokens,
|
|
40
|
+
cacheWriteTokens: usage.cacheWriteTokens,
|
|
41
|
+
}
|
|
42
|
+
: null,
|
|
43
|
+
responseModel: usage.model,
|
|
44
|
+
stopReason: usage.finishReasons[0] || null,
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=projection.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projection.js","sourceRoot":"","sources":["../../src/server/projection.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AAGhD;;;;;GAKG;AACH,MAAM,UAAU,YAAY,CAC1B,CAAgB,EAChB,WAAwB;IAExB,mEAAmE;IACnE,MAAM,KAAK,GAAG,kBAAkB,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;IAE7C,iEAAiE;IACjE,MAAM,YAAY,GAChB,KAAK,CAAC,WAAW,GAAG,CAAC;QACrB,KAAK,CAAC,YAAY,GAAG,CAAC;QACtB,KAAK,CAAC,eAAe,GAAG,CAAC;QACzB,KAAK,CAAC,gBAAgB,GAAG,CAAC,CAAC;IAE7B,OAAO;QACL,EAAE,EAAE,CAAC,CAAC,EAAE;QACR,SAAS,EAAE,CAAC,CAAC,SAAS;QACtB,WAAW;QACX,QAAQ,EAAE,CAAC,CAAC,QAAQ;QACpB,YAAY,EAAE,CAAC,CAAC,YAAY;QAC5B,MAAM,EAAE,CAAC,CAAC,MAAM;QAChB,cAAc,EAAE,CAAC,CAAC,cAAc;QAChC,QAAQ,EAAE,CAAC,CAAC,QAAQ;QACpB,UAAU,EAAE,CAAC,CAAC,UAAU;QACxB,UAAU,EAAE,CAAC,CAAC,UAAU;QACxB,OAAO,EAAE,CAAC,CAAC,OAAO;QAClB,YAAY,EAAE,CAAC,CAAC,YAAY;QAC5B,aAAa,EAAE,CAAC,CAAC,aAAa;QAC9B,SAAS,EAAE,CAAC,CAAC,SAAS;QACtB,WAAW,EAAE,CAAC,CAAC,WAAW;QAC1B,OAAO,EAAE,CAAC,CAAC,OAAO;QAClB,WAAW,EAAE,CAAC,CAAC,WAAW;QAC1B,cAAc,EAAE,CAAC,CAAC,cAAc,IAAI,EAAE;QACtC,KAAK,EAAE,YAAY;YACjB,CAAC,CAAC;gBACE,WAAW,EAAE,KAAK,CAAC,WAAW;gBAC9B,YAAY,EAAE,KAAK,CAAC,YAAY;gBAChC,eAAe,EAAE,KAAK,CAAC,eAAe;gBACtC,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;aACzC;YACH,CAAC,CAAC,IAAI;QACR,aAAa,EAAE,KAAK,CAAC,KAAK;QAC1B,UAAU,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,IAAI;KAC3C,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import http from "node:http";
|
|
2
|
+
import url from "node:url";
|
|
3
|
+
import type { Upstreams } from "../types.js";
|
|
4
|
+
import type { Store } from "./store.js";
|
|
5
|
+
export declare function forwardRequest(req: http.IncomingMessage, res: http.ServerResponse, parsedUrl: url.UrlWithStringQuery, body: Buffer | null, opts: {
|
|
6
|
+
upstreams: Upstreams;
|
|
7
|
+
allowTargetOverride: boolean;
|
|
8
|
+
}): void;
|
|
9
|
+
export declare function createProxyHandler(store: Store, opts: {
|
|
10
|
+
upstreams: Upstreams;
|
|
11
|
+
allowTargetOverride: boolean;
|
|
12
|
+
}): (req: http.IncomingMessage, res: http.ServerResponse) => void;
|
|
13
|
+
//# sourceMappingURL=proxy.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"proxy.d.ts","sourceRoot":"","sources":["../../src/server/proxy.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAE7B,OAAO,GAAG,MAAM,UAAU,CAAC;AAS3B,OAAO,KAAK,EAA4B,SAAS,EAAE,MAAM,aAAa,CAAC;AACvE,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AA6CxC,wBAAgB,cAAc,CAC5B,GAAG,EAAE,IAAI,CAAC,eAAe,EACzB,GAAG,EAAE,IAAI,CAAC,cAAc,EACxB,SAAS,EAAE,GAAG,CAAC,kBAAkB,EACjC,IAAI,EAAE,MAAM,GAAG,IAAI,EACnB,IAAI,EAAE;IAAE,SAAS,EAAE,SAAS,CAAC;IAAC,mBAAmB,EAAE,OAAO,CAAA;CAAE,GAC3D,IAAI,CA2CN;AAED,wBAAgB,kBAAkB,CAChC,KAAK,EAAE,KAAK,EACZ,IAAI,EAAE;IAAE,SAAS,EAAE,SAAS,CAAC;IAAC,mBAAmB,EAAE,OAAO,CAAA;CAAE,GAC3D,CAAC,GAAG,EAAE,IAAI,CAAC,eAAe,EAAE,GAAG,EAAE,IAAI,CAAC,cAAc,KAAK,IAAI,CA8N/D"}
|