oh-my-claude-sisyphus 3.4.1 → 3.4.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.
- package/commands/cancel-ultraqa.md +1 -1
- package/dist/__tests__/analytics/analytics-summary.test.d.ts +2 -0
- package/dist/__tests__/analytics/analytics-summary.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/analytics-summary.test.js +267 -0
- package/dist/__tests__/analytics/analytics-summary.test.js.map +1 -0
- package/dist/__tests__/analytics/backfill-dedup.test.d.ts +2 -0
- package/dist/__tests__/analytics/backfill-dedup.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/backfill-dedup.test.js +179 -0
- package/dist/__tests__/analytics/backfill-dedup.test.js.map +1 -0
- package/dist/__tests__/analytics/backfill-engine.test.d.ts +2 -0
- package/dist/__tests__/analytics/backfill-engine.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/backfill-engine.test.js +362 -0
- package/dist/__tests__/analytics/backfill-engine.test.js.map +1 -0
- package/dist/__tests__/analytics/cost-estimator.test.d.ts +2 -0
- package/dist/__tests__/analytics/cost-estimator.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/cost-estimator.test.js +212 -0
- package/dist/__tests__/analytics/cost-estimator.test.js.map +1 -0
- package/dist/__tests__/analytics/output-estimator.test.d.ts +2 -0
- package/dist/__tests__/analytics/output-estimator.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/output-estimator.test.js +106 -0
- package/dist/__tests__/analytics/output-estimator.test.js.map +1 -0
- package/dist/__tests__/analytics/token-extractor.test.d.ts +2 -0
- package/dist/__tests__/analytics/token-extractor.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/token-extractor.test.js +121 -0
- package/dist/__tests__/analytics/token-extractor.test.js.map +1 -0
- package/dist/__tests__/analytics/transcript-parser.test.d.ts +2 -0
- package/dist/__tests__/analytics/transcript-parser.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/transcript-parser.test.js +285 -0
- package/dist/__tests__/analytics/transcript-parser.test.js.map +1 -0
- package/dist/__tests__/analytics/transcript-scanner.test.d.ts +2 -0
- package/dist/__tests__/analytics/transcript-scanner.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/transcript-scanner.test.js +401 -0
- package/dist/__tests__/analytics/transcript-scanner.test.js.map +1 -0
- package/dist/__tests__/analytics/transcript-token-extractor.test.d.ts +2 -0
- package/dist/__tests__/analytics/transcript-token-extractor.test.d.ts.map +1 -0
- package/dist/__tests__/analytics/transcript-token-extractor.test.js +175 -0
- package/dist/__tests__/analytics/transcript-token-extractor.test.js.map +1 -0
- package/dist/__tests__/hud/auto-tracking.integration.test.d.ts +2 -0
- package/dist/__tests__/hud/auto-tracking.integration.test.d.ts.map +1 -0
- package/dist/__tests__/hud/auto-tracking.integration.test.js +12 -0
- package/dist/__tests__/hud/auto-tracking.integration.test.js.map +1 -0
- package/dist/analytics/analytics-summary.d.ts +47 -0
- package/dist/analytics/analytics-summary.d.ts.map +1 -0
- package/dist/analytics/analytics-summary.js +170 -0
- package/dist/analytics/analytics-summary.js.map +1 -0
- package/dist/analytics/backfill-dedup.d.ts +49 -0
- package/dist/analytics/backfill-dedup.d.ts.map +1 -0
- package/dist/analytics/backfill-dedup.js +118 -0
- package/dist/analytics/backfill-dedup.js.map +1 -0
- package/dist/analytics/backfill-engine.d.ts +59 -0
- package/dist/analytics/backfill-engine.d.ts.map +1 -0
- package/dist/analytics/backfill-engine.js +163 -0
- package/dist/analytics/backfill-engine.js.map +1 -0
- package/dist/analytics/index.d.ts +8 -0
- package/dist/analytics/index.d.ts.map +1 -1
- package/dist/analytics/index.js +10 -0
- package/dist/analytics/index.js.map +1 -1
- package/dist/analytics/output-estimator.d.ts +26 -0
- package/dist/analytics/output-estimator.d.ts.map +1 -0
- package/dist/analytics/output-estimator.js +61 -0
- package/dist/analytics/output-estimator.js.map +1 -0
- package/dist/analytics/token-extractor.d.ts +31 -0
- package/dist/analytics/token-extractor.d.ts.map +1 -0
- package/dist/analytics/token-extractor.js +57 -0
- package/dist/analytics/token-extractor.js.map +1 -0
- package/dist/analytics/token-tracker.d.ts +7 -1
- package/dist/analytics/token-tracker.d.ts.map +1 -1
- package/dist/analytics/token-tracker.js +81 -0
- package/dist/analytics/token-tracker.js.map +1 -1
- package/dist/analytics/transcript-parser.d.ts +42 -0
- package/dist/analytics/transcript-parser.d.ts.map +1 -0
- package/dist/analytics/transcript-parser.js +90 -0
- package/dist/analytics/transcript-parser.js.map +1 -0
- package/dist/analytics/transcript-scanner.d.ts +50 -0
- package/dist/analytics/transcript-scanner.d.ts.map +1 -0
- package/dist/analytics/transcript-scanner.js +149 -0
- package/dist/analytics/transcript-scanner.js.map +1 -0
- package/dist/analytics/transcript-token-extractor.d.ts +19 -0
- package/dist/analytics/transcript-token-extractor.d.ts.map +1 -0
- package/dist/analytics/transcript-token-extractor.js +89 -0
- package/dist/analytics/transcript-token-extractor.js.map +1 -0
- package/dist/analytics/types.d.ts +52 -0
- package/dist/analytics/types.d.ts.map +1 -1
- package/dist/analytics/types.js.map +1 -1
- package/dist/cli/analytics.js +26 -1
- package/dist/cli/analytics.js.map +1 -1
- package/dist/cli/commands/backfill.d.ts +15 -0
- package/dist/cli/commands/backfill.d.ts.map +1 -0
- package/dist/cli/commands/backfill.js +146 -0
- package/dist/cli/commands/backfill.js.map +1 -0
- package/dist/cli/commands/stats.d.ts +1 -0
- package/dist/cli/commands/stats.d.ts.map +1 -1
- package/dist/cli/commands/stats.js +67 -31
- package/dist/cli/commands/stats.js.map +1 -1
- package/dist/cli/components/CostDashboard.d.ts +15 -0
- package/dist/cli/components/CostDashboard.d.ts.map +1 -0
- package/dist/cli/components/CostDashboard.js +15 -0
- package/dist/cli/components/CostDashboard.js.map +1 -0
- package/dist/cli/components/LiveStats.d.ts +16 -0
- package/dist/cli/components/LiveStats.d.ts.map +1 -0
- package/dist/cli/components/LiveStats.js +16 -0
- package/dist/cli/components/LiveStats.js.map +1 -0
- package/dist/cli/components/SessionBrowser.d.ts +14 -0
- package/dist/cli/components/SessionBrowser.d.ts.map +1 -0
- package/dist/cli/components/SessionBrowser.js +14 -0
- package/dist/cli/components/SessionBrowser.js.map +1 -0
- package/dist/cli/index.js +159 -3
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/tui.d.ts +21 -0
- package/dist/cli/tui.d.ts.map +1 -0
- package/dist/cli/tui.js +21 -0
- package/dist/cli/tui.js.map +1 -0
- package/dist/hud/analytics-display.d.ts +16 -0
- package/dist/hud/analytics-display.d.ts.map +1 -1
- package/dist/hud/analytics-display.js +42 -0
- package/dist/hud/analytics-display.js.map +1 -1
- package/dist/hud/index.js +90 -3
- package/dist/hud/index.js.map +1 -1
- package/dist/hud/render.d.ts.map +1 -1
- package/dist/hud/render.js +27 -1
- package/dist/hud/render.js.map +1 -1
- package/dist/hud/types.d.ts +2 -0
- package/dist/hud/types.d.ts.map +1 -1
- package/dist/hud/types.js.map +1 -1
- package/docs/ANALYTICS-SYSTEM.md +150 -0
- package/docs/FULL-README.md +2 -2
- package/hooks/keyword-detector.sh +1 -1
- package/package.json +1 -1
- package/scripts/keyword-detector.mjs +1 -1
- package/scripts/persistent-mode.mjs +1 -1
- package/scripts/test-mutual-exclusion.ts +4 -4
- package/scripts/test-remember-tags.ts +6 -6
- package/scripts/test-session-injection.ts +4 -4
- package/skills/cancel-ultraqa/SKILL.md +1 -1
- package/skills/omc-setup/SKILL.md +30 -5
|
@@ -0,0 +1,401 @@
|
|
|
1
|
+
import { vi, describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import * as fs from 'fs/promises';
|
|
3
|
+
import { mkdirSync, rmSync, existsSync } from 'fs';
|
|
4
|
+
import { join } from 'path';
|
|
5
|
+
import { homedir, tmpdir } from 'os';
|
|
6
|
+
import { scanTranscripts, decodeProjectPath } from '../../analytics/transcript-scanner.js';
|
|
7
|
+
vi.mock('fs/promises');
|
|
8
|
+
vi.mock('os');
|
|
9
|
+
describe('transcript-scanner', () => {
|
|
10
|
+
const mockHomedir = '/home/testuser';
|
|
11
|
+
const projectsDir = join(mockHomedir, '.claude', 'projects');
|
|
12
|
+
beforeEach(() => {
|
|
13
|
+
vi.mocked(homedir).mockReturnValue(mockHomedir);
|
|
14
|
+
});
|
|
15
|
+
afterEach(() => {
|
|
16
|
+
vi.clearAllMocks();
|
|
17
|
+
});
|
|
18
|
+
describe('scanTranscripts()', () => {
|
|
19
|
+
it('discovers .jsonl files in project directories', async () => {
|
|
20
|
+
const mockEntries = [
|
|
21
|
+
{ name: '-home-testuser-project1', isDirectory: () => true },
|
|
22
|
+
{ name: '-home-testuser-project2', isDirectory: () => true },
|
|
23
|
+
];
|
|
24
|
+
const mockProjectFiles1 = [
|
|
25
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
26
|
+
'b2c3d4e5-f6a7-8901-bcde-f12345678901.jsonl',
|
|
27
|
+
'sessions-index.json',
|
|
28
|
+
];
|
|
29
|
+
const mockProjectFiles2 = [
|
|
30
|
+
'c3d4e5f6-a7b8-9012-cdef-123456789012.jsonl',
|
|
31
|
+
];
|
|
32
|
+
vi.mocked(fs.readdir)
|
|
33
|
+
.mockResolvedValueOnce(mockEntries)
|
|
34
|
+
.mockResolvedValueOnce(mockProjectFiles1)
|
|
35
|
+
.mockResolvedValueOnce(mockProjectFiles2);
|
|
36
|
+
vi.mocked(fs.stat).mockImplementation(async (path) => {
|
|
37
|
+
const stats = {
|
|
38
|
+
size: 1024,
|
|
39
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
40
|
+
};
|
|
41
|
+
return stats;
|
|
42
|
+
});
|
|
43
|
+
const result = await scanTranscripts();
|
|
44
|
+
expect(result.transcripts).toHaveLength(3);
|
|
45
|
+
expect(result.projectCount).toBe(2);
|
|
46
|
+
expect(result.totalSize).toBe(3072); // 1024 * 3
|
|
47
|
+
expect(result.transcripts[0]).toMatchObject({
|
|
48
|
+
projectPath: '/home/testuser/project1',
|
|
49
|
+
projectDir: '-home-testuser-project1',
|
|
50
|
+
sessionId: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890',
|
|
51
|
+
fileSize: 1024,
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
it('filters by project pattern', async () => {
|
|
55
|
+
const mockEntries = [
|
|
56
|
+
{ name: '-home-testuser-workspace-foo', isDirectory: () => true },
|
|
57
|
+
{ name: '-home-testuser-workspace-bar', isDirectory: () => true },
|
|
58
|
+
{ name: '-home-testuser-other-baz', isDirectory: () => true },
|
|
59
|
+
];
|
|
60
|
+
const mockProjectFiles = [
|
|
61
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
62
|
+
];
|
|
63
|
+
vi.mocked(fs.readdir)
|
|
64
|
+
.mockResolvedValueOnce(mockEntries)
|
|
65
|
+
.mockResolvedValue(mockProjectFiles);
|
|
66
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
67
|
+
size: 512,
|
|
68
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
69
|
+
});
|
|
70
|
+
const result = await scanTranscripts({
|
|
71
|
+
projectFilter: '/home/testuser/workspace/*',
|
|
72
|
+
});
|
|
73
|
+
expect(result.transcripts).toHaveLength(2);
|
|
74
|
+
expect(result.transcripts[0].projectPath).toBe('/home/testuser/workspace/foo');
|
|
75
|
+
expect(result.transcripts[1].projectPath).toBe('/home/testuser/workspace/bar');
|
|
76
|
+
});
|
|
77
|
+
it('filters by date', async () => {
|
|
78
|
+
const mockEntries = [
|
|
79
|
+
{ name: '-home-testuser-project', isDirectory: () => true },
|
|
80
|
+
];
|
|
81
|
+
const mockProjectFiles = [
|
|
82
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
83
|
+
'b2c3d4e5-f6a7-8901-bcde-f12345678901.jsonl',
|
|
84
|
+
'c3d4e5f6-a7b8-9012-cdef-123456789012.jsonl',
|
|
85
|
+
];
|
|
86
|
+
vi.mocked(fs.readdir)
|
|
87
|
+
.mockResolvedValueOnce(mockEntries)
|
|
88
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
89
|
+
let callIndex = 0;
|
|
90
|
+
vi.mocked(fs.stat).mockImplementation(async () => {
|
|
91
|
+
const dates = [
|
|
92
|
+
new Date('2026-01-20T00:00:00.000Z'), // Old
|
|
93
|
+
new Date('2026-01-23T00:00:00.000Z'), // Recent
|
|
94
|
+
new Date('2026-01-24T00:00:00.000Z'), // Recent
|
|
95
|
+
];
|
|
96
|
+
const stats = {
|
|
97
|
+
size: 256,
|
|
98
|
+
mtime: dates[callIndex++],
|
|
99
|
+
};
|
|
100
|
+
return stats;
|
|
101
|
+
});
|
|
102
|
+
const result = await scanTranscripts({
|
|
103
|
+
minDate: new Date('2026-01-22T00:00:00.000Z'),
|
|
104
|
+
});
|
|
105
|
+
expect(result.transcripts).toHaveLength(2);
|
|
106
|
+
expect(result.transcripts[0].sessionId).toBe('b2c3d4e5-f6a7-8901-bcde-f12345678901');
|
|
107
|
+
expect(result.transcripts[1].sessionId).toBe('c3d4e5f6-a7b8-9012-cdef-123456789012');
|
|
108
|
+
});
|
|
109
|
+
it('excludes non-UUID filenames', async () => {
|
|
110
|
+
const mockEntries = [
|
|
111
|
+
{ name: '-home-testuser-project', isDirectory: () => true },
|
|
112
|
+
];
|
|
113
|
+
const mockProjectFiles = [
|
|
114
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl', // Valid UUID
|
|
115
|
+
'invalid-session-id.jsonl', // Invalid
|
|
116
|
+
'not-a-uuid.jsonl', // Invalid
|
|
117
|
+
'sessions-index.json', // Excluded anyway
|
|
118
|
+
'readme.txt', // Not .jsonl
|
|
119
|
+
];
|
|
120
|
+
vi.mocked(fs.readdir)
|
|
121
|
+
.mockResolvedValueOnce(mockEntries)
|
|
122
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
123
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
124
|
+
size: 128,
|
|
125
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
126
|
+
});
|
|
127
|
+
const result = await scanTranscripts();
|
|
128
|
+
expect(result.transcripts).toHaveLength(1);
|
|
129
|
+
expect(result.transcripts[0].sessionId).toBe('a1b2c3d4-e5f6-7890-abcd-ef1234567890');
|
|
130
|
+
});
|
|
131
|
+
it('handles missing directories gracefully', async () => {
|
|
132
|
+
const error = new Error('ENOENT');
|
|
133
|
+
error.code = 'ENOENT';
|
|
134
|
+
vi.mocked(fs.readdir).mockRejectedValue(error);
|
|
135
|
+
const result = await scanTranscripts();
|
|
136
|
+
expect(result).toEqual({
|
|
137
|
+
transcripts: [],
|
|
138
|
+
totalSize: 0,
|
|
139
|
+
projectCount: 0,
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
it('throws on other file system errors', async () => {
|
|
143
|
+
const error = new Error('EACCES');
|
|
144
|
+
error.code = 'EACCES';
|
|
145
|
+
vi.mocked(fs.readdir).mockRejectedValue(error);
|
|
146
|
+
await expect(scanTranscripts()).rejects.toThrow('EACCES');
|
|
147
|
+
});
|
|
148
|
+
it('skips non-directory entries', async () => {
|
|
149
|
+
const mockEntries = [
|
|
150
|
+
{ name: '-home-testuser-project', isDirectory: () => true },
|
|
151
|
+
{ name: 'some-file.txt', isDirectory: () => false },
|
|
152
|
+
];
|
|
153
|
+
const mockProjectFiles = [
|
|
154
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
155
|
+
];
|
|
156
|
+
vi.mocked(fs.readdir)
|
|
157
|
+
.mockResolvedValueOnce(mockEntries)
|
|
158
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
159
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
160
|
+
size: 64,
|
|
161
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
162
|
+
});
|
|
163
|
+
const result = await scanTranscripts();
|
|
164
|
+
expect(result.transcripts).toHaveLength(1);
|
|
165
|
+
expect(result.projectCount).toBe(1);
|
|
166
|
+
});
|
|
167
|
+
it('calculates total size correctly', async () => {
|
|
168
|
+
const mockEntries = [
|
|
169
|
+
{ name: '-home-testuser-project', isDirectory: () => true },
|
|
170
|
+
];
|
|
171
|
+
const mockProjectFiles = [
|
|
172
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
173
|
+
'b2c3d4e5-f6a7-8901-bcde-f12345678901.jsonl',
|
|
174
|
+
];
|
|
175
|
+
vi.mocked(fs.readdir)
|
|
176
|
+
.mockResolvedValueOnce(mockEntries)
|
|
177
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
178
|
+
let callIndex = 0;
|
|
179
|
+
vi.mocked(fs.stat).mockImplementation(async () => {
|
|
180
|
+
const sizes = [2048, 4096];
|
|
181
|
+
const stats = {
|
|
182
|
+
size: sizes[callIndex++],
|
|
183
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
184
|
+
};
|
|
185
|
+
return stats;
|
|
186
|
+
});
|
|
187
|
+
const result = await scanTranscripts();
|
|
188
|
+
expect(result.totalSize).toBe(6144);
|
|
189
|
+
});
|
|
190
|
+
it('handles empty project directories', async () => {
|
|
191
|
+
const mockEntries = [
|
|
192
|
+
{ name: '-home-testuser-empty-project', isDirectory: () => true },
|
|
193
|
+
];
|
|
194
|
+
vi.mocked(fs.readdir)
|
|
195
|
+
.mockResolvedValueOnce(mockEntries)
|
|
196
|
+
.mockResolvedValueOnce([]); // Empty directory
|
|
197
|
+
const result = await scanTranscripts();
|
|
198
|
+
expect(result.transcripts).toHaveLength(0);
|
|
199
|
+
expect(result.projectCount).toBe(0);
|
|
200
|
+
});
|
|
201
|
+
it('combines project and date filters', async () => {
|
|
202
|
+
const mockEntries = [
|
|
203
|
+
{ name: '-home-testuser-workspace-foo', isDirectory: () => true },
|
|
204
|
+
{ name: '-home-testuser-other-bar', isDirectory: () => true },
|
|
205
|
+
];
|
|
206
|
+
const mockProjectFiles = [
|
|
207
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
208
|
+
];
|
|
209
|
+
vi.mocked(fs.readdir)
|
|
210
|
+
.mockResolvedValueOnce(mockEntries)
|
|
211
|
+
.mockResolvedValue(mockProjectFiles);
|
|
212
|
+
let callIndex = 0;
|
|
213
|
+
vi.mocked(fs.stat).mockImplementation(async () => {
|
|
214
|
+
const dates = [
|
|
215
|
+
new Date('2026-01-24T00:00:00.000Z'), // workspace-foo: recent
|
|
216
|
+
new Date('2026-01-20T00:00:00.000Z'), // other-bar: old (but filtered by project anyway)
|
|
217
|
+
];
|
|
218
|
+
const stats = {
|
|
219
|
+
size: 512,
|
|
220
|
+
mtime: dates[callIndex++],
|
|
221
|
+
};
|
|
222
|
+
return stats;
|
|
223
|
+
});
|
|
224
|
+
const result = await scanTranscripts({
|
|
225
|
+
projectFilter: '/home/testuser/workspace/*',
|
|
226
|
+
minDate: new Date('2026-01-23T00:00:00.000Z'),
|
|
227
|
+
});
|
|
228
|
+
expect(result.transcripts).toHaveLength(1);
|
|
229
|
+
expect(result.transcripts[0].projectPath).toBe('/home/testuser/workspace/foo');
|
|
230
|
+
});
|
|
231
|
+
});
|
|
232
|
+
describe('decodeProjectPath()', () => {
|
|
233
|
+
it('decodes standard encoded paths', () => {
|
|
234
|
+
// We need to test this indirectly through scanTranscripts
|
|
235
|
+
const mockEntries = [
|
|
236
|
+
{ name: '-home-user-workspace-project', isDirectory: () => true },
|
|
237
|
+
];
|
|
238
|
+
const mockProjectFiles = [
|
|
239
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
240
|
+
];
|
|
241
|
+
vi.mocked(fs.readdir)
|
|
242
|
+
.mockResolvedValueOnce(mockEntries)
|
|
243
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
244
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
245
|
+
size: 128,
|
|
246
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
247
|
+
});
|
|
248
|
+
return scanTranscripts().then(result => {
|
|
249
|
+
expect(result.transcripts[0].projectPath).toBe('/home/user/workspace/project');
|
|
250
|
+
});
|
|
251
|
+
});
|
|
252
|
+
it('handles paths without leading dash', () => {
|
|
253
|
+
const mockEntries = [
|
|
254
|
+
{ name: 'relative-path-project', isDirectory: () => true },
|
|
255
|
+
];
|
|
256
|
+
const mockProjectFiles = [
|
|
257
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
258
|
+
];
|
|
259
|
+
vi.mocked(fs.readdir)
|
|
260
|
+
.mockResolvedValueOnce(mockEntries)
|
|
261
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
262
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
263
|
+
size: 128,
|
|
264
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
265
|
+
});
|
|
266
|
+
return scanTranscripts().then(result => {
|
|
267
|
+
// Should return unchanged if no leading dash
|
|
268
|
+
expect(result.transcripts[0].projectPath).toBe('relative-path-project');
|
|
269
|
+
});
|
|
270
|
+
});
|
|
271
|
+
it('handles root path', () => {
|
|
272
|
+
const mockEntries = [
|
|
273
|
+
{ name: '-root', isDirectory: () => true },
|
|
274
|
+
];
|
|
275
|
+
const mockProjectFiles = [
|
|
276
|
+
'a1b2c3d4-e5f6-7890-abcd-ef1234567890.jsonl',
|
|
277
|
+
];
|
|
278
|
+
vi.mocked(fs.readdir)
|
|
279
|
+
.mockResolvedValueOnce(mockEntries)
|
|
280
|
+
.mockResolvedValueOnce(mockProjectFiles);
|
|
281
|
+
vi.mocked(fs.stat).mockResolvedValue({
|
|
282
|
+
size: 128,
|
|
283
|
+
mtime: new Date('2026-01-24T00:00:00.000Z'),
|
|
284
|
+
});
|
|
285
|
+
return scanTranscripts().then(result => {
|
|
286
|
+
expect(result.transcripts[0].projectPath).toBe('/root');
|
|
287
|
+
});
|
|
288
|
+
});
|
|
289
|
+
});
|
|
290
|
+
});
|
|
291
|
+
/**
|
|
292
|
+
* Tests for decodeProjectPath with actual filesystem checks.
|
|
293
|
+
* These tests verify the smart path resolution works correctly with real directories.
|
|
294
|
+
*/
|
|
295
|
+
describe('decodeProjectPath (filesystem-aware)', () => {
|
|
296
|
+
let testDir;
|
|
297
|
+
beforeEach(() => {
|
|
298
|
+
// Restore tmpdir for this test suite
|
|
299
|
+
vi.mocked(tmpdir).mockReturnValue(require('os').tmpdir());
|
|
300
|
+
// Create a temporary test directory
|
|
301
|
+
testDir = join(tmpdir(), `test-decode-path-${Date.now()}`);
|
|
302
|
+
mkdirSync(testDir, { recursive: true });
|
|
303
|
+
});
|
|
304
|
+
afterEach(() => {
|
|
305
|
+
// Clean up test directory
|
|
306
|
+
if (existsSync(testDir)) {
|
|
307
|
+
rmSync(testDir, { recursive: true, force: true });
|
|
308
|
+
}
|
|
309
|
+
});
|
|
310
|
+
it('should return non-encoded paths as-is', () => {
|
|
311
|
+
const result = decodeProjectPath('my-project');
|
|
312
|
+
expect(result).toBe('my-project');
|
|
313
|
+
});
|
|
314
|
+
it('should decode simple paths without hyphens when path exists', () => {
|
|
315
|
+
// Create: /tmp/test-xxx/home/user/project
|
|
316
|
+
const projectPath = join(testDir, 'home', 'user', 'project');
|
|
317
|
+
mkdirSync(projectPath, { recursive: true });
|
|
318
|
+
const encoded = `-${testDir.slice(1)}-home-user-project`;
|
|
319
|
+
const result = decodeProjectPath(encoded);
|
|
320
|
+
expect(result).toBe(`${testDir}/home/user/project`);
|
|
321
|
+
});
|
|
322
|
+
it('should decode paths with legitimate hyphens in directory names', () => {
|
|
323
|
+
// Create: /tmp/test-xxx/home/user/my-project
|
|
324
|
+
const projectPath = join(testDir, 'home', 'user', 'my-project');
|
|
325
|
+
mkdirSync(projectPath, { recursive: true });
|
|
326
|
+
const encoded = `-${testDir.slice(1)}-home-user-my-project`;
|
|
327
|
+
const result = decodeProjectPath(encoded);
|
|
328
|
+
// Should preserve "my-project" as one directory
|
|
329
|
+
expect(result).toBe(`${testDir}/home/user/my-project`);
|
|
330
|
+
});
|
|
331
|
+
it('should handle multiple hyphens in a single directory name', () => {
|
|
332
|
+
// Create: /tmp/test-xxx/home/user/my-cool-project
|
|
333
|
+
const projectPath = join(testDir, 'home', 'user', 'my-cool-project');
|
|
334
|
+
mkdirSync(projectPath, { recursive: true });
|
|
335
|
+
const encoded = `-${testDir.slice(1)}-home-user-my-cool-project`;
|
|
336
|
+
const result = decodeProjectPath(encoded);
|
|
337
|
+
expect(result).toBe(`${testDir}/home/user/my-cool-project`);
|
|
338
|
+
});
|
|
339
|
+
it('should handle hyphens at multiple levels', () => {
|
|
340
|
+
// Create: /tmp/test-xxx/my-workspace/my-project
|
|
341
|
+
const projectPath = join(testDir, 'my-workspace', 'my-project');
|
|
342
|
+
mkdirSync(projectPath, { recursive: true });
|
|
343
|
+
const encoded = `-${testDir.slice(1)}-my-workspace-my-project`;
|
|
344
|
+
const result = decodeProjectPath(encoded);
|
|
345
|
+
expect(result).toBe(`${testDir}/my-workspace/my-project`);
|
|
346
|
+
});
|
|
347
|
+
it('should fall back to simple decode if no matching filesystem path exists', () => {
|
|
348
|
+
// Don't create any directories - test fallback behavior
|
|
349
|
+
const encoded = '-home-user-nonexistent-project';
|
|
350
|
+
const result = decodeProjectPath(encoded);
|
|
351
|
+
// Should fall back to simple decode (all dashes -> slashes)
|
|
352
|
+
expect(result).toBe('/home/user/nonexistent/project');
|
|
353
|
+
});
|
|
354
|
+
it('should handle root-level project directories', () => {
|
|
355
|
+
// Create: /tmp/test-xxx/my-project
|
|
356
|
+
const projectPath = join(testDir, 'my-project');
|
|
357
|
+
mkdirSync(projectPath, { recursive: true });
|
|
358
|
+
const encoded = `-${testDir.slice(1)}-my-project`;
|
|
359
|
+
const result = decodeProjectPath(encoded);
|
|
360
|
+
expect(result).toBe(`${testDir}/my-project`);
|
|
361
|
+
});
|
|
362
|
+
it('should prefer filesystem-verified paths over simple decode', () => {
|
|
363
|
+
// Create: /tmp/test-xxx/a/b-c (the correct interpretation)
|
|
364
|
+
// Don't create /tmp/test-xxx/a/b/c
|
|
365
|
+
const correctPath = join(testDir, 'a', 'b-c');
|
|
366
|
+
mkdirSync(correctPath, { recursive: true });
|
|
367
|
+
const encoded = `-${testDir.slice(1)}-a-b-c`;
|
|
368
|
+
const result = decodeProjectPath(encoded);
|
|
369
|
+
// Should choose /a/b-c over /a/b/c
|
|
370
|
+
expect(result).toBe(`${testDir}/a/b-c`);
|
|
371
|
+
});
|
|
372
|
+
it('should handle deeply nested paths with hyphens', () => {
|
|
373
|
+
// Create: /tmp/test-xxx/home/user/workspace/my-project/sub-folder
|
|
374
|
+
const projectPath = join(testDir, 'home', 'user', 'workspace', 'my-project', 'sub-folder');
|
|
375
|
+
mkdirSync(projectPath, { recursive: true });
|
|
376
|
+
const encoded = `-${testDir.slice(1)}-home-user-workspace-my-project-sub-folder`;
|
|
377
|
+
const result = decodeProjectPath(encoded);
|
|
378
|
+
expect(result).toBe(`${testDir}/home/user/workspace/my-project/sub-folder`);
|
|
379
|
+
});
|
|
380
|
+
it('should handle paths with consecutive hyphens', () => {
|
|
381
|
+
// Create: /tmp/test-xxx/my--project (unusual but valid)
|
|
382
|
+
const projectPath = join(testDir, 'my--project');
|
|
383
|
+
mkdirSync(projectPath, { recursive: true });
|
|
384
|
+
const encoded = `-${testDir.slice(1)}-my--project`;
|
|
385
|
+
const result = decodeProjectPath(encoded);
|
|
386
|
+
expect(result).toBe(`${testDir}/my--project`);
|
|
387
|
+
});
|
|
388
|
+
it('should find first matching path when multiple interpretations exist', () => {
|
|
389
|
+
// Create both possible interpretations
|
|
390
|
+
const path1 = join(testDir, 'a-b', 'c');
|
|
391
|
+
const path2 = join(testDir, 'a', 'b-c');
|
|
392
|
+
mkdirSync(path1, { recursive: true });
|
|
393
|
+
mkdirSync(path2, { recursive: true });
|
|
394
|
+
const encoded = `-${testDir.slice(1)}-a-b-c`;
|
|
395
|
+
const result = decodeProjectPath(encoded);
|
|
396
|
+
// Should match one of the valid paths
|
|
397
|
+
const isValid = result === `${testDir}/a-b/c` || result === `${testDir}/a/b-c`;
|
|
398
|
+
expect(isValid).toBe(true);
|
|
399
|
+
});
|
|
400
|
+
});
|
|
401
|
+
//# sourceMappingURL=transcript-scanner.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript-scanner.test.js","sourceRoot":"","sources":["../../../src/__tests__/analytics/transcript-scanner.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript-token-extractor.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/analytics/transcript-token-extractor.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { extractTokenUsage } from '../../analytics/transcript-token-extractor.js';
|
|
3
|
+
describe('extractTokenUsage', () => {
|
|
4
|
+
it('should extract token usage from assistant entry', () => {
|
|
5
|
+
const entry = {
|
|
6
|
+
type: 'assistant',
|
|
7
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
8
|
+
sessionId: 'test-session-123',
|
|
9
|
+
message: {
|
|
10
|
+
model: 'claude-sonnet-4-5-20250929',
|
|
11
|
+
usage: {
|
|
12
|
+
input_tokens: 1000,
|
|
13
|
+
output_tokens: 500,
|
|
14
|
+
cache_creation_input_tokens: 200,
|
|
15
|
+
cache_read_input_tokens: 300
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
20
|
+
expect(result).not.toBeNull();
|
|
21
|
+
expect(result?.usage.modelName).toBe('claude-sonnet-4.5');
|
|
22
|
+
expect(result?.usage.inputTokens).toBe(1000);
|
|
23
|
+
expect(result?.usage.outputTokens).toBe(500);
|
|
24
|
+
expect(result?.usage.cacheCreationTokens).toBe(200);
|
|
25
|
+
expect(result?.usage.cacheReadTokens).toBe(300);
|
|
26
|
+
expect(result?.usage.sessionId).toBe('test-session-123');
|
|
27
|
+
expect(result?.sourceFile).toBe('test.jsonl');
|
|
28
|
+
expect(result?.entryId).toBeDefined();
|
|
29
|
+
expect(result?.entryId.length).toBe(64); // SHA256 hex length
|
|
30
|
+
});
|
|
31
|
+
it('should return null for non-assistant entries', () => {
|
|
32
|
+
const entry = {
|
|
33
|
+
type: 'user',
|
|
34
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
35
|
+
sessionId: 'test-session-123'
|
|
36
|
+
};
|
|
37
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
38
|
+
expect(result).toBeNull();
|
|
39
|
+
});
|
|
40
|
+
it('should return null for assistant entries without usage', () => {
|
|
41
|
+
const entry = {
|
|
42
|
+
type: 'assistant',
|
|
43
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
44
|
+
sessionId: 'test-session-123',
|
|
45
|
+
message: {
|
|
46
|
+
model: 'claude-sonnet-4-5-20250929'
|
|
47
|
+
}
|
|
48
|
+
};
|
|
49
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
50
|
+
expect(result).toBeNull();
|
|
51
|
+
});
|
|
52
|
+
it('should detect agent name from agentId and slug', () => {
|
|
53
|
+
const entry = {
|
|
54
|
+
type: 'assistant',
|
|
55
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
56
|
+
sessionId: 'test-session-123',
|
|
57
|
+
agentId: 'a61283e',
|
|
58
|
+
slug: 'smooth-swinging-avalanche',
|
|
59
|
+
message: {
|
|
60
|
+
model: 'claude-haiku-4-5-20251001',
|
|
61
|
+
usage: {
|
|
62
|
+
input_tokens: 100,
|
|
63
|
+
output_tokens: 50
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
68
|
+
expect(result).not.toBeNull();
|
|
69
|
+
expect(result?.usage.agentName).toBe('smooth-swinging-avalanche');
|
|
70
|
+
});
|
|
71
|
+
it('should normalize model names correctly', () => {
|
|
72
|
+
const testCases = [
|
|
73
|
+
{ model: 'claude-opus-4-5-20251101', expected: 'claude-opus-4.5' },
|
|
74
|
+
{ model: 'claude-sonnet-4-5-20250929', expected: 'claude-sonnet-4.5' },
|
|
75
|
+
{ model: 'claude-haiku-4-5-20251001', expected: 'claude-haiku-4' }
|
|
76
|
+
];
|
|
77
|
+
testCases.forEach(({ model, expected }) => {
|
|
78
|
+
const entry = {
|
|
79
|
+
type: 'assistant',
|
|
80
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
81
|
+
sessionId: 'test-session-123',
|
|
82
|
+
message: {
|
|
83
|
+
model,
|
|
84
|
+
usage: {
|
|
85
|
+
input_tokens: 100,
|
|
86
|
+
output_tokens: 50
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
};
|
|
90
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
91
|
+
expect(result?.usage.modelName).toBe(expected);
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
it('should detect agent from Task tool usage', () => {
|
|
95
|
+
const entry = {
|
|
96
|
+
type: 'assistant',
|
|
97
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
98
|
+
sessionId: 'test-session-123',
|
|
99
|
+
message: {
|
|
100
|
+
model: 'claude-sonnet-4-5-20250929',
|
|
101
|
+
usage: {
|
|
102
|
+
input_tokens: 100,
|
|
103
|
+
output_tokens: 50
|
|
104
|
+
},
|
|
105
|
+
content: [
|
|
106
|
+
{
|
|
107
|
+
type: 'tool_use',
|
|
108
|
+
name: 'Task',
|
|
109
|
+
input: {
|
|
110
|
+
subagent_type: 'oh-my-claudecode:executor',
|
|
111
|
+
model: 'sonnet'
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
]
|
|
115
|
+
}
|
|
116
|
+
};
|
|
117
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
118
|
+
expect(result).not.toBeNull();
|
|
119
|
+
expect(result?.usage.agentName).toBe('oh-my-claudecode:executor');
|
|
120
|
+
});
|
|
121
|
+
it('should use ACTUAL output_tokens from transcript', () => {
|
|
122
|
+
const entry = {
|
|
123
|
+
type: 'assistant',
|
|
124
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
125
|
+
sessionId: 'test-session-123',
|
|
126
|
+
message: {
|
|
127
|
+
model: 'claude-sonnet-4-5-20250929',
|
|
128
|
+
usage: {
|
|
129
|
+
input_tokens: 1000,
|
|
130
|
+
output_tokens: 1234 // ACTUAL value, not estimate
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
};
|
|
134
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
135
|
+
expect(result).not.toBeNull();
|
|
136
|
+
expect(result?.usage.outputTokens).toBe(1234);
|
|
137
|
+
});
|
|
138
|
+
it('should generate consistent entry IDs for same data', () => {
|
|
139
|
+
const entry = {
|
|
140
|
+
type: 'assistant',
|
|
141
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
142
|
+
sessionId: 'test-session-123',
|
|
143
|
+
message: {
|
|
144
|
+
model: 'claude-sonnet-4-5-20250929',
|
|
145
|
+
usage: {
|
|
146
|
+
input_tokens: 100,
|
|
147
|
+
output_tokens: 50
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
};
|
|
151
|
+
const result1 = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
152
|
+
const result2 = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
153
|
+
expect(result1?.entryId).toBe(result2?.entryId);
|
|
154
|
+
});
|
|
155
|
+
it('should handle missing cache tokens gracefully', () => {
|
|
156
|
+
const entry = {
|
|
157
|
+
type: 'assistant',
|
|
158
|
+
timestamp: '2026-01-24T05:07:46.325Z',
|
|
159
|
+
sessionId: 'test-session-123',
|
|
160
|
+
message: {
|
|
161
|
+
model: 'claude-sonnet-4-5-20250929',
|
|
162
|
+
usage: {
|
|
163
|
+
input_tokens: 100,
|
|
164
|
+
output_tokens: 50
|
|
165
|
+
// No cache tokens
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
};
|
|
169
|
+
const result = extractTokenUsage(entry, 'test-session-123', 'test.jsonl');
|
|
170
|
+
expect(result).not.toBeNull();
|
|
171
|
+
expect(result?.usage.cacheCreationTokens).toBe(0);
|
|
172
|
+
expect(result?.usage.cacheReadTokens).toBe(0);
|
|
173
|
+
});
|
|
174
|
+
});
|
|
175
|
+
//# sourceMappingURL=transcript-token-extractor.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transcript-token-extractor.test.js","sourceRoot":"","sources":["../../../src/__tests__/analytics/transcript-token-extractor.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auto-tracking.integration.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/hud/auto-tracking.integration.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
describe('Auto-Tracking Integration', () => {
|
|
3
|
+
it('should record tokens on HUD render', async () => {
|
|
4
|
+
// TODO: Test end-to-end auto-recording
|
|
5
|
+
expect(true).toBe(true);
|
|
6
|
+
});
|
|
7
|
+
it('should associate tokens with running agents', async () => {
|
|
8
|
+
// TODO: Test agent correlation
|
|
9
|
+
expect(true).toBe(true);
|
|
10
|
+
});
|
|
11
|
+
});
|
|
12
|
+
//# sourceMappingURL=auto-tracking.integration.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auto-tracking.integration.test.js","sourceRoot":"","sources":["../../../src/__tests__/hud/auto-tracking.integration.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAE9C,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;IACzC,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,uCAAuC;QACvC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,KAAK,IAAI,EAAE;QAC3D,+BAA+B;QAC/B,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|