@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.
- package/LICENSE +21 -0
- package/README.md +526 -0
- package/dist/tokenscope-lib/analyzer.d.ts +29 -0
- package/dist/tokenscope-lib/analyzer.d.ts.map +1 -0
- package/dist/tokenscope-lib/analyzer.js +321 -0
- package/dist/tokenscope-lib/analyzer.js.map +1 -0
- package/dist/tokenscope-lib/config.d.ts +10 -0
- package/dist/tokenscope-lib/config.d.ts.map +1 -0
- package/dist/tokenscope-lib/config.js +102 -0
- package/dist/tokenscope-lib/config.js.map +1 -0
- package/dist/tokenscope-lib/context.d.ts +81 -0
- package/dist/tokenscope-lib/context.d.ts.map +1 -0
- package/dist/tokenscope-lib/context.js +453 -0
- package/dist/tokenscope-lib/context.js.map +1 -0
- package/dist/tokenscope-lib/cost.d.ts +9 -0
- package/dist/tokenscope-lib/cost.d.ts.map +1 -0
- package/dist/tokenscope-lib/cost.js +51 -0
- package/dist/tokenscope-lib/cost.js.map +1 -0
- package/dist/tokenscope-lib/formatter.d.ts +26 -0
- package/dist/tokenscope-lib/formatter.d.ts.map +1 -0
- package/dist/tokenscope-lib/formatter.js +415 -0
- package/dist/tokenscope-lib/formatter.js.map +1 -0
- package/dist/tokenscope-lib/subagent.d.ts +13 -0
- package/dist/tokenscope-lib/subagent.d.ts.map +1 -0
- package/dist/tokenscope-lib/subagent.js +134 -0
- package/dist/tokenscope-lib/subagent.js.map +1 -0
- package/dist/tokenscope-lib/tokenizer.d.ts +17 -0
- package/dist/tokenscope-lib/tokenizer.d.ts.map +1 -0
- package/dist/tokenscope-lib/tokenizer.js +117 -0
- package/dist/tokenscope-lib/tokenizer.js.map +1 -0
- package/dist/tokenscope-lib/types.d.ts +252 -0
- package/dist/tokenscope-lib/types.d.ts.map +1 -0
- package/dist/tokenscope-lib/types.js +12 -0
- package/dist/tokenscope-lib/types.js.map +1 -0
- package/dist/tokenscope.d.ts +4 -0
- package/dist/tokenscope.d.ts.map +1 -0
- package/dist/tokenscope.js +98 -0
- package/dist/tokenscope.js.map +1 -0
- package/models.json +290 -0
- package/package.json +54 -0
- 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
|