416 lines
17 KiB
TypeScript
416 lines
17 KiB
TypeScript
import { describe, expect, it, vi } from 'vitest';
|
|
import {
|
|
AndroidCompanionClient,
|
|
IOSCompanionClient,
|
|
MacOSCompanionClient,
|
|
} from './platformClients.js';
|
|
import type { CompanionRuntimeClient } from './runtimeClient.js';
|
|
|
|
function createRuntimeMock(): {
|
|
runtime: CompanionRuntimeClient;
|
|
connect: ReturnType<typeof vi.fn>;
|
|
disconnect: ReturnType<typeof vi.fn>;
|
|
dispose: ReturnType<typeof vi.fn>;
|
|
registerNode: ReturnType<typeof vi.fn>;
|
|
bootstrapNode: ReturnType<typeof vi.fn>;
|
|
getNodeCapabilities: ReturnType<typeof vi.fn>;
|
|
setNodeStatus: ReturnType<typeof vi.fn>;
|
|
setNodeLocation: ReturnType<typeof vi.fn>;
|
|
getNodeLocation: ReturnType<typeof vi.fn>;
|
|
setNodePushToken: ReturnType<typeof vi.fn>;
|
|
getSystemCapabilities: ReturnType<typeof vi.fn>;
|
|
listSystemNodes: ReturnType<typeof vi.fn>;
|
|
putCanvasArtifact: ReturnType<typeof vi.fn>;
|
|
getCanvasArtifact: ReturnType<typeof vi.fn>;
|
|
listCanvasArtifacts: ReturnType<typeof vi.fn>;
|
|
deleteCanvasArtifact: ReturnType<typeof vi.fn>;
|
|
clearCanvasArtifacts: ReturnType<typeof vi.fn>;
|
|
subscribeAgentStream: ReturnType<typeof vi.fn>;
|
|
subscribeAgentTyping: ReturnType<typeof vi.fn>;
|
|
subscribeContextWarning: ReturnType<typeof vi.fn>;
|
|
subscribeEvents: ReturnType<typeof vi.fn>;
|
|
subscribeEvent: ReturnType<typeof vi.fn>;
|
|
clearEventSubscriptions: ReturnType<typeof vi.fn>;
|
|
listKnownEventNames: ReturnType<typeof vi.fn>;
|
|
waitForEvent: ReturnType<typeof vi.fn>;
|
|
waitForAgentStream: ReturnType<typeof vi.fn>;
|
|
waitForAgentTyping: ReturnType<typeof vi.fn>;
|
|
waitForContextWarning: ReturnType<typeof vi.fn>;
|
|
waitForAnyEvent: ReturnType<typeof vi.fn>;
|
|
eventSubscriptionCount: number;
|
|
} {
|
|
const connect = vi.fn(async () => undefined);
|
|
const disconnect = vi.fn(() => undefined);
|
|
const dispose = vi.fn(() => undefined);
|
|
const registerNode = vi.fn(async () => ({ registered: true }));
|
|
const bootstrapNode = vi.fn(async () => ({
|
|
register: { registered: true },
|
|
capabilities: {
|
|
node: { id: 'n1', role: 'companion', registeredAt: Date.now() },
|
|
protocol: { serverVersion: 1, nodeVersion: 1, negotiatedVersion: 1 },
|
|
capabilities: { declared: [], enabled: [], featureGates: {} },
|
|
},
|
|
}));
|
|
const getNodeCapabilities = vi.fn(async () => ({ node: { id: 'n1', role: 'companion', registeredAt: Date.now() }, protocol: { serverVersion: 1, nodeVersion: 1, negotiatedVersion: 1 }, capabilities: { declared: [], enabled: [], featureGates: {} } }));
|
|
const setNodeStatus = vi.fn(async () => ({ updated: true }));
|
|
const setNodeLocation = vi.fn(async () => ({ updated: true }));
|
|
const getNodeLocation = vi.fn(async () => ({ node: { id: 'n1', role: 'companion' }, location: null }));
|
|
const setNodePushToken = vi.fn(async () => ({ updated: true }));
|
|
const getSystemCapabilities = vi.fn(async () => ({ protocol: { version: 1 }, nodes: { enabled: true, locationEnabled: true, pushEnabled: true, allowedRoles: ['companion'], registered: true }, featureGates: {} }));
|
|
const listSystemNodes = vi.fn(async () => ({ nodes: [], summary: { total: 0 } }));
|
|
const putCanvasArtifact = vi.fn(async () => ({ upserted: true, artifact: { id: 'a1' } }));
|
|
const getCanvasArtifact = vi.fn(async () => ({ artifact: { id: 'a1' } }));
|
|
const listCanvasArtifacts = vi.fn(async () => ({ artifacts: [{ id: 'a1' }] }));
|
|
const deleteCanvasArtifact = vi.fn(async () => ({ deleted: true }));
|
|
const clearCanvasArtifacts = vi.fn(async () => ({ cleared: 1 }));
|
|
const subscribeAgentStream = vi.fn(() => () => undefined);
|
|
const subscribeAgentTyping = vi.fn(() => () => undefined);
|
|
const subscribeContextWarning = vi.fn(() => () => undefined);
|
|
const subscribeEvents = vi.fn(() => () => undefined);
|
|
const subscribeEvent = vi.fn(() => () => undefined);
|
|
const clearEventSubscriptions = vi.fn(() => undefined);
|
|
const listKnownEventNames = vi.fn(() => ['agent.stream', 'agent.typing', 'context_warning']);
|
|
const waitForEvent = vi.fn(async () => ({ token: 'evented' }));
|
|
const waitForAgentStream = vi.fn(async () => ({ token: 'streamed' }));
|
|
const waitForAgentTyping = vi.fn(async () => ({ active: true }));
|
|
const waitForContextWarning = vi.fn(async () => ({ thresholdPct: 75, estimatedPct: 90 }));
|
|
const waitForAnyEvent = vi.fn(async () => ({ event: 'agent.stream', data: { token: 'any' } }));
|
|
const eventSubscriptionCount = 3;
|
|
|
|
const runtime = {
|
|
connect,
|
|
disconnect,
|
|
dispose,
|
|
registerNode,
|
|
bootstrapNode,
|
|
getNodeCapabilities,
|
|
setNodeStatus,
|
|
setNodeLocation,
|
|
getNodeLocation,
|
|
setNodePushToken,
|
|
getSystemCapabilities,
|
|
listSystemNodes,
|
|
putCanvasArtifact,
|
|
getCanvasArtifact,
|
|
listCanvasArtifacts,
|
|
deleteCanvasArtifact,
|
|
clearCanvasArtifacts,
|
|
subscribeAgentStream,
|
|
subscribeAgentTyping,
|
|
subscribeContextWarning,
|
|
subscribeEvents,
|
|
subscribeEvent,
|
|
clearEventSubscriptions,
|
|
listKnownEventNames,
|
|
waitForEvent,
|
|
waitForAgentStream,
|
|
waitForAgentTyping,
|
|
waitForContextWarning,
|
|
waitForAnyEvent,
|
|
get eventSubscriptionCount() {
|
|
return eventSubscriptionCount;
|
|
},
|
|
} as unknown as CompanionRuntimeClient;
|
|
|
|
return {
|
|
runtime,
|
|
connect,
|
|
disconnect,
|
|
dispose,
|
|
registerNode,
|
|
bootstrapNode,
|
|
getNodeCapabilities,
|
|
setNodeStatus,
|
|
setNodeLocation,
|
|
getNodeLocation,
|
|
setNodePushToken,
|
|
getSystemCapabilities,
|
|
listSystemNodes,
|
|
putCanvasArtifact,
|
|
getCanvasArtifact,
|
|
listCanvasArtifacts,
|
|
deleteCanvasArtifact,
|
|
clearCanvasArtifacts,
|
|
subscribeAgentStream,
|
|
subscribeAgentTyping,
|
|
subscribeContextWarning,
|
|
subscribeEvents,
|
|
subscribeEvent,
|
|
clearEventSubscriptions,
|
|
listKnownEventNames,
|
|
waitForEvent,
|
|
waitForAgentStream,
|
|
waitForAgentTyping,
|
|
waitForContextWarning,
|
|
waitForAnyEvent,
|
|
eventSubscriptionCount,
|
|
};
|
|
}
|
|
|
|
describe('platform companion clients', () => {
|
|
it('macOS client uses macos platform status and APNs push', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new MacOSCompanionClient({ runtime: mock.runtime, nodeId: 'mac-node' });
|
|
|
|
await client.connect();
|
|
await client.register();
|
|
await client.setStatus({ appVersion: '1.0.0', powerSource: 'ac' });
|
|
await client.setLocation({ latitude: 10, longitude: 20, source: 'manual' });
|
|
await client.registerPushToken({ token: 'a'.repeat(64), topic: 'dev.flynn.macos', environment: 'production' });
|
|
await client.listNodes();
|
|
client.disconnect();
|
|
|
|
expect(mock.connect).toHaveBeenCalledOnce();
|
|
expect(mock.registerNode).toHaveBeenCalledWith(expect.objectContaining({ nodeId: 'mac-node', role: 'companion' }));
|
|
expect(mock.setNodeStatus).toHaveBeenCalledWith(expect.objectContaining({ platform: 'macos' }));
|
|
expect(mock.setNodePushToken).toHaveBeenCalledWith(expect.objectContaining({ provider: 'apns', topic: 'dev.flynn.macos' }));
|
|
expect(mock.listSystemNodes).toHaveBeenCalledWith({ platform: 'macos', role: 'companion' });
|
|
expect(mock.disconnect).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it('iOS client uses ios platform status and APNs push', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
await client.register();
|
|
await client.setStatus({ statusText: 'foreground', batteryPct: 52, powerSource: 'battery' });
|
|
await client.registerPushToken({ token: 'b'.repeat(64), topic: 'dev.flynn.ios', environment: 'sandbox' });
|
|
await client.listNodes();
|
|
|
|
expect(mock.setNodeStatus).toHaveBeenCalledWith(expect.objectContaining({ platform: 'ios' }));
|
|
expect(mock.setNodePushToken).toHaveBeenCalledWith(expect.objectContaining({ provider: 'apns', environment: 'sandbox' }));
|
|
expect(mock.listSystemNodes).toHaveBeenCalledWith({ platform: 'ios', role: 'companion' });
|
|
});
|
|
|
|
it('android client uses android platform status and FCM push', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new AndroidCompanionClient({ runtime: mock.runtime, nodeId: 'android-node' });
|
|
|
|
await client.register();
|
|
await client.setStatus({ appVersion: '2.0.0', powerSource: 'battery' });
|
|
await client.registerPushToken('c'.repeat(64));
|
|
await client.listNodes();
|
|
|
|
expect(mock.setNodeStatus).toHaveBeenCalledWith(expect.objectContaining({ platform: 'android' }));
|
|
expect(mock.setNodePushToken).toHaveBeenCalledWith({ provider: 'fcm', token: 'c'.repeat(64) });
|
|
expect(mock.listSystemNodes).toHaveBeenCalledWith({ platform: 'android', role: 'companion' });
|
|
});
|
|
|
|
it('platform dispose forwards to runtime dispose', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new MacOSCompanionClient({ runtime: mock.runtime, nodeId: 'mac-node' });
|
|
|
|
client.dispose();
|
|
expect(mock.dispose).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it('platform stream helper methods forward to runtime client', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new AndroidCompanionClient({ runtime: mock.runtime, nodeId: 'android-node' });
|
|
const streamHandler = vi.fn();
|
|
const typingHandler = vi.fn();
|
|
const warningHandler = vi.fn();
|
|
const eventHandler = vi.fn();
|
|
const streamEventHandler = vi.fn();
|
|
|
|
const unsubscribeStream = client.subscribeAgentStream(streamHandler);
|
|
const unsubscribeTyping = client.subscribeAgentTyping(typingHandler);
|
|
const unsubscribeWarning = client.subscribeContextWarning(warningHandler);
|
|
const unsubscribeEvents = client.subscribeEvents(eventHandler);
|
|
const unsubscribeEvent = client.subscribeEvent('agent.stream', streamEventHandler);
|
|
const waitedEvent = await client.waitForEvent<{ token: string }>('agent.stream', { timeoutMs: 500 });
|
|
const awaited = await client.waitForAgentStream<{ token: string }>({ timeoutMs: 500 });
|
|
const waitedTyping = await client.waitForAgentTyping<{ active: boolean }>({ timeoutMs: 500 });
|
|
const waitedWarning = await client.waitForContextWarning<{ thresholdPct: number; estimatedPct: number }>({ timeoutMs: 500 });
|
|
const waitedAny = await client.waitForAnyEvent<{ token: string }>(['agent.stream'], { timeoutMs: 500 });
|
|
|
|
expect(mock.subscribeAgentStream).toHaveBeenCalledWith(streamHandler);
|
|
expect(mock.subscribeAgentTyping).toHaveBeenCalledWith(typingHandler);
|
|
expect(mock.subscribeContextWarning).toHaveBeenCalledWith(warningHandler);
|
|
expect(mock.subscribeEvents).toHaveBeenCalledWith(eventHandler);
|
|
expect(mock.subscribeEvent).toHaveBeenCalledWith('agent.stream', streamEventHandler);
|
|
expect(mock.waitForEvent).toHaveBeenCalledWith('agent.stream', { timeoutMs: 500 });
|
|
expect(mock.waitForAgentStream).toHaveBeenCalledWith({ timeoutMs: 500 });
|
|
expect(mock.waitForAgentTyping).toHaveBeenCalledWith({ timeoutMs: 500 });
|
|
expect(mock.waitForContextWarning).toHaveBeenCalledWith({ timeoutMs: 500 });
|
|
expect(mock.waitForAnyEvent).toHaveBeenCalledWith(['agent.stream'], { timeoutMs: 500 });
|
|
expect(waitedEvent).toEqual({ token: 'evented' });
|
|
expect(awaited).toEqual({ token: 'streamed' });
|
|
expect(waitedTyping).toEqual({ active: true });
|
|
expect(waitedWarning).toEqual({ thresholdPct: 75, estimatedPct: 90 });
|
|
expect(waitedAny).toEqual({ event: 'agent.stream', data: { token: 'any' } });
|
|
unsubscribeStream();
|
|
unsubscribeTyping();
|
|
unsubscribeWarning();
|
|
unsubscribeEvents();
|
|
unsubscribeEvent();
|
|
});
|
|
|
|
it('platform clearEventSubscriptions forwards to runtime client', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
client.clearEventSubscriptions();
|
|
|
|
expect(mock.clearEventSubscriptions).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it('platform listKnownEventNames forwards to runtime client', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
const events = client.listKnownEventNames();
|
|
|
|
expect(mock.listKnownEventNames).toHaveBeenCalledOnce();
|
|
expect(events).toEqual(['agent.stream', 'agent.typing', 'context_warning']);
|
|
});
|
|
|
|
it('platform eventSubscriptionCount forwards runtime getter value', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
expect(client.eventSubscriptionCount).toBe(mock.eventSubscriptionCount);
|
|
});
|
|
|
|
it('macOS client forwards canvas methods to runtime client', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new MacOSCompanionClient({ runtime: mock.runtime, nodeId: 'mac-node' });
|
|
|
|
await client.putCanvasArtifact({
|
|
sessionId: 'ws:test-canvas',
|
|
type: 'markdown',
|
|
content: { body: 'hello' },
|
|
});
|
|
await client.listCanvasArtifacts('ws:test-canvas');
|
|
await client.getCanvasArtifact({ sessionId: 'ws:test-canvas', artifactId: 'a1' });
|
|
await client.deleteCanvasArtifact({ sessionId: 'ws:test-canvas', artifactId: 'a1' });
|
|
await client.clearCanvasArtifacts('ws:test-canvas');
|
|
|
|
expect(mock.putCanvasArtifact).toHaveBeenCalledWith({
|
|
sessionId: 'ws:test-canvas',
|
|
type: 'markdown',
|
|
content: { body: 'hello' },
|
|
});
|
|
expect(mock.listCanvasArtifacts).toHaveBeenCalledWith('ws:test-canvas');
|
|
expect(mock.getCanvasArtifact).toHaveBeenCalledWith({ sessionId: 'ws:test-canvas', artifactId: 'a1' });
|
|
expect(mock.deleteCanvasArtifact).toHaveBeenCalledWith({ sessionId: 'ws:test-canvas', artifactId: 'a1' });
|
|
expect(mock.clearCanvasArtifacts).toHaveBeenCalledWith('ws:test-canvas');
|
|
});
|
|
|
|
it('bootstrap registers node and then fetches capabilities', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
const result = await client.bootstrap();
|
|
|
|
expect(mock.bootstrapNode).toHaveBeenCalledOnce();
|
|
expect(result).toEqual({
|
|
register: { registered: true },
|
|
capabilities: expect.any(Object),
|
|
systemCapabilities: undefined,
|
|
});
|
|
});
|
|
|
|
it('publishHeartbeat uses safe defaults for status payload', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new AndroidCompanionClient({ runtime: mock.runtime, nodeId: 'android-node' });
|
|
|
|
await client.publishHeartbeat();
|
|
|
|
expect(mock.setNodeStatus).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
platform: 'android',
|
|
statusText: 'heartbeat',
|
|
powerSource: 'unknown',
|
|
}),
|
|
);
|
|
});
|
|
|
|
it('uses defaultSessionId for canvas operations when sessionId is omitted', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({
|
|
runtime: mock.runtime,
|
|
nodeId: 'ios-node',
|
|
defaultSessionId: 'ws:default-canvas',
|
|
});
|
|
|
|
await client.putCanvasArtifact({
|
|
type: 'markdown',
|
|
content: { body: 'default session' },
|
|
});
|
|
await client.listCanvasArtifacts();
|
|
await client.getCanvasArtifact({ artifactId: 'a1' });
|
|
await client.deleteCanvasArtifact({ artifactId: 'a1' });
|
|
await client.clearCanvasArtifacts();
|
|
|
|
expect(mock.putCanvasArtifact).toHaveBeenCalledWith({
|
|
sessionId: 'ws:default-canvas',
|
|
type: 'markdown',
|
|
content: { body: 'default session' },
|
|
artifactId: undefined,
|
|
title: undefined,
|
|
metadata: undefined,
|
|
});
|
|
expect(mock.listCanvasArtifacts).toHaveBeenCalledWith('ws:default-canvas');
|
|
expect(mock.getCanvasArtifact).toHaveBeenCalledWith({ sessionId: 'ws:default-canvas', artifactId: 'a1' });
|
|
expect(mock.deleteCanvasArtifact).toHaveBeenCalledWith({ sessionId: 'ws:default-canvas', artifactId: 'a1' });
|
|
expect(mock.clearCanvasArtifacts).toHaveBeenCalledWith('ws:default-canvas');
|
|
});
|
|
|
|
it('throws when canvas sessionId is missing and no defaultSessionId is configured', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new MacOSCompanionClient({ runtime: mock.runtime, nodeId: 'mac-node' });
|
|
|
|
expect(() => client.listCanvasArtifacts()).toThrow(
|
|
'sessionId is required (provide one or configure defaultSessionId)',
|
|
);
|
|
});
|
|
|
|
it('creates a bound heartbeat loop helper from platform clients', async () => {
|
|
const mock = createRuntimeMock();
|
|
const client = new IOSCompanionClient({ runtime: mock.runtime, nodeId: 'ios-node' });
|
|
|
|
const loop = client.createHeartbeatLoop();
|
|
await loop.tickNow();
|
|
|
|
expect(mock.setNodeStatus).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
platform: 'ios',
|
|
statusText: 'heartbeat',
|
|
}),
|
|
);
|
|
});
|
|
|
|
it('bootstrap can request system capabilities snapshot', async () => {
|
|
const mock = createRuntimeMock();
|
|
mock.bootstrapNode.mockResolvedValueOnce({
|
|
register: { registered: true },
|
|
capabilities: {
|
|
node: { id: 'n1', role: 'companion', registeredAt: Date.now() },
|
|
protocol: { serverVersion: 1, nodeVersion: 1, negotiatedVersion: 1 },
|
|
capabilities: { declared: [], enabled: [], featureGates: {} },
|
|
},
|
|
systemCapabilities: {
|
|
protocol: { version: 1 },
|
|
nodes: {
|
|
enabled: true,
|
|
locationEnabled: true,
|
|
pushEnabled: true,
|
|
allowedRoles: ['companion'],
|
|
registered: true,
|
|
},
|
|
featureGates: {},
|
|
},
|
|
});
|
|
const client = new MacOSCompanionClient({ runtime: mock.runtime, nodeId: 'mac-node' });
|
|
|
|
const result = await client.bootstrap({ includeSystemCapabilities: true });
|
|
|
|
expect(mock.bootstrapNode).toHaveBeenCalledWith(
|
|
expect.objectContaining({ nodeId: 'mac-node' }),
|
|
{ includeSystemCapabilities: true },
|
|
);
|
|
expect(result.systemCapabilities?.nodes.enabled).toBe(true);
|
|
});
|
|
});
|