@amaster.ai/pi-computer-use 0.1.1-beta.0

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.
Files changed (42) hide show
  1. package/LICENSE +201 -0
  2. package/dist/__tests__/computer-client.test.d.ts +2 -0
  3. package/dist/__tests__/computer-client.test.d.ts.map +1 -0
  4. package/dist/__tests__/computer-client.test.js +174 -0
  5. package/dist/__tests__/computer-client.test.js.map +1 -0
  6. package/dist/__tests__/index.test.d.ts +2 -0
  7. package/dist/__tests__/index.test.d.ts.map +1 -0
  8. package/dist/__tests__/index.test.js +385 -0
  9. package/dist/__tests__/index.test.js.map +1 -0
  10. package/dist/__tests__/server-process.test.d.ts +2 -0
  11. package/dist/__tests__/server-process.test.d.ts.map +1 -0
  12. package/dist/__tests__/server-process.test.js +127 -0
  13. package/dist/__tests__/server-process.test.js.map +1 -0
  14. package/dist/__tests__/vision.test.d.ts +2 -0
  15. package/dist/__tests__/vision.test.d.ts.map +1 -0
  16. package/dist/__tests__/vision.test.js +36 -0
  17. package/dist/__tests__/vision.test.js.map +1 -0
  18. package/dist/actions.d.ts +15 -0
  19. package/dist/actions.d.ts.map +1 -0
  20. package/dist/actions.js +45 -0
  21. package/dist/actions.js.map +1 -0
  22. package/dist/computer-client.d.ts +13 -0
  23. package/dist/computer-client.d.ts.map +1 -0
  24. package/dist/computer-client.js +109 -0
  25. package/dist/computer-client.js.map +1 -0
  26. package/dist/config.d.ts +31 -0
  27. package/dist/config.d.ts.map +1 -0
  28. package/dist/config.js +25 -0
  29. package/dist/config.js.map +1 -0
  30. package/dist/index.d.ts +6 -0
  31. package/dist/index.d.ts.map +1 -0
  32. package/dist/index.js +102 -0
  33. package/dist/index.js.map +1 -0
  34. package/dist/server-process.d.ts +9 -0
  35. package/dist/server-process.d.ts.map +1 -0
  36. package/dist/server-process.js +76 -0
  37. package/dist/server-process.js.map +1 -0
  38. package/dist/vision.d.ts +6 -0
  39. package/dist/vision.d.ts.map +1 -0
  40. package/dist/vision.js +57 -0
  41. package/dist/vision.js.map +1 -0
  42. package/package.json +52 -0
@@ -0,0 +1,385 @@
1
+ import { beforeEach, describe, expect, test, vi } from 'vitest';
2
+ // --- Mocks ---
3
+ const mockWsSend = vi.fn();
4
+ const mockWsClose = vi.fn();
5
+ const mockWsOn = vi.fn();
6
+ const mockWsOff = vi.fn();
7
+ let wsOpenCallback = null;
8
+ let wsMessageCallback = null;
9
+ vi.mock('ws', () => ({
10
+ default: class MockWebSocket {
11
+ static OPEN = 1;
12
+ readyState = 1;
13
+ send = mockWsSend;
14
+ close = mockWsClose;
15
+ on = vi.fn((event, handler) => {
16
+ if (event === 'open')
17
+ wsOpenCallback = handler;
18
+ if (event === 'message')
19
+ wsMessageCallback = handler;
20
+ mockWsOn(event, handler);
21
+ });
22
+ off = mockWsOff;
23
+ terminate = vi.fn();
24
+ constructor() {
25
+ setTimeout(() => wsOpenCallback?.(), 0);
26
+ }
27
+ },
28
+ }));
29
+ const mockSpawn = vi.fn((_cmd, _args) => ({
30
+ exitCode: null,
31
+ stderr: { on: vi.fn() },
32
+ on: vi.fn(),
33
+ kill: vi.fn(),
34
+ }));
35
+ vi.mock('node:child_process', () => ({
36
+ spawn: (cmd, args) => mockSpawn(cmd, args),
37
+ }));
38
+ let mockConfigContent = null;
39
+ vi.mock('node:fs', async (importOriginal) => {
40
+ const actual = (await importOriginal());
41
+ return {
42
+ ...actual,
43
+ readFileSync: vi.fn((...args) => {
44
+ if (mockConfigContent !== null)
45
+ return mockConfigContent;
46
+ throw new Error('ENOENT');
47
+ }),
48
+ };
49
+ });
50
+ const registeredTools = new Map();
51
+ const sessionStartHandlers = [];
52
+ const sessionShutdownHandlers = [];
53
+ const mockPi = {
54
+ registerTool: vi.fn((tool) => {
55
+ registeredTools.set(tool.name, tool);
56
+ }),
57
+ on: vi.fn((event, handler) => {
58
+ if (event === 'session_start')
59
+ sessionStartHandlers.push(handler);
60
+ if (event === 'session_shutdown')
61
+ sessionShutdownHandlers.push(handler);
62
+ }),
63
+ };
64
+ const { default: computerUseExtension } = await import('../index.js');
65
+ function registerExtension(config) {
66
+ if (config) {
67
+ mockConfigContent = JSON.stringify({ 'pi-computer-use': config });
68
+ }
69
+ else {
70
+ mockConfigContent = null;
71
+ }
72
+ registeredTools.clear();
73
+ sessionStartHandlers.length = 0;
74
+ sessionShutdownHandlers.length = 0;
75
+ computerUseExtension(mockPi);
76
+ }
77
+ // --- Tests ---
78
+ describe('computerUseExtension', () => {
79
+ beforeEach(() => {
80
+ registeredTools.clear();
81
+ sessionStartHandlers.length = 0;
82
+ sessionShutdownHandlers.length = 0;
83
+ mockPi.registerTool.mockClear();
84
+ mockPi.on.mockClear();
85
+ mockWsSend.mockClear();
86
+ mockWsClose.mockClear();
87
+ mockSpawn.mockClear();
88
+ wsOpenCallback = null;
89
+ wsMessageCallback = null;
90
+ });
91
+ test('registers session_start and session_shutdown handlers', () => {
92
+ registerExtension();
93
+ expect(mockPi.on).toHaveBeenCalledWith('session_start', expect.any(Function));
94
+ expect(mockPi.on).toHaveBeenCalledWith('session_shutdown', expect.any(Function));
95
+ });
96
+ test('registers computer_use tool', () => {
97
+ registerExtension();
98
+ expect(registeredTools.has('computer_use')).toBe(true);
99
+ });
100
+ describe('tool registration', () => {
101
+ test('tool has correct name and description', () => {
102
+ registerExtension();
103
+ const tool = registeredTools.get('computer_use');
104
+ expect(tool.name).toBe('computer_use');
105
+ expect(tool.description).toContain('screenshot');
106
+ expect(tool.description).toContain('click');
107
+ expect(tool.description).toContain('type');
108
+ });
109
+ test('only registers a single tool', () => {
110
+ registerExtension();
111
+ expect(registeredTools.size).toBe(1);
112
+ });
113
+ });
114
+ describe('config loading', () => {
115
+ test('uses defaults when config.json is missing', () => {
116
+ registerExtension();
117
+ expect(registeredTools.size).toBe(1);
118
+ });
119
+ test('reads pi-computer-use section from config.json', () => {
120
+ registerExtension({ mode: 'external', host: '192.168.1.100' });
121
+ expect(registeredTools.size).toBe(1);
122
+ });
123
+ });
124
+ describe('lifecycle (session_shutdown)', () => {
125
+ test('no-ops when shutdown called without prior tool use', async () => {
126
+ registerExtension();
127
+ for (const handler of sessionShutdownHandlers) {
128
+ await handler();
129
+ }
130
+ expect(mockWsClose).not.toHaveBeenCalled();
131
+ });
132
+ });
133
+ });
134
+ describe('action dispatch', () => {
135
+ test('click sends left_click command', async () => {
136
+ const { dispatchAction } = await import('../actions.js');
137
+ const mockClient = {
138
+ sendCommand: vi.fn(() => Promise.resolve({})),
139
+ screenshot: vi.fn(),
140
+ };
141
+ const result = await dispatchAction(mockClient, {
142
+ type: 'click',
143
+ x: 100,
144
+ y: 200,
145
+ });
146
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('left_click', { x: 100, y: 200 });
147
+ expect(result).toContain('100');
148
+ expect(result).toContain('200');
149
+ });
150
+ test('right click sends right_click command', async () => {
151
+ const { dispatchAction } = await import('../actions.js');
152
+ const mockClient = {
153
+ sendCommand: vi.fn(() => Promise.resolve({})),
154
+ screenshot: vi.fn(),
155
+ };
156
+ await dispatchAction(mockClient, {
157
+ type: 'click',
158
+ x: 50,
159
+ y: 75,
160
+ button: 'right',
161
+ });
162
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('right_click', { x: 50, y: 75 });
163
+ });
164
+ test('type sends type_text command', async () => {
165
+ const { dispatchAction } = await import('../actions.js');
166
+ const mockClient = {
167
+ sendCommand: vi.fn(() => Promise.resolve({})),
168
+ screenshot: vi.fn(),
169
+ };
170
+ const result = await dispatchAction(mockClient, {
171
+ type: 'type',
172
+ text: 'hello',
173
+ });
174
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('type_text', { text: 'hello' });
175
+ expect(result).toContain('hello');
176
+ });
177
+ test('keypress sends hotkey command', async () => {
178
+ const { dispatchAction } = await import('../actions.js');
179
+ const mockClient = {
180
+ sendCommand: vi.fn(() => Promise.resolve({})),
181
+ screenshot: vi.fn(),
182
+ };
183
+ await dispatchAction(mockClient, {
184
+ type: 'keypress',
185
+ keys: ['ctrl', 'c'],
186
+ });
187
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('hotkey', { keys: ['ctrl', 'c'] });
188
+ });
189
+ test('scroll sends scroll command', async () => {
190
+ const { dispatchAction } = await import('../actions.js');
191
+ const mockClient = {
192
+ sendCommand: vi.fn(() => Promise.resolve({})),
193
+ screenshot: vi.fn(),
194
+ };
195
+ await dispatchAction(mockClient, {
196
+ type: 'scroll',
197
+ scroll_x: 0,
198
+ scroll_y: -3,
199
+ });
200
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('scroll', { x: 0, y: -3 });
201
+ });
202
+ test('run_command sends run_command and returns output', async () => {
203
+ const { dispatchAction } = await import('../actions.js');
204
+ const mockClient = {
205
+ sendCommand: vi.fn(() => Promise.resolve({ stdout: 'file.txt', stderr: '' })),
206
+ screenshot: vi.fn(),
207
+ };
208
+ const result = await dispatchAction(mockClient, {
209
+ type: 'run_command',
210
+ command: 'ls',
211
+ });
212
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('run_command', { command: 'ls' });
213
+ expect(result).toContain('file.txt');
214
+ });
215
+ test('wait sleeps for 1 second', async () => {
216
+ const { dispatchAction } = await import('../actions.js');
217
+ const mockClient = {
218
+ sendCommand: vi.fn(() => Promise.resolve({})),
219
+ screenshot: vi.fn(),
220
+ };
221
+ const start = Date.now();
222
+ await dispatchAction(mockClient, { type: 'wait' });
223
+ const elapsed = Date.now() - start;
224
+ expect(elapsed).toBeGreaterThanOrEqual(900);
225
+ expect(mockClient.sendCommand).not.toHaveBeenCalled();
226
+ });
227
+ test('double_click sends double_click command', async () => {
228
+ const { dispatchAction } = await import('../actions.js');
229
+ const mockClient = {
230
+ sendCommand: vi.fn(() => Promise.resolve({})),
231
+ screenshot: vi.fn(),
232
+ };
233
+ const result = await dispatchAction(mockClient, {
234
+ type: 'double_click',
235
+ x: 300,
236
+ y: 400,
237
+ });
238
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('double_click', { x: 300, y: 400 });
239
+ expect(result).toContain('300');
240
+ expect(result).toContain('400');
241
+ });
242
+ test('move sends move_cursor command', async () => {
243
+ const { dispatchAction } = await import('../actions.js');
244
+ const mockClient = {
245
+ sendCommand: vi.fn(() => Promise.resolve({})),
246
+ screenshot: vi.fn(),
247
+ };
248
+ const result = await dispatchAction(mockClient, {
249
+ type: 'move',
250
+ x: 500,
251
+ y: 600,
252
+ });
253
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('move_cursor', { x: 500, y: 600 });
254
+ expect(result).toContain('500');
255
+ expect(result).toContain('600');
256
+ });
257
+ test('drag sends drag command with path and button', async () => {
258
+ const { dispatchAction } = await import('../actions.js');
259
+ const mockClient = {
260
+ sendCommand: vi.fn(() => Promise.resolve({})),
261
+ screenshot: vi.fn(),
262
+ };
263
+ const path = [
264
+ [10, 10],
265
+ [50, 50],
266
+ [100, 100],
267
+ ];
268
+ const result = await dispatchAction(mockClient, {
269
+ type: 'drag',
270
+ path,
271
+ button: 'left',
272
+ });
273
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('drag', { path, button: 'left' });
274
+ expect(result).toContain('3');
275
+ });
276
+ test('run_command with stderr includes stderr in output', async () => {
277
+ const { dispatchAction } = await import('../actions.js');
278
+ const mockClient = {
279
+ sendCommand: vi.fn(() => Promise.resolve({ stdout: 'output', stderr: 'warning: deprecated' })),
280
+ screenshot: vi.fn(),
281
+ };
282
+ const result = await dispatchAction(mockClient, {
283
+ type: 'run_command',
284
+ command: 'make build',
285
+ });
286
+ expect(result).toContain('output');
287
+ expect(result).toContain('warning: deprecated');
288
+ });
289
+ test('run_command with no output returns fallback message', async () => {
290
+ const { dispatchAction } = await import('../actions.js');
291
+ const mockClient = {
292
+ sendCommand: vi.fn(() => Promise.resolve({ stdout: '', stderr: '' })),
293
+ screenshot: vi.fn(),
294
+ };
295
+ const result = await dispatchAction(mockClient, {
296
+ type: 'run_command',
297
+ command: 'true',
298
+ });
299
+ expect(result).toBe('Command executed (no output)');
300
+ });
301
+ test('screenshot action returns undefined', async () => {
302
+ const { dispatchAction } = await import('../actions.js');
303
+ const mockClient = {
304
+ sendCommand: vi.fn(() => Promise.resolve({})),
305
+ screenshot: vi.fn(),
306
+ };
307
+ const result = await dispatchAction(mockClient, { type: 'screenshot' });
308
+ expect(result).toBeUndefined();
309
+ expect(mockClient.sendCommand).not.toHaveBeenCalled();
310
+ });
311
+ test('scroll defaults x and y to 0', async () => {
312
+ const { dispatchAction } = await import('../actions.js');
313
+ const mockClient = {
314
+ sendCommand: vi.fn(() => Promise.resolve({})),
315
+ screenshot: vi.fn(),
316
+ };
317
+ await dispatchAction(mockClient, { type: 'scroll' });
318
+ expect(mockClient.sendCommand).toHaveBeenCalledWith('scroll', { x: 0, y: 0 });
319
+ });
320
+ test('unknown action throws', async () => {
321
+ const { dispatchAction } = await import('../actions.js');
322
+ const mockClient = {
323
+ sendCommand: vi.fn(() => Promise.resolve({})),
324
+ screenshot: vi.fn(),
325
+ };
326
+ await expect(dispatchAction(mockClient, { type: 'unknown_action' })).rejects.toThrow('Unknown action type');
327
+ });
328
+ });
329
+ describe('config', () => {
330
+ test('provides sane defaults', async () => {
331
+ const { resolveConfig } = await import('../config.js');
332
+ const config = resolveConfig();
333
+ expect(config.mode).toBe('managed');
334
+ expect(config.host).toBe('127.0.0.1');
335
+ expect(config.port).toBe(8000);
336
+ expect(config.command).toBe('uvx');
337
+ expect(config.package).toBe('cua-computer-server');
338
+ expect(config.autoScreenshot).toBe(true);
339
+ });
340
+ test('merges user config over defaults', async () => {
341
+ const { resolveConfig } = await import('../config.js');
342
+ const config = resolveConfig({ mode: 'external', host: '10.0.0.1', port: 9000 });
343
+ expect(config.mode).toBe('external');
344
+ expect(config.host).toBe('10.0.0.1');
345
+ expect(config.port).toBe(9000);
346
+ expect(config.command).toBe('uvx');
347
+ });
348
+ test('preserves visionModel config', async () => {
349
+ const { resolveConfig } = await import('../config.js');
350
+ const config = resolveConfig({
351
+ visionModel: { provider: 'openai', model: 'gpt-4o' },
352
+ });
353
+ expect(config.visionModel).toEqual({ provider: 'openai', model: 'gpt-4o' });
354
+ });
355
+ test('preserves apiKey and vmName for cloud mode', async () => {
356
+ const { resolveConfig } = await import('../config.js');
357
+ const config = resolveConfig({
358
+ mode: 'external',
359
+ apiKey: 'my-secret-key',
360
+ vmName: 'production-vm',
361
+ });
362
+ expect(config.apiKey).toBe('my-secret-key');
363
+ expect(config.vmName).toBe('production-vm');
364
+ });
365
+ test('autoScreenshot can be disabled', async () => {
366
+ const { resolveConfig } = await import('../config.js');
367
+ const config = resolveConfig({ autoScreenshot: false });
368
+ expect(config.autoScreenshot).toBe(false);
369
+ });
370
+ test('loadConfigFromFile returns empty on missing file', async () => {
371
+ mockConfigContent = null;
372
+ const { loadConfigFromFile } = await import('../config.js');
373
+ const config = loadConfigFromFile();
374
+ expect(config).toEqual({});
375
+ });
376
+ test('loadConfigFromFile reads pi-computer-use section', async () => {
377
+ mockConfigContent = JSON.stringify({
378
+ 'pi-computer-use': { mode: 'external', port: 4000 },
379
+ });
380
+ const { loadConfigFromFile } = await import('../config.js');
381
+ const config = loadConfigFromFile();
382
+ expect(config).toEqual({ mode: 'external', port: 4000 });
383
+ });
384
+ });
385
+ //# sourceMappingURL=index.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.test.js","sourceRoot":"","sources":["../../src/__tests__/index.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=server-process.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-process.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/server-process.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,127 @@
1
+ import { beforeEach, describe, expect, test, vi } from 'vitest';
2
+ const mockKill = vi.fn();
3
+ let processExitHandler = null;
4
+ let processErrorHandler = null;
5
+ let mockExitCode = null;
6
+ const mockSpawn = vi.fn((_cmd, _args, _opts) => ({
7
+ get exitCode() {
8
+ return mockExitCode;
9
+ },
10
+ stderr: {
11
+ on: vi.fn(),
12
+ },
13
+ on: vi.fn((event, handler) => {
14
+ if (event === 'exit')
15
+ processExitHandler = handler;
16
+ if (event === 'error')
17
+ processErrorHandler = handler;
18
+ }),
19
+ kill: mockKill,
20
+ }));
21
+ vi.mock('node:child_process', () => ({
22
+ spawn: (cmd, args, opts) => mockSpawn(cmd, args, opts),
23
+ }));
24
+ let wsProbeCallCount = 0;
25
+ let wsProbeSuccess = () => true;
26
+ vi.mock('ws', () => ({
27
+ default: class MockWsProbe {
28
+ static OPEN = 1;
29
+ readyState = 1;
30
+ send = vi.fn();
31
+ close = vi.fn();
32
+ terminate = vi.fn();
33
+ off = vi.fn();
34
+ on;
35
+ constructor() {
36
+ const idx = wsProbeCallCount++;
37
+ const shouldSucceed = wsProbeSuccess(idx);
38
+ this.on = (event, handler) => {
39
+ if (event === 'open' && shouldSucceed) {
40
+ setTimeout(() => handler(), 0);
41
+ }
42
+ if (event === 'error' && !shouldSucceed) {
43
+ setTimeout(() => handler(new Error('ECONNREFUSED')), 0);
44
+ }
45
+ };
46
+ }
47
+ },
48
+ }));
49
+ const { ComputerServerProcess } = await import('../server-process.js');
50
+ describe('ComputerServerProcess', () => {
51
+ beforeEach(() => {
52
+ mockSpawn.mockClear();
53
+ mockKill.mockClear();
54
+ processExitHandler = null;
55
+ processErrorHandler = null;
56
+ mockExitCode = null;
57
+ wsProbeCallCount = 0;
58
+ wsProbeSuccess = () => true;
59
+ });
60
+ describe('start()', () => {
61
+ test('spawns subprocess with default args', async () => {
62
+ const server = new ComputerServerProcess();
63
+ await server.start({});
64
+ expect(mockSpawn).toHaveBeenCalledWith('uvx', ['cua-computer-server', '--host', '127.0.0.1', '--port', '8000'], { stdio: ['ignore', 'pipe', 'pipe'] });
65
+ });
66
+ test('uses custom command, package, host, port', async () => {
67
+ const server = new ComputerServerProcess();
68
+ await server.start({
69
+ command: 'pipx',
70
+ package: 'my-server',
71
+ host: '0.0.0.0',
72
+ port: 9090,
73
+ });
74
+ expect(mockSpawn).toHaveBeenCalledWith('pipx', ['my-server', '--host', '0.0.0.0', '--port', '9090'], { stdio: ['ignore', 'pipe', 'pipe'] });
75
+ });
76
+ test('appends extraArgs', async () => {
77
+ const server = new ComputerServerProcess();
78
+ await server.start({ extraArgs: ['--verbose', '--no-sandbox'] });
79
+ expect(mockSpawn).toHaveBeenCalledWith('uvx', [
80
+ 'cua-computer-server',
81
+ '--host',
82
+ '127.0.0.1',
83
+ '--port',
84
+ '8000',
85
+ '--verbose',
86
+ '--no-sandbox',
87
+ ], { stdio: ['ignore', 'pipe', 'pipe'] });
88
+ });
89
+ test('retries WS probe until success', async () => {
90
+ wsProbeSuccess = (idx) => idx >= 2;
91
+ const server = new ComputerServerProcess();
92
+ await server.start({});
93
+ expect(wsProbeCallCount).toBeGreaterThanOrEqual(3);
94
+ });
95
+ test('throws if process exits before ready', async () => {
96
+ mockExitCode = 1;
97
+ wsProbeSuccess = () => false;
98
+ const server = new ComputerServerProcess();
99
+ await expect(server.start({})).rejects.toThrow('cua-computer-server exited with code 1 before becoming ready');
100
+ });
101
+ });
102
+ describe('stop()', () => {
103
+ test('sends SIGTERM to subprocess', async () => {
104
+ const server = new ComputerServerProcess();
105
+ await server.start({});
106
+ const stopPromise = server.stop();
107
+ setTimeout(() => processExitHandler?.(0), 10);
108
+ await stopPromise;
109
+ expect(mockKill).toHaveBeenCalledWith('SIGTERM');
110
+ });
111
+ test('no-ops when no process is running', async () => {
112
+ const server = new ComputerServerProcess();
113
+ await server.stop();
114
+ expect(mockKill).not.toHaveBeenCalled();
115
+ });
116
+ test('sends SIGKILL if process does not exit within grace period', async () => {
117
+ const server = new ComputerServerProcess();
118
+ await server.start({});
119
+ const stopPromise = server.stop();
120
+ // Don't call processExitHandler — simulate process not exiting
121
+ await stopPromise;
122
+ expect(mockKill).toHaveBeenCalledWith('SIGTERM');
123
+ expect(mockKill).toHaveBeenCalledWith('SIGKILL');
124
+ }, 10_000);
125
+ });
126
+ });
127
+ //# sourceMappingURL=server-process.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-process.test.js","sourceRoot":"","sources":["../../src/__tests__/server-process.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAEhE,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;AACzB,IAAI,kBAAkB,GAA2C,IAAI,CAAC;AACtE,IAAI,mBAAmB,GAAkC,IAAI,CAAC;AAC9D,IAAI,YAAY,GAAkB,IAAI,CAAC;AAEvC,MAAM,SAAS,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC,IAAY,EAAE,KAAe,EAAE,KAAa,EAAE,EAAE,CAAC,CAAC;IACzE,IAAI,QAAQ;QACV,OAAO,YAAY,CAAC;IACtB,CAAC;IACD,MAAM,EAAE;QACN,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE;KACZ;IACD,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC,KAAa,EAAE,OAAqC,EAAE,EAAE;QACjE,IAAI,KAAK,KAAK,MAAM;YAAE,kBAAkB,GAAG,OAAwC,CAAC;QACpF,IAAI,KAAK,KAAK,OAAO;YAAE,mBAAmB,GAAG,OAA+B,CAAC;IAC/E,CAAC,CAAC;IACF,IAAI,EAAE,QAAQ;CACf,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,GAAG,EAAE,CAAC,CAAC;IACnC,KAAK,EAAE,CAAC,GAAW,EAAE,IAAc,EAAE,IAAY,EAAE,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,IAAI,EAAE,IAAI,CAAC;CACjF,CAAC,CAAC,CAAC;AAEJ,IAAI,gBAAgB,GAAG,CAAC,CAAC;AACzB,IAAI,cAAc,GAA+B,GAAG,EAAE,CAAC,IAAI,CAAC;AAE5D,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,CAAC,CAAC;IACnB,OAAO,EAAE,MAAM,WAAW;QACxB,MAAM,CAAC,IAAI,GAAG,CAAC,CAAC;QAChB,UAAU,GAAG,CAAC,CAAC;QACf,IAAI,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACf,KAAK,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QAChB,SAAS,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACpB,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;QACd,EAAE,CAAiE;QAEnE;YACE,MAAM,GAAG,GAAG,gBAAgB,EAAE,CAAC;YAC/B,MAAM,aAAa,GAAG,cAAc,CAAC,GAAG,CAAC,CAAC;YAC1C,IAAI,CAAC,EAAE,GAAG,CAAC,KAAa,EAAE,OAAqC,EAAE,EAAE;gBACjE,IAAI,KAAK,KAAK,MAAM,IAAI,aAAa,EAAE,CAAC;oBACtC,UAAU,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;gBACjC,CAAC;gBACD,IAAI,KAAK,KAAK,OAAO,IAAI,CAAC,aAAa,EAAE,CAAC;oBACxC,UAAU,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,KAAK,CAAC,cAAc,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC1D,CAAC;YACH,CAAC,CAAC;QACJ,CAAC;KACF;CACF,CAAC,CAAC,CAAC;AAEJ,MAAM,EAAE,qBAAqB,EAAE,GAAG,MAAM,MAAM,CAAC,sBAAsB,CAAC,CAAC;AAEvE,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;IACrC,UAAU,CAAC,GAAG,EAAE;QACd,SAAS,CAAC,SAAS,EAAE,CAAC;QACtB,QAAQ,CAAC,SAAS,EAAE,CAAC;QACrB,kBAAkB,GAAG,IAAI,CAAC;QAC1B,mBAAmB,GAAG,IAAI,CAAC;QAC3B,YAAY,GAAG,IAAI,CAAC;QACpB,gBAAgB,GAAG,CAAC,CAAC;QACrB,cAAc,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC;IAC9B,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;QACvB,IAAI,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;YACrD,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAEvB,MAAM,CAAC,SAAS,CAAC,CAAC,oBAAoB,CACpC,KAAK,EACL,CAAC,qBAAqB,EAAE,QAAQ,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,CAAC,EAChE,EAAE,KAAK,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,CACtC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,0CAA0C,EAAE,KAAK,IAAI,EAAE;YAC1D,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC;gBACjB,OAAO,EAAE,MAAM;gBACf,OAAO,EAAE,WAAW;gBACpB,IAAI,EAAE,SAAS;gBACf,IAAI,EAAE,IAAI;aACX,CAAC,CAAC;YAEH,MAAM,CAAC,SAAS,CAAC,CAAC,oBAAoB,CACpC,MAAM,EACN,CAAC,WAAW,EAAE,QAAQ,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,CAAC,EACpD,EAAE,KAAK,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,CACtC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,mBAAmB,EAAE,KAAK,IAAI,EAAE;YACnC,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC,EAAE,SAAS,EAAE,CAAC,WAAW,EAAE,cAAc,CAAC,EAAE,CAAC,CAAC;YAEjE,MAAM,CAAC,SAAS,CAAC,CAAC,oBAAoB,CACpC,KAAK,EACL;gBACE,qBAAqB;gBACrB,QAAQ;gBACR,WAAW;gBACX,QAAQ;gBACR,MAAM;gBACN,WAAW;gBACX,cAAc;aACf,EACD,EAAE,KAAK,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,CACtC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,gCAAgC,EAAE,KAAK,IAAI,EAAE;YAChD,cAAc,GAAG,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,IAAI,CAAC,CAAC;YAEnC,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAEvB,MAAM,CAAC,gBAAgB,CAAC,CAAC,sBAAsB,CAAC,CAAC,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,sCAAsC,EAAE,KAAK,IAAI,EAAE;YACtD,YAAY,GAAG,CAAC,CAAC;YACjB,cAAc,GAAG,GAAG,EAAE,CAAC,KAAK,CAAC;YAE7B,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAC5C,8DAA8D,CAC/D,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;QACtB,IAAI,CAAC,6BAA6B,EAAE,KAAK,IAAI,EAAE;YAC7C,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAEvB,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;YAClC,UAAU,CAAC,GAAG,EAAE,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC9C,MAAM,WAAW,CAAC;YAElB,MAAM,CAAC,QAAQ,CAAC,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,mCAAmC,EAAE,KAAK,IAAI,EAAE;YACnD,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;YACpB,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;QAC1C,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,4DAA4D,EAAE,KAAK,IAAI,EAAE;YAC5E,MAAM,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;YAC3C,MAAM,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAEvB,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;YAClC,+DAA+D;YAC/D,MAAM,WAAW,CAAC;YAElB,MAAM,CAAC,QAAQ,CAAC,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;YACjD,MAAM,CAAC,QAAQ,CAAC,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;QACnD,CAAC,EAAE,MAAM,CAAC,CAAC;IACb,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=vision.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vision.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/vision.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,36 @@
1
+ import { describe, expect, test, vi } from 'vitest';
2
+ import { createPiVisionCaller, VISUAL_SYSTEM_PROMPT } from '../vision.js';
3
+ describe('vision', () => {
4
+ describe('VISUAL_SYSTEM_PROMPT', () => {
5
+ test('contains key instructions for computer-use analysis', () => {
6
+ expect(VISUAL_SYSTEM_PROMPT).toContain('Visual Analysis Agent');
7
+ expect(VISUAL_SYSTEM_PROMPT).toContain('pixel coordinates');
8
+ expect(VISUAL_SYSTEM_PROMPT).toContain('clickable elements');
9
+ expect(VISUAL_SYSTEM_PROMPT).toContain('CENTER point');
10
+ expect(VISUAL_SYSTEM_PROMPT).toContain('Do NOT perform actions');
11
+ });
12
+ });
13
+ describe('createPiVisionCaller()', () => {
14
+ test('throws when model is not found in registry', async () => {
15
+ const mockCtx = {
16
+ modelRegistry: {
17
+ find: vi.fn(() => null),
18
+ getApiKeyAndHeaders: vi.fn(),
19
+ },
20
+ };
21
+ const caller = createPiVisionCaller({ provider: 'openai', model: 'gpt-4o' }, mockCtx);
22
+ await expect(caller('describe', 'base64data', 'image/png')).rejects.toThrow('not found in model registry');
23
+ });
24
+ test('throws when auth fails', async () => {
25
+ const mockCtx = {
26
+ modelRegistry: {
27
+ find: vi.fn(() => ({ id: 'gpt-4o', provider: 'openai' })),
28
+ getApiKeyAndHeaders: vi.fn(() => ({ ok: false, error: 'No API key configured' })),
29
+ },
30
+ };
31
+ const caller = createPiVisionCaller({ provider: 'openai', model: 'gpt-4o' }, mockCtx);
32
+ await expect(caller('describe', 'base64data', 'image/png')).rejects.toThrow('Auth failed for vision model');
33
+ });
34
+ });
35
+ });
36
+ //# sourceMappingURL=vision.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vision.test.js","sourceRoot":"","sources":["../../src/__tests__/vision.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AACpD,OAAO,EAAE,oBAAoB,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;AAE1E,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;IACtB,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;QACpC,IAAI,CAAC,qDAAqD,EAAE,GAAG,EAAE;YAC/D,MAAM,CAAC,oBAAoB,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;YAChE,MAAM,CAAC,oBAAoB,CAAC,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;YAC5D,MAAM,CAAC,oBAAoB,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;YAC7D,MAAM,CAAC,oBAAoB,CAAC,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;YACvD,MAAM,CAAC,oBAAoB,CAAC,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;QACnE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,wBAAwB,EAAE,GAAG,EAAE;QACtC,IAAI,CAAC,4CAA4C,EAAE,KAAK,IAAI,EAAE;YAC5D,MAAM,OAAO,GAAG;gBACd,aAAa,EAAE;oBACb,IAAI,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC;oBACvB,mBAAmB,EAAE,EAAE,CAAC,EAAE,EAAE;iBAC7B;aACF,CAAC;YAEF,MAAM,MAAM,GAAG,oBAAoB,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,OAAc,CAAC,CAAC;YAE7F,MAAM,MAAM,CAAC,MAAM,CAAC,UAAU,EAAE,YAAY,EAAE,WAAW,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CACzE,6BAA6B,CAC9B,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,wBAAwB,EAAE,KAAK,IAAI,EAAE;YACxC,MAAM,OAAO,GAAG;gBACd,aAAa,EAAE;oBACb,IAAI,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,CAAC;oBACzD,mBAAmB,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;iBAClF;aACF,CAAC;YAEF,MAAM,MAAM,GAAG,oBAAoB,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,OAAc,CAAC,CAAC;YAE7F,MAAM,MAAM,CAAC,MAAM,CAAC,UAAU,EAAE,YAAY,EAAE,WAAW,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CACzE,8BAA8B,CAC/B,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,15 @@
1
+ import type { ComputerClient } from './computer-client.js';
2
+ export interface ComputerAction {
3
+ type: string;
4
+ x?: number;
5
+ y?: number;
6
+ button?: string;
7
+ text?: string;
8
+ keys?: string[];
9
+ scroll_x?: number;
10
+ scroll_y?: number;
11
+ path?: Array<[number, number]>;
12
+ command?: string;
13
+ }
14
+ export declare function dispatchAction(client: ComputerClient, action: ComputerAction): Promise<string | undefined>;
15
+ //# sourceMappingURL=actions.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"actions.d.ts","sourceRoot":"","sources":["../src/actions.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AAE3D,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,CAAC,CAAC,EAAE,MAAM,CAAC;IACX,CAAC,CAAC,EAAE,MAAM,CAAC;IACX,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,KAAK,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC;IAC/B,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,wBAAsB,cAAc,CAClC,MAAM,EAAE,cAAc,EACtB,MAAM,EAAE,cAAc,GACrB,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC,CAqD7B"}