104 lines
3.3 KiB
TypeScript
104 lines
3.3 KiB
TypeScript
import * as os from 'os';
|
|
import * as path from 'path';
|
|
import { afterEach, describe, expect, it } from 'vitest';
|
|
import * as fs from 'fs/promises';
|
|
|
|
import { TeamTaskStallJournal } from '../../../../../src/main/services/team/stallMonitor/TeamTaskStallJournal';
|
|
import { setClaudeBasePathOverride } from '../../../../../src/main/utils/pathDecoder';
|
|
|
|
describe('TeamTaskStallJournal', () => {
|
|
let tmpDir: string | null = null;
|
|
|
|
afterEach(async () => {
|
|
setClaudeBasePathOverride(null);
|
|
if (tmpDir) {
|
|
await fs.rm(tmpDir, { recursive: true, force: true });
|
|
tmpDir = null;
|
|
}
|
|
});
|
|
|
|
it('requires two scans before returning an alert-ready candidate', async () => {
|
|
tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'stall-journal-'));
|
|
setClaudeBasePathOverride(tmpDir);
|
|
await fs.mkdir(path.join(tmpDir, 'teams', 'demo'), { recursive: true });
|
|
|
|
const journal = new TeamTaskStallJournal();
|
|
const evaluation = {
|
|
status: 'alert',
|
|
taskId: 'task-a',
|
|
branch: 'work',
|
|
signal: 'turn_ended_after_touch',
|
|
epochKey: 'task-a:epoch-1',
|
|
reason: 'Potential work stall',
|
|
} as const;
|
|
|
|
const firstReady = await journal.reconcileScan({
|
|
teamName: 'demo',
|
|
evaluations: [evaluation],
|
|
activeTaskIds: ['task-a'],
|
|
now: '2026-04-19T12:10:00.000Z',
|
|
});
|
|
const secondReady = await journal.reconcileScan({
|
|
teamName: 'demo',
|
|
evaluations: [evaluation],
|
|
activeTaskIds: ['task-a'],
|
|
now: '2026-04-19T12:11:00.000Z',
|
|
});
|
|
|
|
expect(firstReady).toEqual([]);
|
|
expect(secondReady).toEqual([evaluation]);
|
|
});
|
|
|
|
it('does not prune journal entries outside an explicit task scope', async () => {
|
|
tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'stall-journal-'));
|
|
setClaudeBasePathOverride(tmpDir);
|
|
const teamDir = path.join(tmpDir, 'teams', 'demo');
|
|
await fs.mkdir(teamDir, { recursive: true });
|
|
const journalPath = path.join(teamDir, 'stall-monitor-journal.json');
|
|
await fs.writeFile(
|
|
journalPath,
|
|
JSON.stringify(
|
|
[
|
|
{
|
|
epochKey: 'task-codex:epoch-1',
|
|
teamName: 'demo',
|
|
taskId: 'task-codex',
|
|
branch: 'work',
|
|
signal: 'turn_ended_after_touch',
|
|
state: 'suspected',
|
|
consecutiveScans: 1,
|
|
createdAt: '2026-04-19T12:00:00.000Z',
|
|
updatedAt: '2026-04-19T12:00:00.000Z',
|
|
},
|
|
{
|
|
epochKey: 'task-opencode:epoch-1',
|
|
teamName: 'demo',
|
|
taskId: 'task-opencode',
|
|
branch: 'work',
|
|
signal: 'turn_ended_after_touch',
|
|
state: 'suspected',
|
|
consecutiveScans: 1,
|
|
createdAt: '2026-04-19T12:00:00.000Z',
|
|
updatedAt: '2026-04-19T12:00:00.000Z',
|
|
},
|
|
],
|
|
null,
|
|
2
|
|
)
|
|
);
|
|
|
|
const journal = new TeamTaskStallJournal();
|
|
await journal.reconcileScan({
|
|
teamName: 'demo',
|
|
evaluations: [],
|
|
activeTaskIds: ['task-codex', 'task-opencode'],
|
|
scopeTaskIds: ['task-opencode'],
|
|
now: '2026-04-19T12:10:00.000Z',
|
|
});
|
|
|
|
const saved = JSON.parse(await fs.readFile(journalPath, 'utf8')) as Array<{
|
|
epochKey: string;
|
|
}>;
|
|
expect(saved.map((entry) => entry.epochKey)).toEqual(['task-codex:epoch-1']);
|
|
});
|
|
});
|