@a-company/paradigm 5.9.0 → 5.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (29) hide show
  1. package/dist/{accept-orchestration-GX2YRWM4.js → accept-orchestration-UQLM7PTQ.js} +4 -4
  2. package/dist/{agent-loader-X7TDYLFL.js → agent-loader-TFIANSF4.js} +1 -1
  3. package/dist/agent-state-S5DAWPTF.js +24 -0
  4. package/dist/{chunk-3UCH56D5.js → chunk-4BLYIB7J.js} +270 -928
  5. package/dist/chunk-4L3UTYQX.js +677 -0
  6. package/dist/chunk-54LTTQBH.js +138 -0
  7. package/dist/chunk-5OUOLN6M.js +659 -0
  8. package/dist/{chunk-SDDCVUCV.js → chunk-CL7JSK52.js} +23 -0
  9. package/dist/{chunk-MA7G4CTI.js → chunk-RJE5G7WO.js} +27 -1
  10. package/dist/{chunk-EI32ZBE6.js → chunk-RTHA3XRE.js} +19 -672
  11. package/dist/{chunk-V7BZBBI6.js → chunk-VPPK3SY4.js} +1 -1
  12. package/dist/{chunk-WQITYKHM.js → chunk-YRZ5RPEB.js} +7 -7
  13. package/dist/{diff-RQLLNAFI.js → diff-D4X53HAC.js} +4 -4
  14. package/dist/{docs-AIY6VNF7.js → docs-QIYKO3BR.js} +1 -1
  15. package/dist/index.js +19 -19
  16. package/dist/mcp.js +140 -66
  17. package/dist/model-discovery-D2H3VBGC.js +8 -0
  18. package/dist/{nomination-engine-LLREC5BZ.js → nomination-engine-RV5CNO5B.js} +2 -2
  19. package/dist/{orchestrate-XZA33TJC.js → orchestrate-JLILBBJE.js} +4 -4
  20. package/dist/{reindex-U2HEB6GW.js → reindex-5LTD53ZC.js} +3 -2
  21. package/dist/{serve-QWWJP2EW.js → serve-CAH3PHE7.js} +1 -1
  22. package/dist/session-tracker-C4BMD5WG.js +13 -0
  23. package/dist/{session-work-log-KDOH4GER.js → session-work-log-MZ47OAPB.js} +1 -1
  24. package/dist/{shift-VJUGMADR.js → shift-D2JOHHBF.js} +33 -5
  25. package/dist/{spawn-AW6GDECS.js → spawn-RCHNXDHE.js} +4 -4
  26. package/dist/{team-7HG7XK5C.js → team-O5MIIFMA.js} +6 -5
  27. package/package.json +1 -1
  28. package/dist/{chunk-LSRABQIY.js → chunk-45MUDW6E.js} +3 -3
  29. /package/dist/{platform-server-U5L2G3EU.js → platform-server-H5YO3DQD.js} +0 -0
@@ -0,0 +1,677 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ init_session_work_log,
4
+ session_work_log_exports
5
+ } from "./chunk-CL7JSK52.js";
6
+ import {
7
+ __toCommonJS
8
+ } from "./chunk-7N7GSU6K.js";
9
+
10
+ // ../paradigm-mcp/src/utils/session-tracker.ts
11
+ import * as fs2 from "fs";
12
+ import * as path2 from "path";
13
+
14
+ // ../paradigm-mcp/src/utils/global-store.ts
15
+ import * as fs from "fs";
16
+ import * as path from "path";
17
+ import * as os from "os";
18
+ import * as crypto from "crypto";
19
+ import * as yaml from "js-yaml";
20
+ function getGlobalDir() {
21
+ const dir = path.join(os.homedir(), ".paradigm");
22
+ if (!fs.existsSync(dir)) {
23
+ fs.mkdirSync(dir, { recursive: true });
24
+ }
25
+ return dir;
26
+ }
27
+ function getProjectHash(rootDir) {
28
+ const absolute = path.resolve(rootDir);
29
+ return crypto.createHash("sha256").update(absolute).digest("hex").slice(0, 12);
30
+ }
31
+ function getSessionDir(rootDir) {
32
+ const hash = getProjectHash(rootDir);
33
+ const dir = path.join(getGlobalDir(), "sessions", hash);
34
+ if (!fs.existsSync(dir)) {
35
+ fs.mkdirSync(dir, { recursive: true });
36
+ }
37
+ const handoffsDir = path.join(dir, "pending-handoffs");
38
+ if (!fs.existsSync(handoffsDir)) {
39
+ fs.mkdirSync(handoffsDir, { recursive: true });
40
+ }
41
+ return dir;
42
+ }
43
+ function writeProjectMeta(rootDir) {
44
+ const sessionDir = getSessionDir(rootDir);
45
+ const metaPath = path.join(sessionDir, "_project-meta.json");
46
+ const projectName = path.basename(path.resolve(rootDir));
47
+ const meta = {
48
+ name: projectName,
49
+ path: path.resolve(rootDir),
50
+ lastSeen: (/* @__PURE__ */ new Date()).toISOString()
51
+ };
52
+ fs.writeFileSync(metaPath, JSON.stringify(meta, null, 2));
53
+ }
54
+ function writePendingHandoff(rootDir, handoff) {
55
+ const sessionDir = getSessionDir(rootDir);
56
+ const filePath = path.join(sessionDir, "pending-handoffs", `${handoff.id}.json`);
57
+ fs.writeFileSync(filePath, JSON.stringify(handoff, null, 2));
58
+ }
59
+ function loadPendingHandoffs(rootDir) {
60
+ const sessionDir = getSessionDir(rootDir);
61
+ const handoffsDir = path.join(sessionDir, "pending-handoffs");
62
+ if (!fs.existsSync(handoffsDir)) {
63
+ return [];
64
+ }
65
+ const handoffs = [];
66
+ try {
67
+ const files = fs.readdirSync(handoffsDir);
68
+ for (const file of files) {
69
+ if (!file.endsWith(".json")) continue;
70
+ try {
71
+ const content = fs.readFileSync(path.join(handoffsDir, file), "utf8");
72
+ const handoff = JSON.parse(content);
73
+ if (handoff.status === "pending") {
74
+ handoffs.push(handoff);
75
+ }
76
+ } catch {
77
+ }
78
+ }
79
+ } catch {
80
+ }
81
+ handoffs.sort((a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime());
82
+ return handoffs;
83
+ }
84
+ function markHandoffDelivered(rootDir, handoffId) {
85
+ const sessionDir = getSessionDir(rootDir);
86
+ const filePath = path.join(sessionDir, "pending-handoffs", `${handoffId}.json`);
87
+ if (!fs.existsSync(filePath)) return;
88
+ try {
89
+ const content = fs.readFileSync(filePath, "utf8");
90
+ const handoff = JSON.parse(content);
91
+ handoff.status = "delivered";
92
+ fs.writeFileSync(filePath, JSON.stringify(handoff, null, 2));
93
+ } catch {
94
+ }
95
+ }
96
+ function getGlobalWisdomDir() {
97
+ const dir = path.join(getGlobalDir(), "wisdom");
98
+ if (!fs.existsSync(dir)) {
99
+ fs.mkdirSync(dir, { recursive: true });
100
+ }
101
+ return dir;
102
+ }
103
+ function loadGlobalAntipatterns() {
104
+ const filePath = path.join(getGlobalWisdomDir(), "antipatterns.yaml");
105
+ if (!fs.existsSync(filePath)) return [];
106
+ try {
107
+ const content = fs.readFileSync(filePath, "utf8");
108
+ const data = yaml.load(content);
109
+ return data?.antipatterns || [];
110
+ } catch {
111
+ return [];
112
+ }
113
+ }
114
+ function loadGlobalDecisions() {
115
+ const decisionsDir = path.join(getGlobalWisdomDir(), "decisions");
116
+ if (!fs.existsSync(decisionsDir)) return [];
117
+ const decisions = [];
118
+ try {
119
+ const files = fs.readdirSync(decisionsDir);
120
+ for (const file of files) {
121
+ if (!file.endsWith(".yaml") && !file.endsWith(".yml")) continue;
122
+ try {
123
+ const content = fs.readFileSync(path.join(decisionsDir, file), "utf8");
124
+ const decision = yaml.load(content);
125
+ decisions.push(decision);
126
+ } catch {
127
+ }
128
+ }
129
+ } catch {
130
+ }
131
+ decisions.sort((a, b) => a.id.localeCompare(b.id));
132
+ return decisions;
133
+ }
134
+ function loadGlobalPreferences() {
135
+ const filePath = path.join(getGlobalWisdomDir(), "preferences.yaml");
136
+ if (!fs.existsSync(filePath)) return null;
137
+ try {
138
+ const content = fs.readFileSync(filePath, "utf8");
139
+ return yaml.load(content);
140
+ } catch {
141
+ return null;
142
+ }
143
+ }
144
+ function recordGlobalAntipattern(antipattern) {
145
+ const filePath = path.join(getGlobalWisdomDir(), "antipatterns.yaml");
146
+ let data = { version: "1.0", antipatterns: [] };
147
+ if (fs.existsSync(filePath)) {
148
+ try {
149
+ const content = fs.readFileSync(filePath, "utf8");
150
+ data = yaml.load(content);
151
+ if (!data.antipatterns) data.antipatterns = [];
152
+ } catch {
153
+ }
154
+ }
155
+ data.antipatterns.push({
156
+ ...antipattern,
157
+ added: (/* @__PURE__ */ new Date()).toISOString()
158
+ });
159
+ fs.writeFileSync(filePath, yaml.dump(data, { lineWidth: -1 }));
160
+ }
161
+ function recordGlobalDecision(decision) {
162
+ const decisionsDir = path.join(getGlobalWisdomDir(), "decisions");
163
+ if (!fs.existsSync(decisionsDir)) {
164
+ fs.mkdirSync(decisionsDir, { recursive: true });
165
+ }
166
+ const slug = decision.title.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/^-|-$/g, "");
167
+ const fileName = `${decision.id}-${slug}.yaml`;
168
+ const filePath = path.join(decisionsDir, fileName);
169
+ fs.writeFileSync(filePath, yaml.dump(decision, { lineWidth: -1 }));
170
+ }
171
+
172
+ // ../paradigm-mcp/src/utils/session-tracker.ts
173
+ var MODEL_PRICING = {
174
+ "claude-opus-4": { input: 15, output: 75, name: "Claude Opus 4" },
175
+ "claude-sonnet-4": { input: 3, output: 15, name: "Claude Sonnet 4" },
176
+ "claude-haiku-3.5": { input: 0.8, output: 4, name: "Claude Haiku 3.5" }
177
+ };
178
+ var MAX_BREADCRUMBS = 50;
179
+ var BREADCRUMBS_FILE = ".paradigm/session-breadcrumbs.json";
180
+ var CHECKPOINT_FILE = ".paradigm/session-checkpoint.json";
181
+ var CHECKPOINT_MAX_AGE_MS = 7 * 24 * 60 * 60 * 1e3;
182
+ var SessionTracker = class {
183
+ session;
184
+ rootDir = null;
185
+ _recovered = false;
186
+ lastLoreEntryId = null;
187
+ constructor() {
188
+ this.session = this.createNewSession();
189
+ }
190
+ /**
191
+ * Set the project root directory (for persisting breadcrumbs)
192
+ */
193
+ setRootDir(rootDir) {
194
+ this.rootDir = rootDir;
195
+ try {
196
+ const { clearSessionWorkLog } = (init_session_work_log(), __toCommonJS(session_work_log_exports));
197
+ clearSessionWorkLog(rootDir);
198
+ } catch {
199
+ }
200
+ }
201
+ createNewSession() {
202
+ return {
203
+ sessionId: `s${Date.now().toString(36)}`,
204
+ startTime: Date.now(),
205
+ lastActivity: Date.now(),
206
+ model: "claude-sonnet-4",
207
+ resourceReads: [],
208
+ toolCalls: [],
209
+ breadcrumbs: [],
210
+ totals: {
211
+ resourceReadCount: 0,
212
+ toolCallCount: 0,
213
+ totalBytes: 0,
214
+ totalTokens: 0,
215
+ estimatedCostUsd: 0
216
+ }
217
+ };
218
+ }
219
+ /**
220
+ * Add a breadcrumb (summarized action for session recovery)
221
+ */
222
+ addBreadcrumb(action, summary, options = {}) {
223
+ this.session.breadcrumbs.push({
224
+ timestamp: Date.now(),
225
+ action,
226
+ tool: options.tool,
227
+ symbol: options.symbol,
228
+ summary
229
+ });
230
+ if (this.session.breadcrumbs.length > MAX_BREADCRUMBS) {
231
+ this.session.breadcrumbs = this.session.breadcrumbs.slice(-MAX_BREADCRUMBS);
232
+ }
233
+ this.persistBreadcrumbs();
234
+ }
235
+ /**
236
+ * Get recent breadcrumbs
237
+ */
238
+ getBreadcrumbs(limit = 20) {
239
+ return this.session.breadcrumbs.slice(-limit);
240
+ }
241
+ /**
242
+ * Persist breadcrumbs to file (dual-write: local + global)
243
+ */
244
+ persistBreadcrumbs() {
245
+ if (!this.rootDir) return;
246
+ const data = {
247
+ sessionId: this.session.sessionId,
248
+ startTime: this.session.startTime,
249
+ lastActivity: this.session.lastActivity,
250
+ breadcrumbs: this.session.breadcrumbs,
251
+ symbolsModified: this.extractSymbolsFromBreadcrumbs(),
252
+ filesExplored: this.extractFilesFromBreadcrumbs()
253
+ };
254
+ let jsonData;
255
+ try {
256
+ jsonData = JSON.stringify(data, null, 2);
257
+ } catch (err) {
258
+ console.error("[paradigm-mcp] persistBreadcrumbs: JSON.stringify failed:", err.message);
259
+ return;
260
+ }
261
+ try {
262
+ const filePath = path2.join(this.rootDir, BREADCRUMBS_FILE);
263
+ const dir = path2.dirname(filePath);
264
+ if (!fs2.existsSync(dir)) {
265
+ fs2.mkdirSync(dir, { recursive: true });
266
+ }
267
+ fs2.writeFileSync(filePath, jsonData);
268
+ } catch (err) {
269
+ console.error("[paradigm-mcp] persistBreadcrumbs: local write failed:", err.message);
270
+ }
271
+ try {
272
+ const globalSessionDir = getSessionDir(this.rootDir);
273
+ fs2.writeFileSync(path2.join(globalSessionDir, "breadcrumbs.json"), jsonData);
274
+ writeProjectMeta(this.rootDir);
275
+ } catch (err) {
276
+ console.error("[paradigm-mcp] persistBreadcrumbs: global write failed:", err.message);
277
+ }
278
+ }
279
+ /**
280
+ * Load previous session breadcrumbs from file.
281
+ * Prefers global path (~/.paradigm/sessions/{hash}/breadcrumbs.json),
282
+ * falls back to local (.paradigm/session-breadcrumbs.json).
283
+ */
284
+ loadPreviousSession() {
285
+ if (!this.rootDir) return null;
286
+ try {
287
+ const globalSessionDir = getSessionDir(this.rootDir);
288
+ const globalPath = path2.join(globalSessionDir, "breadcrumbs.json");
289
+ if (fs2.existsSync(globalPath)) {
290
+ const content = fs2.readFileSync(globalPath, "utf8");
291
+ return JSON.parse(content);
292
+ }
293
+ } catch {
294
+ }
295
+ try {
296
+ const filePath = path2.join(this.rootDir, BREADCRUMBS_FILE);
297
+ if (!fs2.existsSync(filePath)) return null;
298
+ const content = fs2.readFileSync(filePath, "utf8");
299
+ return JSON.parse(content);
300
+ } catch {
301
+ return null;
302
+ }
303
+ }
304
+ /**
305
+ * Save a cognitive-transition checkpoint for crash recovery.
306
+ * Fills in timestamp, sessionId, and snapshots recent breadcrumbs.
307
+ * Returns the checkpoint and whether it was persisted to disk.
308
+ */
309
+ saveCheckpoint(data) {
310
+ const checkpoint = {
311
+ phase: data.phase,
312
+ context: data.context,
313
+ timestamp: Date.now(),
314
+ sessionId: this.session.sessionId,
315
+ externalId: data.externalId,
316
+ plan: data.plan,
317
+ modifiedFiles: data.modifiedFiles,
318
+ symbolsTouched: data.symbolsTouched,
319
+ decisions: data.decisions,
320
+ recentBreadcrumbs: this.session.breadcrumbs.slice(-10)
321
+ };
322
+ const persisted = this.persistCheckpoint(checkpoint);
323
+ return { checkpoint, persisted };
324
+ }
325
+ /**
326
+ * Load the most recent checkpoint.
327
+ * Prefers global path, falls back to local.
328
+ * Returns null for checkpoints older than 7 days.
329
+ */
330
+ loadCheckpoint() {
331
+ if (!this.rootDir) return null;
332
+ let checkpoint = null;
333
+ try {
334
+ const globalSessionDir = getSessionDir(this.rootDir);
335
+ const globalPath = path2.join(globalSessionDir, "checkpoint.json");
336
+ if (fs2.existsSync(globalPath)) {
337
+ const content = fs2.readFileSync(globalPath, "utf8");
338
+ checkpoint = JSON.parse(content);
339
+ }
340
+ } catch {
341
+ }
342
+ if (!checkpoint) {
343
+ try {
344
+ const localPath = path2.join(this.rootDir, CHECKPOINT_FILE);
345
+ if (fs2.existsSync(localPath)) {
346
+ const content = fs2.readFileSync(localPath, "utf8");
347
+ checkpoint = JSON.parse(content);
348
+ }
349
+ } catch {
350
+ }
351
+ }
352
+ if (checkpoint && Date.now() - checkpoint.timestamp > CHECKPOINT_MAX_AGE_MS) {
353
+ return null;
354
+ }
355
+ if (checkpoint) {
356
+ for (const key of ["modifiedFiles", "symbolsTouched", "decisions"]) {
357
+ const val = checkpoint[key];
358
+ if (typeof val === "string") {
359
+ try {
360
+ checkpoint[key] = JSON.parse(val);
361
+ } catch {
362
+ checkpoint[key] = [];
363
+ }
364
+ }
365
+ }
366
+ }
367
+ return checkpoint;
368
+ }
369
+ /**
370
+ * Persist checkpoint to both local and global paths.
371
+ * Returns which writes succeeded so callers can report accurately.
372
+ */
373
+ persistCheckpoint(checkpoint) {
374
+ const result = { local: false, global: false };
375
+ if (!this.rootDir) {
376
+ console.error("[paradigm-mcp] persistCheckpoint: rootDir not set, skipping write");
377
+ return result;
378
+ }
379
+ let jsonData;
380
+ try {
381
+ jsonData = JSON.stringify(checkpoint, null, 2);
382
+ } catch (err) {
383
+ console.error("[paradigm-mcp] persistCheckpoint: JSON.stringify failed:", err.message);
384
+ return result;
385
+ }
386
+ try {
387
+ const filePath = path2.join(this.rootDir, CHECKPOINT_FILE);
388
+ const dir = path2.dirname(filePath);
389
+ if (!fs2.existsSync(dir)) {
390
+ fs2.mkdirSync(dir, { recursive: true });
391
+ }
392
+ fs2.writeFileSync(filePath, jsonData);
393
+ result.local = true;
394
+ } catch (err) {
395
+ console.error("[paradigm-mcp] persistCheckpoint: local write failed:", err.message);
396
+ }
397
+ try {
398
+ const globalSessionDir = getSessionDir(this.rootDir);
399
+ fs2.writeFileSync(path2.join(globalSessionDir, "checkpoint.json"), jsonData);
400
+ writeProjectMeta(this.rootDir);
401
+ result.global = true;
402
+ } catch (err) {
403
+ console.error("[paradigm-mcp] persistCheckpoint: global write failed:", err.message);
404
+ }
405
+ return result;
406
+ }
407
+ /**
408
+ * Set the last lore entry ID recorded in this session
409
+ */
410
+ setLastLoreEntryId(id) {
411
+ this.lastLoreEntryId = id;
412
+ }
413
+ /**
414
+ * Get the last lore entry ID recorded in this session
415
+ */
416
+ getLastLoreEntryId() {
417
+ return this.lastLoreEntryId;
418
+ }
419
+ /**
420
+ * Check whether auto-recovery has already fired this session.
421
+ */
422
+ hasRecoveredThisSession() {
423
+ return this._recovered;
424
+ }
425
+ /**
426
+ * Mark that auto-recovery has fired (so it only fires once per session).
427
+ */
428
+ markRecovered() {
429
+ this._recovered = true;
430
+ }
431
+ /**
432
+ * Extract symbols from breadcrumbs
433
+ */
434
+ extractSymbolsFromBreadcrumbs() {
435
+ const symbols = /* @__PURE__ */ new Set();
436
+ for (const bc of this.session.breadcrumbs) {
437
+ if (bc.symbol) symbols.add(bc.symbol);
438
+ }
439
+ return Array.from(symbols);
440
+ }
441
+ /**
442
+ * Extract files from breadcrumbs
443
+ */
444
+ extractFilesFromBreadcrumbs() {
445
+ const files = /* @__PURE__ */ new Set();
446
+ for (const bc of this.session.breadcrumbs) {
447
+ const matches = bc.summary.match(/\b[\w./]+\.(ts|js|tsx|jsx|py|go|rs|yaml|json|md)\b/g);
448
+ if (matches) {
449
+ for (const m of matches) {
450
+ files.add(m);
451
+ }
452
+ }
453
+ }
454
+ return Array.from(files);
455
+ }
456
+ /**
457
+ * Estimate tokens from text (approx 3.5 chars per token)
458
+ */
459
+ estimateTokens(text) {
460
+ const len = typeof text === "number" ? text : text.length;
461
+ return Math.ceil(len / 3.5);
462
+ }
463
+ /**
464
+ * Calculate cost for tokens using current model pricing
465
+ */
466
+ calculateCost(tokens, isOutput = true) {
467
+ const pricing = MODEL_PRICING[this.session.model];
468
+ const rate = isOutput ? pricing.output : pricing.input;
469
+ return tokens / 1e6 * rate;
470
+ }
471
+ /**
472
+ * Set the model for cost calculations
473
+ */
474
+ setModel(model) {
475
+ this.session.model = model;
476
+ this.recalculateTotals();
477
+ }
478
+ /**
479
+ * Get current model
480
+ */
481
+ getModel() {
482
+ return this.session.model;
483
+ }
484
+ /**
485
+ * Track a resource read
486
+ */
487
+ trackResourceRead(uri, bytes) {
488
+ const resourceType = this.extractResourceType(uri);
489
+ const tokens = this.estimateTokens(bytes);
490
+ this.session.resourceReads.push({
491
+ timestamp: Date.now(),
492
+ resourceType,
493
+ uri,
494
+ bytes,
495
+ tokens
496
+ });
497
+ this.session.lastActivity = Date.now();
498
+ this.updateTotals(bytes, tokens);
499
+ }
500
+ /**
501
+ * Track a tool call
502
+ */
503
+ trackToolCall(toolName, responseBytes) {
504
+ const tokens = this.estimateTokens(responseBytes);
505
+ this.session.toolCalls.push({
506
+ timestamp: Date.now(),
507
+ toolName,
508
+ responseBytes,
509
+ responseTokens: tokens
510
+ });
511
+ this.session.lastActivity = Date.now();
512
+ this.updateTotals(responseBytes, tokens);
513
+ }
514
+ /**
515
+ * Update running totals
516
+ */
517
+ updateTotals(bytes, tokens) {
518
+ this.session.totals.resourceReadCount = this.session.resourceReads.length;
519
+ this.session.totals.toolCallCount = this.session.toolCalls.length;
520
+ this.session.totals.totalBytes += bytes;
521
+ this.session.totals.totalTokens += tokens;
522
+ this.session.totals.estimatedCostUsd = this.calculateCost(this.session.totals.totalTokens);
523
+ }
524
+ /**
525
+ * Recalculate totals (used when model changes)
526
+ */
527
+ recalculateTotals() {
528
+ this.session.totals.estimatedCostUsd = this.calculateCost(this.session.totals.totalTokens);
529
+ }
530
+ /**
531
+ * Extract resource type from URI
532
+ */
533
+ extractResourceType(uri) {
534
+ const path3 = uri.replace("paradigm://", "");
535
+ const firstPart = path3.split("/")[0];
536
+ return firstPart || "unknown";
537
+ }
538
+ /**
539
+ * Get session statistics
540
+ */
541
+ getStats() {
542
+ return { ...this.session };
543
+ }
544
+ /**
545
+ * Get detailed cost breakdown
546
+ */
547
+ getCostBreakdown() {
548
+ const resourcesByType = {};
549
+ let resourceBytes = 0;
550
+ let resourceTokens = 0;
551
+ for (const read of this.session.resourceReads) {
552
+ if (!resourcesByType[read.resourceType]) {
553
+ resourcesByType[read.resourceType] = { count: 0, bytes: 0, tokens: 0 };
554
+ }
555
+ resourcesByType[read.resourceType].count++;
556
+ resourcesByType[read.resourceType].bytes += read.bytes;
557
+ resourcesByType[read.resourceType].tokens += read.tokens;
558
+ resourceBytes += read.bytes;
559
+ resourceTokens += read.tokens;
560
+ }
561
+ const toolsByName = {};
562
+ let toolBytes = 0;
563
+ let toolTokens = 0;
564
+ for (const call of this.session.toolCalls) {
565
+ if (!toolsByName[call.toolName]) {
566
+ toolsByName[call.toolName] = { count: 0, bytes: 0, tokens: 0 };
567
+ }
568
+ toolsByName[call.toolName].count++;
569
+ toolsByName[call.toolName].bytes += call.responseBytes;
570
+ toolsByName[call.toolName].tokens += call.responseTokens;
571
+ toolBytes += call.responseBytes;
572
+ toolTokens += call.responseTokens;
573
+ }
574
+ const totalTokens = resourceTokens + toolTokens;
575
+ const totalCost = this.calculateCost(totalTokens);
576
+ return {
577
+ model: MODEL_PRICING[this.session.model].name,
578
+ modelId: this.session.model,
579
+ pricing: MODEL_PRICING[this.session.model],
580
+ resources: {
581
+ count: this.session.resourceReads.length,
582
+ bytes: resourceBytes,
583
+ tokens: resourceTokens,
584
+ costUsd: this.calculateCost(resourceTokens),
585
+ byType: resourcesByType
586
+ },
587
+ tools: {
588
+ count: this.session.toolCalls.length,
589
+ bytes: toolBytes,
590
+ tokens: toolTokens,
591
+ costUsd: this.calculateCost(toolTokens),
592
+ byName: toolsByName
593
+ },
594
+ total: {
595
+ tokens: totalTokens,
596
+ costUsd: totalCost
597
+ }
598
+ };
599
+ }
600
+ /**
601
+ * Get handoff recommendation based on context usage
602
+ */
603
+ getHandoffRecommendation(contextWindowSize = 2e5, estimatedTotalTokens) {
604
+ const mcpTokens = this.session.totals.totalTokens;
605
+ const estimatedConversationOverhead = mcpTokens * 4;
606
+ const totalEstimate = estimatedTotalTokens || mcpTokens + estimatedConversationOverhead;
607
+ const usagePercent = Math.round(totalEstimate / contextWindowSize * 100);
608
+ let recommendation;
609
+ let message;
610
+ if (usagePercent >= 85) {
611
+ recommendation = "handoff-urgent";
612
+ message = "Context is nearly full. Initiate handoff immediately to preserve session continuity.";
613
+ } else if (usagePercent >= 70) {
614
+ recommendation = "handoff-recommended";
615
+ message = "Context usage is high. Consider initiating handoff soon to ensure smooth transition.";
616
+ } else if (usagePercent >= 50) {
617
+ recommendation = "consider-handoff";
618
+ message = "Context usage is moderate. Plan a good stopping point for potential handoff.";
619
+ } else {
620
+ recommendation = "continue";
621
+ message = "Context usage is healthy. Continue working.";
622
+ }
623
+ const signals = [];
624
+ const durationMin = Math.round((Date.now() - this.session.startTime) / 6e4);
625
+ const totalCalls = this.session.toolCalls.length + this.session.resourceReads.length;
626
+ if (totalCalls > 50) {
627
+ signals.push(`High number of MCP interactions (${totalCalls})`);
628
+ }
629
+ if (durationMin > 30) {
630
+ signals.push(`Session duration >30 min (${durationMin} min)`);
631
+ }
632
+ if (this.session.totals.totalBytes > 5e5) {
633
+ signals.push(`Large data volume (${Math.round(this.session.totals.totalBytes / 1024)}KB)`);
634
+ }
635
+ return { recommendation, message, usagePercent, signals };
636
+ }
637
+ /**
638
+ * Get session duration in minutes
639
+ */
640
+ getDurationMinutes() {
641
+ return Math.round((Date.now() - this.session.startTime) / 6e4);
642
+ }
643
+ /**
644
+ * Reset session (for handoff or new session)
645
+ */
646
+ reset() {
647
+ this.session = this.createNewSession();
648
+ this._recovered = false;
649
+ this.lastLoreEntryId = null;
650
+ }
651
+ };
652
+ var tracker = null;
653
+ function getSessionTracker() {
654
+ if (!tracker) {
655
+ tracker = new SessionTracker();
656
+ }
657
+ return tracker;
658
+ }
659
+ function resetSessionTracker() {
660
+ if (tracker) {
661
+ tracker.reset();
662
+ }
663
+ }
664
+
665
+ export {
666
+ writePendingHandoff,
667
+ loadPendingHandoffs,
668
+ markHandoffDelivered,
669
+ loadGlobalAntipatterns,
670
+ loadGlobalDecisions,
671
+ loadGlobalPreferences,
672
+ recordGlobalAntipattern,
673
+ recordGlobalDecision,
674
+ MODEL_PRICING,
675
+ getSessionTracker,
676
+ resetSessionTracker
677
+ };