@philippwassibauer/agentlens 0.1.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/.next/BUILD_ID +1 -0
- package/.next/app-build-manifest.json +44 -0
- package/.next/app-path-routes-manifest.json +1 -0
- package/.next/build-manifest.json +32 -0
- package/.next/export-marker.json +1 -0
- package/.next/images-manifest.json +1 -0
- package/.next/next-minimal-server.js.nft.json +1 -0
- package/.next/next-server.js.nft.json +1 -0
- package/.next/package.json +1 -0
- package/.next/prerender-manifest.json +1 -0
- package/.next/react-loadable-manifest.json +1 -0
- package/.next/required-server-files.json +1 -0
- package/.next/routes-manifest.json +1 -0
- package/.next/server/app/_not-found/page.js +1 -0
- package/.next/server/app/_not-found/page.js.nft.json +1 -0
- package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -0
- package/.next/server/app/_not-found.html +1 -0
- package/.next/server/app/_not-found.meta +6 -0
- package/.next/server/app/_not-found.rsc +9 -0
- package/.next/server/app/api/check-prs/route.js +86 -0
- package/.next/server/app/api/check-prs/route.js.nft.json +1 -0
- package/.next/server/app/api/events/route.js +86 -0
- package/.next/server/app/api/events/route.js.nft.json +1 -0
- package/.next/server/app/api/sessions/[id]/route.js +86 -0
- package/.next/server/app/api/sessions/[id]/route.js.nft.json +1 -0
- package/.next/server/app/api/sessions/route.js +86 -0
- package/.next/server/app/api/sessions/route.js.nft.json +1 -0
- package/.next/server/app/api/stats/route.js +86 -0
- package/.next/server/app/api/stats/route.js.nft.json +1 -0
- package/.next/server/app/page.js +1 -0
- package/.next/server/app/page.js.nft.json +1 -0
- package/.next/server/app/page_client-reference-manifest.js +1 -0
- package/.next/server/app/sessions/[id]/page.js +1 -0
- package/.next/server/app/sessions/[id]/page.js.nft.json +1 -0
- package/.next/server/app/sessions/[id]/page_client-reference-manifest.js +1 -0
- package/.next/server/app/sessions/page.js +1 -0
- package/.next/server/app/sessions/page.js.nft.json +1 -0
- package/.next/server/app/sessions/page_client-reference-manifest.js +1 -0
- package/.next/server/app-paths-manifest.json +11 -0
- package/.next/server/chunks/111.js +1 -0
- package/.next/server/chunks/19.js +1 -0
- package/.next/server/chunks/218.js +1 -0
- package/.next/server/chunks/267.js +13 -0
- package/.next/server/chunks/449.js +12 -0
- package/.next/server/chunks/522.js +2 -0
- package/.next/server/chunks/590.js +6 -0
- package/.next/server/chunks/619.js +2 -0
- package/.next/server/chunks/780.js +1 -0
- package/.next/server/chunks/787.js +1 -0
- package/.next/server/chunks/938.js +1 -0
- package/.next/server/chunks/95.js +1 -0
- package/.next/server/chunks/98.js +86 -0
- package/.next/server/chunks/font-manifest.json +1 -0
- package/.next/server/font-manifest.json +1 -0
- package/.next/server/functions-config-manifest.json +1 -0
- package/.next/server/interception-route-rewrite-manifest.js +1 -0
- package/.next/server/middleware-build-manifest.js +1 -0
- package/.next/server/middleware-manifest.json +6 -0
- package/.next/server/middleware-react-loadable-manifest.js +1 -0
- package/.next/server/next-font-manifest.js +1 -0
- package/.next/server/next-font-manifest.json +1 -0
- package/.next/server/pages/404.html +1 -0
- package/.next/server/pages/500.html +1 -0
- package/.next/server/pages/_app.js +1 -0
- package/.next/server/pages/_app.js.nft.json +1 -0
- package/.next/server/pages/_document.js +1 -0
- package/.next/server/pages/_document.js.nft.json +1 -0
- package/.next/server/pages/_error.js +1 -0
- package/.next/server/pages/_error.js.nft.json +1 -0
- package/.next/server/pages-manifest.json +1 -0
- package/.next/server/server-reference-manifest.js +1 -0
- package/.next/server/server-reference-manifest.json +1 -0
- package/.next/server/webpack-runtime.js +1 -0
- package/.next/static/VcvMciURiwijLrVt2jf_P/_buildManifest.js +1 -0
- package/.next/static/VcvMciURiwijLrVt2jf_P/_ssgManifest.js +1 -0
- package/.next/static/chunks/2200cc46-96120a9be9a790d7.js +1 -0
- package/.next/static/chunks/838-ee1ac53cffa08ff2.js +1 -0
- package/.next/static/chunks/919-13111e16ca5aad12.js +1 -0
- package/.next/static/chunks/945-08d88d38313883d1.js +2 -0
- package/.next/static/chunks/app/_not-found/page-52828565ce3bf1a5.js +1 -0
- package/.next/static/chunks/app/layout-d3c17bc5a9ba9afe.js +1 -0
- package/.next/static/chunks/app/page-f02e37b80f020816.js +1 -0
- package/.next/static/chunks/app/sessions/[id]/page-9cd58c2b3cbdb58c.js +1 -0
- package/.next/static/chunks/app/sessions/page-381d2eb994edfbe5.js +1 -0
- package/.next/static/chunks/framework-6e06c675866dc992.js +1 -0
- package/.next/static/chunks/main-921b2ae56ade90f2.js +1 -0
- package/.next/static/chunks/main-app-f7575201e7f66c72.js +1 -0
- package/.next/static/chunks/pages/_app-0c3037849002a4aa.js +1 -0
- package/.next/static/chunks/pages/_error-a647cd2c75dc4dc7.js +1 -0
- package/.next/static/chunks/polyfills-42372ed130431b0a.js +1 -0
- package/.next/static/chunks/webpack-5a21843c4f0469c6.js +1 -0
- package/.next/static/css/e0ed3c3bd07a0247.css +3 -0
- package/.next/static/media/011e180705008d6f-s.woff2 +0 -0
- package/.next/static/media/0aa834ed78bf6d07-s.woff2 +0 -0
- package/.next/static/media/20535187d867b7b9-s.woff2 +0 -0
- package/.next/static/media/37786be940ec402b-s.p.woff2 +0 -0
- package/.next/static/media/46e154b2fcbd6033-s.woff2 +0 -0
- package/.next/static/media/5356a6a4f2c8c8d8-s.woff2 +0 -0
- package/.next/static/media/58f386aa6b1a2a92-s.woff2 +0 -0
- package/.next/static/media/656feb427634a431-s.woff2 +0 -0
- package/.next/static/media/67957d42bae0796d-s.woff2 +0 -0
- package/.next/static/media/704b853f32d191d5-s.woff2 +0 -0
- package/.next/static/media/73cb51aac9c97f90-s.woff2 +0 -0
- package/.next/static/media/7ba5fb2a8c88521c-s.woff2 +0 -0
- package/.next/static/media/886030b0b59bc5a7-s.woff2 +0 -0
- package/.next/static/media/92eeb95d069020cc-s.woff2 +0 -0
- package/.next/static/media/939c4f875ee75fbb-s.woff2 +0 -0
- package/.next/static/media/98e207f02528a563-s.p.woff2 +0 -0
- package/.next/static/media/991629005c80bdf1-s.woff2 +0 -0
- package/.next/static/media/99dcf268bda04fe5-s.woff2 +0 -0
- package/.next/static/media/bb3ef058b751a6ad-s.p.woff2 +0 -0
- package/.next/static/media/d26bbd13d6b70f89-s.woff2 +0 -0
- package/.next/static/media/d29838c109ef09b4-s.woff2 +0 -0
- package/.next/static/media/d3ebbfd689654d3a-s.p.woff2 +0 -0
- package/.next/static/media/db96af6b531dc71f-s.p.woff2 +0 -0
- package/.next/static/media/e40af3453d7c920a-s.woff2 +0 -0
- package/.next/static/media/ef4d5661765d0e49-s.woff2 +0 -0
- package/.next/static/media/f911b923c6adde36-s.woff2 +0 -0
- package/README.md +152 -0
- package/dist/bin/agentlens.mjs +1104 -0
- package/next.config.mjs +10 -0
- package/package.json +88 -0
|
@@ -0,0 +1,1104 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
// bin/cli.ts
|
|
4
|
+
import { join as join2, dirname, resolve } from "path";
|
|
5
|
+
import { existsSync as existsSync2, mkdirSync, readFileSync as readFileSync2 } from "fs";
|
|
6
|
+
import { homedir as homedir2 } from "os";
|
|
7
|
+
import { fileURLToPath } from "url";
|
|
8
|
+
import { spawn, exec } from "child_process";
|
|
9
|
+
import { createRequire } from "module";
|
|
10
|
+
|
|
11
|
+
// collector/indexer.ts
|
|
12
|
+
import { readdirSync, readFileSync, existsSync } from "fs";
|
|
13
|
+
import { join } from "path";
|
|
14
|
+
import { execSync } from "child_process";
|
|
15
|
+
import { userInfo } from "os";
|
|
16
|
+
import { eq as eq2 } from "drizzle-orm";
|
|
17
|
+
|
|
18
|
+
// lib/db/schema.ts
|
|
19
|
+
import { sql, eq, and, gte, lte, desc, asc, count } from "drizzle-orm";
|
|
20
|
+
import { sqliteTable, text, integer } from "drizzle-orm/sqlite-core";
|
|
21
|
+
var sessions = sqliteTable("sessions", {
|
|
22
|
+
id: text("id").primaryKey(),
|
|
23
|
+
teamId: text("team_id").notNull(),
|
|
24
|
+
userId: text("user_id").notNull(),
|
|
25
|
+
project: text("project").notNull(),
|
|
26
|
+
gitRepo: text("git_repo"),
|
|
27
|
+
gitBranch: text("git_branch"),
|
|
28
|
+
piVersion: text("pi_version").notNull().default(""),
|
|
29
|
+
skillsUsed: text("skills_used").notNull().default("[]"),
|
|
30
|
+
model: text("model"),
|
|
31
|
+
harness: text("harness").notNull().default("pi"),
|
|
32
|
+
startedAt: text("started_at").notNull(),
|
|
33
|
+
endedAt: text("ended_at"),
|
|
34
|
+
status: text("status").notNull().default("active"),
|
|
35
|
+
createdAt: text("created_at").notNull().default(sql`(datetime('now'))`)
|
|
36
|
+
});
|
|
37
|
+
var events = sqliteTable("events", {
|
|
38
|
+
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
39
|
+
sessionId: text("session_id").notNull(),
|
|
40
|
+
eventType: text("event_type").notNull(),
|
|
41
|
+
timestamp: text("timestamp").notNull(),
|
|
42
|
+
payload: text("payload").notNull().default("{}"),
|
|
43
|
+
createdAt: text("created_at").notNull().default(sql`(datetime('now'))`)
|
|
44
|
+
});
|
|
45
|
+
var pullRequests = sqliteTable("pull_requests", {
|
|
46
|
+
id: integer("id").primaryKey({ autoIncrement: true }),
|
|
47
|
+
sessionId: text("session_id").notNull(),
|
|
48
|
+
repo: text("repo").notNull(),
|
|
49
|
+
branch: text("branch").notNull(),
|
|
50
|
+
prNumber: integer("pr_number").notNull(),
|
|
51
|
+
prUrl: text("pr_url").notNull(),
|
|
52
|
+
prTitle: text("pr_title").notNull().default(""),
|
|
53
|
+
status: text("status").notNull().default("open"),
|
|
54
|
+
mergedAt: text("merged_at"),
|
|
55
|
+
reviewsApproved: integer("reviews_approved").notNull().default(0),
|
|
56
|
+
reviewsChangesRequested: integer("reviews_changes_requested").notNull().default(0),
|
|
57
|
+
reviewComments: integer("review_comments").notNull().default(0),
|
|
58
|
+
commitsAfterReview: integer("commits_after_review").notNull().default(0),
|
|
59
|
+
firstReviewAt: text("first_review_at"),
|
|
60
|
+
timeToMergeHours: text("time_to_merge_hours"),
|
|
61
|
+
// stored as text for SQLite real compat
|
|
62
|
+
checkedAt: text("checked_at"),
|
|
63
|
+
createdAt: text("created_at").notNull().default(sql`(datetime('now'))`)
|
|
64
|
+
});
|
|
65
|
+
function insertSession(db, input) {
|
|
66
|
+
db.insert(sessions).values({
|
|
67
|
+
id: input.id,
|
|
68
|
+
teamId: input.teamId,
|
|
69
|
+
userId: input.userId,
|
|
70
|
+
project: input.project,
|
|
71
|
+
gitRepo: input.gitRepo,
|
|
72
|
+
gitBranch: input.gitBranch,
|
|
73
|
+
piVersion: input.piVersion,
|
|
74
|
+
skillsUsed: JSON.stringify(input.skillsUsed),
|
|
75
|
+
model: input.model,
|
|
76
|
+
harness: input.harness || "pi",
|
|
77
|
+
startedAt: input.startedAt
|
|
78
|
+
}).run();
|
|
79
|
+
}
|
|
80
|
+
function insertEvent(db, input) {
|
|
81
|
+
db.insert(events).values({
|
|
82
|
+
sessionId: input.sessionId,
|
|
83
|
+
eventType: input.eventType,
|
|
84
|
+
timestamp: input.timestamp,
|
|
85
|
+
payload: JSON.stringify(input.payload)
|
|
86
|
+
}).run();
|
|
87
|
+
}
|
|
88
|
+
function upsertPullRequest(db, input) {
|
|
89
|
+
const existing = db.select().from(pullRequests).where(
|
|
90
|
+
and(eq(pullRequests.sessionId, input.sessionId), eq(pullRequests.prNumber, input.prNumber))
|
|
91
|
+
).get();
|
|
92
|
+
if (existing) {
|
|
93
|
+
db.update(pullRequests).set({
|
|
94
|
+
status: input.status,
|
|
95
|
+
prTitle: input.prTitle,
|
|
96
|
+
checkedAt: (/* @__PURE__ */ new Date()).toISOString()
|
|
97
|
+
}).where(eq(pullRequests.id, existing.id)).run();
|
|
98
|
+
} else {
|
|
99
|
+
db.insert(pullRequests).values({
|
|
100
|
+
sessionId: input.sessionId,
|
|
101
|
+
repo: input.repo,
|
|
102
|
+
branch: input.branch,
|
|
103
|
+
prNumber: input.prNumber,
|
|
104
|
+
prUrl: input.prUrl,
|
|
105
|
+
prTitle: input.prTitle,
|
|
106
|
+
status: input.status
|
|
107
|
+
}).run();
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// collector/model-context.ts
|
|
112
|
+
var CONTEXT_WINDOWS = {
|
|
113
|
+
// Claude 4
|
|
114
|
+
"claude-opus-4-6": 2e5,
|
|
115
|
+
"claude-sonnet-4-6": 2e5,
|
|
116
|
+
// Claude 3.5+
|
|
117
|
+
"claude-opus-4-5": 2e5,
|
|
118
|
+
"claude-sonnet-4-5": 2e5,
|
|
119
|
+
"claude-haiku-4-5-20251001": 2e5,
|
|
120
|
+
// Claude 3
|
|
121
|
+
"claude-3-opus-20240229": 2e5,
|
|
122
|
+
"claude-3-sonnet-20240229": 2e5,
|
|
123
|
+
"claude-3-haiku-20240307": 2e5,
|
|
124
|
+
// OpenAI
|
|
125
|
+
"gpt-4o": 128e3,
|
|
126
|
+
"gpt-4-turbo": 128e3,
|
|
127
|
+
"gpt-4": 8192,
|
|
128
|
+
"gpt-5.3-codex": 2e5,
|
|
129
|
+
"o1-preview": 128e3,
|
|
130
|
+
"o1-mini": 128e3
|
|
131
|
+
};
|
|
132
|
+
function getContextWindow(model) {
|
|
133
|
+
if (!model) return null;
|
|
134
|
+
if (CONTEXT_WINDOWS[model]) return CONTEXT_WINDOWS[model];
|
|
135
|
+
for (const [key, value] of Object.entries(CONTEXT_WINDOWS)) {
|
|
136
|
+
if (model.startsWith(key)) return value;
|
|
137
|
+
}
|
|
138
|
+
if (model.includes("claude")) return 2e5;
|
|
139
|
+
if (model.includes("gpt-4")) return 128e3;
|
|
140
|
+
return null;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
// collector/claude-code/parser.ts
|
|
144
|
+
var SUBAGENT_TOOLS = /* @__PURE__ */ new Set(["Task", "Agent"]);
|
|
145
|
+
function projectFromCwd(cwd) {
|
|
146
|
+
const parts = cwd.replace(/\/+$/, "").split("/");
|
|
147
|
+
return parts[parts.length - 1] || cwd;
|
|
148
|
+
}
|
|
149
|
+
function getFilePath(input) {
|
|
150
|
+
return input.file_path || input.path || null;
|
|
151
|
+
}
|
|
152
|
+
function summarizeToolInput(name, input) {
|
|
153
|
+
if (name === "Bash" && input.command) {
|
|
154
|
+
return String(input.command).slice(0, 200);
|
|
155
|
+
}
|
|
156
|
+
const filePath = getFilePath(input);
|
|
157
|
+
if ((name === "Write" || name === "Edit" || name === "MultiEdit") && filePath) {
|
|
158
|
+
return filePath;
|
|
159
|
+
}
|
|
160
|
+
if (name === "Read" && filePath) {
|
|
161
|
+
return filePath;
|
|
162
|
+
}
|
|
163
|
+
if ((name === "Task" || name === "Agent") && input.description) {
|
|
164
|
+
return String(input.description).slice(0, 200);
|
|
165
|
+
}
|
|
166
|
+
if ((name === "Task" || name === "Agent") && input.prompt) {
|
|
167
|
+
return String(input.prompt).slice(0, 200);
|
|
168
|
+
}
|
|
169
|
+
for (const val of Object.values(input)) {
|
|
170
|
+
if (typeof val === "string" && val.length > 0) {
|
|
171
|
+
return val.slice(0, 200);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
return "";
|
|
175
|
+
}
|
|
176
|
+
function parseClaudeCodeSession(jsonl) {
|
|
177
|
+
const lines = jsonl.split("\n").filter((l) => l.trim()).map((l) => JSON.parse(l));
|
|
178
|
+
const events2 = [];
|
|
179
|
+
let sessionId = "";
|
|
180
|
+
let project = "";
|
|
181
|
+
let gitBranch = null;
|
|
182
|
+
let model = null;
|
|
183
|
+
let version = "";
|
|
184
|
+
let userId = "";
|
|
185
|
+
let startedAt = "";
|
|
186
|
+
let endedAt = "";
|
|
187
|
+
let usedSubagents = false;
|
|
188
|
+
let totalInputTokens = 0;
|
|
189
|
+
let totalOutputTokens = 0;
|
|
190
|
+
let totalCacheReadTokens = 0;
|
|
191
|
+
let totalCacheWriteTokens = 0;
|
|
192
|
+
const toolResults = /* @__PURE__ */ new Map();
|
|
193
|
+
for (const line of lines) {
|
|
194
|
+
if (line.type === "user") {
|
|
195
|
+
const content = line.message?.content;
|
|
196
|
+
if (Array.isArray(content)) {
|
|
197
|
+
for (const item of content) {
|
|
198
|
+
if (item.type === "tool_result" && item.tool_use_id) {
|
|
199
|
+
toolResults.set(item.tool_use_id, item);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
for (const line of lines) {
|
|
206
|
+
const ts = line.timestamp;
|
|
207
|
+
if (!ts) continue;
|
|
208
|
+
if (!startedAt || ts < startedAt) startedAt = ts;
|
|
209
|
+
if (!endedAt || ts > endedAt) endedAt = ts;
|
|
210
|
+
if (line.type === "user" && line.sessionId && !sessionId) {
|
|
211
|
+
sessionId = line.sessionId;
|
|
212
|
+
project = projectFromCwd(line.cwd || "");
|
|
213
|
+
gitBranch = line.gitBranch || null;
|
|
214
|
+
version = line.version || "";
|
|
215
|
+
userId = line.message?.content?.user_id || "";
|
|
216
|
+
}
|
|
217
|
+
if (line.type === "user" && line.userType === "external") {
|
|
218
|
+
const content = line.message?.content;
|
|
219
|
+
if (typeof content === "string" && content.trim()) {
|
|
220
|
+
events2.push({
|
|
221
|
+
eventType: "user_prompt",
|
|
222
|
+
timestamp: ts,
|
|
223
|
+
payload: {
|
|
224
|
+
prompt_text: content,
|
|
225
|
+
prompt_length: content.length
|
|
226
|
+
}
|
|
227
|
+
});
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
if (line.type === "assistant") {
|
|
231
|
+
const msg = line.message || {};
|
|
232
|
+
const usage = msg.usage || {};
|
|
233
|
+
if (msg.model && !model) {
|
|
234
|
+
model = msg.model;
|
|
235
|
+
}
|
|
236
|
+
const inputTokens = usage.input_tokens || 0;
|
|
237
|
+
const outputTokens = usage.output_tokens || 0;
|
|
238
|
+
const cacheRead = usage.cache_read_input_tokens || 0;
|
|
239
|
+
const cacheWrite = usage.cache_creation_input_tokens || 0;
|
|
240
|
+
totalInputTokens += inputTokens;
|
|
241
|
+
totalOutputTokens += outputTokens;
|
|
242
|
+
totalCacheReadTokens += cacheRead;
|
|
243
|
+
totalCacheWriteTokens += cacheWrite;
|
|
244
|
+
const responseModel = msg.model || model;
|
|
245
|
+
const contextWindow = getContextWindow(responseModel);
|
|
246
|
+
events2.push({
|
|
247
|
+
eventType: "llm_response",
|
|
248
|
+
timestamp: ts,
|
|
249
|
+
payload: {
|
|
250
|
+
model: responseModel,
|
|
251
|
+
input_tokens: inputTokens,
|
|
252
|
+
output_tokens: outputTokens,
|
|
253
|
+
cache_read_tokens: cacheRead,
|
|
254
|
+
cache_write_tokens: cacheWrite,
|
|
255
|
+
total_tokens: inputTokens + outputTokens,
|
|
256
|
+
stop_reason: msg.stop_reason || null,
|
|
257
|
+
...contextWindow ? { context_window: contextWindow } : {}
|
|
258
|
+
}
|
|
259
|
+
});
|
|
260
|
+
const content = msg.content || [];
|
|
261
|
+
if (Array.isArray(content)) {
|
|
262
|
+
for (const item of content) {
|
|
263
|
+
if (item.type === "tool_use") {
|
|
264
|
+
const toolUse = item;
|
|
265
|
+
const toolResult = toolResults.get(toolUse.id);
|
|
266
|
+
const isError = toolResult?.is_error === true;
|
|
267
|
+
const isSubagent = SUBAGENT_TOOLS.has(toolUse.name);
|
|
268
|
+
if (isSubagent) usedSubagents = true;
|
|
269
|
+
const argsSummary = summarizeToolInput(toolUse.name, toolUse.input || {});
|
|
270
|
+
events2.push({
|
|
271
|
+
eventType: "tool_call",
|
|
272
|
+
timestamp: ts,
|
|
273
|
+
payload: {
|
|
274
|
+
tool: toolUse.name,
|
|
275
|
+
args_summary: argsSummary,
|
|
276
|
+
result_summary: toolResult?.content ? String(toolResult.content).slice(0, 200) : "",
|
|
277
|
+
success: !isError,
|
|
278
|
+
is_error: isError,
|
|
279
|
+
...isSubagent ? { is_subagent: true } : {}
|
|
280
|
+
}
|
|
281
|
+
});
|
|
282
|
+
const changeFilePath = getFilePath(toolUse.input || {});
|
|
283
|
+
if (toolUse.name === "Write" && changeFilePath) {
|
|
284
|
+
events2.push({
|
|
285
|
+
eventType: "file_change",
|
|
286
|
+
timestamp: ts,
|
|
287
|
+
payload: { path: changeFilePath, action: "create" }
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
if ((toolUse.name === "Edit" || toolUse.name === "MultiEdit") && changeFilePath) {
|
|
291
|
+
events2.push({
|
|
292
|
+
eventType: "file_change",
|
|
293
|
+
timestamp: ts,
|
|
294
|
+
payload: { path: changeFilePath, action: "edit" }
|
|
295
|
+
});
|
|
296
|
+
}
|
|
297
|
+
if (toolUse.name === "Bash" && toolUse.input?.command) {
|
|
298
|
+
const cmd = String(toolUse.input.command);
|
|
299
|
+
const resultText = toolResult?.content ? String(toolResult.content) : "";
|
|
300
|
+
if (cmd.includes("git commit")) {
|
|
301
|
+
events2.push({
|
|
302
|
+
eventType: "git_commit",
|
|
303
|
+
timestamp: ts,
|
|
304
|
+
payload: {
|
|
305
|
+
command: cmd,
|
|
306
|
+
result_summary: resultText.slice(0, 200)
|
|
307
|
+
}
|
|
308
|
+
});
|
|
309
|
+
}
|
|
310
|
+
if (cmd.includes("git push")) {
|
|
311
|
+
events2.push({
|
|
312
|
+
eventType: "git_push",
|
|
313
|
+
timestamp: ts,
|
|
314
|
+
payload: {
|
|
315
|
+
command: cmd,
|
|
316
|
+
result_summary: resultText.slice(0, 200)
|
|
317
|
+
}
|
|
318
|
+
});
|
|
319
|
+
}
|
|
320
|
+
if (cmd.includes("gh pr create") || cmd.includes("gh pr merge")) {
|
|
321
|
+
const prMatch = resultText.match(
|
|
322
|
+
/https:\/\/github\.com\/([^/]+\/[^/]+)\/pull\/(\d+)/
|
|
323
|
+
);
|
|
324
|
+
if (prMatch) {
|
|
325
|
+
events2.push({
|
|
326
|
+
eventType: "pr_opened",
|
|
327
|
+
timestamp: ts,
|
|
328
|
+
payload: {
|
|
329
|
+
command: cmd.slice(0, 200),
|
|
330
|
+
pr_url: prMatch[0],
|
|
331
|
+
repo: prMatch[1],
|
|
332
|
+
pr_number: parseInt(prMatch[2], 10)
|
|
333
|
+
}
|
|
334
|
+
});
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
if (isError && toolResult?.content) {
|
|
339
|
+
events2.push({
|
|
340
|
+
eventType: "error",
|
|
341
|
+
timestamp: ts,
|
|
342
|
+
payload: {
|
|
343
|
+
tool: toolUse.name,
|
|
344
|
+
message: String(toolResult.content).slice(0, 500)
|
|
345
|
+
}
|
|
346
|
+
});
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
events2.sort((a, b) => a.timestamp.localeCompare(b.timestamp));
|
|
354
|
+
events2.unshift({
|
|
355
|
+
eventType: "session_start",
|
|
356
|
+
timestamp: startedAt,
|
|
357
|
+
payload: {
|
|
358
|
+
model,
|
|
359
|
+
git_branch: gitBranch,
|
|
360
|
+
project,
|
|
361
|
+
harness: "claude-code",
|
|
362
|
+
version,
|
|
363
|
+
user_id: userId
|
|
364
|
+
}
|
|
365
|
+
});
|
|
366
|
+
events2.push({
|
|
367
|
+
eventType: "session_end",
|
|
368
|
+
timestamp: endedAt,
|
|
369
|
+
payload: {
|
|
370
|
+
reason: "session_end_inferred"
|
|
371
|
+
}
|
|
372
|
+
});
|
|
373
|
+
return {
|
|
374
|
+
sessionId,
|
|
375
|
+
project,
|
|
376
|
+
gitBranch,
|
|
377
|
+
harness: "claude-code",
|
|
378
|
+
model,
|
|
379
|
+
version,
|
|
380
|
+
userId,
|
|
381
|
+
startedAt,
|
|
382
|
+
endedAt,
|
|
383
|
+
events: events2,
|
|
384
|
+
totalInputTokens,
|
|
385
|
+
totalOutputTokens,
|
|
386
|
+
totalCacheReadTokens,
|
|
387
|
+
totalCacheWriteTokens,
|
|
388
|
+
usedSubagents
|
|
389
|
+
};
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
// collector/pi/parser.ts
|
|
393
|
+
var SUBAGENT_TOOLS2 = /* @__PURE__ */ new Set(["subagent", "delegate", "Task", "Agent"]);
|
|
394
|
+
function projectFromCwd2(cwd) {
|
|
395
|
+
const parts = cwd.replace(/\/+$/, "").split("/");
|
|
396
|
+
return parts[parts.length - 1] || cwd;
|
|
397
|
+
}
|
|
398
|
+
function getFilePath2(args2) {
|
|
399
|
+
return args2.path || args2.file_path || null;
|
|
400
|
+
}
|
|
401
|
+
function summarizeToolInput2(name, args2) {
|
|
402
|
+
const lname = name.toLowerCase();
|
|
403
|
+
if (lname === "bash" && args2.command) {
|
|
404
|
+
return String(args2.command).slice(0, 200);
|
|
405
|
+
}
|
|
406
|
+
const filePath = getFilePath2(args2);
|
|
407
|
+
if ((lname === "write" || lname === "edit" || lname === "read") && filePath) {
|
|
408
|
+
return filePath;
|
|
409
|
+
}
|
|
410
|
+
if (lname === "edit" && Array.isArray(args2.multi) && args2.multi.length > 0) {
|
|
411
|
+
const firstPath = args2.multi[0].path || args2.multi[0].file_path || "";
|
|
412
|
+
const count2 = args2.multi.length;
|
|
413
|
+
return count2 > 1 ? `${firstPath} (+${count2 - 1} more)` : firstPath;
|
|
414
|
+
}
|
|
415
|
+
if (SUBAGENT_TOOLS2.has(name)) {
|
|
416
|
+
return String(args2.task || args2.description || args2.prompt || "").slice(0, 200);
|
|
417
|
+
}
|
|
418
|
+
for (const [key, val] of Object.entries(args2)) {
|
|
419
|
+
if (key === "content") continue;
|
|
420
|
+
if (typeof val === "string" && val.length > 0) {
|
|
421
|
+
return val.slice(0, 200);
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
return "";
|
|
425
|
+
}
|
|
426
|
+
function parsePiSession(jsonl, sessionDir) {
|
|
427
|
+
const lines = jsonl.split("\n").filter((l) => l.trim()).map((l) => JSON.parse(l));
|
|
428
|
+
const events2 = [];
|
|
429
|
+
let sessionId = "";
|
|
430
|
+
let project = "";
|
|
431
|
+
let model = null;
|
|
432
|
+
let version = "";
|
|
433
|
+
let startedAt = "";
|
|
434
|
+
let endedAt = "";
|
|
435
|
+
let usedSubagents = false;
|
|
436
|
+
let totalInputTokens = 0;
|
|
437
|
+
let totalOutputTokens = 0;
|
|
438
|
+
let totalCacheReadTokens = 0;
|
|
439
|
+
let totalCacheWriteTokens = 0;
|
|
440
|
+
const toolResultById = /* @__PURE__ */ new Map();
|
|
441
|
+
const toolResultsByIndex = [];
|
|
442
|
+
for (const line of lines) {
|
|
443
|
+
if (line.type !== "message") continue;
|
|
444
|
+
const msg = line.message || {};
|
|
445
|
+
if (msg.role !== "toolResult") continue;
|
|
446
|
+
const text2 = (msg.content || []).filter((c) => c.type === "text").map((c) => c.text).join("\n");
|
|
447
|
+
const info = {
|
|
448
|
+
content: text2,
|
|
449
|
+
isError: msg.isError === true,
|
|
450
|
+
ts: line.timestamp
|
|
451
|
+
};
|
|
452
|
+
if (msg.toolCallId) {
|
|
453
|
+
toolResultById.set(msg.toolCallId, info);
|
|
454
|
+
}
|
|
455
|
+
toolResultsByIndex.push(info);
|
|
456
|
+
}
|
|
457
|
+
let toolCallSeqIndex = 0;
|
|
458
|
+
for (const line of lines) {
|
|
459
|
+
const ts = line.timestamp;
|
|
460
|
+
if (!ts) continue;
|
|
461
|
+
if (!startedAt || ts < startedAt) startedAt = ts;
|
|
462
|
+
if (!endedAt || ts > endedAt) endedAt = ts;
|
|
463
|
+
if (line.type === "session") {
|
|
464
|
+
sessionId = line.id;
|
|
465
|
+
version = String(line.version || "");
|
|
466
|
+
if (line.cwd) project = projectFromCwd2(line.cwd);
|
|
467
|
+
}
|
|
468
|
+
if (line.type === "model_change" && line.modelId) {
|
|
469
|
+
model = line.modelId;
|
|
470
|
+
}
|
|
471
|
+
if (line.type !== "message") continue;
|
|
472
|
+
const msg = line.message || {};
|
|
473
|
+
if (msg.role === "user") {
|
|
474
|
+
const text2 = (msg.content || []).filter((c) => c.type === "text").map((c) => c.text).join("\n");
|
|
475
|
+
if (text2.trim()) {
|
|
476
|
+
events2.push({
|
|
477
|
+
eventType: "user_prompt",
|
|
478
|
+
timestamp: ts,
|
|
479
|
+
payload: { prompt_text: text2, prompt_length: text2.length }
|
|
480
|
+
});
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
if (msg.role === "assistant") {
|
|
484
|
+
const usage = msg.usage || {};
|
|
485
|
+
const inputTokens = usage.input || 0;
|
|
486
|
+
const outputTokens = usage.output || 0;
|
|
487
|
+
const cacheRead = usage.cacheRead || 0;
|
|
488
|
+
const cacheWrite = usage.cacheWrite || 0;
|
|
489
|
+
const costTotal = usage.cost?.total || 0;
|
|
490
|
+
totalInputTokens += inputTokens;
|
|
491
|
+
totalOutputTokens += outputTokens;
|
|
492
|
+
totalCacheReadTokens += cacheRead;
|
|
493
|
+
totalCacheWriteTokens += cacheWrite;
|
|
494
|
+
const responseModel = msg.model || model;
|
|
495
|
+
const contextWindow = getContextWindow(responseModel);
|
|
496
|
+
events2.push({
|
|
497
|
+
eventType: "llm_response",
|
|
498
|
+
timestamp: ts,
|
|
499
|
+
payload: {
|
|
500
|
+
model: responseModel,
|
|
501
|
+
input_tokens: inputTokens,
|
|
502
|
+
output_tokens: outputTokens,
|
|
503
|
+
cache_read_tokens: cacheRead,
|
|
504
|
+
cache_write_tokens: cacheWrite,
|
|
505
|
+
total_tokens: usage.totalTokens || inputTokens + outputTokens,
|
|
506
|
+
total_cost_usd: costTotal,
|
|
507
|
+
stop_reason: msg.stop_reason || null,
|
|
508
|
+
...contextWindow ? { context_window: contextWindow } : {}
|
|
509
|
+
}
|
|
510
|
+
});
|
|
511
|
+
for (const c of msg.content || []) {
|
|
512
|
+
if (c.type !== "toolCall") continue;
|
|
513
|
+
const toolId = c.id;
|
|
514
|
+
const args2 = c.arguments || c.input || {};
|
|
515
|
+
const toolResult = toolResultById.get(toolId) || toolResultsByIndex[toolCallSeqIndex] || null;
|
|
516
|
+
const isError = toolResult?.isError === true;
|
|
517
|
+
const isSubagent = SUBAGENT_TOOLS2.has(c.name);
|
|
518
|
+
if (isSubagent) usedSubagents = true;
|
|
519
|
+
const argsSummary = summarizeToolInput2(c.name, args2);
|
|
520
|
+
events2.push({
|
|
521
|
+
eventType: "tool_call",
|
|
522
|
+
timestamp: ts,
|
|
523
|
+
payload: {
|
|
524
|
+
tool: c.name,
|
|
525
|
+
args_summary: argsSummary,
|
|
526
|
+
result_summary: toolResult?.content ? toolResult.content.slice(0, 200) : "",
|
|
527
|
+
success: !isError,
|
|
528
|
+
is_error: isError,
|
|
529
|
+
...isSubagent ? { is_subagent: true } : {}
|
|
530
|
+
}
|
|
531
|
+
});
|
|
532
|
+
const lname = c.name.toLowerCase();
|
|
533
|
+
const filePath = getFilePath2(args2);
|
|
534
|
+
if (lname === "write" && filePath) {
|
|
535
|
+
events2.push({
|
|
536
|
+
eventType: "file_change",
|
|
537
|
+
timestamp: ts,
|
|
538
|
+
payload: { path: filePath, action: "create" }
|
|
539
|
+
});
|
|
540
|
+
}
|
|
541
|
+
if (lname === "edit" || lname === "multiedit") {
|
|
542
|
+
if (filePath) {
|
|
543
|
+
events2.push({
|
|
544
|
+
eventType: "file_change",
|
|
545
|
+
timestamp: ts,
|
|
546
|
+
payload: { path: filePath, action: "edit" }
|
|
547
|
+
});
|
|
548
|
+
}
|
|
549
|
+
if (Array.isArray(args2.multi)) {
|
|
550
|
+
const seen = /* @__PURE__ */ new Set();
|
|
551
|
+
for (const item of args2.multi) {
|
|
552
|
+
const p = item.path || item.file_path;
|
|
553
|
+
if (p && !seen.has(p)) {
|
|
554
|
+
seen.add(p);
|
|
555
|
+
events2.push({
|
|
556
|
+
eventType: "file_change",
|
|
557
|
+
timestamp: ts,
|
|
558
|
+
payload: { path: p, action: "edit" }
|
|
559
|
+
});
|
|
560
|
+
}
|
|
561
|
+
}
|
|
562
|
+
}
|
|
563
|
+
}
|
|
564
|
+
if (lname === "bash" && args2.command) {
|
|
565
|
+
const cmd = String(args2.command);
|
|
566
|
+
if (cmd.includes("git commit")) {
|
|
567
|
+
events2.push({
|
|
568
|
+
eventType: "git_commit",
|
|
569
|
+
timestamp: ts,
|
|
570
|
+
payload: {
|
|
571
|
+
command: cmd,
|
|
572
|
+
result_summary: toolResult?.content?.slice(0, 200) || ""
|
|
573
|
+
}
|
|
574
|
+
});
|
|
575
|
+
}
|
|
576
|
+
if (cmd.includes("git push")) {
|
|
577
|
+
events2.push({
|
|
578
|
+
eventType: "git_push",
|
|
579
|
+
timestamp: ts,
|
|
580
|
+
payload: {
|
|
581
|
+
command: cmd,
|
|
582
|
+
result_summary: toolResult?.content?.slice(0, 200) || ""
|
|
583
|
+
}
|
|
584
|
+
});
|
|
585
|
+
}
|
|
586
|
+
if (cmd.includes("gh pr create") || cmd.includes("gh pr merge")) {
|
|
587
|
+
const resultText = toolResult?.content || "";
|
|
588
|
+
const prMatch = resultText.match(/https:\/\/github\.com\/([^/]+\/[^/]+)\/pull\/(\d+)/);
|
|
589
|
+
if (prMatch) {
|
|
590
|
+
events2.push({
|
|
591
|
+
eventType: "pr_opened",
|
|
592
|
+
timestamp: ts,
|
|
593
|
+
payload: {
|
|
594
|
+
command: cmd.slice(0, 200),
|
|
595
|
+
pr_url: prMatch[0],
|
|
596
|
+
repo: prMatch[1],
|
|
597
|
+
pr_number: parseInt(prMatch[2], 10)
|
|
598
|
+
}
|
|
599
|
+
});
|
|
600
|
+
}
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
if (isError && toolResult?.content) {
|
|
604
|
+
events2.push({
|
|
605
|
+
eventType: "error",
|
|
606
|
+
timestamp: ts,
|
|
607
|
+
payload: { tool: c.name, message: toolResult.content.slice(0, 500) }
|
|
608
|
+
});
|
|
609
|
+
}
|
|
610
|
+
toolCallSeqIndex++;
|
|
611
|
+
}
|
|
612
|
+
}
|
|
613
|
+
}
|
|
614
|
+
events2.sort((a, b) => a.timestamp.localeCompare(b.timestamp));
|
|
615
|
+
events2.unshift({
|
|
616
|
+
eventType: "session_start",
|
|
617
|
+
timestamp: startedAt,
|
|
618
|
+
payload: { model, project, harness: "pi", version }
|
|
619
|
+
});
|
|
620
|
+
events2.push({
|
|
621
|
+
eventType: "session_end",
|
|
622
|
+
timestamp: endedAt,
|
|
623
|
+
payload: { reason: "session_end_inferred" }
|
|
624
|
+
});
|
|
625
|
+
return {
|
|
626
|
+
sessionId,
|
|
627
|
+
project,
|
|
628
|
+
gitBranch: null,
|
|
629
|
+
harness: "pi",
|
|
630
|
+
model,
|
|
631
|
+
version,
|
|
632
|
+
userId: "",
|
|
633
|
+
startedAt,
|
|
634
|
+
endedAt,
|
|
635
|
+
events: events2,
|
|
636
|
+
totalInputTokens,
|
|
637
|
+
totalOutputTokens,
|
|
638
|
+
totalCacheReadTokens,
|
|
639
|
+
totalCacheWriteTokens,
|
|
640
|
+
usedSubagents
|
|
641
|
+
};
|
|
642
|
+
}
|
|
643
|
+
|
|
644
|
+
// collector/indexer.ts
|
|
645
|
+
var _cachedUserId = null;
|
|
646
|
+
function detectUserId() {
|
|
647
|
+
if (_cachedUserId) return _cachedUserId;
|
|
648
|
+
try {
|
|
649
|
+
_cachedUserId = execSync("git config user.email", { encoding: "utf-8" }).trim();
|
|
650
|
+
if (_cachedUserId) return _cachedUserId;
|
|
651
|
+
} catch {
|
|
652
|
+
}
|
|
653
|
+
_cachedUserId = userInfo().username || "unknown";
|
|
654
|
+
return _cachedUserId;
|
|
655
|
+
}
|
|
656
|
+
function discoverClaudeCodeSessions(projectsDir) {
|
|
657
|
+
if (!existsSync(projectsDir)) return [];
|
|
658
|
+
const files = [];
|
|
659
|
+
try {
|
|
660
|
+
const projectDirs = readdirSync(projectsDir, { withFileTypes: true });
|
|
661
|
+
for (const dir of projectDirs) {
|
|
662
|
+
if (!dir.isDirectory()) continue;
|
|
663
|
+
const dirPath = join(projectsDir, dir.name);
|
|
664
|
+
const dirEntries = readdirSync(dirPath, { withFileTypes: true });
|
|
665
|
+
for (const entry of dirEntries) {
|
|
666
|
+
if (entry.isFile() && entry.name.endsWith(".jsonl")) {
|
|
667
|
+
files.push(join(dirPath, entry.name));
|
|
668
|
+
}
|
|
669
|
+
}
|
|
670
|
+
}
|
|
671
|
+
} catch {
|
|
672
|
+
}
|
|
673
|
+
return files;
|
|
674
|
+
}
|
|
675
|
+
function discoverPiSessions(sessionsDir) {
|
|
676
|
+
if (!existsSync(sessionsDir)) return [];
|
|
677
|
+
const files = [];
|
|
678
|
+
try {
|
|
679
|
+
const projectDirs = readdirSync(sessionsDir, { withFileTypes: true });
|
|
680
|
+
for (const dir of projectDirs) {
|
|
681
|
+
if (!dir.isDirectory()) continue;
|
|
682
|
+
const dirPath = join(sessionsDir, dir.name);
|
|
683
|
+
const dirEntries = readdirSync(dirPath, { withFileTypes: true });
|
|
684
|
+
for (const entry of dirEntries) {
|
|
685
|
+
if (entry.isFile() && entry.name.endsWith(".jsonl")) {
|
|
686
|
+
files.push(join(dirPath, entry.name));
|
|
687
|
+
}
|
|
688
|
+
}
|
|
689
|
+
}
|
|
690
|
+
} catch {
|
|
691
|
+
}
|
|
692
|
+
return files;
|
|
693
|
+
}
|
|
694
|
+
function importParsedSession(db, parsed, result) {
|
|
695
|
+
if (!parsed.sessionId) {
|
|
696
|
+
return false;
|
|
697
|
+
}
|
|
698
|
+
const existing = db.select().from(sessions).where(eq2(sessions.id, parsed.sessionId)).get();
|
|
699
|
+
if (existing) {
|
|
700
|
+
result.skipped++;
|
|
701
|
+
return false;
|
|
702
|
+
}
|
|
703
|
+
const userId = parsed.userId || detectUserId();
|
|
704
|
+
insertSession(db, {
|
|
705
|
+
id: parsed.sessionId,
|
|
706
|
+
teamId: "default",
|
|
707
|
+
userId,
|
|
708
|
+
project: parsed.project,
|
|
709
|
+
gitRepo: null,
|
|
710
|
+
gitBranch: parsed.gitBranch,
|
|
711
|
+
piVersion: parsed.version,
|
|
712
|
+
skillsUsed: [],
|
|
713
|
+
model: parsed.model,
|
|
714
|
+
harness: parsed.harness,
|
|
715
|
+
startedAt: parsed.startedAt
|
|
716
|
+
});
|
|
717
|
+
if (parsed.endedAt) {
|
|
718
|
+
db.update(sessions).set({ endedAt: parsed.endedAt, status: "completed" }).where(eq2(sessions.id, parsed.sessionId)).run();
|
|
719
|
+
}
|
|
720
|
+
for (const event of parsed.events) {
|
|
721
|
+
insertEvent(db, {
|
|
722
|
+
sessionId: parsed.sessionId,
|
|
723
|
+
eventType: event.eventType,
|
|
724
|
+
timestamp: event.timestamp,
|
|
725
|
+
payload: event.payload
|
|
726
|
+
});
|
|
727
|
+
if (event.eventType === "pr_opened" && event.payload.pr_url) {
|
|
728
|
+
upsertPullRequest(db, {
|
|
729
|
+
sessionId: parsed.sessionId,
|
|
730
|
+
repo: event.payload.repo || "",
|
|
731
|
+
branch: parsed.gitBranch || "",
|
|
732
|
+
prNumber: event.payload.pr_number || 0,
|
|
733
|
+
prUrl: event.payload.pr_url,
|
|
734
|
+
prTitle: "",
|
|
735
|
+
// not available from bash output
|
|
736
|
+
status: "open"
|
|
737
|
+
});
|
|
738
|
+
}
|
|
739
|
+
}
|
|
740
|
+
result.imported++;
|
|
741
|
+
return true;
|
|
742
|
+
}
|
|
743
|
+
function indexSessions(db, options = {}) {
|
|
744
|
+
const result = { imported: 0, skipped: 0, errors: [] };
|
|
745
|
+
const harness = options.harness;
|
|
746
|
+
if (!harness || harness === "claude-code") {
|
|
747
|
+
const projectsDir = options.claudeProjectsDir || join(process.env.HOME || "~", ".claude", "projects");
|
|
748
|
+
const files = discoverClaudeCodeSessions(projectsDir);
|
|
749
|
+
for (const filePath of files) {
|
|
750
|
+
try {
|
|
751
|
+
const jsonl = readFileSync(filePath, "utf-8");
|
|
752
|
+
const parsed = parseClaudeCodeSession(jsonl);
|
|
753
|
+
if (!parsed.sessionId) {
|
|
754
|
+
result.errors.push(`No session ID found in ${filePath}`);
|
|
755
|
+
continue;
|
|
756
|
+
}
|
|
757
|
+
importParsedSession(db, parsed, result);
|
|
758
|
+
} catch (err) {
|
|
759
|
+
result.errors.push(`Error processing ${filePath}: ${err}`);
|
|
760
|
+
}
|
|
761
|
+
}
|
|
762
|
+
}
|
|
763
|
+
if (!harness || harness === "pi") {
|
|
764
|
+
const sessionsDir = options.piSessionsDir || join(process.env.HOME || "~", ".pi", "agent", "sessions");
|
|
765
|
+
const files = discoverPiSessions(sessionsDir);
|
|
766
|
+
for (const filePath of files) {
|
|
767
|
+
try {
|
|
768
|
+
const jsonl = readFileSync(filePath, "utf-8");
|
|
769
|
+
const dirName = join(filePath, "..").split("/").pop() || "";
|
|
770
|
+
const parsed = parsePiSession(jsonl, dirName);
|
|
771
|
+
if (!parsed.sessionId) {
|
|
772
|
+
result.errors.push(`No session ID found in ${filePath}`);
|
|
773
|
+
continue;
|
|
774
|
+
}
|
|
775
|
+
importParsedSession(db, parsed, result);
|
|
776
|
+
} catch (err) {
|
|
777
|
+
result.errors.push(`Error processing ${filePath}: ${err}`);
|
|
778
|
+
}
|
|
779
|
+
}
|
|
780
|
+
}
|
|
781
|
+
return result;
|
|
782
|
+
}
|
|
783
|
+
|
|
784
|
+
// collector/watcher.ts
|
|
785
|
+
function createWatcher(db, options = {}) {
|
|
786
|
+
const interval = options.pollIntervalMs ?? 5e3;
|
|
787
|
+
let stopped = false;
|
|
788
|
+
let timer = null;
|
|
789
|
+
function poll() {
|
|
790
|
+
if (stopped) return;
|
|
791
|
+
try {
|
|
792
|
+
indexSessions(db, options);
|
|
793
|
+
} catch {
|
|
794
|
+
}
|
|
795
|
+
if (!stopped) {
|
|
796
|
+
timer = setTimeout(poll, interval);
|
|
797
|
+
}
|
|
798
|
+
}
|
|
799
|
+
poll();
|
|
800
|
+
return {
|
|
801
|
+
stop() {
|
|
802
|
+
stopped = true;
|
|
803
|
+
if (timer) {
|
|
804
|
+
clearTimeout(timer);
|
|
805
|
+
timer = null;
|
|
806
|
+
}
|
|
807
|
+
}
|
|
808
|
+
};
|
|
809
|
+
}
|
|
810
|
+
|
|
811
|
+
// lib/db/index.ts
|
|
812
|
+
import Database from "better-sqlite3";
|
|
813
|
+
import { drizzle } from "drizzle-orm/better-sqlite3";
|
|
814
|
+
import { sql as sql2 } from "drizzle-orm";
|
|
815
|
+
import path from "path";
|
|
816
|
+
import fs from "fs";
|
|
817
|
+
import { homedir } from "os";
|
|
818
|
+
var _db = null;
|
|
819
|
+
function getDb() {
|
|
820
|
+
if (_db) return _db;
|
|
821
|
+
const dbPath2 = process.env.AGENTLENS_DB_PATH || path.join(homedir(), ".agentlens", "agentlens.db");
|
|
822
|
+
const dataDir2 = path.dirname(dbPath2);
|
|
823
|
+
if (!fs.existsSync(dataDir2)) {
|
|
824
|
+
fs.mkdirSync(dataDir2, { recursive: true });
|
|
825
|
+
}
|
|
826
|
+
const sqlite = new Database(dbPath2);
|
|
827
|
+
sqlite.pragma("journal_mode = WAL");
|
|
828
|
+
const db = drizzle(sqlite);
|
|
829
|
+
db.run(sql2`
|
|
830
|
+
CREATE TABLE IF NOT EXISTS sessions (
|
|
831
|
+
id TEXT PRIMARY KEY,
|
|
832
|
+
team_id TEXT NOT NULL,
|
|
833
|
+
user_id TEXT NOT NULL,
|
|
834
|
+
project TEXT NOT NULL,
|
|
835
|
+
git_repo TEXT,
|
|
836
|
+
git_branch TEXT,
|
|
837
|
+
pi_version TEXT NOT NULL DEFAULT '',
|
|
838
|
+
skills_used TEXT NOT NULL DEFAULT '[]',
|
|
839
|
+
model TEXT,
|
|
840
|
+
harness TEXT NOT NULL DEFAULT 'pi',
|
|
841
|
+
started_at TEXT NOT NULL,
|
|
842
|
+
ended_at TEXT,
|
|
843
|
+
status TEXT NOT NULL DEFAULT 'active',
|
|
844
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
845
|
+
)
|
|
846
|
+
`);
|
|
847
|
+
db.run(sql2`
|
|
848
|
+
CREATE TABLE IF NOT EXISTS events (
|
|
849
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
850
|
+
session_id TEXT NOT NULL,
|
|
851
|
+
event_type TEXT NOT NULL,
|
|
852
|
+
timestamp TEXT NOT NULL,
|
|
853
|
+
payload TEXT NOT NULL DEFAULT '{}',
|
|
854
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
855
|
+
)
|
|
856
|
+
`);
|
|
857
|
+
try {
|
|
858
|
+
db.run(sql2`ALTER TABLE sessions ADD COLUMN model TEXT`);
|
|
859
|
+
} catch {
|
|
860
|
+
}
|
|
861
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_sessions_team_id ON sessions(team_id)`);
|
|
862
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_sessions_user_id ON sessions(user_id)`);
|
|
863
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_sessions_project ON sessions(project)`);
|
|
864
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_sessions_started_at ON sessions(started_at)`);
|
|
865
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_sessions_status ON sessions(status)`);
|
|
866
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_events_session_id ON events(session_id)`);
|
|
867
|
+
db.run(
|
|
868
|
+
sql2`CREATE INDEX IF NOT EXISTS idx_events_session_timestamp ON events(session_id, timestamp)`
|
|
869
|
+
);
|
|
870
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_events_event_type ON events(event_type)`);
|
|
871
|
+
db.run(sql2`
|
|
872
|
+
CREATE TABLE IF NOT EXISTS pull_requests (
|
|
873
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
874
|
+
session_id TEXT NOT NULL,
|
|
875
|
+
repo TEXT NOT NULL,
|
|
876
|
+
branch TEXT NOT NULL,
|
|
877
|
+
pr_number INTEGER NOT NULL,
|
|
878
|
+
pr_url TEXT NOT NULL,
|
|
879
|
+
pr_title TEXT NOT NULL DEFAULT '',
|
|
880
|
+
status TEXT NOT NULL DEFAULT 'open',
|
|
881
|
+
merged_at TEXT,
|
|
882
|
+
reviews_approved INTEGER NOT NULL DEFAULT 0,
|
|
883
|
+
reviews_changes_requested INTEGER NOT NULL DEFAULT 0,
|
|
884
|
+
review_comments INTEGER NOT NULL DEFAULT 0,
|
|
885
|
+
commits_after_review INTEGER NOT NULL DEFAULT 0,
|
|
886
|
+
first_review_at TEXT,
|
|
887
|
+
time_to_merge_hours REAL,
|
|
888
|
+
checked_at TEXT,
|
|
889
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
890
|
+
UNIQUE(session_id, pr_number)
|
|
891
|
+
)
|
|
892
|
+
`);
|
|
893
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_prs_session_id ON pull_requests(session_id)`);
|
|
894
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_prs_status ON pull_requests(status)`);
|
|
895
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_prs_repo_branch ON pull_requests(repo, branch)`);
|
|
896
|
+
try {
|
|
897
|
+
db.run(sql2`ALTER TABLE sessions ADD COLUMN model TEXT`);
|
|
898
|
+
} catch {
|
|
899
|
+
}
|
|
900
|
+
try {
|
|
901
|
+
db.run(sql2`ALTER TABLE sessions ADD COLUMN harness TEXT NOT NULL DEFAULT 'pi'`);
|
|
902
|
+
} catch {
|
|
903
|
+
}
|
|
904
|
+
db.run(sql2`
|
|
905
|
+
CREATE TABLE IF NOT EXISTS eval_checks (
|
|
906
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
907
|
+
session_id TEXT NOT NULL,
|
|
908
|
+
check_id TEXT NOT NULL,
|
|
909
|
+
check_version TEXT NOT NULL,
|
|
910
|
+
score REAL NOT NULL,
|
|
911
|
+
raw_score INTEGER NOT NULL,
|
|
912
|
+
reasoning TEXT NOT NULL DEFAULT '',
|
|
913
|
+
evidence TEXT,
|
|
914
|
+
model TEXT NOT NULL,
|
|
915
|
+
cost_usd REAL NOT NULL DEFAULT 0,
|
|
916
|
+
evaluated_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
917
|
+
UNIQUE(session_id, check_id, check_version)
|
|
918
|
+
)
|
|
919
|
+
`);
|
|
920
|
+
db.run(sql2`
|
|
921
|
+
CREATE TABLE IF NOT EXISTS eval_summaries (
|
|
922
|
+
session_id TEXT PRIMARY KEY,
|
|
923
|
+
qual_score REAL NOT NULL,
|
|
924
|
+
checks_passed INTEGER NOT NULL DEFAULT 0,
|
|
925
|
+
checks_total INTEGER NOT NULL DEFAULT 0,
|
|
926
|
+
total_eval_cost_usd REAL NOT NULL DEFAULT 0,
|
|
927
|
+
evaluated_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
928
|
+
)
|
|
929
|
+
`);
|
|
930
|
+
db.run(sql2`CREATE INDEX IF NOT EXISTS idx_eval_checks_session ON eval_checks(session_id)`);
|
|
931
|
+
db.run(
|
|
932
|
+
sql2`CREATE INDEX IF NOT EXISTS idx_eval_checks_lookup ON eval_checks(session_id, check_id, check_version)`
|
|
933
|
+
);
|
|
934
|
+
db.run(sql2`
|
|
935
|
+
CREATE TABLE IF NOT EXISTS session_insights (
|
|
936
|
+
session_id TEXT PRIMARY KEY,
|
|
937
|
+
task_category TEXT NOT NULL,
|
|
938
|
+
task_size TEXT NOT NULL,
|
|
939
|
+
workflow_tags TEXT NOT NULL DEFAULT '[]',
|
|
940
|
+
prompting_tags TEXT NOT NULL DEFAULT '[]',
|
|
941
|
+
failure_modes TEXT NOT NULL DEFAULT '[]',
|
|
942
|
+
complexity_factors TEXT NOT NULL DEFAULT '[]',
|
|
943
|
+
key_learning TEXT,
|
|
944
|
+
extracted_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
945
|
+
model TEXT NOT NULL,
|
|
946
|
+
synced_at TEXT
|
|
947
|
+
)
|
|
948
|
+
`);
|
|
949
|
+
for (const col of [
|
|
950
|
+
"failure_modes TEXT NOT NULL DEFAULT '[]'",
|
|
951
|
+
"complexity_factors TEXT NOT NULL DEFAULT '[]'",
|
|
952
|
+
"key_learning TEXT"
|
|
953
|
+
]) {
|
|
954
|
+
try {
|
|
955
|
+
db.run(sql2.raw(`ALTER TABLE session_insights ADD COLUMN ${col}`));
|
|
956
|
+
} catch {
|
|
957
|
+
}
|
|
958
|
+
}
|
|
959
|
+
_db = db;
|
|
960
|
+
return db;
|
|
961
|
+
}
|
|
962
|
+
|
|
963
|
+
// bin/cli.ts
|
|
964
|
+
var args = process.argv.slice(2);
|
|
965
|
+
function getFlag(name) {
|
|
966
|
+
return args.includes(`--${name}`);
|
|
967
|
+
}
|
|
968
|
+
function getFlagValue(name) {
|
|
969
|
+
const idx = args.indexOf(`--${name}`);
|
|
970
|
+
if (idx >= 0 && idx + 1 < args.length) return args[idx + 1];
|
|
971
|
+
return void 0;
|
|
972
|
+
}
|
|
973
|
+
if (getFlag("help") || getFlag("h")) {
|
|
974
|
+
console.log(`
|
|
975
|
+
agentlens \u2014 observability for AI coding agents
|
|
976
|
+
|
|
977
|
+
Usage:
|
|
978
|
+
npx agentlens [options]
|
|
979
|
+
|
|
980
|
+
Options:
|
|
981
|
+
--port <number> Port for the dashboard (default: 3333)
|
|
982
|
+
--data <path> Data directory for the SQLite DB (default: ~/.agentlens)
|
|
983
|
+
--no-open Don't auto-open the browser
|
|
984
|
+
--help, -h Show this help message
|
|
985
|
+
`);
|
|
986
|
+
process.exit(0);
|
|
987
|
+
}
|
|
988
|
+
var port = parseInt(getFlagValue("port") || "3333", 10);
|
|
989
|
+
var dataDir = getFlagValue("data") || join2(homedir2(), ".agentlens");
|
|
990
|
+
var noOpen = getFlag("no-open");
|
|
991
|
+
var dbPath = join2(dataDir, "agentlens.db");
|
|
992
|
+
var __filename = fileURLToPath(import.meta.url);
|
|
993
|
+
var __dirname = dirname(__filename);
|
|
994
|
+
var pkgRoot = existsSync2(join2(__dirname, "..", ".next")) ? resolve(__dirname, "..") : resolve(__dirname, "..", "..");
|
|
995
|
+
var claudeProjectsDir = join2(homedir2(), ".claude", "projects");
|
|
996
|
+
var piSessionsDir = join2(homedir2(), ".pi", "agent", "sessions");
|
|
997
|
+
async function main() {
|
|
998
|
+
if (!existsSync2(dataDir)) {
|
|
999
|
+
mkdirSync(dataDir, { recursive: true });
|
|
1000
|
+
}
|
|
1001
|
+
process.env.AGENTLENS_DB_PATH = dbPath;
|
|
1002
|
+
console.log(`
|
|
1003
|
+
\x1B[1magentlens\x1B[0m v${getVersion()}
|
|
1004
|
+
|
|
1005
|
+
\u25B8 Dashboard: \x1B[36mhttp://localhost:${port}\x1B[0m
|
|
1006
|
+
\u25B8 Data: ${dbPath}
|
|
1007
|
+
\u25B8 Watching: ${claudeProjectsDir}
|
|
1008
|
+
${piSessionsDir}
|
|
1009
|
+
`);
|
|
1010
|
+
const db = getDb();
|
|
1011
|
+
const startTime = Date.now();
|
|
1012
|
+
process.stdout.write(" Indexing sessions...");
|
|
1013
|
+
const result = indexSessions(db, { claudeProjectsDir, piSessionsDir });
|
|
1014
|
+
const elapsed = ((Date.now() - startTime) / 1e3).toFixed(1);
|
|
1015
|
+
const total = result.imported + result.skipped;
|
|
1016
|
+
console.log(
|
|
1017
|
+
` found ${total} session${total !== 1 ? "s" : ""}, ${result.imported} new (${elapsed}s)`
|
|
1018
|
+
);
|
|
1019
|
+
if (result.errors.length > 0) {
|
|
1020
|
+
console.log(` \u26A0 ${result.errors.length} error${result.errors.length !== 1 ? "s" : ""}`);
|
|
1021
|
+
}
|
|
1022
|
+
const watcher = createWatcher(db, {
|
|
1023
|
+
claudeProjectsDir,
|
|
1024
|
+
piSessionsDir,
|
|
1025
|
+
pollIntervalMs: 5e3
|
|
1026
|
+
});
|
|
1027
|
+
console.log(" Watching for new sessions...\n");
|
|
1028
|
+
const require2 = createRequire(import.meta.url);
|
|
1029
|
+
const nextBin = require2.resolve("next/dist/bin/next");
|
|
1030
|
+
const server = spawn(process.execPath, [nextBin, "start", "--port", String(port)], {
|
|
1031
|
+
cwd: pkgRoot,
|
|
1032
|
+
env: {
|
|
1033
|
+
...process.env,
|
|
1034
|
+
AGENTLENS_DB_PATH: dbPath,
|
|
1035
|
+
PORT: String(port),
|
|
1036
|
+
HOSTNAME: "localhost"
|
|
1037
|
+
},
|
|
1038
|
+
stdio: ["ignore", "pipe", "pipe"]
|
|
1039
|
+
});
|
|
1040
|
+
let serverReady = false;
|
|
1041
|
+
server.stdout?.on("data", (data) => {
|
|
1042
|
+
const text2 = data.toString().trim();
|
|
1043
|
+
if (text2 && !serverReady) {
|
|
1044
|
+
serverReady = true;
|
|
1045
|
+
if (!noOpen) {
|
|
1046
|
+
openBrowser(`http://localhost:${port}`);
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
1049
|
+
});
|
|
1050
|
+
server.stderr?.on("data", (data) => {
|
|
1051
|
+
const text2 = data.toString().trim();
|
|
1052
|
+
if (text2) {
|
|
1053
|
+
if (!text2.includes("ExperimentalWarning")) {
|
|
1054
|
+
console.error(` [server] ${text2}`);
|
|
1055
|
+
}
|
|
1056
|
+
}
|
|
1057
|
+
});
|
|
1058
|
+
server.on("error", (err) => {
|
|
1059
|
+
console.error(` Failed to start server: ${err.message}`);
|
|
1060
|
+
cleanup(watcher, server);
|
|
1061
|
+
process.exit(1);
|
|
1062
|
+
});
|
|
1063
|
+
server.on("exit", (code) => {
|
|
1064
|
+
if (code !== 0 && code !== null) {
|
|
1065
|
+
console.error(` Server exited with code ${code}`);
|
|
1066
|
+
}
|
|
1067
|
+
cleanup(watcher, null);
|
|
1068
|
+
process.exit(code ?? 0);
|
|
1069
|
+
});
|
|
1070
|
+
function shutdown() {
|
|
1071
|
+
console.log("\n Shutting down...");
|
|
1072
|
+
cleanup(watcher, server);
|
|
1073
|
+
process.exit(0);
|
|
1074
|
+
}
|
|
1075
|
+
process.on("SIGINT", shutdown);
|
|
1076
|
+
process.on("SIGTERM", shutdown);
|
|
1077
|
+
}
|
|
1078
|
+
function cleanup(watcher, server) {
|
|
1079
|
+
if (watcher) watcher.stop();
|
|
1080
|
+
if (server && !server.killed) {
|
|
1081
|
+
server.kill("SIGTERM");
|
|
1082
|
+
}
|
|
1083
|
+
}
|
|
1084
|
+
function getVersion() {
|
|
1085
|
+
try {
|
|
1086
|
+
const pkgPath = join2(pkgRoot, "package.json");
|
|
1087
|
+
if (existsSync2(pkgPath)) {
|
|
1088
|
+
const pkg = JSON.parse(readFileSync2(pkgPath, "utf-8"));
|
|
1089
|
+
return pkg.version || "0.0.0";
|
|
1090
|
+
}
|
|
1091
|
+
} catch {
|
|
1092
|
+
}
|
|
1093
|
+
return "0.0.0";
|
|
1094
|
+
}
|
|
1095
|
+
function openBrowser(url) {
|
|
1096
|
+
const cmd = process.platform === "darwin" ? "open" : process.platform === "win32" ? "start" : "xdg-open";
|
|
1097
|
+
exec(`${cmd} ${url}`, () => {
|
|
1098
|
+
});
|
|
1099
|
+
}
|
|
1100
|
+
main().catch((err) => {
|
|
1101
|
+
console.error("Fatal error:", err);
|
|
1102
|
+
process.exit(1);
|
|
1103
|
+
});
|
|
1104
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../bin/cli.ts", "../../collector/indexer.ts", "../../lib/db/schema.ts", "../../collector/model-context.ts", "../../collector/claude-code/parser.ts", "../../collector/pi/parser.ts", "../../collector/watcher.ts", "../../lib/db/index.ts"],
  "sourcesContent": ["/**\n * agentlens CLI \u2014 single command to index sessions, watch for new ones, and serve the dashboard.\n *\n * Usage:\n *   npx agentlens                     # start everything on :3333\n *   npx agentlens --port 4000         # custom port\n *   npx agentlens --data ~/my-data    # custom data directory\n *   npx agentlens --no-open           # don't auto-open browser\n */\n\nimport { join, dirname, resolve } from \"path\";\nimport { existsSync, mkdirSync, readFileSync } from \"fs\";\nimport { homedir } from \"os\";\nimport { fileURLToPath } from \"url\";\nimport { spawn, exec } from \"child_process\";\nimport { createRequire } from \"module\";\nimport { indexSessions } from \"../collector/indexer\";\nimport { createWatcher, type Watcher } from \"../collector/watcher\";\nimport { getDb } from \"../lib/db/index\";\n\n// --- Parse CLI args ---\n\nconst args = process.argv.slice(2);\n\nfunction getFlag(name: string): boolean {\n  return args.includes(`--${name}`);\n}\n\nfunction getFlagValue(name: string): string | undefined {\n  const idx = args.indexOf(`--${name}`);\n  if (idx >= 0 && idx + 1 < args.length) return args[idx + 1];\n  return undefined;\n}\n\nif (getFlag(\"help\") || getFlag(\"h\")) {\n  console.log(`\n  agentlens \u2014 observability for AI coding agents\n\n  Usage:\n    npx agentlens [options]\n\n  Options:\n    --port <number>    Port for the dashboard (default: 3333)\n    --data <path>      Data directory for the SQLite DB (default: ~/.agentlens)\n    --no-open          Don't auto-open the browser\n    --help, -h         Show this help message\n`);\n  process.exit(0);\n}\n\nconst port = parseInt(getFlagValue(\"port\") || \"3333\", 10);\nconst dataDir = getFlagValue(\"data\") || join(homedir(), \".agentlens\");\nconst noOpen = getFlag(\"no-open\");\nconst dbPath = join(dataDir, \"agentlens.db\");\n\n// --- Resolve package root (where .next/ lives) ---\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\n// In production (dist/bin/agentlens.mjs), package root is ../../\n// In dev (bin/cli.ts), package root is ../\nconst pkgRoot = existsSync(join(__dirname, \"..\", \".next\"))\n  ? resolve(__dirname, \"..\")\n  : resolve(__dirname, \"..\", \"..\");\n\n// --- Session source directories ---\n\nconst claudeProjectsDir = join(homedir(), \".claude\", \"projects\");\nconst piSessionsDir = join(homedir(), \".pi\", \"agent\", \"sessions\");\n\n// --- Main ---\n\nasync function main() {\n  // Ensure data directory exists\n  if (!existsSync(dataDir)) {\n    mkdirSync(dataDir, { recursive: true });\n  }\n\n  // Set env for the Next.js server process\n  process.env.AGENTLENS_DB_PATH = dbPath;\n\n  console.log(`\n  \\x1b[1magentlens\\x1b[0m v${getVersion()}\n\n  \u25B8 Dashboard:   \\x1b[36mhttp://localhost:${port}\\x1b[0m\n  \u25B8 Data:        ${dbPath}\n  \u25B8 Watching:    ${claudeProjectsDir}\n                 ${piSessionsDir}\n`);\n\n  // --- Initial index ---\n\n  const db = getDb();\n  const startTime = Date.now();\n\n  process.stdout.write(\"  Indexing sessions...\");\n  const result = indexSessions(db, { claudeProjectsDir, piSessionsDir });\n  const elapsed = ((Date.now() - startTime) / 1000).toFixed(1);\n\n  const total = result.imported + result.skipped;\n  console.log(\n    ` found ${total} session${total !== 1 ? \"s\" : \"\"}, ${result.imported} new (${elapsed}s)`,\n  );\n\n  if (result.errors.length > 0) {\n    console.log(`  \u26A0 ${result.errors.length} error${result.errors.length !== 1 ? \"s\" : \"\"}`);\n  }\n\n  // --- Start watcher ---\n\n  const watcher = createWatcher(db, {\n    claudeProjectsDir,\n    piSessionsDir,\n    pollIntervalMs: 5000,\n  });\n\n  console.log(\"  Watching for new sessions...\\n\");\n\n  // --- Start Next.js server ---\n\n  const require = createRequire(import.meta.url);\n  const nextBin = require.resolve(\"next/dist/bin/next\");\n\n  const server = spawn(process.execPath, [nextBin, \"start\", \"--port\", String(port)], {\n    cwd: pkgRoot,\n    env: {\n      ...process.env,\n      AGENTLENS_DB_PATH: dbPath,\n      PORT: String(port),\n      HOSTNAME: \"localhost\",\n    },\n    stdio: [\"ignore\", \"pipe\", \"pipe\"],\n  });\n\n  let serverReady = false;\n\n  server.stdout?.on(\"data\", (data: Buffer) => {\n    const text = data.toString().trim();\n    if (text && !serverReady) {\n      serverReady = true;\n      if (!noOpen) {\n        openBrowser(`http://localhost:${port}`);\n      }\n    }\n  });\n\n  server.stderr?.on(\"data\", (data: Buffer) => {\n    const text = data.toString().trim();\n    if (text) {\n      // Filter out noisy Next.js warnings\n      if (!text.includes(\"ExperimentalWarning\")) {\n        console.error(`  [server] ${text}`);\n      }\n    }\n  });\n\n  server.on(\"error\", (err) => {\n    console.error(`  Failed to start server: ${err.message}`);\n    cleanup(watcher, server);\n    process.exit(1);\n  });\n\n  server.on(\"exit\", (code) => {\n    if (code !== 0 && code !== null) {\n      console.error(`  Server exited with code ${code}`);\n    }\n    cleanup(watcher, null);\n    process.exit(code ?? 0);\n  });\n\n  // --- Graceful shutdown ---\n\n  function shutdown() {\n    console.log(\"\\n  Shutting down...\");\n    cleanup(watcher, server);\n    process.exit(0);\n  }\n\n  process.on(\"SIGINT\", shutdown);\n  process.on(\"SIGTERM\", shutdown);\n}\n\nfunction cleanup(watcher: Watcher | null, server: ReturnType<typeof spawn> | null) {\n  if (watcher) watcher.stop();\n  if (server && !server.killed) {\n    server.kill(\"SIGTERM\");\n  }\n}\n\nfunction getVersion(): string {\n  try {\n    const pkgPath = join(pkgRoot, \"package.json\");\n    if (existsSync(pkgPath)) {\n      const pkg = JSON.parse(readFileSync(pkgPath, \"utf-8\"));\n      return pkg.version || \"0.0.0\";\n    }\n  } catch {}\n  return \"0.0.0\";\n}\n\nfunction openBrowser(url: string) {\n  const cmd =\n    process.platform === \"darwin\" ? \"open\" : process.platform === \"win32\" ? \"start\" : \"xdg-open\";\n\n  exec(`${cmd} ${url}`, () => {\n    // Silently ignore errors (e.g., no display server)\n  });\n}\n\nmain().catch((err) => {\n  console.error(\"Fatal error:\", err);\n  process.exit(1);\n});\n", "/**\n * Unified session indexer: discovers and imports agent sessions into agentlens DB.\n */\n\nimport { readdirSync, readFileSync, existsSync } from \"fs\";\nimport { join } from \"path\";\nimport { execSync } from \"child_process\";\nimport { userInfo } from \"os\";\nimport type { BetterSQLite3Database } from \"drizzle-orm/better-sqlite3\";\nimport { eq } from \"drizzle-orm\";\nimport { sessions, insertSession, insertEvent, upsertPullRequest } from \"@/lib/db/schema\";\nimport { parseClaudeCodeSession } from \"./claude-code/parser\";\nimport { parsePiSession } from \"./pi/parser\";\nimport type { ParsedSession } from \"./claude-code/parser\";\n\n/** Detect user identity from git config or OS, cached. */\nlet _cachedUserId: string | null = null;\nexport function detectUserId(): string {\n  if (_cachedUserId) return _cachedUserId;\n  try {\n    _cachedUserId = execSync(\"git config user.email\", { encoding: \"utf-8\" }).trim();\n    if (_cachedUserId) return _cachedUserId;\n  } catch {\n    /* git not available */\n  }\n  _cachedUserId = userInfo().username || \"unknown\";\n  return _cachedUserId;\n}\n\nexport interface IndexOptions {\n  /** Path to Claude Code projects directory (default: ~/.claude/projects) */\n  claudeProjectsDir?: string;\n  /** Path to Pi sessions directory (default: ~/.pi/agent/sessions) */\n  piSessionsDir?: string;\n  /** Filter to a specific harness */\n  harness?: string;\n}\n\nexport interface IndexResult {\n  imported: number;\n  skipped: number;\n  errors: string[];\n}\n\n/**\n * Discover Claude Code session JSONL files in the projects directory.\n */\nexport function discoverClaudeCodeSessions(projectsDir: string): string[] {\n  if (!existsSync(projectsDir)) return [];\n\n  const files: string[] = [];\n  try {\n    const projectDirs = readdirSync(projectsDir, { withFileTypes: true });\n    for (const dir of projectDirs) {\n      if (!dir.isDirectory()) continue;\n      const dirPath = join(projectsDir, dir.name);\n      const dirEntries = readdirSync(dirPath, { withFileTypes: true });\n      for (const entry of dirEntries) {\n        if (entry.isFile() && entry.name.endsWith(\".jsonl\")) {\n          files.push(join(dirPath, entry.name));\n        }\n      }\n    }\n  } catch {\n    // Directory not readable\n  }\n  return files;\n}\n\n/**\n * Discover Pi session JSONL files in the sessions directory.\n * Pi stores sessions at ~/.pi/agent/sessions/<project-dir>/<timestamp>_<uuid>.jsonl\n */\nexport function discoverPiSessions(sessionsDir: string): string[] {\n  if (!existsSync(sessionsDir)) return [];\n\n  const files: string[] = [];\n  try {\n    const projectDirs = readdirSync(sessionsDir, { withFileTypes: true });\n    for (const dir of projectDirs) {\n      if (!dir.isDirectory()) continue;\n      const dirPath = join(sessionsDir, dir.name);\n      const dirEntries = readdirSync(dirPath, { withFileTypes: true });\n      for (const entry of dirEntries) {\n        if (entry.isFile() && entry.name.endsWith(\".jsonl\")) {\n          files.push(join(dirPath, entry.name));\n        }\n      }\n    }\n  } catch {\n    // Directory not readable\n  }\n  return files;\n}\n\n/**\n * Import a parsed session into the DB. Returns true if imported, false if skipped.\n */\nfunction importParsedSession(\n  db: BetterSQLite3Database,\n  parsed: ParsedSession,\n  result: IndexResult,\n): boolean {\n  if (!parsed.sessionId) {\n    return false;\n  }\n\n  // Check if already imported\n  const existing = db.select().from(sessions).where(eq(sessions.id, parsed.sessionId)).get();\n  if (existing) {\n    result.skipped++;\n    return false;\n  }\n\n  // Insert session\n  const userId = parsed.userId || detectUserId();\n  insertSession(db, {\n    id: parsed.sessionId,\n    teamId: \"default\",\n    userId,\n    project: parsed.project,\n    gitRepo: null,\n    gitBranch: parsed.gitBranch,\n    piVersion: parsed.version,\n    skillsUsed: [],\n    model: parsed.model,\n    harness: parsed.harness,\n    startedAt: parsed.startedAt,\n  });\n\n  // Update session end if we have it\n  if (parsed.endedAt) {\n    db.update(sessions)\n      .set({ endedAt: parsed.endedAt, status: \"completed\" })\n      .where(eq(sessions.id, parsed.sessionId))\n      .run();\n  }\n\n  // Insert all events\n  for (const event of parsed.events) {\n    insertEvent(db, {\n      sessionId: parsed.sessionId,\n      eventType: event.eventType,\n      timestamp: event.timestamp,\n      payload: event.payload,\n    });\n\n    // Populate pull_requests table from pr_opened events\n    if (event.eventType === \"pr_opened\" && event.payload.pr_url) {\n      upsertPullRequest(db, {\n        sessionId: parsed.sessionId,\n        repo: event.payload.repo || \"\",\n        branch: parsed.gitBranch || \"\",\n        prNumber: event.payload.pr_number || 0,\n        prUrl: event.payload.pr_url,\n        prTitle: \"\", // not available from bash output\n        status: \"open\",\n      });\n    }\n  }\n\n  result.imported++;\n  return true;\n}\n\n/**\n * Import sessions from discovered files into the agentlens DB.\n */\nexport function indexSessions(db: BetterSQLite3Database, options: IndexOptions = {}): IndexResult {\n  const result: IndexResult = { imported: 0, skipped: 0, errors: [] };\n  const harness = options.harness;\n\n  // Claude Code sessions\n  if (!harness || harness === \"claude-code\") {\n    const projectsDir =\n      options.claudeProjectsDir || join(process.env.HOME || \"~\", \".claude\", \"projects\");\n    const files = discoverClaudeCodeSessions(projectsDir);\n\n    for (const filePath of files) {\n      try {\n        const jsonl = readFileSync(filePath, \"utf-8\");\n        const parsed = parseClaudeCodeSession(jsonl);\n        if (!parsed.sessionId) {\n          result.errors.push(`No session ID found in ${filePath}`);\n          continue;\n        }\n        importParsedSession(db, parsed, result);\n      } catch (err) {\n        result.errors.push(`Error processing ${filePath}: ${err}`);\n      }\n    }\n  }\n\n  // Pi sessions\n  if (!harness || harness === \"pi\") {\n    const sessionsDir =\n      options.piSessionsDir || join(process.env.HOME || \"~\", \".pi\", \"agent\", \"sessions\");\n    const files = discoverPiSessions(sessionsDir);\n\n    for (const filePath of files) {\n      try {\n        const jsonl = readFileSync(filePath, \"utf-8\");\n        const dirName = join(filePath, \"..\").split(\"/\").pop() || \"\";\n        const parsed = parsePiSession(jsonl, dirName);\n        if (!parsed.sessionId) {\n          result.errors.push(`No session ID found in ${filePath}`);\n          continue;\n        }\n        importParsedSession(db, parsed, result);\n      } catch (err) {\n        result.errors.push(`Error processing ${filePath}: ${err}`);\n      }\n    }\n  }\n\n  return result;\n}\n", "import { sql, eq, and, gte, lte, desc, asc, count } from \"drizzle-orm\";\nimport { sqliteTable, text, integer } from \"drizzle-orm/sqlite-core\";\nimport type { BetterSQLite3Database } from \"drizzle-orm/better-sqlite3\";\n\n// --- Table Definitions ---\n\nexport const sessions = sqliteTable(\"sessions\", {\n  id: text(\"id\").primaryKey(),\n  teamId: text(\"team_id\").notNull(),\n  userId: text(\"user_id\").notNull(),\n  project: text(\"project\").notNull(),\n  gitRepo: text(\"git_repo\"),\n  gitBranch: text(\"git_branch\"),\n  piVersion: text(\"pi_version\").notNull().default(\"\"),\n  skillsUsed: text(\"skills_used\").notNull().default(\"[]\"),\n  model: text(\"model\"),\n  harness: text(\"harness\").notNull().default(\"pi\"),\n  startedAt: text(\"started_at\").notNull(),\n  endedAt: text(\"ended_at\"),\n  status: text(\"status\").notNull().default(\"active\"),\n  createdAt: text(\"created_at\")\n    .notNull()\n    .default(sql`(datetime('now'))`),\n});\n\nexport const events = sqliteTable(\"events\", {\n  id: integer(\"id\").primaryKey({ autoIncrement: true }),\n  sessionId: text(\"session_id\").notNull(),\n  eventType: text(\"event_type\").notNull(),\n  timestamp: text(\"timestamp\").notNull(),\n  payload: text(\"payload\").notNull().default(\"{}\"),\n  createdAt: text(\"created_at\")\n    .notNull()\n    .default(sql`(datetime('now'))`),\n});\n\nexport const pullRequests = sqliteTable(\"pull_requests\", {\n  id: integer(\"id\").primaryKey({ autoIncrement: true }),\n  sessionId: text(\"session_id\").notNull(),\n  repo: text(\"repo\").notNull(),\n  branch: text(\"branch\").notNull(),\n  prNumber: integer(\"pr_number\").notNull(),\n  prUrl: text(\"pr_url\").notNull(),\n  prTitle: text(\"pr_title\").notNull().default(\"\"),\n  status: text(\"status\").notNull().default(\"open\"),\n  mergedAt: text(\"merged_at\"),\n  reviewsApproved: integer(\"reviews_approved\").notNull().default(0),\n  reviewsChangesRequested: integer(\"reviews_changes_requested\").notNull().default(0),\n  reviewComments: integer(\"review_comments\").notNull().default(0),\n  commitsAfterReview: integer(\"commits_after_review\").notNull().default(0),\n  firstReviewAt: text(\"first_review_at\"),\n  timeToMergeHours: text(\"time_to_merge_hours\"), // stored as text for SQLite real compat\n  checkedAt: text(\"checked_at\"),\n  createdAt: text(\"created_at\")\n    .notNull()\n    .default(sql`(datetime('now'))`),\n});\n\n// --- Types ---\n\nexport interface SessionInput {\n  id: string;\n  teamId: string;\n  userId: string;\n  project: string;\n  gitRepo: string | null;\n  gitBranch: string | null;\n  piVersion: string;\n  skillsUsed: string[];\n  model?: string | null;\n  harness?: string;\n  startedAt: string;\n}\n\nexport interface EventInput {\n  sessionId: string;\n  eventType: string;\n  timestamp: string;\n  payload: object;\n}\n\nexport interface SessionRow {\n  id: string;\n  teamId: string;\n  userId: string;\n  project: string;\n  gitRepo: string | null;\n  gitBranch: string | null;\n  piVersion: string;\n  skillsUsed: string[];\n  model: string | null;\n  harness: string;\n  startedAt: string;\n  endedAt: string | null;\n  status: string;\n}\n\nexport interface EventRow {\n  id: number;\n  sessionId: string;\n  eventType: string;\n  timestamp: string;\n  payload: any;\n}\n\n// --- Query Helpers ---\n\ntype DB = BetterSQLite3Database;\n\nexport function insertSession(db: DB, input: SessionInput): void {\n  db.insert(sessions)\n    .values({\n      id: input.id,\n      teamId: input.teamId,\n      userId: input.userId,\n      project: input.project,\n      gitRepo: input.gitRepo,\n      gitBranch: input.gitBranch,\n      piVersion: input.piVersion,\n      skillsUsed: JSON.stringify(input.skillsUsed),\n      model: input.model,\n      harness: input.harness || \"pi\",\n      startedAt: input.startedAt,\n    })\n    .run();\n}\n\nexport function updateSessionEnd(db: DB, sessionId: string, endedAt: string, status: string): void {\n  db.update(sessions).set({ endedAt, status }).where(eq(sessions.id, sessionId)).run();\n}\n\nexport function updateSessionModel(db: DB, sessionId: string, model: string): void {\n  db.update(sessions).set({ model }).where(eq(sessions.id, sessionId)).run();\n}\n\nexport function upsertSession(db: DB, input: SessionInput): void {\n  // Try insert, on conflict update\n  const existing = db.select().from(sessions).where(eq(sessions.id, input.id)).get();\n\n  if (existing) {\n    db.update(sessions)\n      .set({\n        userId: input.userId,\n        project: input.project,\n        gitRepo: input.gitRepo,\n        gitBranch: input.gitBranch,\n        piVersion: input.piVersion,\n        skillsUsed: JSON.stringify(input.skillsUsed),\n        model: input.model,\n      })\n      .where(eq(sessions.id, input.id))\n      .run();\n  } else {\n    insertSession(db, input);\n  }\n}\n\nexport function insertEvent(db: DB, input: EventInput): void {\n  db.insert(events)\n    .values({\n      sessionId: input.sessionId,\n      eventType: input.eventType,\n      timestamp: input.timestamp,\n      payload: JSON.stringify(input.payload),\n    })\n    .run();\n}\n\nfunction parseSessionRow(row: any): SessionRow {\n  return {\n    id: row.id,\n    teamId: row.teamId,\n    userId: row.userId,\n    project: row.project,\n    gitRepo: row.gitRepo,\n    gitBranch: row.gitBranch,\n    piVersion: row.piVersion,\n    skillsUsed: JSON.parse(row.skillsUsed || \"[]\"),\n    model: row.model || null,\n    harness: row.harness || \"pi\",\n    startedAt: row.startedAt,\n    endedAt: row.endedAt,\n    status: row.status,\n  };\n}\n\nfunction parseEventRow(row: any): EventRow {\n  return {\n    id: row.id,\n    sessionId: row.sessionId,\n    eventType: row.eventType,\n    timestamp: row.timestamp,\n    payload: JSON.parse(row.payload || \"{}\"),\n  };\n}\n\nexport function getSessionWithEvents(\n  db: DB,\n  sessionId: string,\n): { session: SessionRow; events: EventRow[] } | null {\n  const sessionRow = db.select().from(sessions).where(eq(sessions.id, sessionId)).get();\n\n  if (!sessionRow) return null;\n\n  const eventRows = db\n    .select()\n    .from(events)\n    .where(eq(events.sessionId, sessionId))\n    .orderBy(asc(events.timestamp), asc(events.id))\n    .all();\n\n  return {\n    session: parseSessionRow(sessionRow),\n    events: eventRows.map(parseEventRow),\n  };\n}\n\nexport interface ListSessionsFilter {\n  teamId: string;\n  userId?: string;\n  project?: string;\n  harness?: string;\n  status?: string;\n  from?: string;\n  to?: string;\n  limit?: number;\n  offset?: number;\n}\n\nexport function listSessions(\n  db: DB,\n  filter: ListSessionsFilter,\n): { sessions: SessionRow[]; total: number } {\n  const conditions = [eq(sessions.teamId, filter.teamId)];\n\n  if (filter.userId) conditions.push(eq(sessions.userId, filter.userId));\n  if (filter.project) conditions.push(eq(sessions.project, filter.project));\n  if (filter.harness) conditions.push(eq(sessions.harness, filter.harness));\n  if (filter.status) conditions.push(eq(sessions.status, filter.status));\n  if (filter.from) conditions.push(gte(sessions.startedAt, filter.from));\n  if (filter.to) conditions.push(lte(sessions.startedAt, filter.to));\n\n  const where = conditions.length === 1 ? conditions[0] : and(...conditions);\n\n  // Count total\n  const countResult = db.select({ count: count() }).from(sessions).where(where!).get();\n  const total = countResult?.count ?? 0;\n\n  // Fetch page\n  const limit = Math.min(filter.limit ?? 50, 200);\n  const offset = filter.offset ?? 0;\n\n  const rows = db\n    .select()\n    .from(sessions)\n    .where(where!)\n    .orderBy(desc(sessions.startedAt))\n    .limit(limit)\n    .offset(offset)\n    .all();\n\n  return {\n    sessions: rows.map(parseSessionRow),\n    total,\n  };\n}\n\nexport interface TokenStats {\n  totalInputTokens: number;\n  totalOutputTokens: number;\n  totalCacheReadTokens: number;\n  totalCacheWriteTokens: number;\n  totalTokens: number;\n  totalCostUsd: number;\n  llmCallCount: number;\n  avgInputTokensPerCall: number;\n  peakInputTokens: number;\n  contextWindow: number | null;\n  contextUtilizationPct: number | null;\n  model: string | null;\n}\n\nexport function getTokenStats(db: DB, sessionId: string): TokenStats {\n  const eventRows = db\n    .select()\n    .from(events)\n    .where(and(eq(events.sessionId, sessionId), eq(events.eventType, \"llm_response\")))\n    .orderBy(asc(events.timestamp))\n    .all();\n\n  let totalInput = 0;\n  let totalOutput = 0;\n  let totalCacheRead = 0;\n  let totalCacheWrite = 0;\n  let totalTokensSum = 0;\n  let totalCost = 0;\n  let peakInput = 0;\n  let contextWindow: number | null = null;\n  const modelCounts = new Map<string, number>();\n\n  for (const row of eventRows) {\n    const payload = JSON.parse(row.payload || \"{}\");\n    totalInput += payload.input_tokens || 0;\n    totalOutput += payload.output_tokens || 0;\n    totalCacheRead += payload.cache_read_tokens || 0;\n    totalCacheWrite += payload.cache_write_tokens || 0;\n    totalTokensSum += payload.total_tokens || 0;\n    totalCost += payload.total_cost_usd || 0;\n\n    if ((payload.input_tokens || 0) > peakInput) {\n      peakInput = payload.input_tokens || 0;\n    }\n\n    if (payload.context_window) {\n      contextWindow = payload.context_window;\n    }\n\n    if (payload.model) {\n      modelCounts.set(payload.model, (modelCounts.get(payload.model) || 0) + 1);\n    }\n  }\n\n  const llmCallCount = eventRows.length;\n\n  // Find most-used model\n  let model: string | null = null;\n  let maxModelCount = 0;\n  modelCounts.forEach((c, m) => {\n    if (c > maxModelCount) {\n      model = m;\n      maxModelCount = c;\n    }\n  });\n\n  return {\n    totalInputTokens: totalInput,\n    totalOutputTokens: totalOutput,\n    totalCacheReadTokens: totalCacheRead,\n    totalCacheWriteTokens: totalCacheWrite,\n    totalTokens: totalTokensSum,\n    totalCostUsd: totalCost,\n    llmCallCount,\n    avgInputTokensPerCall: llmCallCount > 0 ? Math.round(totalInput / llmCallCount) : 0,\n    peakInputTokens: peakInput,\n    contextWindow,\n    contextUtilizationPct:\n      contextWindow && peakInput > 0 ? Math.round((peakInput / contextWindow) * 10000) / 100 : null,\n    model,\n  };\n}\n\nexport function getEventCountsByType(db: DB, sessionId: string): Record<string, number> {\n  const rows = db\n    .select({\n      eventType: events.eventType,\n      count: count(),\n    })\n    .from(events)\n    .where(eq(events.sessionId, sessionId))\n    .groupBy(events.eventType)\n    .all();\n\n  const counts: Record<string, number> = {\n    user_prompt: 0,\n    tool_call: 0,\n    error: 0,\n    file_change: 0,\n    heartbeat: 0,\n    session_start: 0,\n    session_end: 0,\n  };\n\n  for (const row of rows) {\n    counts[row.eventType] = row.count;\n  }\n\n  return counts;\n}\n\n// --- PR Helpers ---\n\nexport interface PullRequestInput {\n  sessionId: string;\n  repo: string;\n  branch: string;\n  prNumber: number;\n  prUrl: string;\n  prTitle: string;\n  status: string;\n}\n\nexport interface PullRequestRow {\n  id: number;\n  sessionId: string;\n  repo: string;\n  branch: string;\n  prNumber: number;\n  prUrl: string;\n  prTitle: string;\n  status: string;\n  mergedAt: string | null;\n  reviewsApproved: number;\n  reviewsChangesRequested: number;\n  reviewComments: number;\n  commitsAfterReview: number;\n  firstReviewAt: string | null;\n  timeToMergeHours: number | null;\n  checkedAt: string | null;\n}\n\nexport function upsertPullRequest(db: DB, input: PullRequestInput): void {\n  const existing = db\n    .select()\n    .from(pullRequests)\n    .where(\n      and(eq(pullRequests.sessionId, input.sessionId), eq(pullRequests.prNumber, input.prNumber)),\n    )\n    .get();\n\n  if (existing) {\n    db.update(pullRequests)\n      .set({\n        status: input.status,\n        prTitle: input.prTitle,\n        checkedAt: new Date().toISOString(),\n      })\n      .where(eq(pullRequests.id, existing.id))\n      .run();\n  } else {\n    db.insert(pullRequests)\n      .values({\n        sessionId: input.sessionId,\n        repo: input.repo,\n        branch: input.branch,\n        prNumber: input.prNumber,\n        prUrl: input.prUrl,\n        prTitle: input.prTitle,\n        status: input.status,\n      })\n      .run();\n  }\n}\n\nexport interface PrUpdateData {\n  status?: string;\n  mergedAt?: string;\n  reviewsApproved?: number;\n  reviewsChangesRequested?: number;\n  reviewComments?: number;\n  commitsAfterReview?: number;\n  firstReviewAt?: string;\n  timeToMergeHours?: string;\n}\n\nexport function updatePullRequest(db: DB, id: number, data: PrUpdateData): void {\n  db.update(pullRequests)\n    .set({\n      ...data,\n      checkedAt: new Date().toISOString(),\n    })\n    .where(eq(pullRequests.id, id))\n    .run();\n}\n\nexport function getOpenPullRequests(db: DB): PullRequestRow[] {\n  return db\n    .select()\n    .from(pullRequests)\n    .where(eq(pullRequests.status, \"open\"))\n    .all() as PullRequestRow[];\n}\n\nexport function getPullRequestsForSession(db: DB, sessionId: string): PullRequestRow[] {\n  return db\n    .select()\n    .from(pullRequests)\n    .where(eq(pullRequests.sessionId, sessionId))\n    .all() as PullRequestRow[];\n}\n", "/**\n * Known context window sizes for common models.\n * Used to compute context utilization percentage.\n */\n\nconst CONTEXT_WINDOWS: Record<string, number> = {\n  // Claude 4\n  \"claude-opus-4-6\": 200000,\n  \"claude-sonnet-4-6\": 200000,\n\n  // Claude 3.5+\n  \"claude-opus-4-5\": 200000,\n  \"claude-sonnet-4-5\": 200000,\n  \"claude-haiku-4-5-20251001\": 200000,\n\n  // Claude 3\n  \"claude-3-opus-20240229\": 200000,\n  \"claude-3-sonnet-20240229\": 200000,\n  \"claude-3-haiku-20240307\": 200000,\n\n  // OpenAI\n  \"gpt-4o\": 128000,\n  \"gpt-4-turbo\": 128000,\n  \"gpt-4\": 8192,\n  \"gpt-5.3-codex\": 200000,\n  \"o1-preview\": 128000,\n  \"o1-mini\": 128000,\n};\n\n/**\n * Look up known context window for a model.\n * Does prefix matching: \"claude-sonnet-4-6\" matches \"claude-sonnet-4-6\".\n * Returns null if model is unknown.\n */\nexport function getContextWindow(model: string | null): number | null {\n  if (!model) return null;\n\n  // Exact match\n  if (CONTEXT_WINDOWS[model]) return CONTEXT_WINDOWS[model];\n\n  // Prefix match (handles versioned model names like \"claude-sonnet-4-6-20260301\")\n  for (const [key, value] of Object.entries(CONTEXT_WINDOWS)) {\n    if (model.startsWith(key)) return value;\n  }\n\n  // Family match\n  if (model.includes(\"claude\")) return 200000;\n  if (model.includes(\"gpt-4\")) return 128000;\n\n  return null;\n}\n", "/**\n * Parse a Claude Code session JSONL string into agentlens events.\n */\n\nexport interface ParsedEvent {\n  eventType: string;\n  timestamp: string;\n  payload: Record<string, any>;\n}\n\nexport interface ParsedSession {\n  sessionId: string;\n  project: string;\n  gitBranch: string | null;\n  harness: string;\n  model: string | null;\n  version: string;\n  userId: string;\n  startedAt: string;\n  endedAt: string;\n  events: ParsedEvent[];\n  totalInputTokens: number;\n  totalOutputTokens: number;\n  totalCacheReadTokens: number;\n  totalCacheWriteTokens: number;\n  usedSubagents: boolean;\n}\n\ninterface ToolUseItem {\n  type: \"tool_use\";\n  id: string;\n  name: string;\n  input: Record<string, any>;\n}\n\ninterface ToolResultItem {\n  type: \"tool_result\";\n  tool_use_id: string;\n  content: string;\n  is_error?: boolean;\n}\n\nimport { getContextWindow } from \"../model-context\";\n\nconst SUBAGENT_TOOLS = new Set([\"Task\", \"Agent\"]);\n\n/**\n * Derive a project name from the cwd path.\n * e.g. \"/Users/dev/my-project\" \u2192 \"my-project\"\n */\nfunction projectFromCwd(cwd: string): string {\n  const parts = cwd.replace(/\\/+$/, \"\").split(\"/\");\n  return parts[parts.length - 1] || cwd;\n}\n\n/**\n * Summarize tool input for display.\n */\n/** Get file path from tool args \u2014 CC uses file_path, Pi uses path. */\nfunction getFilePath(input: Record<string, any>): string | null {\n  return input.file_path || input.path || null;\n}\n\nfunction summarizeToolInput(name: string, input: Record<string, any>): string {\n  if (name === \"Bash\" && input.command) {\n    return String(input.command).slice(0, 200);\n  }\n  const filePath = getFilePath(input);\n  if ((name === \"Write\" || name === \"Edit\" || name === \"MultiEdit\") && filePath) {\n    return filePath;\n  }\n  if (name === \"Read\" && filePath) {\n    return filePath;\n  }\n  if ((name === \"Task\" || name === \"Agent\") && input.description) {\n    return String(input.description).slice(0, 200);\n  }\n  if ((name === \"Task\" || name === \"Agent\") && input.prompt) {\n    return String(input.prompt).slice(0, 200);\n  }\n  // Generic: first string value\n  for (const val of Object.values(input)) {\n    if (typeof val === \"string\" && val.length > 0) {\n      return val.slice(0, 200);\n    }\n  }\n  return \"\";\n}\n\nexport function parseClaudeCodeSession(jsonl: string): ParsedSession {\n  const lines = jsonl\n    .split(\"\\n\")\n    .filter((l) => l.trim())\n    .map((l) => JSON.parse(l));\n\n  const events: ParsedEvent[] = [];\n  let sessionId = \"\";\n  let project = \"\";\n  let gitBranch: string | null = null;\n  let model: string | null = null;\n  let version = \"\";\n  let userId = \"\";\n  let startedAt = \"\";\n  let endedAt = \"\";\n  let usedSubagents = false;\n\n  let totalInputTokens = 0;\n  let totalOutputTokens = 0;\n  let totalCacheReadTokens = 0;\n  let totalCacheWriteTokens = 0;\n\n  // Build a map of tool_use_id \u2192 tool_result for matching\n  const toolResults = new Map<string, ToolResultItem>();\n  for (const line of lines) {\n    if (line.type === \"user\") {\n      const content = line.message?.content;\n      if (Array.isArray(content)) {\n        for (const item of content) {\n          if (item.type === \"tool_result\" && item.tool_use_id) {\n            toolResults.set(item.tool_use_id, item as ToolResultItem);\n          }\n        }\n      }\n    }\n  }\n\n  for (const line of lines) {\n    const ts = line.timestamp;\n    if (!ts) continue;\n\n    // Track timestamps\n    if (!startedAt || ts < startedAt) startedAt = ts;\n    if (!endedAt || ts > endedAt) endedAt = ts;\n\n    // Extract session metadata from first user message\n    if (line.type === \"user\" && line.sessionId && !sessionId) {\n      sessionId = line.sessionId;\n      project = projectFromCwd(line.cwd || \"\");\n      gitBranch = line.gitBranch || null;\n      version = line.version || \"\";\n      userId = line.message?.content?.user_id || \"\";\n    }\n\n    // User prompts: only external, with string content (not tool_result arrays)\n    if (line.type === \"user\" && line.userType === \"external\") {\n      const content = line.message?.content;\n      if (typeof content === \"string\" && content.trim()) {\n        events.push({\n          eventType: \"user_prompt\",\n          timestamp: ts,\n          payload: {\n            prompt_text: content,\n            prompt_length: content.length,\n          },\n        });\n      }\n    }\n\n    // Assistant messages: extract llm_response + tool_call events\n    if (line.type === \"assistant\") {\n      const msg = line.message || {};\n      const usage = msg.usage || {};\n\n      // Extract model\n      if (msg.model && !model) {\n        model = msg.model;\n      }\n\n      // LLM response event\n      const inputTokens = usage.input_tokens || 0;\n      const outputTokens = usage.output_tokens || 0;\n      const cacheRead = usage.cache_read_input_tokens || 0;\n      const cacheWrite = usage.cache_creation_input_tokens || 0;\n\n      totalInputTokens += inputTokens;\n      totalOutputTokens += outputTokens;\n      totalCacheReadTokens += cacheRead;\n      totalCacheWriteTokens += cacheWrite;\n\n      const responseModel = msg.model || model;\n      const contextWindow = getContextWindow(responseModel);\n\n      events.push({\n        eventType: \"llm_response\",\n        timestamp: ts,\n        payload: {\n          model: responseModel,\n          input_tokens: inputTokens,\n          output_tokens: outputTokens,\n          cache_read_tokens: cacheRead,\n          cache_write_tokens: cacheWrite,\n          total_tokens: inputTokens + outputTokens,\n          stop_reason: msg.stop_reason || null,\n          ...(contextWindow ? { context_window: contextWindow } : {}),\n        },\n      });\n\n      // Extract tool_use items from content\n      const content = msg.content || [];\n      if (Array.isArray(content)) {\n        for (const item of content) {\n          if (item.type === \"tool_use\") {\n            const toolUse = item as ToolUseItem;\n            const toolResult = toolResults.get(toolUse.id);\n            const isError = toolResult?.is_error === true;\n            const isSubagent = SUBAGENT_TOOLS.has(toolUse.name);\n\n            if (isSubagent) usedSubagents = true;\n\n            const argsSummary = summarizeToolInput(toolUse.name, toolUse.input || {});\n\n            events.push({\n              eventType: \"tool_call\",\n              timestamp: ts,\n              payload: {\n                tool: toolUse.name,\n                args_summary: argsSummary,\n                result_summary: toolResult?.content ? String(toolResult.content).slice(0, 200) : \"\",\n                success: !isError,\n                is_error: isError,\n                ...(isSubagent ? { is_subagent: true } : {}),\n              },\n            });\n\n            // File change events for Write/Edit\n            const changeFilePath = getFilePath(toolUse.input || {});\n            if (toolUse.name === \"Write\" && changeFilePath) {\n              events.push({\n                eventType: \"file_change\",\n                timestamp: ts,\n                payload: { path: changeFilePath, action: \"create\" },\n              });\n            }\n            if ((toolUse.name === \"Edit\" || toolUse.name === \"MultiEdit\") && changeFilePath) {\n              events.push({\n                eventType: \"file_change\",\n                timestamp: ts,\n                payload: { path: changeFilePath, action: \"edit\" },\n              });\n            }\n\n            // Git commit detection from Bash commands\n            if (toolUse.name === \"Bash\" && toolUse.input?.command) {\n              const cmd = String(toolUse.input.command);\n              const resultText = toolResult?.content ? String(toolResult.content) : \"\";\n              if (cmd.includes(\"git commit\")) {\n                events.push({\n                  eventType: \"git_commit\",\n                  timestamp: ts,\n                  payload: {\n                    command: cmd,\n                    result_summary: resultText.slice(0, 200),\n                  },\n                });\n              }\n              if (cmd.includes(\"git push\")) {\n                events.push({\n                  eventType: \"git_push\",\n                  timestamp: ts,\n                  payload: {\n                    command: cmd,\n                    result_summary: resultText.slice(0, 200),\n                  },\n                });\n              }\n              // PR detection from gh pr create / result URLs\n              if (cmd.includes(\"gh pr create\") || cmd.includes(\"gh pr merge\")) {\n                const prMatch = resultText.match(\n                  /https:\\/\\/github\\.com\\/([^/]+\\/[^/]+)\\/pull\\/(\\d+)/,\n                );\n                if (prMatch) {\n                  events.push({\n                    eventType: \"pr_opened\",\n                    timestamp: ts,\n                    payload: {\n                      command: cmd.slice(0, 200),\n                      pr_url: prMatch[0],\n                      repo: prMatch[1],\n                      pr_number: parseInt(prMatch[2], 10),\n                    },\n                  });\n                }\n              }\n            }\n\n            // Error events for failed tool calls\n            if (isError && toolResult?.content) {\n              events.push({\n                eventType: \"error\",\n                timestamp: ts,\n                payload: {\n                  tool: toolUse.name,\n                  message: String(toolResult.content).slice(0, 500),\n                },\n              });\n            }\n          }\n        }\n      }\n    }\n  }\n\n  // Sort events by timestamp\n  events.sort((a, b) => a.timestamp.localeCompare(b.timestamp));\n\n  // Add session_start as first event\n  events.unshift({\n    eventType: \"session_start\",\n    timestamp: startedAt,\n    payload: {\n      model,\n      git_branch: gitBranch,\n      project,\n      harness: \"claude-code\",\n      version,\n      user_id: userId,\n    },\n  });\n\n  // Add session_end as last event\n  events.push({\n    eventType: \"session_end\",\n    timestamp: endedAt,\n    payload: {\n      reason: \"session_end_inferred\",\n    },\n  });\n\n  return {\n    sessionId,\n    project,\n    gitBranch,\n    harness: \"claude-code\",\n    model,\n    version,\n    userId,\n    startedAt,\n    endedAt,\n    events,\n    totalInputTokens,\n    totalOutputTokens,\n    totalCacheReadTokens,\n    totalCacheWriteTokens,\n    usedSubagents,\n  };\n}\n", "/**\n * Parse a Pi session JSONL string into agentlens events.\n *\n * Pi session format (real data from ~/.pi/agent/sessions/):\n * - type: \"session\" \u2014 metadata: { id, version, cwd, timestamp }\n * - type: \"model_change\" \u2014 { provider, modelId }\n * - type: \"thinking_level_change\" \u2014 { thinkingLevel }\n * - type: \"message\" \u2014 conversation turns:\n *   - role: \"user\"       \u2014 content: [{type:\"text\", text:\"...\"}]\n *   - role: \"assistant\"  \u2014 content: [{type:\"toolCall\"|\"text\"|\"thinking\", ...}],\n *                           usage: {input, output, cacheRead, cacheWrite, totalTokens, cost:{total,...}}\n *   - role: \"toolResult\" \u2014 content: [{type:\"text\", text:\"...\"}],\n *                           toolCallId: \"...\", toolName: \"...\", isError: bool\n *\n * Tool argument keys (from real Pi data):\n *   write/edit/read \u2192 args.path (NOT file_path \u2014 that's Claude Code)\n *   bash            \u2192 args.command\n */\n\nimport type { ParsedEvent, ParsedSession } from \"../claude-code/parser\";\nimport { getContextWindow } from \"../model-context\";\n\nconst SUBAGENT_TOOLS = new Set([\"subagent\", \"delegate\", \"Task\", \"Agent\"]);\n\n/** Extract the last path segment as the project name. */\nfunction projectFromCwd(cwd: string): string {\n  const parts = cwd.replace(/\\/+$/, \"\").split(\"/\");\n  return parts[parts.length - 1] || cwd;\n}\n\n/** Get the file path from tool args \u2014 Pi uses `path`, Claude Code uses `file_path`. */\nfunction getFilePath(args: Record<string, any>): string | null {\n  return args.path || args.file_path || null;\n}\n\n/** Build a one-line summary of tool arguments for display. */\nfunction summarizeToolInput(name: string, args: Record<string, any>): string {\n  const lname = name.toLowerCase();\n  if (lname === \"bash\" && args.command) {\n    return String(args.command).slice(0, 200);\n  }\n  const filePath = getFilePath(args);\n  if ((lname === \"write\" || lname === \"edit\" || lname === \"read\") && filePath) {\n    return filePath;\n  }\n  // multi-edit: summarize first path\n  if (lname === \"edit\" && Array.isArray(args.multi) && args.multi.length > 0) {\n    const firstPath = args.multi[0].path || args.multi[0].file_path || \"\";\n    const count = args.multi.length;\n    return count > 1 ? `${firstPath} (+${count - 1} more)` : firstPath;\n  }\n  if (SUBAGENT_TOOLS.has(name)) {\n    return String(args.task || args.description || args.prompt || \"\").slice(0, 200);\n  }\n  // Generic: first non-content string value\n  for (const [key, val] of Object.entries(args)) {\n    if (key === \"content\") continue; // skip file content\n    if (typeof val === \"string\" && val.length > 0) {\n      return val.slice(0, 200);\n    }\n  }\n  return \"\";\n}\n\n// \u2500\u2500 Types for first-pass collection \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\ninterface ToolResultInfo {\n  content: string;\n  isError: boolean;\n  ts: string;\n}\n\n// \u2500\u2500 Main parser \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nexport function parsePiSession(jsonl: string, sessionDir?: string): ParsedSession {\n  const lines = jsonl\n    .split(\"\\n\")\n    .filter((l) => l.trim())\n    .map((l) => JSON.parse(l));\n\n  const events: ParsedEvent[] = [];\n  let sessionId = \"\";\n  let project = \"\";\n  let model: string | null = null;\n  let version = \"\";\n  let startedAt = \"\";\n  let endedAt = \"\";\n  let usedSubagents = false;\n\n  let totalInputTokens = 0;\n  let totalOutputTokens = 0;\n  let totalCacheReadTokens = 0;\n  let totalCacheWriteTokens = 0;\n\n  // \u2500\u2500 First pass: index toolResult messages by toolCallId \u2500\u2500\u2500\u2500\u2500\u2500\n  // Real Pi data has toolCallId on toolResult messages for exact matching.\n  // Fall back to sequential index matching when toolCallId is absent.\n\n  const toolResultById = new Map<string, ToolResultInfo>();\n  const toolResultsByIndex: ToolResultInfo[] = [];\n\n  for (const line of lines) {\n    if (line.type !== \"message\") continue;\n    const msg = line.message || {};\n    if (msg.role !== \"toolResult\") continue;\n\n    const text = (msg.content || [])\n      .filter((c: any) => c.type === \"text\")\n      .map((c: any) => c.text)\n      .join(\"\\n\");\n    const info: ToolResultInfo = {\n      content: text,\n      isError: msg.isError === true,\n      ts: line.timestamp,\n    };\n\n    if (msg.toolCallId) {\n      toolResultById.set(msg.toolCallId, info);\n    }\n    toolResultsByIndex.push(info);\n  }\n\n  // \u2500\u2500 Second pass: generate events \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  let toolCallSeqIndex = 0; // fallback sequential counter\n\n  for (const line of lines) {\n    const ts = line.timestamp;\n    if (!ts) continue;\n\n    if (!startedAt || ts < startedAt) startedAt = ts;\n    if (!endedAt || ts > endedAt) endedAt = ts;\n\n    // Session metadata\n    if (line.type === \"session\") {\n      sessionId = line.id;\n      version = String(line.version || \"\");\n      if (line.cwd) project = projectFromCwd(line.cwd);\n    }\n\n    // Model changes\n    if (line.type === \"model_change\" && line.modelId) {\n      model = line.modelId;\n    }\n\n    // Skip non-message lines from here on\n    if (line.type !== \"message\") continue;\n    const msg = line.message || {};\n\n    // \u2500\u2500 User prompts \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    if (msg.role === \"user\") {\n      const text = (msg.content || [])\n        .filter((c: any) => c.type === \"text\")\n        .map((c: any) => c.text)\n        .join(\"\\n\");\n      if (text.trim()) {\n        events.push({\n          eventType: \"user_prompt\",\n          timestamp: ts,\n          payload: { prompt_text: text, prompt_length: text.length },\n        });\n      }\n    }\n\n    // \u2500\u2500 Assistant messages \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    if (msg.role === \"assistant\") {\n      const usage = msg.usage || {};\n      const inputTokens = usage.input || 0;\n      const outputTokens = usage.output || 0;\n      const cacheRead = usage.cacheRead || 0;\n      const cacheWrite = usage.cacheWrite || 0;\n      const costTotal = usage.cost?.total || 0;\n\n      totalInputTokens += inputTokens;\n      totalOutputTokens += outputTokens;\n      totalCacheReadTokens += cacheRead;\n      totalCacheWriteTokens += cacheWrite;\n\n      const responseModel = msg.model || model;\n      const contextWindow = getContextWindow(responseModel);\n\n      events.push({\n        eventType: \"llm_response\",\n        timestamp: ts,\n        payload: {\n          model: responseModel,\n          input_tokens: inputTokens,\n          output_tokens: outputTokens,\n          cache_read_tokens: cacheRead,\n          cache_write_tokens: cacheWrite,\n          total_tokens: usage.totalTokens || inputTokens + outputTokens,\n          total_cost_usd: costTotal,\n          stop_reason: msg.stop_reason || null,\n          ...(contextWindow ? { context_window: contextWindow } : {}),\n        },\n      });\n\n      // \u2500\u2500 Tool calls within assistant content \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n      for (const c of msg.content || []) {\n        if (c.type !== \"toolCall\") continue;\n\n        const toolId = c.id;\n        const args = c.arguments || c.input || {};\n\n        // Match to result: prefer exact id match, fall back to index\n        const toolResult =\n          toolResultById.get(toolId) || toolResultsByIndex[toolCallSeqIndex] || null;\n        const isError = toolResult?.isError === true;\n        const isSubagent = SUBAGENT_TOOLS.has(c.name);\n        if (isSubagent) usedSubagents = true;\n\n        const argsSummary = summarizeToolInput(c.name, args);\n\n        events.push({\n          eventType: \"tool_call\",\n          timestamp: ts,\n          payload: {\n            tool: c.name,\n            args_summary: argsSummary,\n            result_summary: toolResult?.content ? toolResult.content.slice(0, 200) : \"\",\n            success: !isError,\n            is_error: isError,\n            ...(isSubagent ? { is_subagent: true } : {}),\n          },\n        });\n\n        // \u2500\u2500 File changes \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n        const lname = c.name.toLowerCase();\n        const filePath = getFilePath(args);\n\n        if (lname === \"write\" && filePath) {\n          events.push({\n            eventType: \"file_change\",\n            timestamp: ts,\n            payload: { path: filePath, action: \"create\" },\n          });\n        }\n        if (lname === \"edit\" || lname === \"multiedit\") {\n          if (filePath) {\n            events.push({\n              eventType: \"file_change\",\n              timestamp: ts,\n              payload: { path: filePath, action: \"edit\" },\n            });\n          }\n          // Handle multi-edit: array of {path, oldText, newText}\n          if (Array.isArray(args.multi)) {\n            const seen = new Set<string>();\n            for (const item of args.multi) {\n              const p = item.path || item.file_path;\n              if (p && !seen.has(p)) {\n                seen.add(p);\n                events.push({\n                  eventType: \"file_change\",\n                  timestamp: ts,\n                  payload: { path: p, action: \"edit\" },\n                });\n              }\n            }\n          }\n        }\n\n        // \u2500\u2500 Git detection \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n        if (lname === \"bash\" && args.command) {\n          const cmd = String(args.command);\n          if (cmd.includes(\"git commit\")) {\n            events.push({\n              eventType: \"git_commit\",\n              timestamp: ts,\n              payload: {\n                command: cmd,\n                result_summary: toolResult?.content?.slice(0, 200) || \"\",\n              },\n            });\n          }\n          if (cmd.includes(\"git push\")) {\n            events.push({\n              eventType: \"git_push\",\n              timestamp: ts,\n              payload: {\n                command: cmd,\n                result_summary: toolResult?.content?.slice(0, 200) || \"\",\n              },\n            });\n          }\n          // PR detection from gh pr create / result URLs\n          if (cmd.includes(\"gh pr create\") || cmd.includes(\"gh pr merge\")) {\n            const resultText = toolResult?.content || \"\";\n            const prMatch = resultText.match(/https:\\/\\/github\\.com\\/([^/]+\\/[^/]+)\\/pull\\/(\\d+)/);\n            if (prMatch) {\n              events.push({\n                eventType: \"pr_opened\",\n                timestamp: ts,\n                payload: {\n                  command: cmd.slice(0, 200),\n                  pr_url: prMatch[0],\n                  repo: prMatch[1],\n                  pr_number: parseInt(prMatch[2], 10),\n                },\n              });\n            }\n          }\n        }\n\n        // \u2500\u2500 Error events \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n        if (isError && toolResult?.content) {\n          events.push({\n            eventType: \"error\",\n            timestamp: ts,\n            payload: { tool: c.name, message: toolResult.content.slice(0, 500) },\n          });\n        }\n\n        toolCallSeqIndex++;\n      }\n    }\n  }\n\n  // Sort events by timestamp\n  events.sort((a, b) => a.timestamp.localeCompare(b.timestamp));\n\n  // Bookend events\n  events.unshift({\n    eventType: \"session_start\",\n    timestamp: startedAt,\n    payload: { model, project, harness: \"pi\", version },\n  });\n  events.push({\n    eventType: \"session_end\",\n    timestamp: endedAt,\n    payload: { reason: \"session_end_inferred\" },\n  });\n\n  return {\n    sessionId,\n    project,\n    gitBranch: null,\n    harness: \"pi\",\n    model,\n    version,\n    userId: \"\",\n    startedAt,\n    endedAt,\n    events,\n    totalInputTokens,\n    totalOutputTokens,\n    totalCacheReadTokens,\n    totalCacheWriteTokens,\n    usedSubagents,\n  };\n}\n", "/**\n * File watcher: polls for new/updated session JSONL files and imports them.\n * Uses polling (not fs.watch) for reliability across platforms and network drives.\n */\n\nimport type { BetterSQLite3Database } from \"drizzle-orm/better-sqlite3\";\nimport { indexSessions, type IndexOptions } from \"./indexer\";\n\nexport interface WatcherOptions extends IndexOptions {\n  /** Poll interval in milliseconds (default: 5000) */\n  pollIntervalMs?: number;\n}\n\nexport interface Watcher {\n  stop: () => void;\n}\n\nexport function createWatcher(db: BetterSQLite3Database, options: WatcherOptions = {}): Watcher {\n  const interval = options.pollIntervalMs ?? 5000;\n  let stopped = false;\n  let timer: ReturnType<typeof setTimeout> | null = null;\n\n  function poll() {\n    if (stopped) return;\n\n    try {\n      indexSessions(db, options);\n    } catch {\n      // Swallow errors \u2014 keep polling\n    }\n\n    if (!stopped) {\n      timer = setTimeout(poll, interval);\n    }\n  }\n\n  // Start first poll immediately\n  poll();\n\n  return {\n    stop() {\n      stopped = true;\n      if (timer) {\n        clearTimeout(timer);\n        timer = null;\n      }\n    },\n  };\n}\n", "import Database from \"better-sqlite3\";\nimport { drizzle } from \"drizzle-orm/better-sqlite3\";\nimport { sql } from \"drizzle-orm\";\nimport type { BetterSQLite3Database } from \"drizzle-orm/better-sqlite3\";\nimport path from \"path\";\nimport fs from \"fs\";\nimport { homedir } from \"os\";\n\nlet _db: BetterSQLite3Database | null = null;\n\nexport function getDb(): BetterSQLite3Database {\n  if (_db) return _db;\n\n  const dbPath =\n    process.env.AGENTLENS_DB_PATH || path.join(homedir(), \".agentlens\", \"agentlens.db\");\n  const dataDir = path.dirname(dbPath);\n  if (!fs.existsSync(dataDir)) {\n    fs.mkdirSync(dataDir, { recursive: true });\n  }\n  const sqlite = new Database(dbPath);\n\n  // Enable WAL mode for better concurrent read/write\n  sqlite.pragma(\"journal_mode = WAL\");\n\n  const db = drizzle(sqlite);\n\n  // Auto-create tables if they don't exist\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS sessions (\n      id TEXT PRIMARY KEY,\n      team_id TEXT NOT NULL,\n      user_id TEXT NOT NULL,\n      project TEXT NOT NULL,\n      git_repo TEXT,\n      git_branch TEXT,\n      pi_version TEXT NOT NULL DEFAULT '',\n      skills_used TEXT NOT NULL DEFAULT '[]',\n      model TEXT,\n      harness TEXT NOT NULL DEFAULT 'pi',\n      started_at TEXT NOT NULL,\n      ended_at TEXT,\n      status TEXT NOT NULL DEFAULT 'active',\n      created_at TEXT NOT NULL DEFAULT (datetime('now'))\n    )\n  `);\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS events (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      session_id TEXT NOT NULL,\n      event_type TEXT NOT NULL,\n      timestamp TEXT NOT NULL,\n      payload TEXT NOT NULL DEFAULT '{}',\n      created_at TEXT NOT NULL DEFAULT (datetime('now'))\n    )\n  `);\n\n  // Migrations for existing databases\n  try {\n    db.run(sql`ALTER TABLE sessions ADD COLUMN model TEXT`);\n  } catch {\n    // Column already exists\n  }\n\n  // Create indexes\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_sessions_team_id ON sessions(team_id)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_sessions_user_id ON sessions(user_id)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_sessions_project ON sessions(project)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_sessions_started_at ON sessions(started_at)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_sessions_status ON sessions(status)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_events_session_id ON events(session_id)`);\n  db.run(\n    sql`CREATE INDEX IF NOT EXISTS idx_events_session_timestamp ON events(session_id, timestamp)`,\n  );\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_events_event_type ON events(event_type)`);\n\n  // PR tracking table\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS pull_requests (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      session_id TEXT NOT NULL,\n      repo TEXT NOT NULL,\n      branch TEXT NOT NULL,\n      pr_number INTEGER NOT NULL,\n      pr_url TEXT NOT NULL,\n      pr_title TEXT NOT NULL DEFAULT '',\n      status TEXT NOT NULL DEFAULT 'open',\n      merged_at TEXT,\n      reviews_approved INTEGER NOT NULL DEFAULT 0,\n      reviews_changes_requested INTEGER NOT NULL DEFAULT 0,\n      review_comments INTEGER NOT NULL DEFAULT 0,\n      commits_after_review INTEGER NOT NULL DEFAULT 0,\n      first_review_at TEXT,\n      time_to_merge_hours REAL,\n      checked_at TEXT,\n      created_at TEXT NOT NULL DEFAULT (datetime('now')),\n      UNIQUE(session_id, pr_number)\n    )\n  `);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_prs_session_id ON pull_requests(session_id)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_prs_status ON pull_requests(status)`);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_prs_repo_branch ON pull_requests(repo, branch)`);\n\n  // Migrations: add columns that may be missing in existing DBs\n  try {\n    db.run(sql`ALTER TABLE sessions ADD COLUMN model TEXT`);\n  } catch {\n    // Column already exists\n  }\n  try {\n    db.run(sql`ALTER TABLE sessions ADD COLUMN harness TEXT NOT NULL DEFAULT 'pi'`);\n  } catch {\n    // Column already exists\n  }\n\n  // Evaluation tables\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS eval_checks (\n      id INTEGER PRIMARY KEY AUTOINCREMENT,\n      session_id TEXT NOT NULL,\n      check_id TEXT NOT NULL,\n      check_version TEXT NOT NULL,\n      score REAL NOT NULL,\n      raw_score INTEGER NOT NULL,\n      reasoning TEXT NOT NULL DEFAULT '',\n      evidence TEXT,\n      model TEXT NOT NULL,\n      cost_usd REAL NOT NULL DEFAULT 0,\n      evaluated_at TEXT NOT NULL DEFAULT (datetime('now')),\n      UNIQUE(session_id, check_id, check_version)\n    )\n  `);\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS eval_summaries (\n      session_id TEXT PRIMARY KEY,\n      qual_score REAL NOT NULL,\n      checks_passed INTEGER NOT NULL DEFAULT 0,\n      checks_total INTEGER NOT NULL DEFAULT 0,\n      total_eval_cost_usd REAL NOT NULL DEFAULT 0,\n      evaluated_at TEXT NOT NULL DEFAULT (datetime('now'))\n    )\n  `);\n  db.run(sql`CREATE INDEX IF NOT EXISTS idx_eval_checks_session ON eval_checks(session_id)`);\n  db.run(\n    sql`CREATE INDEX IF NOT EXISTS idx_eval_checks_lookup ON eval_checks(session_id, check_id, check_version)`,\n  );\n\n  // Session insights table (team insights feature)\n  db.run(sql`\n    CREATE TABLE IF NOT EXISTS session_insights (\n      session_id TEXT PRIMARY KEY,\n      task_category TEXT NOT NULL,\n      task_size TEXT NOT NULL,\n      workflow_tags TEXT NOT NULL DEFAULT '[]',\n      prompting_tags TEXT NOT NULL DEFAULT '[]',\n      failure_modes TEXT NOT NULL DEFAULT '[]',\n      complexity_factors TEXT NOT NULL DEFAULT '[]',\n      key_learning TEXT,\n      extracted_at TEXT NOT NULL DEFAULT (datetime('now')),\n      model TEXT NOT NULL,\n      synced_at TEXT\n    )\n  `);\n\n  // Migrations for session_insights new columns\n  for (const col of [\n    \"failure_modes TEXT NOT NULL DEFAULT '[]'\",\n    \"complexity_factors TEXT NOT NULL DEFAULT '[]'\",\n    \"key_learning TEXT\",\n  ]) {\n    try {\n      db.run(sql.raw(`ALTER TABLE session_insights ADD COLUMN ${col}`));\n    } catch {\n      // Column already exists\n    }\n  }\n\n  _db = db;\n  return db;\n}\n"],
  "mappings": "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",
  "names": ["join", "existsSync", "readFileSync", "homedir", "eq", "events", "SUBAGENT_TOOLS", "projectFromCwd", "getFilePath", "args", "summarizeToolInput", "count", "events", "text", "eq", "sql", "dbPath", "dataDir", "join", "homedir", "existsSync", "require", "text", "readFileSync"]
}

|