@roberttlange/agentlens 0.2.1 → 0.2.3

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 (57) hide show
  1. package/node_modules/@agentlens/contracts/dist/index.d.ts +14 -1
  2. package/node_modules/@agentlens/core/dist/__tests__/compaction.test.d.ts +1 -0
  3. package/node_modules/@agentlens/core/dist/__tests__/compaction.test.js +139 -0
  4. package/node_modules/@agentlens/core/dist/__tests__/compaction.test.js.map +1 -0
  5. package/node_modules/@agentlens/core/dist/__tests__/config.test.js +50 -2
  6. package/node_modules/@agentlens/core/dist/__tests__/config.test.js.map +1 -1
  7. package/node_modules/@agentlens/core/dist/__tests__/index.test.js +220 -0
  8. package/node_modules/@agentlens/core/dist/__tests__/index.test.js.map +1 -1
  9. package/node_modules/@agentlens/core/dist/config.js +62 -5
  10. package/node_modules/@agentlens/core/dist/config.js.map +1 -1
  11. package/node_modules/@agentlens/core/dist/discovery.js +7 -0
  12. package/node_modules/@agentlens/core/dist/discovery.js.map +1 -1
  13. package/node_modules/@agentlens/core/dist/generatedPricing.d.ts +3 -0
  14. package/node_modules/@agentlens/core/dist/generatedPricing.js +131 -0
  15. package/node_modules/@agentlens/core/dist/generatedPricing.js.map +1 -0
  16. package/node_modules/@agentlens/core/dist/metrics.js +131 -54
  17. package/node_modules/@agentlens/core/dist/metrics.js.map +1 -1
  18. package/node_modules/@agentlens/core/dist/parsers/claude.js +21 -0
  19. package/node_modules/@agentlens/core/dist/parsers/claude.js.map +1 -1
  20. package/node_modules/@agentlens/core/dist/parsers/claude.test.js +32 -0
  21. package/node_modules/@agentlens/core/dist/parsers/claude.test.js.map +1 -1
  22. package/node_modules/@agentlens/core/dist/parsers/codex.js +43 -0
  23. package/node_modules/@agentlens/core/dist/parsers/codex.js.map +1 -1
  24. package/node_modules/@agentlens/core/dist/parsers/codex.test.d.ts +1 -0
  25. package/node_modules/@agentlens/core/dist/parsers/codex.test.js +44 -0
  26. package/node_modules/@agentlens/core/dist/parsers/codex.test.js.map +1 -0
  27. package/node_modules/@agentlens/core/dist/pricing.d.ts +15 -0
  28. package/node_modules/@agentlens/core/dist/pricing.js +133 -0
  29. package/node_modules/@agentlens/core/dist/pricing.js.map +1 -0
  30. package/node_modules/@agentlens/core/dist/pricing.test.d.ts +1 -0
  31. package/node_modules/@agentlens/core/dist/pricing.test.js +109 -0
  32. package/node_modules/@agentlens/core/dist/pricing.test.js.map +1 -0
  33. package/node_modules/@agentlens/core/dist/sourceProfiles.js +4 -68
  34. package/node_modules/@agentlens/core/dist/sourceProfiles.js.map +1 -1
  35. package/node_modules/@agentlens/core/dist/traceIndex.d.ts +12 -0
  36. package/node_modules/@agentlens/core/dist/traceIndex.js +70 -0
  37. package/node_modules/@agentlens/core/dist/traceIndex.js.map +1 -1
  38. package/node_modules/@agentlens/server/dist/activity-cache.d.ts +29 -0
  39. package/node_modules/@agentlens/server/dist/activity-cache.js +57 -0
  40. package/node_modules/@agentlens/server/dist/activity-cache.js.map +1 -0
  41. package/node_modules/@agentlens/server/dist/activity-cache.test.d.ts +1 -0
  42. package/node_modules/@agentlens/server/dist/activity-cache.test.js +84 -0
  43. package/node_modules/@agentlens/server/dist/activity-cache.test.js.map +1 -0
  44. package/node_modules/@agentlens/server/dist/activity.d.ts +12 -1
  45. package/node_modules/@agentlens/server/dist/activity.js +58 -35
  46. package/node_modules/@agentlens/server/dist/activity.js.map +1 -1
  47. package/node_modules/@agentlens/server/dist/app.js +6 -0
  48. package/node_modules/@agentlens/server/dist/app.js.map +1 -1
  49. package/node_modules/@agentlens/server/dist/app.test.js +1 -1
  50. package/node_modules/@agentlens/server/dist/app.test.js.map +1 -1
  51. package/node_modules/@agentlens/server/dist/web/assets/index-DjwZvHl6.css +1 -0
  52. package/node_modules/@agentlens/server/dist/web/assets/index-Ei_qfgA9.js +52 -0
  53. package/node_modules/@agentlens/server/dist/web/favicon.png +0 -0
  54. package/node_modules/@agentlens/server/dist/web/index.html +2 -2
  55. package/package.json +1 -1
  56. package/node_modules/@agentlens/server/dist/web/assets/index-Cnpi6Bab.css +0 -1
  57. package/node_modules/@agentlens/server/dist/web/assets/index-D9NAFTEt.js +0 -52
@@ -1,5 +1,5 @@
1
1
  export type AgentKind = "claude" | "codex" | "cursor" | "opencode" | "gemini" | "pi" | "unknown";
2
- export type EventKind = "system" | "assistant" | "user" | "tool_use" | "tool_result" | "reasoning" | "meta";
2
+ export type EventKind = "system" | "assistant" | "user" | "tool_use" | "tool_result" | "reasoning" | "compaction" | "meta";
3
3
  export type SessionActivityStatus = "running" | "waiting_input" | "idle";
4
4
  export type ActivityBinsMode = "time" | "event_index";
5
5
  export type CostUnknownModelPolicy = "n_a" | "zero";
@@ -49,7 +49,18 @@ export interface CostModelRate {
49
49
  outputPer1MUsd: number;
50
50
  cachedReadPer1MUsd: number;
51
51
  cachedCreatePer1MUsd: number;
52
+ cachedCreate5mPer1MUsd?: number;
53
+ cachedCreate1hPer1MUsd?: number;
52
54
  reasoningOutputPer1MUsd: number;
55
+ longContextThresholdTokens?: number;
56
+ longContextInputPer1MUsd?: number;
57
+ longContextOutputPer1MUsd?: number;
58
+ longContextCachedReadPer1MUsd?: number;
59
+ longContextCachedCreatePer1MUsd?: number;
60
+ longContextCachedCreate5mPer1MUsd?: number;
61
+ longContextCachedCreate1hPer1MUsd?: number;
62
+ longContextReasoningOutputPer1MUsd?: number;
63
+ contextWindowTokens?: number;
53
64
  }
54
65
  export interface CostConfig {
55
66
  enabled: boolean;
@@ -144,6 +155,8 @@ export interface TraceSummary {
144
155
  errorCount: number;
145
156
  toolUseCount: number;
146
157
  toolResultCount: number;
158
+ compactionCount: number;
159
+ lastCompactionTs: number | null;
147
160
  unmatchedToolUses: number;
148
161
  unmatchedToolResults: number;
149
162
  activityStatus: SessionActivityStatus;
@@ -0,0 +1,139 @@
1
+ import { mkdtemp, mkdir, writeFile } from "node:fs/promises";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ import { describe, expect, it } from "vitest";
5
+ import { mergeConfig } from "../config.js";
6
+ import { TraceIndex } from "../traceIndex.js";
7
+ async function createTempRoot() {
8
+ return mkdtemp(path.join(os.tmpdir(), "agentlens-core-compaction-"));
9
+ }
10
+ describe("compaction regression coverage", () => {
11
+ it("tracks compaction count and last compaction timestamp in session summary", async () => {
12
+ const root = await createTempRoot();
13
+ const codexDir = path.join(root, ".codex", "sessions", "2026", "02", "28");
14
+ await mkdir(codexDir, { recursive: true });
15
+ const codexPath = path.join(codexDir, "compaction-summary.jsonl");
16
+ const lines = [
17
+ JSON.stringify({
18
+ timestamp: "2026-02-28T09:06:40.000Z",
19
+ type: "session_meta",
20
+ payload: { id: "sess-compaction", cwd: "/tmp/project", cli_version: "0.1.0" },
21
+ }),
22
+ JSON.stringify({
23
+ timestamp: "2026-02-28T09:06:42.000Z",
24
+ type: "compacted",
25
+ payload: { message: "Context compacted by agent", replacement_history: [] },
26
+ }),
27
+ JSON.stringify({
28
+ timestamp: "2026-02-28T09:06:45.255Z",
29
+ type: "event_msg",
30
+ payload: { type: "context_compacted", message: "Auto compaction complete" },
31
+ }),
32
+ ];
33
+ await writeFile(codexPath, lines.join("\n"), "utf8");
34
+ const config = mergeConfig({
35
+ sessionLogDirectories: [],
36
+ sources: {
37
+ codex_home: {
38
+ name: "codex_home",
39
+ enabled: true,
40
+ roots: [path.join(root, ".codex", "sessions")],
41
+ includeGlobs: ["**/*.jsonl"],
42
+ excludeGlobs: [],
43
+ maxDepth: 8,
44
+ agentHint: "codex",
45
+ },
46
+ },
47
+ });
48
+ const index = new TraceIndex(config);
49
+ await index.refresh();
50
+ const summary = index.getSummaries()[0];
51
+ expect(summary?.sessionId).toBe("sess-compaction");
52
+ expect(summary?.compactionCount).toBe(2);
53
+ expect(summary?.lastCompactionTs).toBe(Date.parse("2026-02-28T09:06:45.255Z"));
54
+ expect(summary?.eventKindCounts.compaction).toBe(2);
55
+ });
56
+ it("ignores claude compact sidechain logs discovered from sessionLogDirectories", async () => {
57
+ const root = await createTempRoot();
58
+ const sessionDir = path.join(root, ".claude", "projects", "proj", "session-1");
59
+ await mkdir(path.join(sessionDir, "subagents"), { recursive: true });
60
+ const mainSessionPath = path.join(sessionDir, "session.jsonl");
61
+ const compactSidechainPath = path.join(sessionDir, "subagents", "agent-acompact-90240fc4860d3bb9.jsonl");
62
+ await writeFile(mainSessionPath, `${JSON.stringify({
63
+ type: "assistant",
64
+ sessionId: "claude-main-session",
65
+ timestamp: "2026-02-28T10:00:00.000Z",
66
+ message: {
67
+ role: "assistant",
68
+ content: [{ type: "text", text: "Main session event" }],
69
+ },
70
+ })}\n`, "utf8");
71
+ await writeFile(compactSidechainPath, `${JSON.stringify({
72
+ type: "assistant",
73
+ sessionId: "claude-compaction-sidechain",
74
+ timestamp: "2026-02-28T10:01:00.000Z",
75
+ message: {
76
+ role: "assistant",
77
+ content: [{ type: "text", text: "Sidechain compact event" }],
78
+ },
79
+ })}\n`, "utf8");
80
+ const config = mergeConfig({
81
+ sessionLogDirectories: [{ directory: path.join(root, ".claude"), logType: "claude" }],
82
+ sources: {},
83
+ });
84
+ const index = new TraceIndex(config);
85
+ await index.refresh();
86
+ const summaries = index.getSummaries();
87
+ expect(summaries).toHaveLength(1);
88
+ expect(summaries[0]?.sessionId).toBe("claude-main-session");
89
+ expect(summaries[0]?.path).toContain(path.sep + "session.jsonl");
90
+ expect(summaries[0]?.path).not.toContain(`subagents${path.sep}agent-acompact-`);
91
+ });
92
+ it("excludes claude compact sidechain logs in source profile discovery", async () => {
93
+ const root = await createTempRoot();
94
+ const sessionDir = path.join(root, ".claude", "projects", "proj", "session-2");
95
+ await mkdir(path.join(sessionDir, "subagents"), { recursive: true });
96
+ const mainSessionPath = path.join(sessionDir, "session.jsonl");
97
+ const compactSidechainPath = path.join(sessionDir, "subagents", "agent-acompact-fedcba123456.jsonl");
98
+ await writeFile(mainSessionPath, `${JSON.stringify({
99
+ type: "assistant",
100
+ sessionId: "claude-source-main-session",
101
+ timestamp: "2026-02-28T11:00:00.000Z",
102
+ message: {
103
+ role: "assistant",
104
+ content: [{ type: "text", text: "Main source-profile session event" }],
105
+ },
106
+ })}\n`, "utf8");
107
+ await writeFile(compactSidechainPath, `${JSON.stringify({
108
+ type: "assistant",
109
+ sessionId: "claude-source-sidechain",
110
+ timestamp: "2026-02-28T11:01:00.000Z",
111
+ message: {
112
+ role: "assistant",
113
+ content: [{ type: "text", text: "Sidechain source-profile event" }],
114
+ },
115
+ })}\n`, "utf8");
116
+ const config = mergeConfig({
117
+ sessionLogDirectories: [],
118
+ sources: {
119
+ claude_projects: {
120
+ name: "claude_projects",
121
+ enabled: true,
122
+ roots: [path.join(root, ".claude", "projects")],
123
+ includeGlobs: ["**/*.jsonl"],
124
+ excludeGlobs: ["**/subagents/agent-acompact-*.jsonl"],
125
+ maxDepth: 8,
126
+ agentHint: "claude",
127
+ },
128
+ },
129
+ });
130
+ const index = new TraceIndex(config);
131
+ await index.refresh();
132
+ const summaries = index.getSummaries();
133
+ expect(summaries).toHaveLength(1);
134
+ expect(summaries[0]?.sessionId).toBe("claude-source-main-session");
135
+ expect(summaries[0]?.path).toContain(path.sep + "session.jsonl");
136
+ expect(summaries[0]?.path).not.toContain(`subagents${path.sep}agent-acompact-`);
137
+ });
138
+ });
139
+ //# sourceMappingURL=compaction.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"compaction.test.js","sourceRoot":"","sources":["../../src/__tests__/compaction.test.ts"],"names":[],"mappings":"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"}
@@ -18,11 +18,17 @@ describe("config", () => {
18
18
  expect(config.cost.unknownModelPolicy).toBe("n_a");
19
19
  expect(config.cost.modelRates.length).toBeGreaterThan(0);
20
20
  expect(config.cost.modelRates.some((rate) => rate.model === "gpt-5.3-codex")).toBe(true);
21
+ expect(config.cost.modelRates.some((rate) => rate.model === "gpt-5.4")).toBe(true);
22
+ expect(config.cost.modelRates.some((rate) => rate.model === "claude-sonnet-4.6")).toBe(true);
21
23
  expect(config.cost.modelRates.some((rate) => rate.model === "claude-opus-4-5-20251101")).toBe(true);
22
24
  expect(config.models.defaultContextWindowTokens).toBeGreaterThan(0);
23
25
  expect(config.models.contextWindows.some((entry) => entry.model === "gpt-5.2-codex")).toBe(true);
24
26
  expect(config.models.contextWindows.some((entry) => entry.model === "gpt-5.2-codex" && entry.contextWindowTokens === 400_000)).toBe(true);
25
27
  expect(config.models.contextWindows.some((entry) => entry.model === "claude-sonnet-4-5-20250929" && entry.contextWindowTokens === 200_000)).toBe(true);
28
+ expect(config.models.contextWindows.some((entry) => entry.model === "gpt-5.4" && entry.contextWindowTokens === 1_050_000)).toBe(true);
29
+ expect(config.models.contextWindows.some((entry) => entry.model === "claude-opus-4.6" && entry.contextWindowTokens === 200_000)).toBe(true);
30
+ expect(config.models.contextWindows.some((entry) => entry.model === "claude-sonnet-4.6" && entry.contextWindowTokens === 200_000)).toBe(true);
31
+ expect(config.models.contextWindows.some((entry) => entry.model === "claude-haiku-4.5" && entry.contextWindowTokens === 200_000)).toBe(true);
26
32
  expect(config.sessionLogDirectories).toContainEqual({ directory: "~/.gemini", logType: "gemini" });
27
33
  expect(config.sessionLogDirectories).toContainEqual({ directory: "~/.pi", logType: "pi" });
28
34
  const defaultEnabledSources = [
@@ -37,6 +43,7 @@ describe("config", () => {
37
43
  for (const sourceName of defaultEnabledSources) {
38
44
  expect(config.sources[sourceName]?.enabled).toBe(true);
39
45
  }
46
+ expect(config.sources.claude_projects?.excludeGlobs).toContain("**/subagents/agent-acompact-*.jsonl");
40
47
  });
41
48
  it("infers gemini log type from legacy sessionJsonlDirectories paths", () => {
42
49
  const config = mergeConfig({
@@ -122,7 +129,12 @@ inputPer1MUsd = 1.25
122
129
  outputPer1MUsd = 2.5
123
130
  cachedReadPer1MUsd = 0.5
124
131
  cachedCreatePer1MUsd = 0.75
132
+ cachedCreate5mPer1MUsd = 0.75
133
+ cachedCreate1hPer1MUsd = 1.25
125
134
  reasoningOutputPer1MUsd = 3.0
135
+ longContextThresholdTokens = 272000
136
+ longContextInputPer1MUsd = 2.5
137
+ contextWindowTokens = 400000
126
138
 
127
139
  [models]
128
140
  defaultContextWindowTokens = 123456
@@ -153,14 +165,50 @@ maxResidentEventsPerWarmTrace = 20
153
165
  const config = await loadConfig(configPath);
154
166
  expect(config.traceInspector.topModelCount).toBe(2);
155
167
  expect(config.redaction.replacement).toBe("[MASK]");
156
- expect(config.cost.modelRates[0]?.model).toBe("gpt-5.3-codex");
168
+ const gpt53CodexRate = config.cost.modelRates.find((rate) => rate.model === "gpt-5.3-codex");
169
+ expect(gpt53CodexRate?.cachedCreate1hPer1MUsd).toBe(1.25);
170
+ expect(gpt53CodexRate?.longContextThresholdTokens).toBe(272000);
171
+ expect(gpt53CodexRate?.contextWindowTokens).toBe(400000);
157
172
  expect(config.models.defaultContextWindowTokens).toBe(123456);
158
- expect(config.models.contextWindows[0]?.contextWindowTokens).toBe(272000);
173
+ expect(config.models.contextWindows.find((entry) => entry.model === "gpt-5.3-codex")?.contextWindowTokens).toBe(272000);
159
174
  expect(config.scan.mode).toBe("fixed");
160
175
  expect(config.scan.intervalSeconds).toBe(7);
161
176
  expect(config.scan.batchDebounceMs).toBe(88);
162
177
  expect(config.retention.strategy).toBe("full_memory");
163
178
  expect(config.retention.hotTraceCount).toBe(9);
164
179
  });
180
+ it("merges legacy explicit pricing and context entries onto new defaults", () => {
181
+ const config = mergeConfig({
182
+ cost: {
183
+ enabled: true,
184
+ currency: "USD",
185
+ unknownModelPolicy: "n_a",
186
+ modelRates: [
187
+ {
188
+ model: "gpt-5.3-codex",
189
+ inputPer1MUsd: 9,
190
+ outputPer1MUsd: 10,
191
+ cachedReadPer1MUsd: 1,
192
+ cachedCreatePer1MUsd: 2,
193
+ reasoningOutputPer1MUsd: 3,
194
+ },
195
+ ],
196
+ },
197
+ models: {
198
+ defaultContextWindowTokens: 200_000,
199
+ contextWindows: [{ model: "gpt-5.3-codex", contextWindowTokens: 123_000 }],
200
+ },
201
+ });
202
+ const codexRate = config.cost.modelRates.find((rate) => rate.model === "gpt-5.3-codex");
203
+ const gpt54Rate = config.cost.modelRates.find((rate) => rate.model === "gpt-5.4");
204
+ const codexWindow = config.models.contextWindows.find((entry) => entry.model === "gpt-5.3-codex");
205
+ const gpt54Window = config.models.contextWindows.find((entry) => entry.model === "gpt-5.4");
206
+ expect(codexRate?.inputPer1MUsd).toBe(9);
207
+ expect(codexRate?.outputPer1MUsd).toBe(10);
208
+ expect(gpt54Rate?.inputPer1MUsd).toBe(1.25);
209
+ expect(gpt54Rate?.longContextThresholdTokens).toBe(272_000);
210
+ expect(codexWindow?.contextWindowTokens).toBe(123_000);
211
+ expect(gpt54Window?.contextWindowTokens).toBe(1_050_000);
212
+ });
165
213
  });
166
214
  //# sourceMappingURL=config.test.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"config.test.js","sourceRoot":"","sources":["../../src/__tests__/config.test.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"config.test.js","sourceRoot":"","sources":["../../src/__tests__/config.test.ts"],"names":[],"mappings":"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"}
@@ -1806,6 +1806,132 @@ describe("trace index", () => {
1806
1806
  const summary = index.getSummaries()[0];
1807
1807
  expect(summary?.costEstimateUsd).toBe(0.0001);
1808
1808
  });
1809
+ it("applies GPT-5.4 long-context pricing per codex token-count row", async () => {
1810
+ const root = await createTempRoot();
1811
+ const codexDir = path.join(root, ".codex", "sessions", "2026", "02", "13");
1812
+ await mkdir(codexDir, { recursive: true });
1813
+ const codexPath = path.join(codexDir, "rollout-gpt54-tiered.jsonl");
1814
+ await writeFile(codexPath, [
1815
+ JSON.stringify({
1816
+ timestamp: "2026-02-13T12:00:00.000Z",
1817
+ type: "session_meta",
1818
+ payload: { id: "sess-gpt54-tiered", cwd: "/tmp/project", cli_version: "0.1.0" },
1819
+ }),
1820
+ JSON.stringify({
1821
+ timestamp: "2026-02-13T12:00:01.000Z",
1822
+ type: "turn_context",
1823
+ payload: { model: "gpt-5.4-2026-02-28" },
1824
+ }),
1825
+ JSON.stringify({
1826
+ timestamp: "2026-02-13T12:00:02.000Z",
1827
+ type: "event_msg",
1828
+ payload: {
1829
+ type: "token_count",
1830
+ info: {
1831
+ total_token_usage: {
1832
+ input_tokens: 100_000,
1833
+ cached_input_tokens: 10_000,
1834
+ output_tokens: 10_000,
1835
+ reasoning_output_tokens: 0,
1836
+ total_tokens: 120_000,
1837
+ },
1838
+ last_token_usage: {
1839
+ input_tokens: 100_000,
1840
+ cached_input_tokens: 10_000,
1841
+ output_tokens: 10_000,
1842
+ reasoning_output_tokens: 0,
1843
+ total_tokens: 120_000,
1844
+ },
1845
+ model_context_window: 1_050_000,
1846
+ },
1847
+ },
1848
+ }),
1849
+ JSON.stringify({
1850
+ timestamp: "2026-02-13T12:00:03.000Z",
1851
+ type: "event_msg",
1852
+ payload: {
1853
+ type: "token_count",
1854
+ info: {
1855
+ total_token_usage: {
1856
+ input_tokens: 400_000,
1857
+ cached_input_tokens: 30_000,
1858
+ output_tokens: 20_000,
1859
+ reasoning_output_tokens: 0,
1860
+ total_tokens: 450_000,
1861
+ },
1862
+ last_token_usage: {
1863
+ input_tokens: 300_000,
1864
+ cached_input_tokens: 20_000,
1865
+ output_tokens: 10_000,
1866
+ reasoning_output_tokens: 0,
1867
+ total_tokens: 330_000,
1868
+ },
1869
+ model_context_window: 1_050_000,
1870
+ },
1871
+ },
1872
+ }),
1873
+ ].join("\n"), "utf8");
1874
+ const config = mergeConfig({
1875
+ sessionLogDirectories: [],
1876
+ cost: {
1877
+ enabled: true,
1878
+ currency: "USD",
1879
+ unknownModelPolicy: "n_a",
1880
+ modelRates: [
1881
+ {
1882
+ model: "gpt-5.4",
1883
+ inputPer1MUsd: 1.25,
1884
+ outputPer1MUsd: 7.5,
1885
+ cachedReadPer1MUsd: 0.125,
1886
+ cachedCreatePer1MUsd: 0,
1887
+ reasoningOutputPer1MUsd: 0,
1888
+ longContextThresholdTokens: 272_000,
1889
+ longContextInputPer1MUsd: 2.5,
1890
+ longContextOutputPer1MUsd: 11.25,
1891
+ contextWindowTokens: 1_050_000,
1892
+ },
1893
+ ],
1894
+ },
1895
+ models: {
1896
+ defaultContextWindowTokens: 200_000,
1897
+ contextWindows: [{ model: "gpt-5.4", contextWindowTokens: 1_050_000 }],
1898
+ },
1899
+ sources: {
1900
+ codex_home: {
1901
+ name: "codex_home",
1902
+ enabled: true,
1903
+ roots: [path.join(root, ".codex", "sessions")],
1904
+ includeGlobs: ["**/*.jsonl"],
1905
+ excludeGlobs: [],
1906
+ maxDepth: 8,
1907
+ agentHint: "codex",
1908
+ },
1909
+ claude_projects: {
1910
+ name: "claude_projects",
1911
+ enabled: false,
1912
+ roots: [],
1913
+ includeGlobs: ["**/*.jsonl"],
1914
+ excludeGlobs: [],
1915
+ maxDepth: 8,
1916
+ agentHint: "claude",
1917
+ },
1918
+ claude_history: {
1919
+ name: "claude_history",
1920
+ enabled: false,
1921
+ roots: [],
1922
+ includeGlobs: ["history.jsonl"],
1923
+ excludeGlobs: [],
1924
+ maxDepth: 8,
1925
+ agentHint: "claude",
1926
+ },
1927
+ },
1928
+ });
1929
+ const index = new TraceIndex(config);
1930
+ await index.refresh();
1931
+ const summary = index.getSummaries()[0];
1932
+ expect(summary?.contextWindowPct).toBeCloseTo(30.47619, 5);
1933
+ expect(summary?.costEstimateUsd).toBe(1.06625);
1934
+ });
1809
1935
  it("deduplicates repeated claude usage rows by request id for cost estimation", async () => {
1810
1936
  const root = await createTempRoot();
1811
1937
  const claudeDir = path.join(root, ".claude", "projects", "proj");
@@ -1925,6 +2051,100 @@ describe("trace index", () => {
1925
2051
  const summary = index.getSummaries()[0];
1926
2052
  expect(summary?.costEstimateUsd).toBe(0.001065);
1927
2053
  });
2054
+ it("applies Claude long-context pricing and alias normalization from assistant usage rows", async () => {
2055
+ const root = await createTempRoot();
2056
+ const claudeDir = path.join(root, ".claude", "projects", "proj");
2057
+ await mkdir(claudeDir, { recursive: true });
2058
+ const claudePath = path.join(claudeDir, "session-cost-tiered.jsonl");
2059
+ await writeFile(claudePath, [
2060
+ JSON.stringify({
2061
+ type: "assistant",
2062
+ sessionId: "claude-tiered-sess",
2063
+ uuid: "u1",
2064
+ requestId: "req_tiered_1",
2065
+ message: {
2066
+ model: "claude-sonnet-4-6-20260219",
2067
+ id: "msg_1",
2068
+ type: "message",
2069
+ role: "assistant",
2070
+ content: [{ type: "text", text: "tiered" }],
2071
+ usage: {
2072
+ input_tokens: 100_000,
2073
+ cache_creation_input_tokens: 10_000,
2074
+ cache_read_input_tokens: 120_000,
2075
+ cache_creation: { ephemeral_5m_input_tokens: 5_000, ephemeral_1h_input_tokens: 5_000 },
2076
+ output_tokens: 50_000,
2077
+ },
2078
+ },
2079
+ }),
2080
+ ].join("\n"), "utf8");
2081
+ const config = mergeConfig({
2082
+ sessionLogDirectories: [],
2083
+ cost: {
2084
+ enabled: true,
2085
+ currency: "USD",
2086
+ unknownModelPolicy: "n_a",
2087
+ modelRates: [
2088
+ {
2089
+ model: "claude-sonnet-4.6",
2090
+ inputPer1MUsd: 3,
2091
+ outputPer1MUsd: 15,
2092
+ cachedReadPer1MUsd: 0.3,
2093
+ cachedCreatePer1MUsd: 3.75,
2094
+ cachedCreate5mPer1MUsd: 3.75,
2095
+ cachedCreate1hPer1MUsd: 6,
2096
+ reasoningOutputPer1MUsd: 0,
2097
+ longContextThresholdTokens: 200_000,
2098
+ longContextInputPer1MUsd: 6,
2099
+ longContextOutputPer1MUsd: 22.5,
2100
+ longContextCachedReadPer1MUsd: 0.6,
2101
+ longContextCachedCreatePer1MUsd: 7.5,
2102
+ longContextCachedCreate5mPer1MUsd: 7.5,
2103
+ longContextCachedCreate1hPer1MUsd: 12,
2104
+ contextWindowTokens: 1_000_000,
2105
+ },
2106
+ ],
2107
+ },
2108
+ models: {
2109
+ defaultContextWindowTokens: 200_000,
2110
+ contextWindows: [{ model: "claude-sonnet-4.6", contextWindowTokens: 1_000_000 }],
2111
+ },
2112
+ sources: {
2113
+ claude_projects: {
2114
+ name: "claude_projects",
2115
+ enabled: true,
2116
+ roots: [path.join(root, ".claude", "projects")],
2117
+ includeGlobs: ["**/*.jsonl"],
2118
+ excludeGlobs: [],
2119
+ maxDepth: 8,
2120
+ agentHint: "claude",
2121
+ },
2122
+ codex_home: {
2123
+ name: "codex_home",
2124
+ enabled: false,
2125
+ roots: [],
2126
+ includeGlobs: ["**/*.jsonl"],
2127
+ excludeGlobs: [],
2128
+ maxDepth: 8,
2129
+ agentHint: "codex",
2130
+ },
2131
+ claude_history: {
2132
+ name: "claude_history",
2133
+ enabled: false,
2134
+ roots: [],
2135
+ includeGlobs: ["history.jsonl"],
2136
+ excludeGlobs: [],
2137
+ maxDepth: 8,
2138
+ agentHint: "claude",
2139
+ },
2140
+ },
2141
+ });
2142
+ const index = new TraceIndex(config);
2143
+ await index.refresh();
2144
+ const summary = index.getSummaries()[0];
2145
+ expect(summary?.contextWindowPct).toBe(23);
2146
+ expect(summary?.costEstimateUsd).toBe(1.8945);
2147
+ });
1928
2148
  it("redacts secret-like values from event previews and raw payloads", async () => {
1929
2149
  const root = await createTempRoot();
1930
2150
  const codexDir = path.join(root, ".codex", "sessions", "2026", "02", "13");