maestro/src/bridge/dashboard-workers.test.ts
2026-06-03 05:08:00 +00:00

174 lines
7.4 KiB
TypeScript

import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { mkdtempSync, rmSync } from 'fs';
import { join } from 'path';
import { tmpdir } from 'os';
import { Repository } from '../db/repository.js';
import { collectWorkerStatuses } from './dashboard-workers.js';
import type { WorkerDef } from '../config.js';
describe('collectWorkerStatuses', () => {
let tmpDir: string;
let repo: Repository;
beforeEach(() => {
tmpDir = mkdtempSync(join(tmpdir(), 'dashboard-workers-test-'));
repo = new Repository(join(tmpDir, 'test.db'));
});
afterEach(() => {
rmSync(tmpDir, { recursive: true, force: true });
});
it('returns idle for all workers when no running jobs exist', async () => {
const workers: WorkerDef[] = [
{ id: 'w1', endpoint: 'x', roles: ['task'] },
{ id: 'w2', endpoint: 'y', roles: ['title'] },
];
const result = await collectWorkerStatuses(repo, workers);
expect(result).toEqual([
{ id: 'w1', name: 'w1', roles: ['task'], state: 'idle', proxy: false },
{ id: 'w2', name: 'w2', roles: ['title'], state: 'idle', proxy: false },
]);
});
it('returns running for workers with active jobs', async () => {
// Seed a running job for w1 via Repository's public API
const j = await repo.createJob({ repo: 'local/task-1', issueNumber: 1, instruction: 'seed' });
await repo.updateJob(j.id, { status: 'running', workerId: 'w1' });
const workers: WorkerDef[] = [
{ id: 'w1', endpoint: 'x', roles: ['task'] },
{ id: 'w2', endpoint: 'y', roles: ['task'] },
];
const result = await collectWorkerStatuses(repo, workers);
expect(result.find(w => w.id === 'w1')!.state).toBe('running');
expect(result.find(w => w.id === 'w2')!.state).toBe('idle');
});
it('does not leak job id/title/owner in the response shape', async () => {
const j = await repo.createJob({ repo: 'local/task-1', issueNumber: 1, instruction: 'seed' });
await repo.updateJob(j.id, { status: 'running', workerId: 'w1' });
const result = await collectWorkerStatuses(repo, [{ id: 'w1', endpoint: 'x' }]);
const keys = Object.keys(result[0]!).sort();
// proxy is a new keyed field; backends is optional and absent for
// direct workers — that's part of the public shape contract.
expect(keys).toEqual(['id', 'name', 'proxy', 'roles', 'state']);
});
it('fans out proxy workers into backends[] when a registry is supplied', async () => {
const fakeRegistry = {
getAll: () => [
// Self-row that the proxy probe surfaces with nodeId === workerId —
// must be filtered out so the proxy doesn't appear as its own child.
{
nodeId: 'gw', workerId: 'gw', source: 'proxy' as const,
online: true, busy: false, busySlots: 0, totalSlots: 0,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
},
{
nodeId: 'backend-a', workerId: 'gw', source: 'proxy' as const,
online: true, busy: true, busySlots: 2, totalSlots: 4,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
},
{
nodeId: 'backend-b', workerId: 'gw', source: 'proxy' as const,
online: true, busy: false, busySlots: 0, totalSlots: 4,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
},
// Belongs to a different worker — must not leak into gw.backends
{
nodeId: 'other-backend', workerId: 'other', source: 'proxy' as const,
online: true, busy: false, busySlots: 0, totalSlots: 4,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
},
],
};
const workers: WorkerDef[] = [
{ id: 'gw', endpoint: 'http://gw/v1', proxy: true },
{ id: 'direct', endpoint: 'http://direct/v1' },
];
const result = await collectWorkerStatuses(repo, workers, fakeRegistry);
const gw = result.find(w => w.id === 'gw')!;
expect(gw.proxy).toBe(true);
expect(gw.backends?.map(b => b.id).sort()).toEqual(['backend-a', 'backend-b']);
expect(gw.backends?.find(b => b.id === 'backend-a')).toMatchObject({
state: 'running', busySlots: 2, totalSlots: 4, online: true,
});
expect(gw.backends?.find(b => b.id === 'backend-b')).toMatchObject({
state: 'idle', busySlots: 0,
});
const direct = result.find(w => w.id === 'direct')!;
expect(direct.proxy).toBe(false);
// direct workers MUST omit `backends` (undefined, not empty) so the
// UI can distinguish "direct" from "proxy with zero backends".
expect(direct.backends).toBeUndefined();
});
it('omits backends[] for proxy workers when no registry is supplied (back-compat)', async () => {
const result = await collectWorkerStatuses(repo, [
{ id: 'gw', endpoint: 'http://gw/v1', proxy: true },
], null);
expect(result[0]!.proxy).toBe(true);
expect(result[0]!.backends).toBeUndefined();
});
it('surfaces busy/total slots on the row for direct workers via the registry self-row', async () => {
// Direct workers don't have a backends[] expansion, so the slot
// pressure has to live at the row level — otherwise the Worker
// widget can't show "(1/3)" for them the way proxy backends do.
const fakeRegistry = {
getAll: () => [
{
nodeId: 'gpu-1', workerId: 'gpu-1', source: 'direct' as const,
online: true, busy: true, busySlots: 1, totalSlots: 3,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
},
],
};
const workers: WorkerDef[] = [
{ id: 'gpu-1', endpoint: 'http://gpu-1:8080/v1' },
];
const result = await collectWorkerStatuses(repo, workers, fakeRegistry);
const row = result[0]!;
expect(row.proxy).toBe(false);
expect(row.backends).toBeUndefined();
expect(row.busySlots).toBe(1);
expect(row.totalSlots).toBe(3);
expect(row.online).toBe(true);
// Probe-derived state wins over the local jobs-table check when
// the probe sees in-flight requests this AAO didn't dispatch.
expect(row.state).toBe('running');
});
it('keeps direct worker slot fields undefined when registry has no matching row', async () => {
const fakeRegistry = {
// Registry knows other workers, but not the one we're asking about.
getAll: () => [{
nodeId: 'other', workerId: 'other', source: 'direct' as const,
online: true, busy: false, busySlots: 0, totalSlots: 4,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
}],
};
const workers: WorkerDef[] = [
{ id: 'gpu-1', endpoint: 'http://gpu-1:8080/v1' },
];
const result = await collectWorkerStatuses(repo, workers, fakeRegistry);
const row = result[0]!;
expect(row.busySlots).toBeUndefined();
expect(row.totalSlots).toBeUndefined();
expect(row.online).toBeUndefined();
});
it('marks online=false for direct workers when probe failed', async () => {
const fakeRegistry = {
getAll: () => [{
nodeId: 'gpu-1', workerId: 'gpu-1', source: 'direct' as const,
online: false, busy: false, busySlots: 0, totalSlots: 0,
loadedModel: null, throughputTps: null, lastSeen: '2026-05-21T00:00:00Z',
lastProbeError: 'connection refused',
}],
};
const result = await collectWorkerStatuses(repo, [{ id: 'gpu-1', endpoint: 'x' }], fakeRegistry);
expect(result[0]!.online).toBe(false);
});
});