@ramtinj95/opencode-tokenscope 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +526 -0
  3. package/dist/tokenscope-lib/analyzer.d.ts +29 -0
  4. package/dist/tokenscope-lib/analyzer.d.ts.map +1 -0
  5. package/dist/tokenscope-lib/analyzer.js +321 -0
  6. package/dist/tokenscope-lib/analyzer.js.map +1 -0
  7. package/dist/tokenscope-lib/config.d.ts +10 -0
  8. package/dist/tokenscope-lib/config.d.ts.map +1 -0
  9. package/dist/tokenscope-lib/config.js +102 -0
  10. package/dist/tokenscope-lib/config.js.map +1 -0
  11. package/dist/tokenscope-lib/context.d.ts +81 -0
  12. package/dist/tokenscope-lib/context.d.ts.map +1 -0
  13. package/dist/tokenscope-lib/context.js +453 -0
  14. package/dist/tokenscope-lib/context.js.map +1 -0
  15. package/dist/tokenscope-lib/cost.d.ts +9 -0
  16. package/dist/tokenscope-lib/cost.d.ts.map +1 -0
  17. package/dist/tokenscope-lib/cost.js +51 -0
  18. package/dist/tokenscope-lib/cost.js.map +1 -0
  19. package/dist/tokenscope-lib/formatter.d.ts +26 -0
  20. package/dist/tokenscope-lib/formatter.d.ts.map +1 -0
  21. package/dist/tokenscope-lib/formatter.js +415 -0
  22. package/dist/tokenscope-lib/formatter.js.map +1 -0
  23. package/dist/tokenscope-lib/subagent.d.ts +13 -0
  24. package/dist/tokenscope-lib/subagent.d.ts.map +1 -0
  25. package/dist/tokenscope-lib/subagent.js +134 -0
  26. package/dist/tokenscope-lib/subagent.js.map +1 -0
  27. package/dist/tokenscope-lib/tokenizer.d.ts +17 -0
  28. package/dist/tokenscope-lib/tokenizer.d.ts.map +1 -0
  29. package/dist/tokenscope-lib/tokenizer.js +117 -0
  30. package/dist/tokenscope-lib/tokenizer.js.map +1 -0
  31. package/dist/tokenscope-lib/types.d.ts +252 -0
  32. package/dist/tokenscope-lib/types.d.ts.map +1 -0
  33. package/dist/tokenscope-lib/types.js +12 -0
  34. package/dist/tokenscope-lib/types.js.map +1 -0
  35. package/dist/tokenscope.d.ts +4 -0
  36. package/dist/tokenscope.d.ts.map +1 -0
  37. package/dist/tokenscope.js +98 -0
  38. package/dist/tokenscope.js.map +1 -0
  39. package/models.json +290 -0
  40. package/package.json +54 -0
  41. package/tokenscope-config.json +6 -0
@@ -0,0 +1,415 @@
1
+ // OutputFormatter - generates visual reports from token analysis
2
+ export class OutputFormatter {
3
+ costCalculator;
4
+ BAR_WIDTH = 30;
5
+ TOKEN_SPACING = 11;
6
+ CATEGORY_LABEL_WIDTH = 9;
7
+ TOOL_LABEL_WIDTH = 20;
8
+ TOP_CONTRIBUTOR_LABEL_WIDTH = 30;
9
+ CONTEXT_LABEL_WIDTH = 22;
10
+ TOOL_ESTIMATE_LABEL_WIDTH = 18;
11
+ config = null;
12
+ constructor(costCalculator) {
13
+ this.costCalculator = costCalculator;
14
+ }
15
+ setConfig(config) {
16
+ this.config = config;
17
+ }
18
+ formatCategoryBar(label, tokens, total, labelWidth = this.CATEGORY_LABEL_WIDTH) {
19
+ if (tokens === 0)
20
+ return "";
21
+ const percentage = total > 0 ? ((tokens / total) * 100).toFixed(1) : "0.0";
22
+ const percentageNum = parseFloat(percentage);
23
+ const barWidth = Math.round((percentageNum / 100) * this.BAR_WIDTH);
24
+ const bar = "\u2588".repeat(barWidth) + "\u2591".repeat(Math.max(0, this.BAR_WIDTH - barWidth));
25
+ const labelPadded = label.padEnd(labelWidth);
26
+ const formattedTokens = this.formatNumber(tokens);
27
+ let pct = percentage;
28
+ if (percentageNum < 10) {
29
+ pct = " " + pct;
30
+ }
31
+ const tokensPart = `(${formattedTokens})`;
32
+ const spacesNeeded = Math.max(1, this.TOKEN_SPACING - tokensPart.length);
33
+ const spacing = " ".repeat(spacesNeeded);
34
+ return `${labelPadded} ${bar} ${spacing}${pct}% ${tokensPart}`;
35
+ }
36
+ format(analysis) {
37
+ const inputCategories = [
38
+ { label: "SYSTEM", tokens: analysis.categories.system.totalTokens },
39
+ { label: "USER", tokens: analysis.categories.user.totalTokens },
40
+ { label: "TOOLS", tokens: analysis.categories.tools.totalTokens },
41
+ ];
42
+ const outputCategories = [
43
+ { label: "ASSISTANT", tokens: analysis.categories.assistant.totalTokens },
44
+ { label: "REASONING", tokens: analysis.categories.reasoning.totalTokens },
45
+ ];
46
+ const topEntries = this.collectTopEntries(analysis, 5);
47
+ const toolStats = new Map();
48
+ for (const [toolName, calls] of analysis.toolCallCounts.entries()) {
49
+ toolStats.set(toolName, { tokens: 0, calls });
50
+ }
51
+ for (const entry of analysis.categories.tools.allEntries) {
52
+ const existing = toolStats.get(entry.label) || { tokens: 0, calls: 0 };
53
+ toolStats.set(entry.label, { ...existing, tokens: entry.tokens });
54
+ }
55
+ const toolEntries = Array.from(toolStats.entries())
56
+ .map(([label, stats]) => ({ label, tokens: stats.tokens, calls: stats.calls }))
57
+ .sort((a, b) => b.tokens - a.tokens);
58
+ const costEstimate = this.costCalculator.calculateCost(analysis);
59
+ return this.formatVisualOutput(analysis.sessionID, analysis.model.name, analysis.totalTokens, analysis.inputTokens, analysis.outputTokens, analysis.reasoningTokens, analysis.cacheReadTokens, analysis.cacheWriteTokens, analysis.assistantMessageCount, analysis.mostRecentInput, analysis.mostRecentOutput, analysis.mostRecentReasoning, analysis.mostRecentCacheRead, analysis.mostRecentCacheWrite, inputCategories, outputCategories, topEntries, toolEntries, costEstimate, analysis.subagentAnalysis, analysis.contextBreakdown, analysis.toolEstimates, analysis.cacheEfficiency);
60
+ }
61
+ formatVisualOutput(sessionID, modelName, totalTokens, inputTokens, outputTokens, reasoningTokens, cacheReadTokens, cacheWriteTokens, assistantMessageCount, mostRecentInput, mostRecentOutput, mostRecentReasoning, mostRecentCacheRead, mostRecentCacheWrite, inputCategories, outputCategories, topEntries, toolEntries, cost, subagentAnalysis, contextBreakdown, toolEstimates, cacheEfficiency) {
62
+ const lines = [];
63
+ const sessionTotal = inputTokens + cacheReadTokens + cacheWriteTokens + outputTokens + reasoningTokens;
64
+ const mainCost = cost.isSubscription ? cost.estimatedSessionCost : cost.apiSessionCost;
65
+ // Header
66
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
67
+ lines.push(`Token Analysis: Session ${sessionID}`);
68
+ lines.push(`Model: ${modelName}`);
69
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
70
+ lines.push(``);
71
+ // 1. TOKEN BREAKDOWN BY CATEGORY
72
+ lines.push(`TOKEN BREAKDOWN BY CATEGORY`);
73
+ lines.push(`\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\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`);
74
+ lines.push(`Estimated using tokenizer analysis of message content:`);
75
+ lines.push(``);
76
+ const inputTotal = inputCategories.reduce((sum, cat) => sum + cat.tokens, 0);
77
+ lines.push(`Input Categories:`);
78
+ for (const category of inputCategories) {
79
+ const barLine = this.formatCategoryBar(category.label, category.tokens, inputTotal);
80
+ if (barLine)
81
+ lines.push(` ${barLine}`);
82
+ }
83
+ lines.push(``);
84
+ lines.push(` Subtotal: ${this.formatNumber(inputTotal)} estimated input tokens`);
85
+ lines.push(``);
86
+ const outputTotal = outputCategories.reduce((sum, cat) => sum + cat.tokens, 0);
87
+ lines.push(`Output Categories:`);
88
+ for (const category of outputCategories) {
89
+ const barLine = this.formatCategoryBar(category.label, category.tokens, outputTotal);
90
+ if (barLine)
91
+ lines.push(` ${barLine}`);
92
+ }
93
+ lines.push(``);
94
+ lines.push(` Subtotal: ${this.formatNumber(outputTotal)} estimated output tokens`);
95
+ lines.push(``);
96
+ lines.push(`Local Total: ${this.formatNumber(totalTokens)} tokens (estimated)`);
97
+ // 2. TOOL USAGE BREAKDOWN (right after token breakdown)
98
+ if (toolEntries.length > 0) {
99
+ const toolsTotalTokens = inputCategories.find((c) => c.label === "TOOLS")?.tokens || 0;
100
+ lines.push(``);
101
+ lines.push(`TOOL USAGE BREAKDOWN`);
102
+ lines.push(`\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\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`);
103
+ for (const tool of toolEntries) {
104
+ const barLine = this.formatCategoryBar(tool.label, tool.tokens, toolsTotalTokens, this.TOOL_LABEL_WIDTH);
105
+ if (barLine) {
106
+ const calls = `${tool.calls}x`.padStart(5);
107
+ lines.push(`${barLine} ${calls}`);
108
+ }
109
+ }
110
+ }
111
+ // 3. TOP CONTRIBUTORS
112
+ if (topEntries.length > 0) {
113
+ lines.push(``);
114
+ lines.push(`TOP CONTRIBUTORS`);
115
+ lines.push(`\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\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`);
116
+ for (const entry of topEntries) {
117
+ const percentage = ((entry.tokens / totalTokens) * 100).toFixed(1);
118
+ const label = `\u2022 ${entry.label}`.padEnd(this.TOP_CONTRIBUTOR_LABEL_WIDTH);
119
+ const formattedTokens = this.formatNumber(entry.tokens);
120
+ lines.push(`${label} ${formattedTokens} tokens (${percentage}%)`);
121
+ }
122
+ }
123
+ // 4. MOST RECENT API CALL
124
+ lines.push(``);
125
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
126
+ lines.push(`MOST RECENT API CALL`);
127
+ lines.push(`\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\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`);
128
+ lines.push(``);
129
+ lines.push(`Raw telemetry from last API response:`);
130
+ lines.push(` Input (fresh): ${this.formatNumber(mostRecentInput).padStart(10)} tokens`);
131
+ lines.push(` Cache read: ${this.formatNumber(mostRecentCacheRead).padStart(10)} tokens`);
132
+ if (mostRecentCacheWrite > 0) {
133
+ lines.push(` Cache write: ${this.formatNumber(mostRecentCacheWrite).padStart(10)} tokens`);
134
+ }
135
+ lines.push(` Output: ${this.formatNumber(mostRecentOutput).padStart(10)} tokens`);
136
+ if (mostRecentReasoning > 0) {
137
+ lines.push(` Reasoning: ${this.formatNumber(mostRecentReasoning).padStart(10)} tokens`);
138
+ }
139
+ lines.push(` \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`);
140
+ lines.push(` Total: ${this.formatNumber(mostRecentInput + mostRecentCacheRead + mostRecentCacheWrite + mostRecentOutput + mostRecentReasoning).padStart(10)} tokens`);
141
+ // 5. SESSION TOTALS
142
+ lines.push(``);
143
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
144
+ lines.push(`SESSION TOTALS (All ${assistantMessageCount} API calls)`);
145
+ lines.push(`\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\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`);
146
+ lines.push(``);
147
+ lines.push(`Total tokens processed across the entire session (for cost calculation):`);
148
+ lines.push(``);
149
+ lines.push(` Input tokens: ${this.formatNumber(inputTokens).padStart(10)} (fresh tokens across all calls)`);
150
+ lines.push(` Cache read: ${this.formatNumber(cacheReadTokens).padStart(10)} (cached tokens across all calls)`);
151
+ lines.push(` Cache write: ${this.formatNumber(cacheWriteTokens).padStart(10)} (tokens written to cache)`);
152
+ lines.push(` Output tokens: ${this.formatNumber(outputTokens).padStart(10)} (all model responses)`);
153
+ if (reasoningTokens > 0) {
154
+ lines.push(` Reasoning tokens: ${this.formatNumber(reasoningTokens).padStart(10)} (thinking/reasoning)`);
155
+ }
156
+ lines.push(` \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`);
157
+ lines.push(` Session Total: ${this.formatNumber(sessionTotal).padStart(10)} tokens (for billing)`);
158
+ // 6. SESSION COST / ESTIMATED SESSION COST
159
+ lines.push(``);
160
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
161
+ if (cost.isSubscription) {
162
+ lines.push(`ESTIMATED SESSION COST (API Key Pricing)`);
163
+ lines.push(`\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\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`);
164
+ lines.push(``);
165
+ lines.push(`You appear to be on a subscription plan (API cost is $0).`);
166
+ lines.push(`Here's what this session would cost with direct API access:`);
167
+ lines.push(``);
168
+ lines.push(` Input tokens: ${this.formatNumber(inputTokens).padStart(10)} \u00d7 $${cost.pricePerMillionInput.toFixed(2)}/M = $${cost.estimatedInputCost.toFixed(4)}`);
169
+ lines.push(` Output tokens: ${this.formatNumber(outputTokens + reasoningTokens).padStart(10)} \u00d7 $${cost.pricePerMillionOutput.toFixed(2)}/M = $${cost.estimatedOutputCost.toFixed(4)}`);
170
+ if (cacheReadTokens > 0 && cost.pricePerMillionCacheRead > 0) {
171
+ lines.push(` Cache read: ${this.formatNumber(cacheReadTokens).padStart(10)} \u00d7 $${cost.pricePerMillionCacheRead.toFixed(2)}/M = $${cost.estimatedCacheReadCost.toFixed(4)}`);
172
+ }
173
+ if (cacheWriteTokens > 0 && cost.pricePerMillionCacheWrite > 0) {
174
+ lines.push(` Cache write: ${this.formatNumber(cacheWriteTokens).padStart(10)} \u00d7 $${cost.pricePerMillionCacheWrite.toFixed(2)}/M = $${cost.estimatedCacheWriteCost.toFixed(4)}`);
175
+ }
176
+ lines.push(`\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\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`);
177
+ lines.push(`ESTIMATED TOTAL: $${cost.estimatedSessionCost.toFixed(4)}`);
178
+ lines.push(``);
179
+ lines.push(`Note: This estimate uses standard API pricing from models.json.`);
180
+ lines.push(`Actual API costs may vary based on provider and context size.`);
181
+ }
182
+ else {
183
+ lines.push(`SESSION COST`);
184
+ lines.push(`\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\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`);
185
+ lines.push(``);
186
+ lines.push(`Token usage breakdown:`);
187
+ lines.push(` Input tokens: ${this.formatNumber(inputTokens).padStart(10)}`);
188
+ lines.push(` Output tokens: ${this.formatNumber(outputTokens).padStart(10)}`);
189
+ if (reasoningTokens > 0) {
190
+ lines.push(` Reasoning tokens: ${this.formatNumber(reasoningTokens).padStart(10)}`);
191
+ }
192
+ if (cacheReadTokens > 0) {
193
+ lines.push(` Cache read: ${this.formatNumber(cacheReadTokens).padStart(10)}`);
194
+ }
195
+ if (cacheWriteTokens > 0) {
196
+ lines.push(` Cache write: ${this.formatNumber(cacheWriteTokens).padStart(10)}`);
197
+ }
198
+ lines.push(``);
199
+ lines.push(`\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\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`);
200
+ lines.push(`ACTUAL COST (from API): $${cost.apiSessionCost.toFixed(4)}`);
201
+ const diff = Math.abs(cost.apiSessionCost - cost.estimatedSessionCost);
202
+ const diffPercent = cost.apiSessionCost > 0 ? (diff / cost.apiSessionCost) * 100 : 0;
203
+ if (diffPercent > 5) {
204
+ lines.push(`Estimated cost: $${cost.estimatedSessionCost.toFixed(4)} (${diffPercent > 0 ? (cost.estimatedSessionCost > cost.apiSessionCost ? "+" : "-") : ""}${diffPercent.toFixed(1)}% diff)`);
205
+ }
206
+ lines.push(``);
207
+ lines.push(`Note: Actual cost from OpenCode includes provider-specific pricing`);
208
+ lines.push(`and 200K+ context adjustments.`);
209
+ }
210
+ // 7. CONTEXT BREAKDOWN (if enabled and available)
211
+ if (this.config?.enableContextBreakdown && contextBreakdown && contextBreakdown.totalCachedContext > 0) {
212
+ lines.push(...this.formatContextBreakdown(contextBreakdown));
213
+ }
214
+ // 8. TOOL DEFINITION COSTS (if enabled and available)
215
+ if (this.config?.enableToolSchemaEstimation && toolEstimates && toolEstimates.length > 0) {
216
+ lines.push(...this.formatToolEstimates(toolEstimates));
217
+ }
218
+ // 9. CACHE EFFICIENCY (if enabled and available)
219
+ if (this.config?.enableCacheEfficiency && cacheEfficiency && cacheEfficiency.totalInputTokens > 0) {
220
+ lines.push(...this.formatCacheEfficiency(cacheEfficiency, cost, modelName));
221
+ }
222
+ // 10. SUBAGENT COSTS (if any)
223
+ if (subagentAnalysis && subagentAnalysis.subagents.length > 0) {
224
+ const subagentLabelWidth = 25;
225
+ const subagentTotalCost = cost.isSubscription
226
+ ? subagentAnalysis.totalEstimatedCost
227
+ : subagentAnalysis.totalApiCost;
228
+ lines.push(``);
229
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
230
+ lines.push(`SUBAGENT COSTS (${subagentAnalysis.subagents.length} child sessions, ${subagentAnalysis.totalApiCalls} API calls)`);
231
+ lines.push(`\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\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`);
232
+ lines.push(``);
233
+ for (const subagent of subagentAnalysis.subagents) {
234
+ const label = `${subagent.agentType}`.padEnd(subagentLabelWidth);
235
+ const costStr = cost.isSubscription
236
+ ? `$${subagent.estimatedCost.toFixed(4)}`
237
+ : `$${subagent.apiCost.toFixed(4)}`;
238
+ const tokensStr = `(${this.formatNumber(subagent.totalTokens)} tokens, ${subagent.assistantMessageCount} calls)`;
239
+ lines.push(` ${label} ${costStr.padStart(10)} ${tokensStr}`);
240
+ }
241
+ lines.push(`\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\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`);
242
+ lines.push(`Subagent Total:${" ".repeat(subagentLabelWidth - 14)} $${subagentTotalCost.toFixed(4)} (${this.formatNumber(subagentAnalysis.totalTokens)} tokens, ${subagentAnalysis.totalApiCalls} calls)`);
243
+ }
244
+ // 11. SUMMARY (always last)
245
+ lines.push(``);
246
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
247
+ lines.push(`SUMMARY`);
248
+ lines.push(`\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\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`);
249
+ lines.push(``);
250
+ lines.push(` Cost Tokens API Calls`);
251
+ if (subagentAnalysis && subagentAnalysis.subagents.length > 0) {
252
+ const subagentTotalCost = cost.isSubscription
253
+ ? subagentAnalysis.totalEstimatedCost
254
+ : subagentAnalysis.totalApiCost;
255
+ const grandTotalCost = mainCost + subagentTotalCost;
256
+ const grandTotalTokens = sessionTotal + subagentAnalysis.totalTokens;
257
+ const grandTotalApiCalls = assistantMessageCount + subagentAnalysis.totalApiCalls;
258
+ lines.push(` Main session: $${mainCost.toFixed(4).padStart(10)} ${this.formatNumber(sessionTotal).padStart(10)} ${assistantMessageCount.toString().padStart(5)}`);
259
+ lines.push(` Subagents: $${subagentTotalCost.toFixed(4).padStart(10)} ${this.formatNumber(subagentAnalysis.totalTokens).padStart(10)} ${subagentAnalysis.totalApiCalls.toString().padStart(5)}`);
260
+ lines.push(`\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\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`);
261
+ lines.push(` TOTAL: $${grandTotalCost.toFixed(4).padStart(10)} ${this.formatNumber(grandTotalTokens).padStart(10)} ${grandTotalApiCalls.toString().padStart(5)}`);
262
+ }
263
+ else {
264
+ lines.push(` Session: $${mainCost.toFixed(4).padStart(10)} ${this.formatNumber(sessionTotal).padStart(10)} ${assistantMessageCount.toString().padStart(5)}`);
265
+ }
266
+ lines.push(``);
267
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
268
+ return lines.join("\n");
269
+ }
270
+ formatContextBreakdown(breakdown) {
271
+ const lines = [];
272
+ const total = breakdown.totalCachedContext;
273
+ // Check if this is estimated from cache tokens vs actual system prompts
274
+ const isEstimated = !breakdown.baseSystemPrompt.identified;
275
+ lines.push(``);
276
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
277
+ if (isEstimated) {
278
+ lines.push(`CONTEXT BREAKDOWN (Estimated from cache_write tokens)`);
279
+ }
280
+ else {
281
+ lines.push(`CONTEXT BREAKDOWN (From system prompt analysis)`);
282
+ }
283
+ lines.push(`\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\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`);
284
+ lines.push(``);
285
+ // Base System Prompt
286
+ if (breakdown.baseSystemPrompt.tokens > 0) {
287
+ const bar = this.formatContextBar("Base System Prompt", breakdown.baseSystemPrompt.tokens, total);
288
+ lines.push(` ${bar}`);
289
+ }
290
+ // Tool Definitions
291
+ if (breakdown.toolDefinitions.tokens > 0) {
292
+ const label = breakdown.toolDefinitions.toolCount > 0
293
+ ? `Tool Definitions (${breakdown.toolDefinitions.toolCount})`
294
+ : "Tool Definitions";
295
+ const bar = this.formatContextBar(label, breakdown.toolDefinitions.tokens, total);
296
+ lines.push(` ${bar}`);
297
+ }
298
+ // Environment Context
299
+ if (breakdown.environmentContext.tokens > 0) {
300
+ const bar = this.formatContextBar("Environment Context", breakdown.environmentContext.tokens, total);
301
+ lines.push(` ${bar}`);
302
+ }
303
+ // Project Tree
304
+ if (breakdown.projectTree.tokens > 0) {
305
+ const label = breakdown.projectTree.fileCount > 0
306
+ ? `Project Tree (~${breakdown.projectTree.fileCount} files)`
307
+ : "Project Tree";
308
+ const bar = this.formatContextBar(label, breakdown.projectTree.tokens, total);
309
+ lines.push(` ${bar}`);
310
+ }
311
+ // Custom Instructions
312
+ if (breakdown.customInstructions.tokens > 0) {
313
+ const bar = this.formatContextBar("Custom Instructions", breakdown.customInstructions.tokens, total);
314
+ lines.push(` ${bar}`);
315
+ if (breakdown.customInstructions.sources.length > 0) {
316
+ for (const source of breakdown.customInstructions.sources.slice(0, 3)) {
317
+ lines.push(` \u2192 ${source}`);
318
+ }
319
+ }
320
+ }
321
+ lines.push(` \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\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500`);
322
+ lines.push(` Total Cached Context:${" ".repeat(34)}~${this.formatNumber(total)} tokens`);
323
+ lines.push(``);
324
+ if (isEstimated) {
325
+ lines.push(` Note: Breakdown estimated from first cache_write. Actual distribution may vary.`);
326
+ }
327
+ else {
328
+ lines.push(` Note: Values from tokenizing actual system prompt content.`);
329
+ }
330
+ return lines;
331
+ }
332
+ formatContextBar(label, tokens, total) {
333
+ const percentage = total > 0 ? ((tokens / total) * 100).toFixed(1) : "0.0";
334
+ const percentageNum = parseFloat(percentage);
335
+ const barWidth = Math.round((percentageNum / 100) * this.BAR_WIDTH);
336
+ const bar = "\u2588".repeat(barWidth) + "\u2591".repeat(Math.max(0, this.BAR_WIDTH - barWidth));
337
+ const labelPadded = label.padEnd(this.CONTEXT_LABEL_WIDTH);
338
+ return `${labelPadded} ${bar} ~${this.formatNumber(tokens).padStart(6)} tokens`;
339
+ }
340
+ formatToolEstimates(estimates) {
341
+ const lines = [];
342
+ const enabledEstimates = estimates.filter((e) => e.enabled);
343
+ const totalTokens = enabledEstimates.reduce((sum, e) => sum + e.estimatedTokens, 0);
344
+ lines.push(``);
345
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
346
+ lines.push(`TOOL DEFINITION COSTS (Estimated from argument analysis)`);
347
+ lines.push(`\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\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`);
348
+ lines.push(``);
349
+ lines.push(` ${"Tool".padEnd(this.TOOL_ESTIMATE_LABEL_WIDTH)} ${"Est. Tokens".padStart(12)} Args Complexity`);
350
+ lines.push(` \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\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500`);
351
+ for (const estimate of enabledEstimates) {
352
+ const name = estimate.name.padEnd(this.TOOL_ESTIMATE_LABEL_WIDTH);
353
+ const tokens = `~${this.formatNumber(estimate.estimatedTokens)}`.padStart(12);
354
+ const args = estimate.argumentCount.toString().padStart(5);
355
+ const complexity = estimate.hasComplexArgs ? "complex (arrays/objects)" : "simple";
356
+ lines.push(` ${name} ${tokens} ${args} ${complexity}`);
357
+ }
358
+ lines.push(` \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\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500`);
359
+ lines.push(` Total:${" ".repeat(this.TOOL_ESTIMATE_LABEL_WIDTH - 6)} ~${this.formatNumber(totalTokens).padStart(11)} tokens (${enabledEstimates.length} enabled tools)`);
360
+ lines.push(``);
361
+ lines.push(` Note: Estimates inferred from tool call arguments in this session.`);
362
+ lines.push(` Actual schema tokens may vary +/-20%.`);
363
+ return lines;
364
+ }
365
+ formatCacheEfficiency(efficiency, cost, modelName) {
366
+ const lines = [];
367
+ const total = efficiency.totalInputTokens;
368
+ lines.push(``);
369
+ lines.push(`\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550\u2550`);
370
+ lines.push(`CACHE EFFICIENCY`);
371
+ lines.push(`\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\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`);
372
+ lines.push(``);
373
+ lines.push(` Token Distribution:`);
374
+ // Cache Read bar
375
+ const cacheReadPct = total > 0 ? ((efficiency.cacheReadTokens / total) * 100).toFixed(1) : "0.0";
376
+ const cacheReadBar = this.formatEfficiencyBar(efficiency.cacheReadTokens, total);
377
+ lines.push(` Cache Read: ${this.formatNumber(efficiency.cacheReadTokens).padStart(10)} tokens ${cacheReadBar} ${cacheReadPct}%`);
378
+ // Fresh Input bar
379
+ const freshPct = total > 0 ? ((efficiency.freshInputTokens / total) * 100).toFixed(1) : "0.0";
380
+ const freshBar = this.formatEfficiencyBar(efficiency.freshInputTokens, total);
381
+ lines.push(` Fresh Input: ${this.formatNumber(efficiency.freshInputTokens).padStart(10)} tokens ${freshBar} ${freshPct}%`);
382
+ lines.push(` \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\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500`);
383
+ lines.push(` Cache Hit Rate: ${efficiency.cacheHitRate.toFixed(1)}%`);
384
+ lines.push(``);
385
+ // Cost analysis
386
+ lines.push(` Cost Analysis (${modelName} @ $${cost.pricePerMillionInput.toFixed(2)}/M input, $${cost.pricePerMillionCacheRead.toFixed(2)}/M cache read):`);
387
+ lines.push(` Without caching: $${efficiency.costWithoutCaching.toFixed(4)} (${this.formatNumber(total)} tokens x $${cost.pricePerMillionInput.toFixed(2)}/M)`);
388
+ lines.push(` With caching: $${efficiency.costWithCaching.toFixed(4)} (fresh x $${cost.pricePerMillionInput.toFixed(2)}/M + cached x $${cost.pricePerMillionCacheRead.toFixed(2)}/M)`);
389
+ lines.push(` \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\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500`);
390
+ lines.push(` Cost Savings: $${efficiency.costSavings.toFixed(4)} (${efficiency.savingsPercent.toFixed(1)}% reduction)`);
391
+ lines.push(` Effective Rate: $${efficiency.effectiveRate.toFixed(2)}/M tokens (vs. $${efficiency.standardRate.toFixed(2)}/M standard)`);
392
+ return lines;
393
+ }
394
+ formatEfficiencyBar(value, total) {
395
+ const percentage = total > 0 ? (value / total) * 100 : 0;
396
+ const barWidth = Math.round((percentage / 100) * this.BAR_WIDTH);
397
+ return "\u2588".repeat(barWidth) + "\u2591".repeat(Math.max(0, this.BAR_WIDTH - barWidth));
398
+ }
399
+ collectTopEntries(analysis, limit) {
400
+ const pool = [
401
+ ...analysis.categories.system.allEntries,
402
+ ...analysis.categories.user.allEntries,
403
+ ...analysis.categories.assistant.allEntries,
404
+ ...analysis.categories.tools.allEntries,
405
+ ...analysis.categories.reasoning.allEntries,
406
+ ]
407
+ .filter((entry) => entry.tokens > 0)
408
+ .sort((a, b) => b.tokens - a.tokens);
409
+ return pool.slice(0, limit);
410
+ }
411
+ formatNumber(value) {
412
+ return new Intl.NumberFormat("en-US").format(value);
413
+ }
414
+ }
415
+ //# sourceMappingURL=formatter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"formatter.js","sourceRoot":"","sources":["../../tokenscope-lib/formatter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ import type { SubagentAnalysis, ModelPricing } from "./types";
2
+ import { CostCalculator } from "./cost";
3
+ export declare class SubagentAnalyzer {
4
+ private client;
5
+ private costCalculator;
6
+ private pricingData;
7
+ constructor(client: any, costCalculator: CostCalculator, pricingData: Record<string, ModelPricing>);
8
+ analyzeChildSessions(parentSessionID: string): Promise<SubagentAnalysis>;
9
+ private analyzeChildSession;
10
+ private extractAgentType;
11
+ private getPricing;
12
+ }
13
+ //# sourceMappingURL=subagent.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"subagent.d.ts","sourceRoot":"","sources":["../../tokenscope-lib/subagent.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAmC,gBAAgB,EAAgB,YAAY,EAAE,MAAM,SAAS,CAAA;AAC5G,OAAO,EAAE,cAAc,EAAE,MAAM,QAAQ,CAAA;AAEvC,qBAAa,gBAAgB;IAEzB,OAAO,CAAC,MAAM;IACd,OAAO,CAAC,cAAc;IACtB,OAAO,CAAC,WAAW;gBAFX,MAAM,EAAE,GAAG,EACX,cAAc,EAAE,cAAc,EAC9B,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC;IAG7C,oBAAoB,CAAC,eAAe,EAAE,MAAM,GAAG,OAAO,CAAC,gBAAgB,CAAC;YAwDhE,mBAAmB;IA6DjC,OAAO,CAAC,gBAAgB;IAOxB,OAAO,CAAC,UAAU;CAWnB"}
@@ -0,0 +1,134 @@
1
+ // SubagentAnalyzer - analyzes child sessions from Task tool calls
2
+ export class SubagentAnalyzer {
3
+ client;
4
+ costCalculator;
5
+ pricingData;
6
+ constructor(client, costCalculator, pricingData) {
7
+ this.client = client;
8
+ this.costCalculator = costCalculator;
9
+ this.pricingData = pricingData;
10
+ }
11
+ async analyzeChildSessions(parentSessionID) {
12
+ const result = {
13
+ subagents: [],
14
+ totalInputTokens: 0,
15
+ totalOutputTokens: 0,
16
+ totalReasoningTokens: 0,
17
+ totalCacheReadTokens: 0,
18
+ totalCacheWriteTokens: 0,
19
+ totalTokens: 0,
20
+ totalApiCost: 0,
21
+ totalEstimatedCost: 0,
22
+ totalApiCalls: 0,
23
+ };
24
+ try {
25
+ const childrenResponse = await this.client.session.children({ path: { id: parentSessionID } });
26
+ const children = (childrenResponse?.data ?? childrenResponse ?? []);
27
+ if (!Array.isArray(children) || children.length === 0)
28
+ return result;
29
+ for (const child of children) {
30
+ const summary = await this.analyzeChildSession(child);
31
+ if (summary) {
32
+ result.subagents.push(summary);
33
+ result.totalInputTokens += summary.inputTokens;
34
+ result.totalOutputTokens += summary.outputTokens;
35
+ result.totalReasoningTokens += summary.reasoningTokens;
36
+ result.totalCacheReadTokens += summary.cacheReadTokens;
37
+ result.totalCacheWriteTokens += summary.cacheWriteTokens;
38
+ result.totalTokens += summary.totalTokens;
39
+ result.totalApiCost += summary.apiCost;
40
+ result.totalEstimatedCost += summary.estimatedCost;
41
+ result.totalApiCalls += summary.assistantMessageCount;
42
+ }
43
+ const nestedAnalysis = await this.analyzeChildSessions(child.id);
44
+ for (const nested of nestedAnalysis.subagents) {
45
+ result.subagents.push(nested);
46
+ }
47
+ result.totalInputTokens += nestedAnalysis.totalInputTokens;
48
+ result.totalOutputTokens += nestedAnalysis.totalOutputTokens;
49
+ result.totalReasoningTokens += nestedAnalysis.totalReasoningTokens;
50
+ result.totalCacheReadTokens += nestedAnalysis.totalCacheReadTokens;
51
+ result.totalCacheWriteTokens += nestedAnalysis.totalCacheWriteTokens;
52
+ result.totalTokens += nestedAnalysis.totalTokens;
53
+ result.totalApiCost += nestedAnalysis.totalApiCost;
54
+ result.totalEstimatedCost += nestedAnalysis.totalEstimatedCost;
55
+ result.totalApiCalls += nestedAnalysis.totalApiCalls;
56
+ }
57
+ }
58
+ catch (error) {
59
+ console.error(`Failed to fetch child sessions for ${parentSessionID}:`, error);
60
+ }
61
+ return result;
62
+ }
63
+ async analyzeChildSession(child) {
64
+ try {
65
+ const messagesResponse = await this.client.session.messages({ path: { id: child.id } });
66
+ const messages = (messagesResponse?.data ?? messagesResponse ?? []);
67
+ if (!Array.isArray(messages) || messages.length === 0)
68
+ return null;
69
+ const agentType = this.extractAgentType(child.title);
70
+ let inputTokens = 0, outputTokens = 0, reasoningTokens = 0;
71
+ let cacheReadTokens = 0, cacheWriteTokens = 0;
72
+ let apiCost = 0, assistantMessageCount = 0, modelName = "unknown";
73
+ for (const message of messages) {
74
+ if (message.info.role === "assistant") {
75
+ assistantMessageCount++;
76
+ const tokens = message.info.tokens;
77
+ if (tokens) {
78
+ inputTokens += Number(tokens.input) || 0;
79
+ outputTokens += Number(tokens.output) || 0;
80
+ reasoningTokens += Number(tokens.reasoning) || 0;
81
+ cacheReadTokens += Number(tokens.cache?.read) || 0;
82
+ cacheWriteTokens += Number(tokens.cache?.write) || 0;
83
+ }
84
+ apiCost += Number(message.info.cost) || 0;
85
+ if (message.info.modelID)
86
+ modelName = message.info.modelID;
87
+ }
88
+ }
89
+ const totalTokens = inputTokens + outputTokens + reasoningTokens + cacheReadTokens + cacheWriteTokens;
90
+ const pricing = this.getPricing(modelName);
91
+ const estimatedCost = (inputTokens / 1_000_000) * pricing.input +
92
+ ((outputTokens + reasoningTokens) / 1_000_000) * pricing.output +
93
+ (cacheReadTokens / 1_000_000) * pricing.cacheRead +
94
+ (cacheWriteTokens / 1_000_000) * pricing.cacheWrite;
95
+ return {
96
+ sessionID: child.id,
97
+ title: child.title,
98
+ agentType,
99
+ inputTokens,
100
+ outputTokens,
101
+ reasoningTokens,
102
+ cacheReadTokens,
103
+ cacheWriteTokens,
104
+ totalTokens,
105
+ apiCost,
106
+ estimatedCost,
107
+ assistantMessageCount,
108
+ };
109
+ }
110
+ catch (error) {
111
+ console.error(`Failed to analyze child session ${child.id}:`, error);
112
+ return null;
113
+ }
114
+ }
115
+ extractAgentType(title) {
116
+ const match = title.match(/@(\w+)\s+subagent/i);
117
+ if (match)
118
+ return match[1];
119
+ const words = title.split(/\s+/);
120
+ return words[0]?.toLowerCase() || "subagent";
121
+ }
122
+ getPricing(modelName) {
123
+ const normalizedName = modelName.includes("/") ? modelName.split("/").pop() || modelName : modelName;
124
+ if (this.pricingData[normalizedName])
125
+ return this.pricingData[normalizedName];
126
+ const lowerModel = normalizedName.toLowerCase();
127
+ for (const [key, pricing] of Object.entries(this.pricingData)) {
128
+ if (lowerModel.startsWith(key.toLowerCase()))
129
+ return pricing;
130
+ }
131
+ return this.pricingData["default"] || { input: 1, output: 3, cacheWrite: 0, cacheRead: 0 };
132
+ }
133
+ }
134
+ //# sourceMappingURL=subagent.js.map