mirror of
https://github.com/AgentSeal/codeburn.git
synced 2026-05-17 03:56:45 +00:00
* Fix mangled project paths in By Project and Top Sessions panels
shortProject() decoded Claude Code slugs by splitting on '-', which
broke directory names containing dashes ('foo-bar' became 'foo/bar').
Switch the dashboard to consume ProjectSummary.projectPath (the
canonical cwd already extracted by parser.ts) and rewrite shortProject
to operate on a real absolute path.
* shortProject: cache homedir, normalize Windows backslashes, fix stale test helper
---------
Co-authored-by: Abdallah Meghraoui <abdallah.meghraoui@outlook.com>
146 lines
5.2 KiB
TypeScript
146 lines
5.2 KiB
TypeScript
import { homedir } from 'os'
|
|
|
|
import { describe, it, expect } from 'vitest'
|
|
|
|
import { shortProject } from '../src/dashboard.js'
|
|
import { formatCost } from '../src/format.js'
|
|
import type { ProjectSummary, SessionSummary } from '../src/types.js'
|
|
|
|
const EMPTY_CATEGORY_BREAKDOWN = {
|
|
coding: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
debugging: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
feature: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
refactoring: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
testing: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
exploration: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
planning: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
delegation: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
git: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
'build/deploy': { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
conversation: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
brainstorming: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
general: { turns: 0, costUSD: 0, retries: 0, editTurns: 0, oneShotTurns: 0 },
|
|
} satisfies SessionSummary['categoryBreakdown']
|
|
|
|
function makeSession(id: string, cost: number, timestamp = '2026-04-14T10:00:00Z'): SessionSummary {
|
|
return {
|
|
sessionId: id,
|
|
project: 'test-project',
|
|
firstTimestamp: timestamp,
|
|
lastTimestamp: timestamp,
|
|
totalCostUSD: cost,
|
|
totalInputTokens: 0,
|
|
totalOutputTokens: 0,
|
|
totalCacheReadTokens: 0,
|
|
totalCacheWriteTokens: 0,
|
|
apiCalls: 1,
|
|
turns: [],
|
|
modelBreakdown: {},
|
|
toolBreakdown: {},
|
|
mcpBreakdown: {},
|
|
bashBreakdown: {},
|
|
categoryBreakdown: { ...EMPTY_CATEGORY_BREAKDOWN },
|
|
skillBreakdown: {},
|
|
}
|
|
}
|
|
|
|
function makeProject(name: string, sessions: SessionSummary[]): ProjectSummary {
|
|
return {
|
|
project: name,
|
|
projectPath: name,
|
|
sessions,
|
|
totalCostUSD: sessions.reduce((s, x) => s + x.totalCostUSD, 0),
|
|
totalApiCalls: sessions.reduce((s, x) => s + x.apiCalls, 0),
|
|
}
|
|
}
|
|
|
|
// Logic replicated from TopSessions component
|
|
function getTopSessions(projects: ProjectSummary[], n = 5) {
|
|
const all = projects.flatMap(p => p.sessions.map(s => ({ ...s, projectPath: p.projectPath })))
|
|
return [...all].sort((a, b) => b.totalCostUSD - a.totalCostUSD).slice(0, n)
|
|
}
|
|
|
|
// Logic replicated from ProjectBreakdown component
|
|
function avgCostLabel(project: ProjectSummary): string {
|
|
return project.sessions.length > 0
|
|
? formatCost(project.totalCostUSD / project.sessions.length)
|
|
: '-'
|
|
}
|
|
|
|
describe('TopSessions - top-5 selection', () => {
|
|
it('returns all sessions when fewer than 5 exist', () => {
|
|
const project = makeProject('proj', [
|
|
makeSession('s1', 1.0),
|
|
makeSession('s2', 2.0),
|
|
])
|
|
const top = getTopSessions([project])
|
|
expect(top).toHaveLength(2)
|
|
expect(top[0].totalCostUSD).toBe(2.0)
|
|
expect(top[1].totalCostUSD).toBe(1.0)
|
|
})
|
|
|
|
it('returns exactly 5 when more than 5 sessions exist', () => {
|
|
const sessions = [0.1, 0.5, 3.0, 1.0, 0.8, 2.0].map((cost, i) =>
|
|
makeSession(`s${i}`, cost)
|
|
)
|
|
const project = makeProject('proj', sessions)
|
|
const top = getTopSessions([project])
|
|
expect(top).toHaveLength(5)
|
|
expect(top[0].totalCostUSD).toBe(3.0)
|
|
expect(top[4].totalCostUSD).toBe(0.5)
|
|
})
|
|
|
|
it('is stable on tied costs - preserves input order for equal values', () => {
|
|
const sessions = [
|
|
makeSession('s1', 1.0),
|
|
makeSession('s2', 1.0),
|
|
makeSession('s3', 1.0),
|
|
]
|
|
const project = makeProject('proj', sessions)
|
|
const top = getTopSessions([project])
|
|
expect(top.map(s => s.sessionId)).toEqual(['s1', 's2', 's3'])
|
|
})
|
|
})
|
|
|
|
describe('shortProject - path shortening', () => {
|
|
const home = homedir()
|
|
|
|
it('preserves directory names containing dashes', () => {
|
|
expect(shortProject(`${home}/work/my-project`)).toBe('work/my-project')
|
|
})
|
|
|
|
it('preserves directory names containing dots', () => {
|
|
expect(shortProject(`${home}/work/my.app.io`)).toBe('work/my.app.io')
|
|
})
|
|
|
|
it('returns "home" for the home dir itself', () => {
|
|
expect(shortProject(home)).toBe('home')
|
|
})
|
|
|
|
it('does not strip a sibling whose name shares the home prefix', () => {
|
|
const sibling = `${home}-backup/proj`
|
|
expect(shortProject(sibling).endsWith('proj')).toBe(true)
|
|
expect(shortProject(sibling)).not.toMatch(/^-/)
|
|
})
|
|
|
|
it('keeps only the last 3 segments for deeply nested paths', () => {
|
|
expect(shortProject(`${home}/a/b/c/d/e/f`)).toBe('d/e/f')
|
|
})
|
|
|
|
it('handles paths outside the home dir', () => {
|
|
expect(shortProject('/opt/myproject')).toBe('opt/myproject')
|
|
})
|
|
})
|
|
|
|
describe('avg/s in ProjectBreakdown', () => {
|
|
it('returns dash for a project with no sessions', () => {
|
|
const project = makeProject('proj', [])
|
|
expect(avgCostLabel(project)).toBe('-')
|
|
})
|
|
|
|
it('returns formatted average cost across sessions', () => {
|
|
const sessions = [makeSession('s1', 2.0), makeSession('s2', 4.0)]
|
|
const project = makeProject('proj', sessions)
|
|
expect(avgCostLabel(project)).toBe(formatCost(3.0))
|
|
})
|
|
})
|