@canaryai/cli 0.2.3 → 0.2.4

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 (57) hide show
  1. package/dist/bin.js +1 -0
  2. package/dist/bin.js.map +1 -1
  3. package/dist/{chunk-DXJNFJ3A.js → chunk-6WWHXWCS.js} +3 -2
  4. package/dist/{chunk-DXJNFJ3A.js.map → chunk-6WWHXWCS.js.map} +1 -1
  5. package/dist/{chunk-HOYYXZPV.js → chunk-DXIAHB72.js} +3 -2
  6. package/dist/{chunk-HOYYXZPV.js.map → chunk-DXIAHB72.js.map} +1 -1
  7. package/dist/chunk-FIQBGAKW.js +373 -0
  8. package/dist/chunk-FIQBGAKW.js.map +1 -0
  9. package/dist/{chunk-JMI7WWPF.js → chunk-ILEPYWZX.js} +3 -2
  10. package/dist/{chunk-JMI7WWPF.js.map → chunk-ILEPYWZX.js.map} +1 -1
  11. package/dist/{chunk-7R4YFGP6.js → chunk-PWWQGYFG.js} +3 -1
  12. package/dist/{chunk-7R4YFGP6.js.map → chunk-PWWQGYFG.js.map} +1 -1
  13. package/dist/chunk-VLFUCAPZ.js +12 -0
  14. package/dist/{debug-workflow-PMLMWKWI.js → debug-workflow-PT3OUR3V.js} +5 -5
  15. package/dist/docs-GCYDTEOY.js +270 -0
  16. package/dist/docs-GCYDTEOY.js.map +1 -0
  17. package/dist/{feature-flag-D3QTHGL6.js → feature-flag-S5B5GLPP.js} +77 -33
  18. package/dist/feature-flag-S5B5GLPP.js.map +1 -0
  19. package/dist/index.js +26 -12
  20. package/dist/index.js.map +1 -1
  21. package/dist/{issues-6MHRFKTU.js → issues-4ZEDHPLW.js} +5 -4
  22. package/dist/{issues-6MHRFKTU.js.map → issues-4ZEDHPLW.js.map} +1 -1
  23. package/dist/{knobs-ED6LXBVM.js → knobs-RKUVK3HC.js} +5 -4
  24. package/dist/{knobs-ED6LXBVM.js.map → knobs-RKUVK3HC.js.map} +1 -1
  25. package/dist/{local-browser-YSE3XCUW.js → local-browser-J6WGFLVD.js} +5 -5
  26. package/dist/mcp-HGYBMDYZ.js +687 -0
  27. package/dist/mcp-HGYBMDYZ.js.map +1 -0
  28. package/dist/{psql-U5LF6ELS.js → psql-WVIHMC6A.js} +4 -3
  29. package/dist/{psql-U5LF6ELS.js.map → psql-WVIHMC6A.js.map} +1 -1
  30. package/dist/record-X4SVNYP3.js +334 -0
  31. package/dist/record-X4SVNYP3.js.map +1 -0
  32. package/dist/{redis-PBQZGU6T.js → redis-RGHECKV5.js} +4 -3
  33. package/dist/{redis-PBQZGU6T.js.map → redis-RGHECKV5.js.map} +1 -1
  34. package/dist/{release-QBSP474D.js → release-ZOD4Y2BF.js} +4 -3
  35. package/dist/{release-QBSP474D.js.map → release-ZOD4Y2BF.js.map} +1 -1
  36. package/dist/runner/preload.js +3 -2
  37. package/dist/runner/preload.js.map +1 -1
  38. package/dist/test.js +3 -2
  39. package/dist/test.js.map +1 -1
  40. package/package.json +2 -2
  41. package/dist/chunk-OE6O7H45.js +0 -92
  42. package/dist/chunk-OE6O7H45.js.map +0 -1
  43. package/dist/chunk-PLDDJCW6.js +0 -49
  44. package/dist/chunk-ZTWHPIXU.js +0 -42798
  45. package/dist/chunk-ZTWHPIXU.js.map +0 -1
  46. package/dist/dist-6NXLJYRZ.js +0 -335
  47. package/dist/dist-6NXLJYRZ.js.map +0 -1
  48. package/dist/feature-flag-D3QTHGL6.js.map +0 -1
  49. package/dist/mcp-LKHFYMA6.js +0 -385
  50. package/dist/mcp-LKHFYMA6.js.map +0 -1
  51. package/dist/pdf-extract-YIDRKYUD.js +0 -12
  52. package/dist/pdf-extract-YIDRKYUD.js.map +0 -1
  53. package/dist/pdfjs-44AOKLEM.js +0 -35242
  54. package/dist/pdfjs-44AOKLEM.js.map +0 -1
  55. /package/dist/{chunk-PLDDJCW6.js.map → chunk-VLFUCAPZ.js.map} +0 -0
  56. /package/dist/{debug-workflow-PMLMWKWI.js.map → debug-workflow-PT3OUR3V.js.map} +0 -0
  57. /package/dist/{local-browser-YSE3XCUW.js.map → local-browser-J6WGFLVD.js.map} +0 -0
@@ -0,0 +1,687 @@
1
+ import { createRequire as __cr } from "module"; const require = __cr(import.meta.url);
2
+ import {
3
+ connectTunnel,
4
+ createLocalRun,
5
+ createTunnel
6
+ } from "./chunk-DXIAHB72.js";
7
+ import {
8
+ LocalBrowserHost
9
+ } from "./chunk-FIQBGAKW.js";
10
+ import {
11
+ readStoredToken
12
+ } from "./chunk-PWWQGYFG.js";
13
+ import "./chunk-VLFUCAPZ.js";
14
+
15
+ // src/mcp/index.ts
16
+ import { Server } from "@modelcontextprotocol/sdk/server/index.js";
17
+ import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js";
18
+ import { CallToolRequestSchema, ListToolsRequestSchema } from "@modelcontextprotocol/sdk/types.js";
19
+
20
+ // src/mcp/tool-helpers.ts
21
+ function toolText(text) {
22
+ return { content: [{ type: "text", text }] };
23
+ }
24
+ function toolJson(data) {
25
+ return { content: [{ type: "text", text: JSON.stringify(data, null, 2) }] };
26
+ }
27
+ function formatMCPContent(result) {
28
+ if (typeof result === "string") {
29
+ return { content: [{ type: "text", text: result }] };
30
+ }
31
+ const content = [];
32
+ if (result.text) {
33
+ content.push({ type: "text", text: result.text });
34
+ }
35
+ for (const img of result.images ?? []) {
36
+ content.push({ type: "image", data: img.data, mimeType: img.mimeType });
37
+ }
38
+ return { content };
39
+ }
40
+
41
+ // src/mcp/session-tools.ts
42
+ import { createParser } from "eventsource-parser";
43
+ import process2 from "process";
44
+ var browserSessions = /* @__PURE__ */ new Map();
45
+ var DEFAULT_API_URL = "https://api.trycanary.ai";
46
+ function resolveApiUrl(input) {
47
+ return input ?? process2.env.CANARY_API_URL ?? DEFAULT_API_URL;
48
+ }
49
+ async function resolveToken() {
50
+ const token = process2.env.CANARY_API_TOKEN ?? await readStoredToken();
51
+ if (!token) {
52
+ throw new Error("Missing token. Run `canary login` first or set CANARY_API_TOKEN.");
53
+ }
54
+ return token;
55
+ }
56
+ function getSessionToolDefinitions() {
57
+ return [
58
+ {
59
+ name: "local_run_tests",
60
+ description: "Start an async local test run. A tunnel is opened automatically. Returns runId and watchUrl.",
61
+ inputSchema: {
62
+ type: "object",
63
+ properties: {
64
+ port: { type: "number" },
65
+ instructions: { type: "string" },
66
+ title: { type: "string" }
67
+ },
68
+ required: ["port", "instructions"]
69
+ }
70
+ },
71
+ {
72
+ name: "local_wait_for_results",
73
+ description: "Wait for a local test run to complete. Streams until completion and returns a compact report.",
74
+ inputSchema: {
75
+ type: "object",
76
+ properties: {
77
+ runId: { type: "string" }
78
+ },
79
+ required: ["runId"]
80
+ }
81
+ },
82
+ {
83
+ name: "local_browser_start",
84
+ description: "Start a local browser session that connects to the cloud agent. The cloud agent can then control this browser to test local applications. Returns sessionId for tracking.",
85
+ inputSchema: {
86
+ type: "object",
87
+ properties: {
88
+ mode: {
89
+ type: "string",
90
+ enum: ["playwright", "cdp"],
91
+ description: "Browser mode: 'playwright' for fresh browser, 'cdp' to connect to existing Chrome"
92
+ },
93
+ cdpUrl: {
94
+ type: "string",
95
+ description: "CDP endpoint URL when mode is 'cdp' (e.g. http://localhost:9222)"
96
+ },
97
+ headless: {
98
+ type: "boolean",
99
+ description: "Run browser headless (default: true for playwright mode)"
100
+ },
101
+ storageStatePath: {
102
+ type: "string",
103
+ description: "Path to Playwright storage state JSON for pre-authenticated sessions"
104
+ },
105
+ instructions: {
106
+ type: "string",
107
+ description: "Instructions for the cloud agent on what to test"
108
+ }
109
+ }
110
+ }
111
+ },
112
+ {
113
+ name: "local_browser_status",
114
+ description: "Check the status of a local browser session.",
115
+ inputSchema: {
116
+ type: "object",
117
+ properties: {
118
+ sessionId: { type: "string" }
119
+ },
120
+ required: ["sessionId"]
121
+ }
122
+ },
123
+ {
124
+ name: "local_browser_stop",
125
+ description: "Stop a local browser session and close the browser.",
126
+ inputSchema: {
127
+ type: "object",
128
+ properties: {
129
+ sessionId: { type: "string" }
130
+ },
131
+ required: ["sessionId"]
132
+ }
133
+ },
134
+ {
135
+ name: "local_browser_list",
136
+ description: "List all active local browser sessions.",
137
+ inputSchema: {
138
+ type: "object",
139
+ properties: {}
140
+ }
141
+ },
142
+ {
143
+ name: "local_browser_run",
144
+ description: "Start a test run on an active local browser session. The cloud agent will control the local browser according to the instructions.",
145
+ inputSchema: {
146
+ type: "object",
147
+ properties: {
148
+ sessionId: {
149
+ type: "string",
150
+ description: "The session ID from local_browser_start"
151
+ },
152
+ instructions: {
153
+ type: "string",
154
+ description: "Instructions for the cloud agent on what to test"
155
+ },
156
+ startUrl: {
157
+ type: "string",
158
+ description: "Optional URL to navigate to before starting"
159
+ }
160
+ },
161
+ required: ["sessionId", "instructions"]
162
+ }
163
+ }
164
+ ];
165
+ }
166
+ async function handleSessionTool(tool, args) {
167
+ const token = await resolveToken();
168
+ if (tool === "local_run_tests") {
169
+ const input = args;
170
+ const apiUrl = resolveApiUrl();
171
+ const tunnel = await createTunnel({ apiUrl, token, port: input.port });
172
+ connectTunnel({
173
+ apiUrl,
174
+ tunnelId: tunnel.tunnelId,
175
+ token: tunnel.token,
176
+ port: input.port
177
+ });
178
+ const tunnelUrl = tunnel.publicUrl;
179
+ const run = await createLocalRun({
180
+ apiUrl,
181
+ token,
182
+ title: input.title ?? "Local MCP run",
183
+ featureSpec: input.instructions,
184
+ startUrl: void 0,
185
+ tunnelUrl
186
+ });
187
+ return toolJson({
188
+ runId: run.runId,
189
+ watchUrl: run.watchUrl,
190
+ tunnelUrl,
191
+ note: "Testing is asynchronous. Use local_wait_for_results with the runId to wait for completion."
192
+ });
193
+ }
194
+ if (tool === "local_wait_for_results") {
195
+ const input = args;
196
+ const apiUrl = resolveApiUrl();
197
+ const report = await waitForResult({ apiUrl, token, runId: input.runId });
198
+ return toolJson(report);
199
+ }
200
+ if (tool === "local_browser_start") {
201
+ const input = args;
202
+ const apiUrl = resolveApiUrl();
203
+ const mode = input.mode ?? "playwright";
204
+ const sessionResponse = await fetch(`${apiUrl}/local-browser/sessions`, {
205
+ method: "POST",
206
+ headers: {
207
+ "Content-Type": "application/json",
208
+ Authorization: `Bearer ${token}`
209
+ },
210
+ body: JSON.stringify({
211
+ browserMode: mode,
212
+ instructions: input.instructions ?? null
213
+ })
214
+ });
215
+ if (!sessionResponse.ok) {
216
+ const text = await sessionResponse.text();
217
+ return toolJson({ ok: false, error: `Failed to create session: ${text}` });
218
+ }
219
+ const session = await sessionResponse.json();
220
+ const host = new LocalBrowserHost({
221
+ apiUrl,
222
+ wsToken: session.wsToken,
223
+ sessionId: session.sessionId,
224
+ browserMode: mode,
225
+ cdpUrl: input.cdpUrl,
226
+ headless: input.headless ?? true,
227
+ storageStatePath: input.storageStatePath,
228
+ onLog: (level, message) => {
229
+ if (level === "error") {
230
+ console.error(`[LocalBrowser] ${message}`);
231
+ }
232
+ }
233
+ });
234
+ host.start().catch((err) => {
235
+ console.error("Failed to start local browser:", err);
236
+ browserSessions.delete(session.sessionId);
237
+ });
238
+ browserSessions.set(session.sessionId, {
239
+ sessionId: session.sessionId,
240
+ host,
241
+ startedAt: Date.now(),
242
+ mode
243
+ });
244
+ return toolJson({
245
+ ok: true,
246
+ sessionId: session.sessionId,
247
+ mode,
248
+ expiresAt: session.expiresAt,
249
+ note: "Browser session started. The cloud agent can now control this browser. Use local_browser_stop to end the session."
250
+ });
251
+ }
252
+ if (tool === "local_browser_status") {
253
+ const input = args;
254
+ const session = browserSessions.get(input.sessionId);
255
+ if (!session) {
256
+ return toolJson({ ok: false, error: "Session not found", sessionId: input.sessionId });
257
+ }
258
+ return toolJson({
259
+ ok: true,
260
+ sessionId: session.sessionId,
261
+ mode: session.mode,
262
+ startedAt: new Date(session.startedAt).toISOString(),
263
+ uptimeMs: Date.now() - session.startedAt
264
+ });
265
+ }
266
+ if (tool === "local_browser_stop") {
267
+ const input = args;
268
+ const session = browserSessions.get(input.sessionId);
269
+ if (!session) {
270
+ return toolJson({ ok: false, error: "Session not found", sessionId: input.sessionId });
271
+ }
272
+ await session.host.stop();
273
+ browserSessions.delete(input.sessionId);
274
+ return toolJson({
275
+ ok: true,
276
+ sessionId: input.sessionId,
277
+ note: "Browser session stopped and browser closed."
278
+ });
279
+ }
280
+ if (tool === "local_browser_list") {
281
+ const sessions = Array.from(browserSessions.values()).map((s) => ({
282
+ sessionId: s.sessionId,
283
+ mode: s.mode,
284
+ startedAt: new Date(s.startedAt).toISOString(),
285
+ uptimeMs: Date.now() - s.startedAt
286
+ }));
287
+ return toolJson({
288
+ ok: true,
289
+ count: sessions.length,
290
+ sessions
291
+ });
292
+ }
293
+ if (tool === "local_browser_run") {
294
+ const input = args;
295
+ const apiUrl = resolveApiUrl();
296
+ const session = browserSessions.get(input.sessionId);
297
+ if (!session) {
298
+ return toolJson({
299
+ ok: false,
300
+ error: "Session not found locally. Make sure you started it with local_browser_start.",
301
+ sessionId: input.sessionId
302
+ });
303
+ }
304
+ const response = await fetch(`${apiUrl}/local-browser/sessions/${input.sessionId}/run`, {
305
+ method: "POST",
306
+ headers: {
307
+ "Content-Type": "application/json",
308
+ Authorization: `Bearer ${token}`
309
+ },
310
+ body: JSON.stringify({
311
+ instructions: input.instructions,
312
+ startUrl: input.startUrl ?? null
313
+ })
314
+ });
315
+ if (!response.ok) {
316
+ const text = await response.text();
317
+ return toolJson({ ok: false, error: `Failed to start run: ${text}` });
318
+ }
319
+ const result = await response.json();
320
+ return toolJson({
321
+ ok: true,
322
+ jobId: result.jobId,
323
+ sessionId: result.sessionId,
324
+ note: "Test run started. The cloud agent is now controlling your local browser. You can watch the browser to see the test in action."
325
+ });
326
+ }
327
+ return null;
328
+ }
329
+ async function waitForResult(input) {
330
+ await streamUntilComplete(input);
331
+ const response = await fetch(`${input.apiUrl}/local-tests/runs/${input.runId}`, {
332
+ credentials: "include",
333
+ headers: { authorization: `Bearer ${input.token}` }
334
+ });
335
+ const data = await response.json();
336
+ const run = data?.data?.run ?? data?.run ?? data?.data;
337
+ const summary = run?.summaryJson;
338
+ return formatReport({ run, summary });
339
+ }
340
+ async function streamUntilComplete(input) {
341
+ const response = await fetch(`${input.apiUrl}/local-tests/runs/${input.runId}/stream`, {
342
+ headers: { authorization: `Bearer ${input.token}` }
343
+ });
344
+ if (!response.body) return;
345
+ const reader = response.body.getReader();
346
+ const decoder = new TextDecoder();
347
+ const parser = createParser({
348
+ onEvent: (event) => {
349
+ if (event.event === "status") {
350
+ try {
351
+ const payload = JSON.parse(event.data);
352
+ if (payload?.status === "completed" || payload?.status === "failed") {
353
+ reader.cancel().catch(() => void 0);
354
+ }
355
+ } catch {
356
+ }
357
+ }
358
+ if (event.event === "complete" || event.event === "error") {
359
+ reader.cancel().catch(() => void 0);
360
+ }
361
+ }
362
+ });
363
+ while (true) {
364
+ const { done, value } = await reader.read();
365
+ if (done) break;
366
+ parser.feed(decoder.decode(value, { stream: true }));
367
+ }
368
+ }
369
+ function formatReport(input) {
370
+ if (!input.summary) {
371
+ return {
372
+ runId: input.run?.id,
373
+ status: input.run?.status ?? "unknown",
374
+ summary: "No final report available."
375
+ };
376
+ }
377
+ const tested = Array.isArray(input.summary.testedItems) ? input.summary.testedItems : [];
378
+ const status = input.summary.status ?? input.run?.status ?? "unknown";
379
+ const issues = status === "issues_found" ? input.summary.notes ? [input.summary.notes] : ["Issues reported."] : [];
380
+ return {
381
+ runId: input.run?.id,
382
+ status,
383
+ summary: input.summary.summary ?? "Run completed.",
384
+ testedItems: tested,
385
+ issues,
386
+ notes: input.summary.notes ?? null
387
+ };
388
+ }
389
+
390
+ // src/mcp/browser-tools.ts
391
+ import { getBrowserToolDefinitionsWithLifecycle } from "@chatsdet/browser-core";
392
+
393
+ // src/mcp/browser-session.ts
394
+ import {
395
+ PlaywrightClient,
396
+ BrowserToolExecutor,
397
+ consoleLogger
398
+ } from "@chatsdet/browser-core";
399
+ var activeSession = null;
400
+ async function startDirectBrowser(opts) {
401
+ if (activeSession) {
402
+ throw new Error("A browser session is already active. Stop it first with browser_stop.");
403
+ }
404
+ const client = new PlaywrightClient({
405
+ logger: consoleLogger
406
+ });
407
+ await client.connect({
408
+ browserMode: opts?.headless ? "headless" : "headed",
409
+ storageStatePath: opts?.storageStatePath
410
+ });
411
+ const executor = new BrowserToolExecutor(client, {
412
+ autoSnapshotAfterAction: true,
413
+ includeScreenshotWithSnapshot: true,
414
+ clickValidation: true,
415
+ logger: consoleLogger,
416
+ ...opts?.executorOptions
417
+ });
418
+ activeSession = {
419
+ client,
420
+ executor,
421
+ startedAt: Date.now()
422
+ };
423
+ const cleanup = async () => {
424
+ if (activeSession) {
425
+ try {
426
+ await activeSession.client.disconnect();
427
+ } catch {
428
+ }
429
+ activeSession = null;
430
+ }
431
+ };
432
+ process.on("exit", () => {
433
+ cleanup();
434
+ });
435
+ process.on("SIGINT", async () => {
436
+ await cleanup();
437
+ process.exit(0);
438
+ });
439
+ process.on("SIGTERM", async () => {
440
+ await cleanup();
441
+ process.exit(0);
442
+ });
443
+ return activeSession;
444
+ }
445
+ async function stopDirectBrowser() {
446
+ if (!activeSession) {
447
+ throw new Error("No active browser session to stop.");
448
+ }
449
+ await activeSession.client.disconnect();
450
+ activeSession = null;
451
+ }
452
+ function requireDirectBrowser() {
453
+ if (!activeSession) {
454
+ throw new Error("No active browser session. Start one with browser_start first.");
455
+ }
456
+ return activeSession;
457
+ }
458
+
459
+ // src/mcp/browser-tools.ts
460
+ function getBrowserMCPToolDefinitions() {
461
+ return getBrowserToolDefinitionsWithLifecycle();
462
+ }
463
+ async function handleBrowserTool(name, args) {
464
+ if (name === "browser_start") {
465
+ try {
466
+ await startDirectBrowser({
467
+ headless: args.headless ?? false,
468
+ storageStatePath: args.storageStatePath
469
+ });
470
+ return toolJson({
471
+ ok: true,
472
+ note: "Browser started. You can now use browser_navigate, browser_click, and other browser tools."
473
+ });
474
+ } catch (err) {
475
+ return toolJson({ ok: false, error: err instanceof Error ? err.message : String(err) });
476
+ }
477
+ }
478
+ if (name === "browser_stop") {
479
+ try {
480
+ await stopDirectBrowser();
481
+ return toolJson({ ok: true, note: "Browser stopped and cleaned up." });
482
+ } catch (err) {
483
+ return toolJson({ ok: false, error: err instanceof Error ? err.message : String(err) });
484
+ }
485
+ }
486
+ if (!name.startsWith("browser_")) return null;
487
+ let session;
488
+ try {
489
+ session = requireDirectBrowser();
490
+ } catch (err) {
491
+ return toolJson({ ok: false, error: err instanceof Error ? err.message : String(err) });
492
+ }
493
+ const executor = session.executor;
494
+ try {
495
+ switch (name) {
496
+ case "browser_navigate":
497
+ return formatMCPContent(await executor.navigate(args.url));
498
+ case "browser_navigate_back":
499
+ return formatMCPContent(await executor.navigateBack());
500
+ case "browser_snapshot":
501
+ return formatMCPContent(
502
+ await executor.snapshot({
503
+ mode: args.mode,
504
+ expand: args.expand,
505
+ search: args.search,
506
+ showCoordinateGrid: args.showCoordinateGrid,
507
+ probeAt: args.probeAt
508
+ })
509
+ );
510
+ case "browser_screenshot":
511
+ return formatMCPContent(
512
+ await executor.screenshot({
513
+ fullPage: args.fullPage,
514
+ element: args.element,
515
+ ref: args.ref,
516
+ label: args.label,
517
+ returnImage: args.returnImage
518
+ })
519
+ );
520
+ case "browser_evaluate":
521
+ return formatMCPContent(
522
+ await executor.evaluate({
523
+ expression: args.function,
524
+ element: args.element,
525
+ ref: args.ref
526
+ })
527
+ );
528
+ case "browser_console_messages":
529
+ return formatMCPContent(
530
+ await executor.consoleMessages({ onlyErrors: args.onlyErrors })
531
+ );
532
+ case "browser_network_requests":
533
+ return formatMCPContent(await executor.networkRequests());
534
+ case "browser_click":
535
+ return formatMCPContent(
536
+ await executor.click({
537
+ ref: args.ref,
538
+ x: args.x,
539
+ y: args.y,
540
+ element: args.element,
541
+ doubleClick: args.doubleClick,
542
+ button: args.button,
543
+ modifiers: args.modifiers
544
+ })
545
+ );
546
+ case "browser_hover":
547
+ return formatMCPContent(
548
+ await executor.hover({
549
+ ref: args.ref,
550
+ element: args.element
551
+ })
552
+ );
553
+ case "browser_drag":
554
+ return formatMCPContent(
555
+ await executor.drag({
556
+ startRef: args.startRef,
557
+ endRef: args.endRef,
558
+ startElement: args.startElement,
559
+ endElement: args.endElement,
560
+ startX: args.startX,
561
+ startY: args.startY,
562
+ endX: args.endX,
563
+ endY: args.endY
564
+ })
565
+ );
566
+ case "browser_type":
567
+ return formatMCPContent(
568
+ await executor.type({
569
+ ref: args.ref,
570
+ text: args.text,
571
+ element: args.element,
572
+ submit: args.submit,
573
+ delay: args.delay
574
+ })
575
+ );
576
+ case "browser_press_key":
577
+ return formatMCPContent(await executor.pressKey(args.key));
578
+ case "browser_fill_form":
579
+ return formatMCPContent(
580
+ await executor.fillForm(
581
+ args.fields
582
+ )
583
+ );
584
+ case "browser_select_option":
585
+ return formatMCPContent(
586
+ await executor.selectOption({
587
+ ref: args.ref,
588
+ value: args.value,
589
+ element: args.element
590
+ })
591
+ );
592
+ case "browser_file_upload":
593
+ return formatMCPContent(await executor.fileUpload(args.paths));
594
+ case "browser_scroll":
595
+ return formatMCPContent(
596
+ await executor.scroll({
597
+ direction: args.direction,
598
+ amount: args.amount,
599
+ withinRef: args.withinRef,
600
+ toRef: args.toRef,
601
+ x: args.x,
602
+ y: args.y
603
+ })
604
+ );
605
+ case "browser_handle_dialog":
606
+ return formatMCPContent(
607
+ await executor.handleDialog({
608
+ action: args.action,
609
+ promptText: args.promptText
610
+ })
611
+ );
612
+ case "browser_dismiss_overlay":
613
+ return formatMCPContent(
614
+ await executor.dismissOverlay({
615
+ preferredStrategy: args.preferredStrategy
616
+ })
617
+ );
618
+ case "browser_wait_for":
619
+ return formatMCPContent(
620
+ await executor.waitFor({
621
+ timeSec: args.time,
622
+ text: args.text,
623
+ textGone: args.textGone,
624
+ selector: args.selector,
625
+ state: args.state,
626
+ timeout: args.timeout
627
+ })
628
+ );
629
+ case "browser_close":
630
+ return formatMCPContent(await executor.close());
631
+ case "browser_resize":
632
+ return formatMCPContent(
633
+ await executor.resize(args.width, args.height)
634
+ );
635
+ case "browser_tabs":
636
+ return formatMCPContent(
637
+ await executor.tabs({
638
+ action: args.action,
639
+ index: args.index
640
+ })
641
+ );
642
+ case "browser_list_downloads":
643
+ return formatMCPContent(await executor.listDownloads());
644
+ case "browser_read_download":
645
+ return formatMCPContent(
646
+ await executor.readDownload(args.downloadId)
647
+ );
648
+ case "browser_pdf_read":
649
+ return formatMCPContent(await executor.fetchPdfText(args.url));
650
+ default:
651
+ return toolText(`Unknown browser tool: ${name}`);
652
+ }
653
+ } catch (err) {
654
+ const message = err instanceof Error ? err.message : String(err);
655
+ return toolJson({ ok: false, error: message, tool: name });
656
+ }
657
+ }
658
+
659
+ // src/mcp/index.ts
660
+ async function runMcp(argv) {
661
+ const server = new Server(
662
+ { name: "canary-cli", version: "0.1.0" },
663
+ { capabilities: { tools: {} } }
664
+ );
665
+ server.setRequestHandler(ListToolsRequestSchema, async () => ({
666
+ tools: [
667
+ ...getSessionToolDefinitions(),
668
+ ...getBrowserMCPToolDefinitions()
669
+ ]
670
+ }));
671
+ server.setRequestHandler(CallToolRequestSchema, async (req) => {
672
+ const tool = req.params.name;
673
+ const args = req.params.arguments ?? {};
674
+ const browserResult = await handleBrowserTool(tool, args);
675
+ if (browserResult) return browserResult;
676
+ const sessionResult = await handleSessionTool(tool, args);
677
+ if (sessionResult) return sessionResult;
678
+ return toolText(`Unknown tool: ${tool}`);
679
+ });
680
+ const transport = new StdioServerTransport();
681
+ await server.connect(transport);
682
+ return new Promise(() => void 0);
683
+ }
684
+ export {
685
+ runMcp
686
+ };
687
+ //# sourceMappingURL=mcp-HGYBMDYZ.js.map