@canaryai/cli 0.2.12 → 0.2.13

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 (73) hide show
  1. package/dist/{chunk-PWWQGYFG.js → chunk-ACRIE2YR.js} +5 -2
  2. package/dist/chunk-ACRIE2YR.js.map +1 -0
  3. package/dist/{chunk-ERSNYLMZ.js → chunk-BOS2YLKH.js} +12 -8
  4. package/dist/chunk-BOS2YLKH.js.map +1 -0
  5. package/dist/{chunk-MSMC6UXW.js → chunk-IFOJT3A5.js} +1002 -253
  6. package/dist/chunk-IFOJT3A5.js.map +1 -0
  7. package/dist/{chunk-Q7WFBG5C.js → chunk-SVU2XTYZ.js} +19 -5
  8. package/dist/chunk-SVU2XTYZ.js.map +1 -0
  9. package/dist/{chunk-A44B2PEA.js → chunk-SYPQF57S.js} +40 -8
  10. package/dist/chunk-SYPQF57S.js.map +1 -0
  11. package/dist/{chunk-CEW4BDXD.js → chunk-Z3F373YR.js} +13 -6
  12. package/dist/chunk-Z3F373YR.js.map +1 -0
  13. package/dist/{debug-workflow-53ULOFJC.js → debug-workflow-K2LL6CO4.js} +10 -12
  14. package/dist/debug-workflow-K2LL6CO4.js.map +1 -0
  15. package/dist/{docs-BEE3LOCO.js → docs-SR7CW24Y.js} +19 -14
  16. package/dist/docs-SR7CW24Y.js.map +1 -0
  17. package/dist/{feature-flag-CYTDV4ZB.js → feature-flag-BIPFVVNC.js} +3 -3
  18. package/dist/index.d.ts +2 -2
  19. package/dist/index.js +36 -30
  20. package/dist/index.js.map +1 -1
  21. package/dist/{init-M6I3MG3D.js → init-KXAVWHYE.js} +4 -4
  22. package/dist/{issues-NLM72HLU.js → issues-EWVB52CA.js} +37 -18
  23. package/dist/issues-EWVB52CA.js.map +1 -0
  24. package/dist/{knobs-O35GAU5M.js → knobs-VYABZESR.js} +3 -3
  25. package/dist/{list-4K4EIGAT.js → list-RCPYLS36.js} +3 -3
  26. package/dist/list-RCPYLS36.js.map +1 -0
  27. package/dist/{local-NHXXPHZ3.js → local-34FX3M5K.js} +6 -6
  28. package/dist/{local-browser-VAZORCO3.js → local-browser-VPOSJS52.js} +4 -4
  29. package/dist/{login-ZLP64YQP.js → login-MSIM2VIH.js} +4 -4
  30. package/dist/{mcp-ZF5G5DCB.js → mcp-YBR7G254.js} +7 -10
  31. package/dist/mcp-YBR7G254.js.map +1 -0
  32. package/dist/{psql-2YPIRMDY.js → psql-XO5BB5L5.js} +2 -2
  33. package/dist/{record-V6QKFFH3.js → record-DXXQHPGT.js} +7 -7
  34. package/dist/record-DXXQHPGT.js.map +1 -0
  35. package/dist/{redis-A7GWM23E.js → redis-CQTBPZ6F.js} +2 -2
  36. package/dist/{release-7TI7EIGD.js → release-DW7RPQSQ.js} +2 -2
  37. package/dist/runner/preload.js +1 -1
  38. package/dist/{session-UGNJXRUW.js → session-XQGCLWNC.js} +33 -12
  39. package/dist/{session-UGNJXRUW.js.map → session-XQGCLWNC.js.map} +1 -1
  40. package/dist/{skill-ORWAPBDW.js → skill-2TXI3IKP.js} +1 -1
  41. package/dist/skill-2TXI3IKP.js.map +1 -0
  42. package/dist/{src-4VIDSK4A.js → src-F7LQ5PY2.js} +8 -2
  43. package/dist/{start-E532F3BU.js → start-ZOMUD6LW.js} +4 -4
  44. package/dist/test.js +1 -1
  45. package/dist/test.js.map +1 -1
  46. package/dist/{workflow-HXIUXRFI.js → workflow-5UZTKX7X.js} +21 -10
  47. package/dist/workflow-5UZTKX7X.js.map +1 -0
  48. package/package.json +1 -2
  49. package/dist/chunk-A44B2PEA.js.map +0 -1
  50. package/dist/chunk-CEW4BDXD.js.map +0 -1
  51. package/dist/chunk-ERSNYLMZ.js.map +0 -1
  52. package/dist/chunk-MSMC6UXW.js.map +0 -1
  53. package/dist/chunk-PWWQGYFG.js.map +0 -1
  54. package/dist/chunk-Q7WFBG5C.js.map +0 -1
  55. package/dist/debug-workflow-53ULOFJC.js.map +0 -1
  56. package/dist/docs-BEE3LOCO.js.map +0 -1
  57. package/dist/issues-NLM72HLU.js.map +0 -1
  58. package/dist/list-4K4EIGAT.js.map +0 -1
  59. package/dist/mcp-ZF5G5DCB.js.map +0 -1
  60. package/dist/record-V6QKFFH3.js.map +0 -1
  61. package/dist/skill-ORWAPBDW.js.map +0 -1
  62. package/dist/workflow-HXIUXRFI.js.map +0 -1
  63. /package/dist/{feature-flag-CYTDV4ZB.js.map → feature-flag-BIPFVVNC.js.map} +0 -0
  64. /package/dist/{init-M6I3MG3D.js.map → init-KXAVWHYE.js.map} +0 -0
  65. /package/dist/{knobs-O35GAU5M.js.map → knobs-VYABZESR.js.map} +0 -0
  66. /package/dist/{local-NHXXPHZ3.js.map → local-34FX3M5K.js.map} +0 -0
  67. /package/dist/{local-browser-VAZORCO3.js.map → local-browser-VPOSJS52.js.map} +0 -0
  68. /package/dist/{login-ZLP64YQP.js.map → login-MSIM2VIH.js.map} +0 -0
  69. /package/dist/{psql-2YPIRMDY.js.map → psql-XO5BB5L5.js.map} +0 -0
  70. /package/dist/{redis-A7GWM23E.js.map → redis-CQTBPZ6F.js.map} +0 -0
  71. /package/dist/{release-7TI7EIGD.js.map → release-DW7RPQSQ.js.map} +0 -0
  72. /package/dist/{src-4VIDSK4A.js.map → src-F7LQ5PY2.js.map} +0 -0
  73. /package/dist/{start-E532F3BU.js.map → start-ZOMUD6LW.js.map} +0 -0
@@ -1,7 +1,7 @@
1
1
  import { createRequire as __cr } from "module"; const require = __cr(import.meta.url);
2
2
  import {
3
3
  PlaywrightClient
4
- } from "./chunk-MSMC6UXW.js";
4
+ } from "./chunk-IFOJT3A5.js";
5
5
 
6
6
  // src/local-browser/host.ts
7
7
  var HEARTBEAT_INTERVAL_MS = 3e4;
@@ -314,10 +314,20 @@ var LocalBrowserHost = class {
314
314
  return client.fileUpload(args[0], args[1]);
315
315
  // Scroll
316
316
  case "scroll":
317
- return client.scroll(args[0], args[1], args[2], args[3], args[4]);
317
+ return client.scroll(
318
+ args[0],
319
+ args[1],
320
+ args[2],
321
+ args[3],
322
+ args[4]
323
+ );
318
324
  // Dialogs
319
325
  case "handleDialog":
320
- return client.handleDialog(args[0], args[1], args[2]);
326
+ return client.handleDialog(
327
+ args[0],
328
+ args[1],
329
+ args[2]
330
+ );
321
331
  // Waiting
322
332
  case "waitFor":
323
333
  return client.waitFor(args[0]);
@@ -327,7 +337,11 @@ var LocalBrowserHost = class {
327
337
  case "resize":
328
338
  return client.resize(args[0], args[1], args[2]);
329
339
  case "tabs":
330
- return client.tabs(args[0], args[1], args[2]);
340
+ return client.tabs(
341
+ args[0],
342
+ args[1],
343
+ args[2]
344
+ );
331
345
  // Context Management
332
346
  case "swapContext":
333
347
  return client.swapContext?.(args[0]);
@@ -370,4 +384,4 @@ var LocalBrowserHost = class {
370
384
  export {
371
385
  LocalBrowserHost
372
386
  };
373
- //# sourceMappingURL=chunk-Q7WFBG5C.js.map
387
+ //# sourceMappingURL=chunk-SVU2XTYZ.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/local-browser/host.ts"],"sourcesContent":["/**\n * Local Browser Host\n *\n * Manages a local browser instance and handles commands from the cloud API\n * via WebSocket. Delegates all browser automation to PlaywrightClient from\n * @chatsdet/browser-core, ensuring the agent experience is identical\n * regardless of whether the browser is local or cloud.\n *\n * @module local-browser-host\n */\n\nimport { PlaywrightClient, type IBrowserClient, type BrowserLogger } from '@chatsdet/browser-core';\nimport type {\n BrowserCommand,\n BrowserResponse,\n HeartbeatMessage,\n SessionMessage,\n LocalBrowserMode,\n} from './protocol';\n\nconst HEARTBEAT_INTERVAL_MS = 30_000;\nconst RECONNECT_DELAY_MS = 1000;\nconst MAX_RECONNECT_DELAY_MS = 30_000;\nconst MAX_RECONNECT_ATTEMPTS = 10;\n\ninterface LocalBrowserHostOptions {\n apiUrl: string;\n wsToken: string;\n sessionId: string;\n browserMode: LocalBrowserMode;\n cdpUrl?: string;\n headless?: boolean;\n storageStatePath?: string;\n onLog?: (level: 'info' | 'warn' | 'error' | 'debug', message: string, data?: unknown) => void;\n}\n\n/**\n * LocalBrowserHost manages the WebSocket connection to the cloud API and\n * delegates all browser operations to a shared PlaywrightClient instance.\n */\nexport class LocalBrowserHost {\n private options: LocalBrowserHostOptions;\n private ws: WebSocket | null = null;\n private client: PlaywrightClient;\n private heartbeatTimer: NodeJS.Timeout | null = null;\n private reconnectAttempts = 0;\n private isShuttingDown = false;\n\n constructor(options: LocalBrowserHostOptions) {\n this.options = options;\n const logger: BrowserLogger = {\n debug: (msg: string, data?: Record<string, unknown>) => this.log('debug', msg, data),\n info: (msg: string, data?: Record<string, unknown>) => this.log('info', msg, data),\n warn: (msg: string, data?: Record<string, unknown>) => this.log('warn', msg, data),\n error: (msg: string, data?: Record<string, unknown>) => this.log('error', msg, data),\n };\n this.client = new PlaywrightClient({ logger });\n }\n\n private log(level: 'info' | 'warn' | 'error' | 'debug', message: string, data?: unknown) {\n if (this.options.onLog) {\n this.options.onLog(level, message, data);\n } else {\n const fn = level === 'error' ? console.error : level === 'warn' ? console.warn : console.log;\n fn(`[LocalBrowserHost] ${message}`, data ?? '');\n }\n }\n\n // =========================================================================\n // Lifecycle\n // =========================================================================\n\n async start(): Promise<void> {\n this.log('info', 'Starting local browser host', {\n browserMode: this.options.browserMode,\n sessionId: this.options.sessionId,\n });\n\n // Connect to WebSocket first\n await this.connectWebSocket();\n\n // Launch browser via PlaywrightClient\n const { browserMode, cdpUrl, headless = true, storageStatePath } = this.options;\n await this.client.connect({\n browserMode: headless ? 'headless' : 'headed',\n cdpUrl: browserMode === 'cdp' ? cdpUrl : undefined,\n storageStatePath,\n });\n\n // Notify cloud that browser is ready\n this.sendSessionEvent('browser_ready');\n }\n\n async stop(): Promise<void> {\n this.isShuttingDown = true;\n this.log('info', 'Stopping local browser host');\n\n this.stopHeartbeat();\n\n if (this.ws) {\n try {\n this.ws.close(1000, 'Shutdown');\n } catch {\n /* intentionally empty */\n }\n this.ws = null;\n }\n\n await this.client.disconnect();\n this.log('info', 'Local browser host stopped');\n }\n\n // =========================================================================\n // WebSocket Connection\n // =========================================================================\n\n private async connectWebSocket(): Promise<void> {\n return new Promise((resolve, reject) => {\n const wsUrl = `${this.options.apiUrl.replace('http', 'ws')}/local-browser/sessions/${this.options.sessionId}/connect?token=${this.options.wsToken}`;\n\n this.log('info', 'Connecting to cloud API', { url: wsUrl.replace(/token=.*/, 'token=***') });\n\n const ws = new WebSocket(wsUrl);\n\n ws.onopen = () => {\n this.log('info', 'Connected to cloud API');\n this.ws = ws;\n this.reconnectAttempts = 0;\n this.startHeartbeat();\n resolve();\n };\n\n ws.onmessage = (event) => {\n this.handleMessage(event.data as string);\n };\n\n ws.onerror = (event) => {\n this.log('error', 'WebSocket error', event);\n };\n\n ws.onclose = () => {\n this.log('info', 'WebSocket closed');\n this.stopHeartbeat();\n this.ws = null;\n\n if (!this.isShuttingDown) {\n this.scheduleReconnect();\n }\n };\n\n // Timeout after 30 seconds\n setTimeout(() => {\n if (!this.ws) {\n reject(new Error('WebSocket connection timeout'));\n }\n }, 30_000);\n });\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectAttempts >= MAX_RECONNECT_ATTEMPTS) {\n this.log('error', 'Max reconnection attempts reached, giving up');\n this.stop();\n return;\n }\n\n const delay = Math.min(\n RECONNECT_DELAY_MS * Math.pow(2, this.reconnectAttempts),\n MAX_RECONNECT_DELAY_MS\n );\n\n this.reconnectAttempts++;\n this.log('info', `Reconnecting in ${delay}ms (attempt ${this.reconnectAttempts})`);\n\n setTimeout(async () => {\n try {\n await this.connectWebSocket();\n this.sendSessionEvent('connected');\n this.sendSessionEvent('browser_ready');\n } catch (error) {\n this.log('error', 'Reconnection failed', error);\n this.scheduleReconnect();\n }\n }, delay);\n }\n\n // =========================================================================\n // Heartbeat\n // =========================================================================\n\n private startHeartbeat(): void {\n this.stopHeartbeat();\n this.heartbeatTimer = setInterval(() => {\n if (this.ws?.readyState === WebSocket.OPEN) {\n const ping: HeartbeatMessage = {\n type: 'heartbeat',\n id: crypto.randomUUID(),\n timestamp: Date.now(),\n direction: 'pong',\n };\n this.ws.send(JSON.stringify(ping));\n }\n }, HEARTBEAT_INTERVAL_MS);\n }\n\n private stopHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n // =========================================================================\n // Message Handling\n // =========================================================================\n\n private handleMessage(data: string): void {\n try {\n const message = JSON.parse(data);\n\n if (message.type === 'heartbeat' && message.direction === 'ping') {\n const pong: HeartbeatMessage = {\n type: 'heartbeat',\n id: crypto.randomUUID(),\n timestamp: Date.now(),\n direction: 'pong',\n };\n this.ws?.send(JSON.stringify(pong));\n return;\n }\n\n if (message.type === 'command') {\n this.handleCommand(message as BrowserCommand);\n return;\n }\n\n this.log('debug', 'Received unknown message type', message);\n } catch (error) {\n this.log('error', 'Failed to parse message', { error, data });\n }\n }\n\n private async handleCommand(command: BrowserCommand): Promise<void> {\n const startTime = Date.now();\n this.log('debug', `Executing command: ${command.method}`, { id: command.id });\n\n try {\n const result = await this.executeMethod(command.method, command.args);\n const response: BrowserResponse = {\n type: 'response',\n id: crypto.randomUUID(),\n timestamp: Date.now(),\n requestId: command.id,\n success: true,\n result,\n contextId: command.contextId,\n };\n this.ws?.send(JSON.stringify(response));\n\n this.log('debug', `Command completed: ${command.method}`, {\n id: command.id,\n durationMs: Date.now() - startTime,\n });\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n const response: BrowserResponse = {\n type: 'response',\n id: crypto.randomUUID(),\n timestamp: Date.now(),\n requestId: command.id,\n success: false,\n error: errorMessage,\n stack: error instanceof Error ? error.stack : undefined,\n contextId: command.contextId,\n };\n this.ws?.send(JSON.stringify(response));\n\n this.log('error', `Command failed: ${command.method}`, {\n id: command.id,\n error: errorMessage,\n });\n }\n }\n\n private sendSessionEvent(\n event: 'connected' | 'disconnected' | 'browser_ready' | 'browser_closed' | 'error',\n error?: string\n ): void {\n if (!this.ws || this.ws.readyState !== WebSocket.OPEN) return;\n\n const message: SessionMessage = {\n type: 'session',\n id: crypto.randomUUID(),\n timestamp: Date.now(),\n event,\n browserMode: this.options.browserMode,\n error,\n };\n this.ws.send(JSON.stringify(message));\n }\n\n // =========================================================================\n // Method Execution — Delegate to PlaywrightClient\n // =========================================================================\n\n /**\n * Maps incoming WebSocket command method names to PlaywrightClient methods.\n * The client implements IBrowserClient, so all standard browser operations\n * are available and behave identically to the cloud environment.\n */\n private async executeMethod(method: string, args: unknown[]): Promise<unknown> {\n const client = this.client as IBrowserClient;\n\n switch (method) {\n // Lifecycle\n case 'connect':\n return client.connect(args[0] as any);\n case 'disconnect':\n return client.disconnect();\n\n // Navigation\n case 'navigate':\n return client.navigate(args[0] as string, args[1] as any);\n case 'navigateBack':\n return client.navigateBack(args[0] as any);\n\n // Page Inspection\n case 'snapshot':\n return client.snapshot(args[0] as any);\n case 'takeScreenshot':\n return client.takeScreenshot(args[0] as any);\n case 'evaluate':\n return client.evaluate(args[0] as string, args[1] as any);\n case 'runCode':\n return client.runCode(args[0] as string, args[1] as any);\n case 'consoleMessages':\n return client.consoleMessages(args[0] as any);\n case 'networkRequests':\n return client.networkRequests(args[0] as any);\n\n // Interaction\n case 'click':\n return client.click(args[0] as string, args[1] as string, args[2] as any);\n case 'clickAtCoordinates':\n return client.clickAtCoordinates(\n args[0] as number,\n args[1] as number,\n args[2] as string,\n args[3] as any\n );\n case 'moveToCoordinates':\n return client.moveToCoordinates(\n args[0] as number,\n args[1] as number,\n args[2] as string,\n args[3] as any\n );\n case 'dragCoordinates':\n return client.dragCoordinates(\n args[0] as number,\n args[1] as number,\n args[2] as number,\n args[3] as number,\n args[4] as string,\n args[5] as any\n );\n case 'hover':\n return client.hover(args[0] as string, args[1] as string, args[2] as any);\n case 'drag':\n return client.drag(\n args[0] as string,\n args[1] as string,\n args[2] as string,\n args[3] as string,\n args[4] as any\n );\n case 'type':\n return client.type(\n args[0] as string,\n args[1] as string,\n args[2] as string,\n args[3] as boolean,\n args[4] as any\n );\n case 'pressKey':\n return client.pressKey(args[0] as string, args[1] as any);\n case 'fillForm':\n return client.fillForm(args[0] as any[], args[1] as any);\n case 'selectOption':\n return client.selectOption(\n args[0] as string,\n args[1] as string,\n args[2] as string,\n args[3] as any\n );\n case 'fileUpload':\n return client.fileUpload(args[0] as string[], args[1] as any);\n\n // Scroll\n case 'scroll':\n return client.scroll(\n args[0] as any,\n args[1] as any,\n args[2] as any,\n args[3] as any,\n args[4] as any\n );\n\n // Dialogs\n case 'handleDialog':\n return client.handleDialog(\n args[0] as 'accept' | 'dismiss',\n args[1] as string,\n args[2] as any\n );\n\n // Waiting\n case 'waitFor':\n return client.waitFor(args[0] as any);\n\n // Browser Management\n case 'close':\n return client.close(args[0] as any);\n case 'resize':\n return client.resize(args[0] as number, args[1] as number, args[2] as any);\n case 'tabs':\n return client.tabs(\n args[0] as 'list' | 'new' | 'close' | 'select',\n args[1] as number,\n args[2] as any\n );\n\n // Context Management\n case 'swapContext':\n return client.swapContext?.(args[0] as any);\n\n // Storage & Page Info\n case 'getStorageState':\n return client.getStorageState(args[0] as any);\n case 'getCurrentUrl':\n return client.getCurrentUrl(args[0] as any);\n case 'getTitle':\n return client.getTitle(args[0] as any);\n case 'getLinks':\n return client.getLinks(args[0] as any);\n case 'getElementBoundingBox':\n return client.getElementBoundingBox(args[0] as string, args[1] as any);\n\n // Tracing\n case 'startTracing':\n return client.startTracing?.(args[0] as any);\n case 'stopTracing':\n return client.stopTracing?.(args[0] as any);\n\n // Video\n case 'isVideoRecordingEnabled':\n return client.isVideoRecordingEnabled?.() ?? false;\n case 'saveVideo':\n return client.saveVideo?.() ?? null;\n case 'getVideoPath':\n return null;\n\n // Screencast\n case 'startScreencast':\n return client.startScreencast?.(args[0] as any, args[1] as any);\n case 'stopScreencast':\n return client.stopScreencast?.();\n case 'isScreencastActive':\n return client.isScreencastActive?.() ?? false;\n\n default:\n throw new Error(`Unknown method: ${method}`);\n }\n }\n}\n"],"mappings":"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","names":[]}
@@ -817,11 +817,27 @@ async function executeHealActions(decision, failure, execCtx) {
817
817
  const logTools = config.debug || process.env.CANARY_TOOL_LOG === "1";
818
818
  const initialContext = await buildInitialPageContext(execCtx.page, config);
819
819
  if (!decision.healable) {
820
- return baseOutcome({ mode, started, healed: false, shouldRetryOriginal: false, reason: decision.reason, actionsRun, errors });
820
+ return baseOutcome({
821
+ mode,
822
+ started,
823
+ healed: false,
824
+ shouldRetryOriginal: false,
825
+ reason: decision.reason,
826
+ actionsRun,
827
+ errors
828
+ });
821
829
  }
822
830
  const { model, modelId, reason: modelReason } = resolveHealerModel(config);
823
831
  if (!model) {
824
- return baseOutcome({ mode, started, healed: false, shouldRetryOriginal: false, reason: modelReason ?? "no_model", actionsRun, errors });
832
+ return baseOutcome({
833
+ mode,
834
+ started,
835
+ healed: false,
836
+ shouldRetryOriginal: false,
837
+ reason: modelReason ?? "no_model",
838
+ actionsRun,
839
+ errors
840
+ });
825
841
  }
826
842
  const testContext = {
827
843
  testFile: execCtx.testContext?.testFile,
@@ -833,7 +849,12 @@ async function executeHealActions(decision, failure, execCtx) {
833
849
  target: failure.target,
834
850
  errorMessage: failure.errorMessage
835
851
  };
836
- const toolset = createAgenticTools({ page: execCtx.page, config, actionsRun, debug: config.debug });
852
+ const toolset = createAgenticTools({
853
+ page: execCtx.page,
854
+ config,
855
+ actionsRun,
856
+ debug: config.debug
857
+ });
837
858
  let completionReason;
838
859
  try {
839
860
  const abort = AbortSignal.timeout(config.healTimeoutMs);
@@ -1019,7 +1040,10 @@ function wrapExpect(expectImpl, options) {
1019
1040
  };
1020
1041
  const proxy = new Proxy(expectImpl, {
1021
1042
  apply(target, thisArg, argArray) {
1022
- const expectation = target.apply(thisArg, argArray);
1043
+ const expectation = target.apply(
1044
+ thisArg,
1045
+ argArray
1046
+ );
1023
1047
  return wrapMatchers(expectation, "hard");
1024
1048
  },
1025
1049
  get(target, prop, receiver) {
@@ -1080,7 +1104,11 @@ function wrapLocator(locator, options, page) {
1080
1104
  };
1081
1105
  }
1082
1106
  if (typeof prop === "string" && LOCATOR_CHAIN_METHODS.has(prop) && typeof value === "function") {
1083
- return (...args) => wrapLocator(value.apply(target, args), options, page);
1107
+ return (...args) => wrapLocator(
1108
+ value.apply(target, args),
1109
+ options,
1110
+ page
1111
+ );
1084
1112
  }
1085
1113
  return value;
1086
1114
  }
@@ -1231,10 +1259,14 @@ function errorInfo(error) {
1231
1259
  return { message: typeof error === "string" ? error : JSON.stringify(error) };
1232
1260
  }
1233
1261
  function isLocatorLike(candidate) {
1234
- return Boolean(candidate && typeof candidate === "object" && "scrollIntoViewIfNeeded" in candidate);
1262
+ return Boolean(
1263
+ candidate && typeof candidate === "object" && "scrollIntoViewIfNeeded" in candidate
1264
+ );
1235
1265
  }
1236
1266
  function isPageLike(candidate) {
1237
- return Boolean(candidate && typeof candidate === "object" && "waitForTimeout" in candidate);
1267
+ return Boolean(
1268
+ candidate && typeof candidate === "object" && "waitForTimeout" in candidate
1269
+ );
1238
1270
  }
1239
1271
  function safeTargetString(target) {
1240
1272
  try {
@@ -1288,4 +1320,4 @@ export {
1288
1320
  wrapExpect,
1289
1321
  wrapPage
1290
1322
  };
1291
- //# sourceMappingURL=chunk-A44B2PEA.js.map
1323
+ //# sourceMappingURL=chunk-SYPQF57S.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/runner/config.ts","../src/runner/env.ts","../src/runner/state.ts","../src/runner/healing-helpers.ts","../src/runner/healer.ts","../src/runner/ai-client.ts","../src/runner/healer-tools.ts","../src/runner/healer-prompt.ts"],"sourcesContent":["import path from \"node:path\";\nimport { loadCanaryEnv } from \"./env\";\n\nexport type CanaryConfig = {\n enabled: boolean;\n allowedPlaywrightVersion?: string;\n aiProvider?: string;\n aiModel?: string;\n apiKey?: string;\n healTimeoutMs: number;\n maxActions: number;\n dryRun: boolean;\n warnOnly: boolean;\n visionEnabled: boolean;\n debug: boolean;\n eventLogPath: string;\n eventLoggingEnabled: boolean;\n readOnly: boolean;\n allowRunCode: boolean;\n allowEvaluate: boolean;\n maxPayloadBytes: number;\n};\n\nexport function loadCanaryConfig(): CanaryConfig {\n loadCanaryEnv();\n\n const eventLogPath = (() => {\n const userPath = process.env.CANARY_EVENT_LOG;\n if (userPath) return userPath;\n const worker = process.env.TEST_WORKER_INDEX ?? \"runner\";\n return path.join(process.cwd(), \"test-results\", \"ai-healer\", `events-worker-${worker}.jsonl`);\n })();\n\n const baseConfig: CanaryConfig = {\n enabled: true,\n allowedPlaywrightVersion: process.env.CANARY_PLAYWRIGHT_VERSION,\n aiProvider: process.env.AI_PROVIDER,\n aiModel: process.env.AI_MODEL,\n apiKey: process.env.AI_API_KEY,\n healTimeoutMs: Number(process.env.AI_TIMEOUT_MS ?? 120000), // 2 minutes for agentic healing\n maxActions: Number(process.env.CANARY_MAX_ACTIONS ?? 50), // Generous step limit for agentic healing\n dryRun: process.env.CANARY_DRY_RUN === \"1\",\n warnOnly: process.env.CANARY_WARN_ONLY === \"1\",\n visionEnabled: process.env.CANARY_VISION === \"1\" || process.env.AI_VISION === \"1\",\n debug: process.env.CANARY_DEBUG === \"1\",\n readOnly: process.env.CANARY_READ_ONLY === \"1\",\n allowRunCode: process.env.CANARY_ALLOW_RUN_CODE === \"1\",\n allowEvaluate: process.env.CANARY_ALLOW_EVALUATE !== \"0\",\n maxPayloadBytes: Number(process.env.CANARY_MAX_PAYLOAD_BYTES ?? 60000), // cap snapshots/screenshots/text\n eventLogPath,\n eventLoggingEnabled: process.env.CANARY_EVENT_LOG !== \"0\",\n };\n\n const disabled =\n process.env.CANARY_ENABLED === \"0\" ||\n process.env.CANARY_DISABLED === \"1\" ||\n process.env.AI_HEALING === \"0\";\n\n return {\n ...baseConfig,\n enabled: !disabled && baseConfig.enabled,\n };\n}\n","import fs from \"node:fs\";\nimport path from \"node:path\";\n\nlet loaded = false;\n\nexport function loadCanaryEnv(): void {\n if (loaded) return;\n loaded = true;\n\n const explicitPath = process.env.CANARY_ENV_FILE;\n const defaultPath = path.join(process.cwd(), \".env\");\n const envPath = explicitPath || defaultPath;\n\n if (fs.existsSync(envPath)) {\n loadFile(envPath);\n }\n}\n\nfunction loadFile(filePath: string): void {\n try {\n const raw = fs.readFileSync(filePath, \"utf-8\");\n for (const line of raw.split(\"\\n\")) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith(\"#\")) continue;\n const idx = trimmed.indexOf(\"=\");\n if (idx === -1) continue;\n const key = trimmed.slice(0, idx).trim();\n const value = trimmed.slice(idx + 1).trim();\n if (!key) continue;\n if (process.env[key] === undefined) {\n process.env[key] = stripQuotes(value);\n }\n }\n } catch {\n // ignore\n }\n}\n\nfunction stripQuotes(value: string): string {\n if ((value.startsWith(\"\\\"\") && value.endsWith(\"\\\"\")) || (value.startsWith(\"'\") && value.endsWith(\"'\"))) {\n return value.slice(1, -1);\n }\n return value;\n}\n","import fs from 'node:fs';\nimport path from 'node:path';\nimport { loadCanaryConfig } from './config';\nimport { loadCanaryEnv } from './env';\n\nexport type HealingEvent = {\n kind: 'locator' | 'page' | 'expect' | 'unknown';\n action?: string;\n target?: string;\n errorMessage?: string;\n healed: boolean;\n workerId?: string;\n timestamp?: string;\n decision?: string;\n reason?: string;\n strategy?: string;\n actions?: Array<{ type: string; detail?: string }>;\n durationMs?: number;\n mode?: 'dry-run' | 'warn' | 'full';\n modelId?: string;\n // Report fields\n summary?: string; // Human-readable one-line healing summary from agent\n testFile?: string; // Test file path for correlation\n testTitle?: string; // Test title for correlation\n};\n\ndeclare global {\n // eslint-disable-next-line no-var\n var __CANARY_EVENTS: HealingEvent[] | undefined;\n // eslint-disable-next-line no-var\n var __CANARY_PATCHED: boolean | undefined;\n // eslint-disable-next-line no-var\n var __CANARY_EVENT_LOG_PATH: string | undefined;\n}\n\nexport function getEventLog(): HealingEvent[] {\n if (!globalThis.__CANARY_EVENTS) {\n globalThis.__CANARY_EVENTS = [];\n }\n return globalThis.__CANARY_EVENTS;\n}\n\nexport function setEventLogPath(path: string): void {\n globalThis.__CANARY_EVENT_LOG_PATH = path;\n}\n\nfunction getEventLogPath(): string | undefined {\n return globalThis.__CANARY_EVENT_LOG_PATH;\n}\n\nexport function recordHealingEvent(event: HealingEvent): void {\n loadCanaryEnv();\n const log = getEventLog();\n const entry: HealingEvent = {\n ...event,\n workerId: process.env.TEST_WORKER_INDEX,\n timestamp: new Date().toISOString(),\n };\n log.push(entry);\n appendEvent(entry);\n}\n\nexport function markPatched(): void {\n globalThis.__CANARY_PATCHED = true;\n}\n\nexport function alreadyPatched(): boolean {\n return globalThis.__CANARY_PATCHED === true;\n}\n\nfunction appendEvent(event: HealingEvent): void {\n const logPath = getEventLogPath();\n if (!logPath) return;\n const config = loadCanaryConfig();\n if (!config.eventLoggingEnabled) return;\n\n try {\n fs.mkdirSync(path.dirname(logPath), { recursive: true });\n fs.appendFileSync(logPath, JSON.stringify(event) + '\\n');\n } catch {\n // Swallow logging errors to avoid impacting tests.\n }\n}\n","/**\n * Shared helpers for Playwright healing instrumentation.\n *\n * Used by both `instrumentation.ts` (Module._load hook approach)\n * and `wrapper.ts` (direct wrap approach).\n */\n\nimport { createRequire } from 'node:module';\nimport fs from 'node:fs';\nimport { classifyFailure, executeHealActions, type FailureContext } from './healer';\nimport { recordHealingEvent } from './state';\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nconst requireFn: any = typeof require !== 'undefined' ? require : createRequire(import.meta.url);\n\n// === Action / Factory Constants ===\n\nconst LOCATOR_ACTIONS = new Set([\n 'click',\n 'dblclick',\n 'fill',\n 'check',\n 'uncheck',\n 'hover',\n 'press',\n 'type',\n 'selectOption',\n 'tap',\n]);\n\nconst PAGE_ACTIONS = new Set([\n 'goto',\n 'click',\n 'dblclick',\n 'fill',\n 'check',\n 'uncheck',\n 'hover',\n 'press',\n 'type',\n 'selectOption',\n 'tap',\n 'waitForSelector',\n]);\n\nconst LOCATOR_FACTORIES = new Set([\n 'locator',\n 'getByRole',\n 'getByText',\n 'getByLabel',\n 'getByPlaceholder',\n 'getByAltText',\n 'getByTitle',\n 'getByTestId',\n 'frameLocator',\n]);\n\nconst LOCATOR_CHAIN_METHODS = new Set([\n 'locator',\n 'first',\n 'last',\n 'nth',\n 'filter',\n 'getByRole',\n 'getByText',\n 'getByLabel',\n 'getByPlaceholder',\n 'getByAltText',\n 'getByTitle',\n 'getByTestId',\n]);\n\n// === Types ===\n\nexport type PlaywrightExports = {\n test: { extend: typeof import('@playwright/test').test.extend };\n expect: typeof import('@playwright/test').expect;\n [key: string]: unknown;\n};\n\nexport type WrapOptions = {\n debug?: boolean;\n};\n\ntype AttemptContext = {\n kind: FailureContext['kind'];\n action: string;\n target?: string;\n locator?: unknown;\n page?: unknown;\n invoke: () => unknown;\n debug?: boolean;\n};\n\ntype MatcherContext = {\n matcher: (...args: unknown[]) => unknown;\n matcherName: string;\n args: unknown[];\n expectTarget: unknown;\n mode: 'hard' | 'soft';\n debug?: boolean;\n};\n\n// === Proxy Wrappers ===\n\nexport function wrapExpect(\n expectImpl: PlaywrightExports['expect'],\n options: WrapOptions\n): PlaywrightExports['expect'] {\n const wrapMatchers = (expectation: unknown, mode: 'hard' | 'soft') => {\n return new Proxy(expectation ?? {}, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n if (typeof prop === 'string' && typeof value === 'function') {\n return (...args: unknown[]) =>\n runMatcherWithHealing({\n matcher: value as (...args: unknown[]) => unknown,\n matcherName: prop,\n args,\n expectTarget: target,\n mode,\n debug: options.debug,\n });\n }\n return value;\n },\n });\n };\n\n const proxy = new Proxy(expectImpl, {\n apply(target, thisArg, argArray) {\n const expectation = (target as (...args: unknown[]) => unknown).apply(\n thisArg,\n argArray as unknown[]\n );\n return wrapMatchers(expectation, 'hard');\n },\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n if (prop === 'soft' && typeof value === 'function') {\n return (...args: unknown[]) =>\n wrapMatchers((value as (...args: unknown[]) => unknown).apply(target, args), 'soft');\n }\n return value;\n },\n });\n\n return proxy as PlaywrightExports['expect'];\n}\n\nexport function wrapPage<PageType extends object>(page: PageType, options: WrapOptions): PageType {\n return new Proxy(page, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n\n if (typeof prop === 'string') {\n if (LOCATOR_FACTORIES.has(prop) && typeof value === 'function') {\n return (...args: unknown[]) => {\n const locator = (value as (...args: unknown[]) => unknown).apply(target, args);\n return wrapLocator(locator, options, target);\n };\n }\n\n if (PAGE_ACTIONS.has(prop) && typeof value === 'function') {\n return async (...args: unknown[]) => {\n return attemptWithHealing({\n kind: 'page',\n action: prop,\n target: safeTargetString(target),\n locator: undefined,\n page: target as unknown,\n invoke: () => (value as (...args: unknown[]) => unknown).apply(target, args),\n debug: options.debug,\n });\n };\n }\n }\n\n return value;\n },\n });\n}\n\nfunction wrapLocator(locator: unknown, options: WrapOptions, page?: unknown): unknown {\n if (!locator || typeof locator !== 'object') return locator;\n\n return new Proxy(locator, {\n get(target, prop, receiver) {\n const value = Reflect.get(target, prop, receiver);\n\n if (typeof prop === 'string' && LOCATOR_ACTIONS.has(prop) && typeof value === 'function') {\n return async (...args: unknown[]) => {\n return attemptWithHealing({\n kind: 'locator',\n action: prop,\n target: safeTargetString(target),\n locator: target as unknown,\n page,\n invoke: () => (value as (...args: unknown[]) => unknown).apply(target, args),\n debug: options.debug,\n });\n };\n }\n\n if (\n typeof prop === 'string' &&\n LOCATOR_CHAIN_METHODS.has(prop) &&\n typeof value === 'function'\n ) {\n return (...args: unknown[]) =>\n wrapLocator(\n (value as (...args: unknown[]) => unknown).apply(target, args),\n options,\n page\n );\n }\n\n return value;\n },\n });\n}\n\n// === Healing Logic ===\n\nasync function attemptWithHealing<T>(ctx: AttemptContext): Promise<T> {\n try {\n return (await Promise.resolve(ctx.invoke())) as T;\n } catch (error) {\n const failure = buildFailureContext(ctx.kind, ctx.action, ctx.target, error);\n recordHealingEvent({ ...failure, healed: false });\n const decision = classifyFailure(failure);\n if (ctx.debug) {\n // eslint-disable-next-line no-console\n console.log(\n `[canary][debug] failure intercepted: kind=${ctx.kind} action=${ctx.action} reason=${\n decision.healable ? (decision.reason ?? 'healable') : decision.reason\n }`\n );\n }\n if (!decision.healable) {\n throw error;\n }\n\n const testContext = getTestContext();\n const outcome = await executeHealActions(decision, failure, {\n kind: ctx.kind,\n action: ctx.action,\n target: ctx.target,\n locator: isLocatorLike(ctx.locator) ? ctx.locator : undefined,\n page: isPageLike(ctx.page) ? ctx.page : undefined,\n testContext: {\n testFile: testContext?.testFile,\n testTitle: testContext?.testTitle,\n testSource: loadTestSource(testContext?.testFile),\n },\n });\n\n const summaryBase = {\n ...failure,\n healed: false,\n strategy: 'agentic',\n reason: outcome.reason ?? decision.reason,\n actions: actionsToEventItems(outcome.actionsRun),\n durationMs: outcome.durationMs,\n mode: outcome.mode,\n decision: decision.reason,\n modelId: outcome.modelId,\n summary: outcome.summary,\n testFile: testContext?.testFile,\n testTitle: testContext?.testTitle,\n } as const;\n\n if (outcome.healed) {\n if (ctx.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] healed via AI tool for action=${ctx.action}`);\n }\n recordHealingEvent({ ...summaryBase, healed: true });\n return undefined as T;\n }\n\n if (outcome.shouldRetryOriginal) {\n try {\n const retried = (await Promise.resolve(ctx.invoke())) as T;\n if (ctx.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] retry_original succeeded for action=${ctx.action}`);\n }\n recordHealingEvent({ ...summaryBase, healed: true });\n return retried;\n } catch (retryError) {\n const retryInfo = errorInfo(retryError);\n recordHealingEvent({ ...summaryBase, healed: false, errorMessage: retryInfo.message });\n if (ctx.debug) {\n // eslint-disable-next-line no-console\n console.log(\n `[canary][debug] retry_original failed for action=${ctx.action}: ${retryInfo.message ?? retryError}`\n );\n }\n throw retryError;\n }\n }\n\n recordHealingEvent(summaryBase);\n throw error;\n }\n}\n\nasync function runMatcherWithHealing(params: MatcherContext): Promise<unknown> {\n const { matcher, matcherName, args, expectTarget, mode } = params;\n const invoke = () => matcher.apply(expectTarget, args);\n\n if (mode === 'soft') {\n return Promise.resolve(invoke());\n }\n\n try {\n return await Promise.resolve(invoke());\n } catch (error) {\n const target = stringifyTarget(args?.[0]);\n const failure = buildFailureContext('expect', matcherName, target, error);\n recordHealingEvent({ ...failure, healed: false });\n const decision = classifyFailure(failure);\n if (!decision.healable) {\n throw error;\n }\n\n const outcome = await executeHealActions(decision, failure, {\n kind: 'expect',\n action: matcherName,\n target,\n locator: isLocatorLike(args?.[0]) ? args[0] : undefined,\n });\n\n const testContextMatcher = getTestContext();\n const summaryBase = {\n ...failure,\n healed: false,\n strategy: 'agentic',\n reason: outcome.reason ?? decision.reason,\n actions: actionsToEventItems(outcome.actionsRun),\n durationMs: outcome.durationMs,\n mode: outcome.mode,\n decision: decision.reason,\n modelId: outcome.modelId,\n summary: outcome.summary,\n testFile: testContextMatcher?.testFile,\n testTitle: testContextMatcher?.testTitle,\n } as const;\n\n if (outcome.shouldRetryOriginal) {\n try {\n const retried = await Promise.resolve(invoke());\n recordHealingEvent({ ...summaryBase, healed: true });\n return retried;\n } catch (retryError) {\n const retryInfo = errorInfo(retryError);\n recordHealingEvent({ ...summaryBase, healed: false, errorMessage: retryInfo.message });\n throw retryError;\n }\n }\n\n recordHealingEvent(summaryBase);\n throw error;\n }\n}\n\n// === Utility Functions ===\n\nfunction buildFailureContext(\n kind: FailureContext['kind'],\n action: string,\n target: string | undefined,\n error: unknown\n): FailureContext {\n const info = errorInfo(error);\n return {\n kind,\n action,\n target,\n errorMessage: info.message,\n errorName: info.name,\n stack: info.stack,\n };\n}\n\nfunction actionsToEventItems(actions: string[]): Array<{ type: string; detail?: string }> {\n return actions.map((action) => ({ type: action }));\n}\n\nfunction errorInfo(error: unknown): { message?: string; name?: string; stack?: string } {\n if (error instanceof Error) {\n return { message: error.message, name: error.name, stack: error.stack };\n }\n return { message: typeof error === 'string' ? error : JSON.stringify(error) };\n}\n\nfunction isLocatorLike(\n candidate: unknown\n): candidate is { scrollIntoViewIfNeeded: () => Promise<unknown> } {\n return Boolean(\n candidate &&\n typeof candidate === 'object' &&\n 'scrollIntoViewIfNeeded' in (candidate as Record<string, unknown>)\n );\n}\n\nfunction isPageLike(\n candidate: unknown\n): candidate is { waitForTimeout: (ms: number) => Promise<unknown> } {\n return Boolean(\n candidate &&\n typeof candidate === 'object' &&\n 'waitForTimeout' in (candidate as Record<string, unknown>)\n );\n}\n\nfunction safeTargetString(target: unknown): string | undefined {\n try {\n if (typeof target === 'object' && target !== null && 'toString' in target) {\n const s = String((target as { toString: () => string }).toString());\n if (s && s !== '[object Object]') return s;\n }\n } catch {\n // ignore\n }\n return undefined;\n}\n\nfunction stringifyTarget(candidate: unknown): string | undefined {\n if (!candidate) return undefined;\n if (typeof candidate === 'string') return candidate;\n if (typeof candidate === 'object') {\n if (\n 'selector' in (candidate as Record<string, unknown>) &&\n typeof (candidate as Record<string, unknown>).selector === 'string'\n ) {\n return String((candidate as Record<string, unknown>).selector);\n }\n }\n return safeTargetString(candidate);\n}\n\nfunction getTestContext(): { testFile?: string; testTitle?: string } | undefined {\n try {\n const playwright = requireFn('@playwright/test') as {\n test?: { info?: () => { file?: string; title?: string } };\n };\n if (playwright?.test?.info) {\n const info = playwright.test.info();\n if (info) {\n return { testFile: info.file, testTitle: info.title };\n }\n }\n } catch {\n // Not in a Playwright test context\n }\n return undefined;\n}\n\nfunction loadTestSource(filePath?: string): string | undefined {\n if (!filePath) return undefined;\n try {\n return fs.readFileSync(filePath, 'utf-8');\n } catch {\n return undefined;\n }\n}\n","/**\n * AI Test Healer — Core types, failure classification, and execution entry point.\n *\n * Tool definitions are in `healer-tools.ts`.\n * Prompt construction is in `healer-prompt.ts`.\n */\n\nimport { stepCountIs, streamText } from 'ai';\nimport { loadCanaryConfig } from './config';\nimport { resolveHealerModel } from './ai-client';\nimport { createAgenticTools, logToolStep } from './healer-tools';\nimport { buildSystemPrompt, buildInitialMessages, buildInitialPageContext } from './healer-prompt';\n\n// === Redaction / Security ===\n\nconst REDACTION_PATTERNS: Array<{ regex: RegExp; replacement: string }> = [\n { regex: /bearer\\s+[a-z0-9._-]+/gi, replacement: '[REDACTED_BEARER]' },\n { regex: /api[_-]?key[:\\s\"']+[a-z0-9._-]+/gi, replacement: '[REDACTED_API_KEY]' },\n { regex: /secret[:\\s\"']+[a-z0-9._-]+/gi, replacement: '[REDACTED_SECRET]' },\n { regex: /[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,}/g, replacement: '[REDACTED_EMAIL]' },\n { regex: /[0-9]{12,}/g, replacement: '[REDACTED_NUMBER]' },\n];\nconst CODE_DENY_PATTERNS = [/process\\.env/i, /child_process/i, /\\brequire\\s*\\(/i, /\\bimport\\s*\\(/i];\n\n// === Types ===\n\nexport type FailureContext = {\n kind: 'locator' | 'page' | 'expect' | 'unknown';\n action?: string;\n target?: string;\n errorMessage?: string;\n errorName?: string;\n stack?: string;\n};\n\nexport type TestContext = {\n testFile?: string;\n testTitle?: string;\n testSource?: string;\n currentStep: string;\n expectedAfter?: string;\n action: string;\n target?: string;\n errorMessage?: string;\n};\n\ntype HealDecision = { healable: false; reason: string } | { healable: true; reason?: string };\n\ntype HealExecutionContext = {\n kind: FailureContext['kind'];\n target?: string;\n action?: string;\n locator?: { scrollIntoViewIfNeeded?: (options?: { timeout?: number }) => Promise<unknown> };\n page?: unknown;\n testContext?: Partial<TestContext>;\n};\n\ntype HealOutcome = {\n healed: boolean;\n shouldRetryOriginal: boolean;\n actionsRun: string[];\n timedOut: boolean;\n durationMs: number;\n errors: string[];\n mode: 'dry-run' | 'warn' | 'full';\n reason?: string;\n modelId?: string;\n summary?: string;\n};\n\n// === Failure Classification ===\n\nexport function classifyFailure(context: FailureContext): HealDecision {\n const message = (context.errorMessage ?? '').toLowerCase();\n\n if (\n message.includes('closed') ||\n message.includes('target page, context or browser has been closed')\n ) {\n return { healable: false, reason: 'context_closed' };\n }\n\n if (message.includes('navigation failed') && message.includes('net::')) {\n return { healable: false, reason: 'navigation_failed' };\n }\n\n return { healable: true, reason: 'agent_healing' };\n}\n\n// === Main Entry Point ===\n\nexport async function executeHealActions(\n decision: HealDecision,\n failure: FailureContext,\n execCtx: HealExecutionContext\n): Promise<HealOutcome> {\n const config = loadCanaryConfig();\n const started = Date.now();\n const mode = resolveMode(config);\n const actionsRun: string[] = [];\n const errors: string[] = [];\n const logTools = config.debug || process.env.CANARY_TOOL_LOG === '1';\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const initialContext = await buildInitialPageContext(execCtx.page as any, config);\n\n if (!decision.healable) {\n return baseOutcome({\n mode,\n started,\n healed: false,\n shouldRetryOriginal: false,\n reason: decision.reason,\n actionsRun,\n errors,\n });\n }\n\n const { model, modelId, reason: modelReason } = resolveHealerModel(config);\n if (!model) {\n return baseOutcome({\n mode,\n started,\n healed: false,\n shouldRetryOriginal: false,\n reason: modelReason ?? 'no_model',\n actionsRun,\n errors,\n });\n }\n\n const testContext: TestContext = {\n testFile: execCtx.testContext?.testFile,\n testTitle: execCtx.testContext?.testTitle,\n testSource: execCtx.testContext?.testSource\n ? sanitizeString(execCtx.testContext.testSource)\n : undefined,\n currentStep: `${failure.action ?? 'unknown'}(${failure.target ?? 'unknown'})`,\n expectedAfter: execCtx.testContext?.expectedAfter,\n action: failure.action ?? 'unknown',\n target: failure.target,\n errorMessage: failure.errorMessage,\n };\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const toolset = createAgenticTools({\n page: execCtx.page as any,\n config,\n actionsRun,\n debug: config.debug,\n });\n let completionReason: string | undefined;\n\n try {\n const abort = AbortSignal.timeout(config.healTimeoutMs);\n const result = await streamText({\n model,\n system: buildSystemPrompt(testContext, config),\n messages: buildInitialMessages(testContext, failure, initialContext),\n tools: toolset.tools,\n stopWhen: stepCountIs(Math.max(1, config.maxActions)),\n abortSignal: abort,\n maxRetries: 0,\n onStepFinish: (step) => {\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(\n `[canary][debug] step finish: finishReason=${step.finishReason}; toolCalls=${step.toolCalls?.length ?? 0}; toolResults=${step.toolResults?.length ?? 0}`\n );\n }\n if (logTools) logToolStep(step);\n },\n });\n\n for await (const _ of result.fullStream) {\n if (toolset.isComplete()) {\n completionReason = toolset.getCompletionReason();\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] agent marked complete: ${completionReason}`);\n }\n break;\n }\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n errors.push(message);\n if (config.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] agent error: ${message}`);\n }\n }\n\n const healed = toolset.isComplete();\n\n return {\n healed,\n shouldRetryOriginal: false,\n actionsRun,\n timedOut: Date.now() - started > config.healTimeoutMs,\n durationMs: Date.now() - started,\n errors,\n mode,\n reason: completionReason ?? (healed ? 'agent_healed' : 'agent_incomplete'),\n modelId,\n summary: toolset.getSummary(),\n };\n}\n\n// === Utilities (exported for use by healer-tools.ts and healer-prompt.ts) ===\n\nfunction resolveMode(config: ReturnType<typeof loadCanaryConfig>): 'dry-run' | 'warn' | 'full' {\n if (config.dryRun) return 'dry-run';\n if (config.warnOnly) return 'warn';\n return 'full';\n}\n\nfunction baseOutcome({\n started,\n healed,\n shouldRetryOriginal,\n mode,\n reason,\n actionsRun = [],\n errors = [],\n}: {\n started: number;\n healed: boolean;\n shouldRetryOriginal: boolean;\n mode: 'dry-run' | 'warn' | 'full';\n reason?: string;\n actionsRun?: string[];\n errors?: string[];\n}): HealOutcome {\n return {\n healed,\n shouldRetryOriginal,\n actionsRun,\n timedOut: false,\n durationMs: Date.now() - started,\n errors,\n mode,\n reason,\n };\n}\n\nexport function sanitizeString(value: string): string {\n let result = value;\n for (const { regex, replacement } of REDACTION_PATTERNS) {\n result = result.replace(regex, replacement);\n }\n return result;\n}\n\nexport function sanitizeUnknown(\n value: unknown,\n maxPayloadBytes: number,\n seen = new WeakSet()\n): unknown {\n if (value === null || value === undefined) return value;\n if (typeof value === 'string') {\n return truncate(sanitizeString(value), maxPayloadBytes);\n }\n if (typeof value === 'number' || typeof value === 'boolean') return value;\n if (typeof value === 'object') {\n if (seen.has(value as object)) return '[REDACTED_CYCLE]';\n seen.add(value as object);\n if (Array.isArray(value)) {\n return value.slice(0, 50).map((v) => sanitizeUnknown(v, maxPayloadBytes, seen));\n }\n const out: Record<string, unknown> = {};\n const entries = Object.entries(value as Record<string, unknown>).slice(0, 50);\n for (const [k, v] of entries) {\n out[k] = sanitizeUnknown(v, maxPayloadBytes, seen);\n }\n return out;\n }\n return '[REDACTED_UNKNOWN]';\n}\n\nexport function violatesCodeDenylist(code: string): boolean {\n return CODE_DENY_PATTERNS.some((p) => p.test(code));\n}\n\nfunction truncate(value: string, max: number): string {\n if (value.length <= max) return value;\n return value.slice(0, max) + '...';\n}\n","import { createOpenAI } from \"@ai-sdk/openai\";\nimport { createAnthropic } from \"@ai-sdk/anthropic\";\nimport { createAzure } from \"@ai-sdk/azure\";\nimport type { LanguageModel } from \"ai\";\nimport { loadCanaryConfig, type CanaryConfig } from \"./config\";\n\ntype ResolvedModel = {\n model: LanguageModel | null;\n modelId?: string;\n reason?: string;\n};\n\nexport function resolveHealerModel(config: CanaryConfig = loadCanaryConfig()): ResolvedModel {\n if (!config.enabled) {\n return { model: null, reason: \"healing_disabled\" };\n }\n\n if (!config.apiKey) {\n return { model: null, reason: \"missing_api_key\" };\n }\n\n const provider = (config.aiProvider ?? \"openai\").toLowerCase();\n const modelId = config.aiModel || (provider === \"anthropic\" ? \"claude-3-5-haiku-20241022\" : \"gpt-4o-mini\");\n\n try {\n switch (provider) {\n case \"openai\": {\n const client = createOpenAI({ apiKey: config.apiKey, baseURL: process.env.AI_BASE_URL });\n return { model: client(modelId), modelId };\n }\n case \"anthropic\": {\n const client = createAnthropic({ apiKey: config.apiKey, baseURL: process.env.AI_BASE_URL });\n return { model: client(modelId), modelId };\n }\n case \"azure\": {\n const resourceName = process.env.AZURE_OPENAI_RESOURCE_NAME;\n const apiVersion = process.env.AZURE_OPENAI_API_VERSION;\n if (!resourceName) {\n return { model: null, reason: \"missing_azure_resource\" };\n }\n const client = createAzure({\n apiKey: config.apiKey,\n resourceName,\n apiVersion,\n });\n return { model: client(modelId), modelId };\n }\n default:\n return { model: null, reason: \"unsupported_provider\" };\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n return { model: null, reason: `model_error:${message}` };\n }\n}\n","/**\n * Agentic tool definitions for the AI test healer.\n *\n * Uses a `makePageTool` factory to eliminate repetitive guard/try-catch boilerplate\n * across the ~15 tool definitions.\n */\n\nimport { tool, type Tool } from \"ai\";\nimport { z } from \"zod\";\nimport type { loadCanaryConfig } from \"./config\";\nimport { sanitizeString, sanitizeUnknown, violatesCodeDenylist } from \"./healer\";\n\ntype PageLike = {\n waitForTimeout?: (ms: number) => Promise<unknown>;\n accessibility?: { snapshot: () => Promise<unknown> };\n screenshot?: (options?: { fullPage?: boolean }) => Promise<Buffer | Uint8Array | string>;\n locator?: (selector: string) => LocatorLike & {\n click?: (opts?: unknown) => Promise<unknown>;\n fill?: (value: string, opts?: unknown) => Promise<unknown>;\n pressSequentially?: (text: string, opts?: unknown) => Promise<unknown>;\n hover?: () => Promise<unknown>;\n selectOption?: (value: string) => Promise<unknown>;\n textContent?: () => Promise<string | null>;\n isVisible?: () => Promise<boolean>;\n waitFor?: (opts?: { state?: string }) => Promise<unknown>;\n };\n getByText?: (text: string) => LocatorLike & {\n click?: (opts?: unknown) => Promise<unknown>;\n };\n keyboard?: { press?: (key: string) => Promise<unknown> };\n goto?: (url: string) => Promise<unknown>;\n goBack?: () => Promise<unknown>;\n reload?: () => Promise<unknown>;\n evaluate?: (script: string) => Promise<unknown>;\n};\n\ntype LocatorLike = {\n scrollIntoViewIfNeeded?: (options?: { timeout?: number }) => Promise<unknown>;\n};\n\ntype ToolConfig = ReturnType<typeof loadCanaryConfig>;\n\ntype ToolContext = {\n page?: PageLike;\n config: ToolConfig;\n actionsRun: string[];\n debug?: boolean;\n};\n\ntype ToolsetResult = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n tools: Record<string, Tool<any, any>>;\n isComplete: () => boolean;\n getCompletionReason: () => string | undefined;\n getSummary: () => string | undefined;\n};\n\n// === Helper Factories ===\n\nfunction truncate(value: string, max: number): string {\n if (value.length <= max) return value;\n return value.slice(0, max) + \"...\";\n}\n\nfunction safeJson(value: unknown): string {\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n}\n\nfunction errMsg(err: unknown): string {\n return err instanceof Error ? err.message : String(err);\n}\n\n/**\n * Factory for tools that interact with the page. Handles:\n * - tracking the action name\n * - debug logging\n * - read-only / dry-run guards\n * - page availability check\n * - try/catch with error formatting\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction makePageTool(\n ctx: ToolContext,\n opts: {\n name: string;\n description: string;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n inputSchema: z.ZodType<any>;\n /** Whether this tool mutates state (subject to readOnly guard). Default true. */\n mutates?: boolean;\n /** Key to check on page for availability (e.g. 'locator', 'goto'). Skips guard if omitted. */\n pageCheck?: keyof PageLike;\n /** The dry-run result to return. */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n dryResult: (input: any) => unknown;\n /** The actual execution. */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n execute: (input: any) => Promise<unknown>;\n }\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n): Tool<any, any> {\n const { page, config, actionsRun, debug } = ctx;\n const mutates = opts.mutates ?? true;\n\n const log = (msg: string) => {\n if (debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] ${msg}`);\n }\n };\n\n return tool({\n description: opts.description,\n inputSchema: opts.inputSchema,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n execute: async (input: any) => {\n const trackSuffix = input?.text ?? input?.selector ?? input?.name ?? input?.key ?? input?.url ?? input?.ms ?? \"\";\n actionsRun.push(trackSuffix ? `${opts.name}:${trackSuffix}` : opts.name);\n log(`${opts.name} called`);\n\n if (mutates && config.readOnly) return { error: \"read_only_mode\" };\n if (config.dryRun) return opts.dryResult(input);\n if (opts.pageCheck && !page?.[opts.pageCheck]) return { error: \"Page unavailable\" };\n\n try {\n const result = await opts.execute(input);\n log(`${opts.name} succeeded`);\n return result;\n } catch (err) {\n const msg = errMsg(err);\n log(`${opts.name} error: ${msg}`);\n return { error: msg };\n }\n },\n });\n}\n\n// === Main Export ===\n\nexport function createAgenticTools(params: {\n page?: PageLike;\n config: ToolConfig;\n actionsRun: string[];\n debug?: boolean;\n}): ToolsetResult {\n const ctx: ToolContext = params;\n const { page, config } = ctx;\n let complete = false;\n let completionReason: string | undefined;\n let completionSummary: string | undefined;\n const maxPayloadBytes = Math.max(1000, config.maxPayloadBytes || 60000);\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const tools: Record<string, Tool<any, any>> = {\n // === Core Action Tools ===\n click: makePageTool(ctx, {\n name: \"click\",\n description: \"Click an element by its visible text\",\n inputSchema: z.object({ text: z.string().describe(\"The visible text to click\") }),\n pageCheck: \"getByText\",\n dryResult: ({ text }) => ({ clicked: text, mode: \"dry-run\" as const }),\n execute: async ({ text }) => {\n const el = page!.getByText!(text);\n if (el?.scrollIntoViewIfNeeded) await el.scrollIntoViewIfNeeded({ timeout: 2000 });\n const clickable = el as { click?: (opts?: unknown) => Promise<unknown> };\n if (clickable?.click) await clickable.click({ timeout: 5000 });\n return { clicked: text };\n },\n }),\n\n click_selector: makePageTool(ctx, {\n name: \"click_selector\",\n description: \"Click an element by CSS or Playwright selector\",\n inputSchema: z.object({ selector: z.string().describe(\"CSS or Playwright selector\") }),\n pageCheck: \"locator\",\n dryResult: ({ selector }) => ({ clicked: selector, mode: \"dry-run\" as const }),\n execute: async ({ selector }) => {\n const el = page!.locator!(selector);\n if (el?.scrollIntoViewIfNeeded) await el.scrollIntoViewIfNeeded({ timeout: 2000 });\n if (el?.click) await el.click({ timeout: 5000 });\n return { clicked: selector };\n },\n }),\n\n fill: makePageTool(ctx, {\n name: \"fill\",\n description: \"Fill an input field with a value using CSS selector (clears existing content)\",\n inputSchema: z.object({\n selector: z.string().describe('CSS selector for the input (e.g. input[placeholder=\"...\"])'),\n value: z.string().describe(\"Value to fill\"),\n }),\n pageCheck: \"locator\",\n dryResult: ({ selector }) => ({ filled: selector, mode: \"dry-run\" as const }),\n execute: async ({ selector, value }) => {\n const el = page!.locator!(selector);\n if (el?.fill) await el.fill(value, { timeout: 5000 });\n return { filled: selector, value };\n },\n }),\n\n fill_by_name: makePageTool(ctx, {\n name: \"fill_by_name\",\n description: 'Fill a textbox by its accessible name (shown in snapshot as textbox \"NAME\"). This is the PREFERRED way to fill inputs - use the name from the snapshot directly.',\n inputSchema: z.object({\n name: z.string().describe('The accessible name of the textbox (from snapshot, e.g. \"Enter 6-digit code\")'),\n value: z.string().describe(\"Value to fill\"),\n }),\n pageCheck: \"locator\",\n dryResult: ({ name }) => ({ filled: name, mode: \"dry-run\" as const }),\n execute: async ({ name, value }) => {\n const el = page!.locator!(`role=textbox[name=\"${name}\"]`);\n if (el?.fill) await el.fill(value, { timeout: 5000 });\n return { filled: name, value };\n },\n }),\n\n type: makePageTool(ctx, {\n name: \"type\",\n description: \"Type text character by character (triggers key events)\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS or Playwright selector\"),\n text: z.string().describe(\"Text to type\"),\n }),\n pageCheck: \"locator\",\n dryResult: ({ text }) => ({ typed: text, mode: \"dry-run\" as const }),\n execute: async ({ selector, text }) => {\n const el = page!.locator!(selector);\n if (el?.pressSequentially) await el.pressSequentially(text);\n return { typed: text, into: selector };\n },\n }),\n\n press_key: makePageTool(ctx, {\n name: \"press_key\",\n description: \"Press a keyboard key (Enter, Tab, Escape, ArrowDown, etc.)\",\n inputSchema: z.object({ key: z.string().describe(\"Key to press\") }),\n pageCheck: \"keyboard\",\n dryResult: ({ key }) => ({ pressed: key, mode: \"dry-run\" as const }),\n execute: async ({ key }) => {\n await page!.keyboard!.press!(key);\n return { pressed: key };\n },\n }),\n\n hover: makePageTool(ctx, {\n name: \"hover\",\n description: \"Hover over an element\",\n inputSchema: z.object({ selector: z.string().describe(\"CSS or Playwright selector\") }),\n pageCheck: \"locator\",\n dryResult: ({ selector }) => ({ hovered: selector, mode: \"dry-run\" as const }),\n execute: async ({ selector }) => {\n const el = page!.locator!(selector);\n if (el?.hover) await el.hover();\n return { hovered: selector };\n },\n }),\n\n select_option: makePageTool(ctx, {\n name: \"select_option\",\n description: \"Select an option from a dropdown\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS or Playwright selector for the select element\"),\n value: z.string().describe(\"Value or label to select\"),\n }),\n pageCheck: \"locator\",\n dryResult: ({ value }) => ({ selected: value, mode: \"dry-run\" as const }),\n execute: async ({ selector, value }) => {\n const el = page!.locator!(selector);\n if (el?.selectOption) await el.selectOption(value);\n return { selected: value, from: selector };\n },\n }),\n\n // === Navigation & Waiting ===\n wait: makePageTool(ctx, {\n name: \"wait\",\n description: \"Wait for a duration in milliseconds\",\n inputSchema: z.object({ ms: z.number().min(100).max(30000).describe(\"Milliseconds to wait\") }),\n mutates: false,\n dryResult: ({ ms }) => ({ waited: ms, mode: \"dry-run\" as const }),\n execute: async ({ ms }) => {\n if (page?.waitForTimeout) {\n await page.waitForTimeout(ms);\n } else {\n await new Promise((resolve) => setTimeout(resolve, ms));\n }\n return { waited: ms };\n },\n }),\n\n wait_for_selector: makePageTool(ctx, {\n name: \"wait_for_selector\",\n description: \"Wait for an element to reach a state (visible, hidden, attached, detached)\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS or Playwright selector\"),\n state: z.enum([\"visible\", \"hidden\", \"attached\", \"detached\"]).optional().describe(\"State to wait for\"),\n }),\n mutates: false,\n pageCheck: \"locator\",\n dryResult: ({ selector }) => ({ found: selector, mode: \"dry-run\" as const }),\n execute: async ({ selector, state }) => {\n const el = page!.locator!(selector);\n if (el?.waitFor) await el.waitFor({ state: state ?? \"visible\" });\n return { found: selector, state: state ?? \"visible\" };\n },\n }),\n\n navigate: makePageTool(ctx, {\n name: \"navigate\",\n description: \"Navigate to a URL\",\n inputSchema: z.object({ url: z.string().describe(\"URL to navigate to\") }),\n pageCheck: \"goto\",\n dryResult: ({ url }) => ({ navigated: url, mode: \"dry-run\" as const }),\n execute: async ({ url }) => {\n await page!.goto!(url);\n return { navigated: url };\n },\n }),\n\n go_back: makePageTool(ctx, {\n name: \"go_back\",\n description: \"Go back in browser history\",\n inputSchema: z.object({}),\n pageCheck: \"goBack\",\n dryResult: () => ({ action: \"back\", mode: \"dry-run\" as const }),\n execute: async () => {\n await page!.goBack!();\n return { action: \"back\" };\n },\n }),\n\n reload: makePageTool(ctx, {\n name: \"reload\",\n description: \"Reload the current page\",\n inputSchema: z.object({}),\n pageCheck: \"reload\",\n dryResult: () => ({ action: \"reload\", mode: \"dry-run\" as const }),\n execute: async () => {\n await page!.reload!();\n return { action: \"reload\" };\n },\n }),\n\n // === Inspection Tools ===\n snapshot: tool({\n description: \"Get the accessibility tree of the current page to understand its structure. ALWAYS call this first!\",\n inputSchema: z.object({}),\n execute: async () => {\n params.actionsRun.push(\"snapshot\");\n if (config.dryRun) return { tree: null, mode: \"dry-run\" };\n if (!page) return { error: \"Page unavailable\" };\n\n let tree: unknown;\n let html: string | undefined;\n\n try {\n if (page.accessibility && typeof page.accessibility.snapshot === \"function\") {\n tree = await page.accessibility.snapshot();\n }\n } catch (err) {\n if (params.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] accessibility snapshot error: ${errMsg(err)}`);\n }\n }\n\n if (!tree && page.evaluate) {\n try {\n html = (await page.evaluate(\"document.body.innerHTML\")) as string;\n } catch (err) {\n if (params.debug) {\n // eslint-disable-next-line no-console\n console.log(`[canary][debug] HTML fallback error: ${errMsg(err)}`);\n }\n }\n }\n\n if (!tree && !html) return { error: \"Could not get page content\" };\n\n const sanitizedTree = sanitizeUnknown(tree, maxPayloadBytes);\n const sanitizedHtml = html ? truncate(sanitizeString(html), maxPayloadBytes) : undefined;\n return { tree: sanitizedTree, html: sanitizedHtml };\n },\n }),\n\n screenshot: tool({\n description: \"Capture a screenshot of the current page\",\n inputSchema: z.object({\n fullPage: z.boolean().optional().describe(\"Whether to capture the full page\"),\n }),\n execute: async ({ fullPage }) => {\n params.actionsRun.push(\"screenshot\");\n if (config.dryRun) return { screenshot: undefined, mode: \"dry-run\" };\n if (!page?.screenshot) return { error: \"Page unavailable\" };\n try {\n const buffer = await page.screenshot({ fullPage: fullPage ?? false });\n const b = typeof buffer === \"string\" ? Buffer.from(buffer) : Buffer.from(buffer);\n const base64 = b.toString(\"base64\");\n if (b.byteLength > maxPayloadBytes || base64.length > maxPayloadBytes) {\n return { error: \"screenshot_too_large\" };\n }\n return { screenshot: `data:image/png;base64,${truncate(base64, maxPayloadBytes)}` };\n } catch (err) {\n return { error: errMsg(err) };\n }\n },\n }),\n\n get_text: makePageTool(ctx, {\n name: \"get_text\",\n description: \"Get the text content of an element\",\n inputSchema: z.object({ selector: z.string().describe(\"CSS or Playwright selector\") }),\n mutates: false,\n pageCheck: \"locator\",\n dryResult: () => ({ text: null, mode: \"dry-run\" as const }),\n execute: async ({ selector }) => {\n const el = page!.locator!(selector);\n const text = el?.textContent ? await el.textContent() : null;\n const sanitized = text ? truncate(sanitizeString(text), maxPayloadBytes) : text;\n return { text: sanitized };\n },\n }),\n\n is_visible: makePageTool(ctx, {\n name: \"is_visible\",\n description: \"Check if an element is visible\",\n inputSchema: z.object({ selector: z.string().describe(\"CSS or Playwright selector\") }),\n mutates: false,\n pageCheck: \"locator\",\n dryResult: () => ({ visible: false, mode: \"dry-run\" as const }),\n execute: async ({ selector }) => {\n const el = page!.locator!(selector);\n const visible = el?.isVisible ? await el.isVisible() : false;\n return { visible };\n },\n }),\n\n evaluate: tool({\n description: \"Run JavaScript in the page context\",\n inputSchema: z.object({ script: z.string().describe(\"JavaScript code to execute\") }),\n execute: async ({ script }) => {\n params.actionsRun.push(\"evaluate\");\n if (!config.allowEvaluate) return { error: \"evaluate_disabled\" };\n if (violatesCodeDenylist(script)) return { error: \"evaluate_blocked\" };\n if (config.dryRun) return { result: null, mode: \"dry-run\" };\n if (!page?.evaluate) return { error: \"Page unavailable\" };\n try {\n const scriptSafe = truncate(script, maxPayloadBytes);\n const result = await page.evaluate(scriptSafe);\n return { result: sanitizeUnknown(result, maxPayloadBytes) };\n } catch (err) {\n return { error: errMsg(err) };\n }\n },\n }),\n\n run_playwright_code: tool({\n description:\n 'Execute arbitrary Playwright code. Has access to `page` object. Use for complex operations not covered by other tools. Example: `await page.getByRole(\"button\", { name: \"Submit\" }).click();`',\n inputSchema: z.object({\n code: z.string().describe(\"Playwright code to execute (has access to `page` object)\"),\n }),\n execute: async ({ code }) => {\n params.actionsRun.push(\"run_playwright_code\");\n if (!config.allowRunCode) return { executed: false, error: \"run_code_disabled\" };\n if (violatesCodeDenylist(code)) return { executed: false, error: \"run_code_blocked\" };\n if (config.dryRun) return { executed: false, mode: \"dry-run\" };\n if (!page) return { executed: false, error: \"Page unavailable\" };\n try {\n const boundedCode = truncate(code, maxPayloadBytes);\n const fn = new Function(\"page\", `return (async () => { ${boundedCode} })();`);\n const result = await fn(page);\n return { executed: true, result: sanitizeUnknown(result, maxPayloadBytes) };\n } catch (err) {\n return { executed: false, error: errMsg(err) };\n }\n },\n }),\n\n // === Completion Tool ===\n mark_complete: tool({\n description:\n \"Call this when the browser is ready for the test to continue. You MUST call this when done healing!\",\n inputSchema: z.object({\n reason: z.string().describe(\"Brief technical description of what was done\"),\n summary: z.string().describe('One-line human-readable summary for the test report, e.g. \"Clicked \\'Send OTP\\' instead of \\'Submit\\'\" or \"Filled email field by accessible name\"'),\n }),\n execute: async ({ reason, summary }) => {\n params.actionsRun.push(\"mark_complete\");\n if (!config.dryRun) {\n complete = true;\n completionReason = reason;\n completionSummary = summary;\n }\n return { complete: !config.dryRun, reason, summary, mode: config.dryRun ? \"dry-run\" : undefined };\n },\n }),\n };\n\n return {\n tools,\n isComplete: () => complete,\n getCompletionReason: () => completionReason,\n getSummary: () => completionSummary,\n };\n}\n\n// === Tool Step Logging ===\n\ntype ToolStep = {\n finishReason?: string;\n toolCalls?: Array<{ toolName?: string; args?: unknown }>;\n toolResults?: Array<{ toolName?: string; result?: unknown; error?: unknown }>;\n};\n\nexport function logToolStep(step: ToolStep): void {\n const color = {\n magenta: \"\\x1b[35m\",\n cyan: \"\\x1b[36m\",\n yellow: \"\\x1b[33m\",\n green: \"\\x1b[32m\",\n reset: \"\\x1b[0m\",\n } as const;\n\n const calls = step.toolCalls ?? [];\n const results = step.toolResults ?? [];\n\n if (calls.length === 0 && results.length === 0) {\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] step${color.reset} (no tool calls/results; finish=${step.finishReason ?? \"unknown\"})`\n );\n }\n\n for (const call of calls) {\n const name = call.toolName ?? \"unknown_tool\";\n const args = call.args ? truncate(safeJson(call.args), 300) : \"\";\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] call${color.reset} ${color.cyan}${name}${color.reset}${args ? ` args=${args}` : \"\"}`\n );\n }\n\n for (const result of results) {\n const name = result.toolName ?? \"unknown_tool\";\n const res = result.result ? truncate(safeJson(result.result), 300) : undefined;\n const err = result.error ? truncate(safeJson(result.error), 300) : undefined;\n const statusColor = err ? color.yellow : color.green;\n // eslint-disable-next-line no-console\n console.log(\n `${color.magenta}[canary][tool] result${color.reset} ${color.cyan}${name}${color.reset}` +\n (res ? ` ${statusColor}->${color.reset} ${res}` : \"\") +\n (err ? ` ${color.yellow}error=${err}${color.reset}` : \"\")\n );\n }\n}\n","/**\n * System prompt and message construction for the AI test healer.\n */\n\nimport type { loadCanaryConfig } from \"./config\";\nimport type { FailureContext, TestContext } from \"./healer\";\nimport { sanitizeString, sanitizeUnknown } from \"./healer\";\n\ntype PageLike = {\n accessibility?: { snapshot: () => Promise<unknown> };\n screenshot?: (options?: { fullPage?: boolean }) => Promise<Buffer | Uint8Array | string>;\n evaluate?: (script: string) => Promise<unknown>;\n};\n\ntype ToolConfig = ReturnType<typeof loadCanaryConfig>;\n\nfunction resolveMode(config: ToolConfig): \"dry-run\" | \"warn\" | \"full\" {\n if (config.dryRun) return \"dry-run\";\n if (config.warnOnly) return \"warn\";\n return \"full\";\n}\n\nfunction truncate(value: string, max: number): string {\n if (value.length <= max) return value;\n return value.slice(0, max) + \"...\";\n}\n\nfunction safeJson(value: unknown): string {\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n}\n\nexport function buildSystemPrompt(\n testContext: TestContext,\n config: ToolConfig\n): string {\n const testInfo = testContext.testFile\n ? `\\n## Test Context\\nFile: ${testContext.testFile}\\nTest: \"${testContext.testTitle ?? \"unknown\"}\"\\n`\n : \"\";\n\n const testSource = testContext.testSource\n ? `\\n### Full Test Code:\\n\\`\\`\\`typescript\\n${testContext.testSource}\\n\\`\\`\\`\\n`\n : \"\";\n\n const expectedAfter = testContext.expectedAfter\n ? `\\n### What the Test Expects After This Step:\\n${testContext.expectedAfter}\\n`\n : \"\";\n\n return `You are an AI test healer. A Playwright test step failed. Your job is to make the browser ready for the next step in the test **without violating the intent of the test**. Heal only the failing step; do not advance beyond it.\n\n## Your Goal\nGet the browser into the state the test expects, then call mark_complete.\n${testInfo}${testSource}\n### Current Step That Failed:\n${testContext.currentStep}\n${expectedAfter}\n## The Failure\nAction attempted: ${testContext.action}\nTarget: ${testContext.target ?? \"unknown\"}\nError: ${testContext.errorMessage ?? \"unknown\"}\n\n## Test Intent and Discipline\n- Adhere to the spirit of the test. If advancing would bypass a required validation or skip a missing field, you must not proceed—report failure instead.\n- Heal only this step. Avoid extra navigation or state changes unrelated to the intended action. If you take a detour, undo it before completion.\n- If you cannot confidently achieve the expected postcondition for this step, do not mark complete; let the test fail.\n\n## Tools Available\n\n### Inspection (use first!)\n- snapshot() - Get accessibility tree to see page structure\n- screenshot() - Capture visual screenshot\n- get_text(selector) - Get text content of an element\n- is_visible(selector) - Check if element is visible\n\n### Actions\n- click(text) - Click element by visible text\n- click_selector(selector) - Click by CSS/Playwright selector\n- fill_by_name(name, value) - PREFERRED: Fill textbox by accessible name from snapshot (e.g. textbox \"Enter 6-digit code\")\n- fill(selector, value) - Fill input by CSS selector (use placeholder attribute, NOT aria-label)\n- type(selector, text) - Type text character by character\n- press_key(key) - Press keyboard key (Enter, Tab, etc.)\n- hover(selector) - Hover over element\n- select_option(selector, value) - Select from dropdown\n\n### Navigation & Waiting\n- wait(ms) - Wait for a duration\n- wait_for_selector(selector, state?) - Wait for element state\n- navigate(url) - Navigate to URL\n- go_back() - Go back in history\n- reload() - Reload page\n\n### Power Tool\n- run_playwright_code(code) - Execute any Playwright code with \\`page\\` object\n Example: \\`await page.getByRole('button', { name: 'Submit' }).click();\\`\n\n### Completion (REQUIRED)\n- mark_complete(reason, summary) - Call when browser is ready for test to continue\n - reason: technical description of what was done\n - summary: one-line human-readable summary for test report (e.g. \"Clicked 'Send OTP' instead of 'Submit'\")\n\n## CRITICAL RULES - READ CAREFULLY\n\n### Rule 1: YOU MUST PERFORM THE ACTION YOURSELF\nThe test tried to ${testContext.action}(\"${testContext.target ?? \"\"}\") but it failed.\nYOU must perform the equivalent action on the correct element.\nDO NOT leave it for \"the next step\" - there is no next step until YOU complete the action.\n\n### Rule 2: The workflow is ALWAYS:\n1. snapshot() - understand the page\n2. click() / fill() / type() - PERFORM THE ACTION on the correct element\n3. mark_complete() - report what you did\n\n### Rule 3: NEVER call mark_complete without performing an action\nIf mark_complete is called without a click/fill/type, the test WILL FAIL.\n\n## Example 1: Test tried click(\"Submit\") but button says \"Send OTP\"\nCORRECT:\n1. snapshot() → see \"Send OTP\" button exists\n2. click(\"Send OTP\") → CLICK THE BUTTON\n3. mark_complete({reason: \"clicked Send OTP button\", summary: \"Clicked 'Send OTP' instead of 'Submit'\"})\n\nWRONG (this will cause test failure):\n1. snapshot() → see \"Send OTP\" button exists\n2. mark_complete(...) ← WRONG! No click performed!\n\n## Example 2: Test tried fill with wrong placeholder\nIf snapshot shows: textbox \"Enter 6-digit code\" [ref=e37]\nCORRECT:\n1. snapshot() → see textbox \"Enter 6-digit code\"\n2. fill_by_name(\"Enter 6-digit code\", \"111222\") → USE THE NAME FROM SNAPSHOT\n3. mark_complete({reason: \"filled by accessible name\", summary: \"Filled 'Enter 6-digit code' field\"})\n\n## If healing would break test intent\n- If the page is missing required UI (e.g., a field not present) or advancing would skip validation, do NOT hack around it. Report failure by not calling mark_complete.\n- If you temporarily navigate or change state, undo it before completion so the test can continue from the intended point.\n\n## Start now\n1. Call snapshot()\n2. Identify the correct element for the intended action\n3. PERFORM THE ACTION (click, fill, type, etc.)\n4. Call mark_complete with what you did\n\nMode: ${resolveMode(config)}.`;\n}\n\nexport function buildInitialMessages(\n testContext: TestContext,\n failure: FailureContext,\n initial: { snapshot?: unknown; html?: string; screenshot?: string }\n): Array<{ role: \"user\"; content: string }> {\n const parts: string[] = [];\n parts.push(\n `Please heal this test step. Start by reviewing the snapshot below. Failing step: ${failure.action ?? \"unknown\"}(${failure.target ?? \"unknown\"}). Error: ${failure.errorMessage ?? \"unknown\"}.`\n );\n\n if (testContext.testSource) {\n parts.push(\n `Full test source (truncated/redacted):\\n\\`\\`\\`typescript\\n${truncate(testContext.testSource, 15000)}\\n\\`\\`\\``\n );\n }\n\n parts.push(`Test file: ${testContext.testFile ?? \"unknown\"}`);\n parts.push(`Test title: ${testContext.testTitle ?? \"unknown\"}`);\n\n if (initial.snapshot) {\n parts.push(`Initial accessibility snapshot (sanitized):\\n\\`\\`\\`json\\n${truncate(safeJson(initial.snapshot), 8000)}\\n\\`\\`\\``);\n }\n if (initial.html) {\n parts.push(`HTML fallback (sanitized):\\n\\`\\`\\`html\\n${truncate(initial.html, 4000)}\\n\\`\\`\\``);\n }\n if (initial.screenshot) {\n parts.push(`Screenshot (base64 data URL, truncated): ${truncate(initial.screenshot, 12000)}`);\n }\n\n parts.push(`Remember: perform only the intended step and mark_complete only when the postcondition for this step is truly met.`);\n\n return [{ role: \"user\", content: parts.join(\"\\n\\n\") }];\n}\n\nexport async function buildInitialPageContext(\n page: PageLike | undefined,\n config: ToolConfig\n): Promise<{ snapshot?: unknown; html?: string; screenshot?: string }> {\n const maxPayloadBytes = Math.max(1000, config.maxPayloadBytes || 60000);\n const result: { snapshot?: unknown; html?: string; screenshot?: string } = {};\n if (!page) return result;\n\n try {\n if (page.accessibility?.snapshot) {\n const tree = await page.accessibility.snapshot();\n if (tree) {\n result.snapshot = sanitizeUnknown(tree, maxPayloadBytes);\n }\n }\n } catch {\n // ignore\n }\n\n if (!result.snapshot && page.evaluate) {\n try {\n const html = (await page.evaluate(\"document.body.innerHTML\")) as string;\n if (html) {\n result.html = truncate(sanitizeString(html), maxPayloadBytes);\n }\n } catch {\n // ignore\n }\n }\n\n if (config.visionEnabled && page.screenshot) {\n try {\n const buffer = await page.screenshot({ fullPage: false });\n const b = typeof buffer === \"string\" ? Buffer.from(buffer) : Buffer.from(buffer);\n const base64 = b.toString(\"base64\");\n if (b.byteLength <= maxPayloadBytes && base64.length <= maxPayloadBytes) {\n result.screenshot = `data:image/png;base64,${truncate(base64, maxPayloadBytes)}`;\n }\n } catch {\n // ignore\n }\n }\n\n return result;\n}\n"],"mappings":"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","names":["path","path","fs","path","path","fs","fs","truncate","safeJson","resolveMode","truncate","fs"]}
@@ -137,11 +137,20 @@ async function swapSessionContext(sessionId, params) {
137
137
  }
138
138
  async function getSessionStorageState(sessionId) {
139
139
  const port = await ensureDaemon();
140
- return daemonFetch(port, "GET", `/sessions/${sessionId}/storage-state`);
140
+ return daemonFetch(
141
+ port,
142
+ "GET",
143
+ `/sessions/${sessionId}/storage-state`
144
+ );
141
145
  }
142
146
  async function callTool(sessionId, toolName, args) {
143
147
  const port = await ensureDaemon();
144
- return daemonFetch(port, "POST", `/sessions/${sessionId}/tools/${toolName}`, args);
148
+ return daemonFetch(
149
+ port,
150
+ "POST",
151
+ `/sessions/${sessionId}/tools/${toolName}`,
152
+ args
153
+ );
145
154
  }
146
155
  async function resolveTargetSession(sessionIdOrName) {
147
156
  const result = await listSessions();
@@ -153,9 +162,7 @@ async function resolveTargetSession(sessionIdOrName) {
153
162
  throw new Error("No active sessions. Start one with: canary session start");
154
163
  }
155
164
  if (sessionIdOrName) {
156
- const match = sessions.find(
157
- (s) => s.id === sessionIdOrName || s.name === sessionIdOrName
158
- );
165
+ const match = sessions.find((s) => s.id === sessionIdOrName || s.name === sessionIdOrName);
159
166
  if (!match) {
160
167
  throw new Error(
161
168
  `Session "${sessionIdOrName}" not found. Active sessions: ${sessions.map((s) => s.id).join(", ")}`
@@ -183,4 +190,4 @@ export {
183
190
  callTool,
184
191
  resolveTargetSession
185
192
  };
186
- //# sourceMappingURL=chunk-CEW4BDXD.js.map
193
+ //# sourceMappingURL=chunk-Z3F373YR.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/session/daemon-client.ts"],"sourcesContent":["/**\n * Daemon client — HTTP client for the session daemon.\n *\n * Handles pidfile read/write, stale PID detection, auto-start,\n * and provides typed HTTP helpers for daemon communication.\n *\n * @module\n */\n\nimport { spawn } from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport { existsSync } from 'node:fs';\nimport path from 'node:path';\nimport os from 'node:os';\nimport type {\n DaemonState,\n DaemonResponse,\n SessionInfo,\n CreateSessionRequest,\n SwapContextRequest,\n ToolResponse,\n StorageStateResponse,\n} from './types.js';\n\nconst PIDFILE_DIR = path.join(os.homedir(), '.config', 'canary-cli');\nconst PIDFILE_PATH = path.join(PIDFILE_DIR, 'daemon.json');\nconst HEALTH_POLL_INTERVAL_MS = 100;\nconst HEALTH_POLL_TIMEOUT_MS = 15_000;\n\n/* ── Pidfile helpers ─────────────────────────────────────────────────── */\n\nasync function readPidfile(): Promise<DaemonState | null> {\n try {\n const content = await fs.readFile(PIDFILE_PATH, 'utf-8');\n return JSON.parse(content) as DaemonState;\n } catch {\n return null;\n }\n}\n\nfunction isProcessAlive(pid: number): boolean {\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n}\n\n/* ── HTTP helpers ────────────────────────────────────────────────────── */\n\nasync function daemonFetch(\n port: number,\n method: string,\n path: string,\n body?: unknown\n): Promise<unknown> {\n const url = `http://127.0.0.1:${port}${path}`;\n const res = await fetch(url, {\n method,\n headers: body ? { 'Content-Type': 'application/json' } : undefined,\n body: body ? JSON.stringify(body) : undefined,\n });\n return res.json();\n}\n\nasync function healthCheck(port: number): Promise<boolean> {\n try {\n const res = await fetch(`http://127.0.0.1:${port}/health`, {\n signal: AbortSignal.timeout(2000),\n });\n return res.ok;\n } catch {\n return false;\n }\n}\n\n/* ── Auto-start ──────────────────────────────────────────────────────── */\n\nasync function spawnDaemon(): Promise<number> {\n // Resolve the canary CLI entry point\n // In source: this file is src/session/daemon-client.ts → ../index.ts\n // In dist: this file is dist/chunk-*.js → ./index.js (flat)\n const dir = path.dirname(new URL(import.meta.url).pathname);\n const candidates = [\n path.resolve(dir, '..', 'index.ts'), // source layout\n path.resolve(dir, 'index.js'), // dist layout (flat)\n path.resolve(dir, '..', 'index.js'), // fallback\n ];\n const cliEntry = candidates.find((p) => existsSync(p));\n if (!cliEntry) {\n throw new Error(`Cannot find CLI entry point. Searched: ${candidates.join(', ')}`);\n }\n\n const child = spawn(process.execPath, [cliEntry, 'session', 'daemon'], {\n detached: true,\n stdio: ['ignore', 'pipe', 'ignore'],\n env: { ...process.env },\n });\n\n child.unref();\n\n // Wait for \"DAEMON_READY:<port>\" on stdout\n return new Promise<number>((resolve, reject) => {\n let output = '';\n const timeout = setTimeout(() => {\n reject(new Error('Daemon startup timed out'));\n }, HEALTH_POLL_TIMEOUT_MS);\n\n child.stdout!.on('data', (data: Buffer) => {\n output += data.toString();\n const match = output.match(/DAEMON_READY:(\\d+)/);\n if (match) {\n clearTimeout(timeout);\n resolve(parseInt(match[1], 10));\n }\n });\n\n child.on('error', (err) => {\n clearTimeout(timeout);\n reject(err);\n });\n\n child.on('exit', (code) => {\n clearTimeout(timeout);\n if (!output.includes('DAEMON_READY')) {\n reject(new Error(`Daemon exited with code ${code} before becoming ready`));\n }\n });\n });\n}\n\n/* ── Public API ──────────────────────────────────────────────────────── */\n\n/**\n * Ensure the daemon is running and return its port.\n * Starts the daemon if needed, cleans up stale pidfiles.\n */\nasync function ensureDaemon(): Promise<number> {\n const state = await readPidfile();\n\n if (state) {\n if (isProcessAlive(state.pid)) {\n // Verify it actually responds\n if (await healthCheck(state.port)) {\n return state.port;\n }\n }\n // Stale pidfile — clean up\n try {\n await fs.unlink(PIDFILE_PATH);\n } catch {\n // ignore\n }\n }\n\n // Spawn new daemon\n const port = await spawnDaemon();\n\n // Poll until healthy\n const deadline = Date.now() + HEALTH_POLL_TIMEOUT_MS;\n while (Date.now() < deadline) {\n if (await healthCheck(port)) return port;\n await new Promise((r) => setTimeout(r, HEALTH_POLL_INTERVAL_MS));\n }\n\n throw new Error('Daemon failed to become healthy after spawn');\n}\n\n/* ── Session operations ──────────────────────────────────────────────── */\n\nexport async function createSession(\n params: CreateSessionRequest\n): Promise<DaemonResponse<SessionInfo>> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'POST', '/sessions', params) as Promise<DaemonResponse<SessionInfo>>;\n}\n\nexport async function listSessions(): Promise<DaemonResponse<SessionInfo[]>> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'GET', '/sessions') as Promise<DaemonResponse<SessionInfo[]>>;\n}\n\nexport async function getSession(sessionId: string): Promise<DaemonResponse<SessionInfo>> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'GET', `/sessions/${sessionId}`) as Promise<DaemonResponse<SessionInfo>>;\n}\n\nexport async function deleteSession(sessionId: string): Promise<DaemonResponse> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'DELETE', `/sessions/${sessionId}`) as Promise<DaemonResponse>;\n}\n\nexport async function deleteAllSessions(): Promise<DaemonResponse> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'DELETE', '/sessions') as Promise<DaemonResponse>;\n}\n\nexport async function swapSessionContext(\n sessionId: string,\n params: SwapContextRequest\n): Promise<DaemonResponse<SessionInfo>> {\n const port = await ensureDaemon();\n return daemonFetch(port, 'POST', `/sessions/${sessionId}/swap-context`, params) as Promise<\n DaemonResponse<SessionInfo>\n >;\n}\n\nexport async function getSessionStorageState(sessionId: string): Promise<StorageStateResponse> {\n const port = await ensureDaemon();\n return daemonFetch(\n port,\n 'GET',\n `/sessions/${sessionId}/storage-state`\n ) as Promise<StorageStateResponse>;\n}\n\nexport async function callTool(\n sessionId: string,\n toolName: string,\n args: Record<string, unknown>\n): Promise<ToolResponse> {\n const port = await ensureDaemon();\n return daemonFetch(\n port,\n 'POST',\n `/sessions/${sessionId}/tools/${toolName}`,\n args\n ) as Promise<ToolResponse>;\n}\n\n/**\n * Resolve the target session for a command.\n * If there's exactly one session, auto-targets it.\n * If a sessionId or name is provided, looks it up.\n */\nexport async function resolveTargetSession(sessionIdOrName?: string): Promise<SessionInfo> {\n const result = await listSessions();\n if (!result.ok || !result.data) {\n throw new Error('Failed to list sessions');\n }\n const sessions = result.data;\n\n if (sessions.length === 0) {\n throw new Error('No active sessions. Start one with: canary session start');\n }\n\n if (sessionIdOrName) {\n const match = sessions.find((s) => s.id === sessionIdOrName || s.name === sessionIdOrName);\n if (!match) {\n throw new Error(\n `Session \"${sessionIdOrName}\" not found. Active sessions: ${sessions.map((s) => s.id).join(', ')}`\n );\n }\n return match;\n }\n\n if (sessions.length === 1) {\n return sessions[0];\n }\n\n throw new Error(\n `Multiple sessions active. Specify one with --session:\\n${sessions.map((s) => ` ${s.id} (${s.name})`).join('\\n')}`\n );\n}\n"],"mappings":"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","names":["path"]}
@@ -1,16 +1,16 @@
1
1
  import { createRequire as __cr } from "module"; const require = __cr(import.meta.url);
2
2
  import {
3
3
  downloadStorageState
4
- } from "./chunk-ERSNYLMZ.js";
4
+ } from "./chunk-BOS2YLKH.js";
5
5
  import {
6
6
  LocalBrowserHost
7
- } from "./chunk-Q7WFBG5C.js";
7
+ } from "./chunk-SVU2XTYZ.js";
8
8
  import {
9
9
  getArgValue,
10
10
  hasFlag,
11
11
  resolveConfig
12
- } from "./chunk-PWWQGYFG.js";
13
- import "./chunk-MSMC6UXW.js";
12
+ } from "./chunk-ACRIE2YR.js";
13
+ import "./chunk-IFOJT3A5.js";
14
14
  import "./chunk-XAA5VQ5N.js";
15
15
  import "./chunk-P5Z2Y5VV.js";
16
16
  import "./chunk-VKVL7WBN.js";
@@ -86,7 +86,7 @@ async function runDebugWorkflow(argv) {
86
86
  console.error(`Failed to create debug session: ${err}`);
87
87
  process.exit(1);
88
88
  }
89
- let debugSessionId = debugSession.sessionId;
89
+ const debugSessionId = debugSession.sessionId;
90
90
  let storageStatePath;
91
91
  if (debugSession.isAuthenticated && debugSession.credential) {
92
92
  console.log("Downloading storage state...");
@@ -181,7 +181,7 @@ async function runDebugWorkflow(argv) {
181
181
  return;
182
182
  }
183
183
  let currentStep = 0;
184
- let hasError = false;
184
+ let _hasError = false;
185
185
  const stepResults = /* @__PURE__ */ new Map();
186
186
  const parser = createParser({
187
187
  onEvent: (event) => {
@@ -209,7 +209,7 @@ async function runDebugWorkflow(argv) {
209
209
  if (e.result.errorMessage) {
210
210
  console.log(` Error: ${e.result.errorMessage.slice(0, 200)}`);
211
211
  }
212
- hasError = true;
212
+ _hasError = true;
213
213
  }
214
214
  stepResults.set(stepIndex, {
215
215
  name: `Step ${stepIndex}`,
@@ -225,9 +225,7 @@ async function runDebugWorkflow(argv) {
225
225
  `Stopped at step ${e.stepsExecuted} of ${e.totalSteps}. ${e.reason === "step-failed" ? "Step failed." : "Error occurred."}`
226
226
  );
227
227
  } else {
228
- console.log(
229
- `All ${e.stepsExecuted} step(s) completed.`
230
- );
228
+ console.log(`All ${e.stepsExecuted} step(s) completed.`);
231
229
  }
232
230
  }
233
231
  if (data.type === "run-error") {
@@ -236,7 +234,7 @@ async function runDebugWorkflow(argv) {
236
234
  if (errorData.error) {
237
235
  console.log(` Error: ${String(errorData.error).slice(0, 200)}`);
238
236
  }
239
- hasError = true;
237
+ _hasError = true;
240
238
  }
241
239
  } catch {
242
240
  }
@@ -262,4 +260,4 @@ async function runDebugWorkflow(argv) {
262
260
  export {
263
261
  runDebugWorkflow
264
262
  };
265
- //# sourceMappingURL=debug-workflow-53ULOFJC.js.map
263
+ //# sourceMappingURL=debug-workflow-K2LL6CO4.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/debug-workflow.ts"],"sourcesContent":["/**\n * Debug Workflow Command\n *\n * Runs a workflow in a local headed browser for debugging. Replays cached\n * actions where possible and falls back to the agent for uncached steps.\n * The browser stays open after execution so the user can inspect page state.\n *\n * Usage:\n * canary debug <workflowId> [--to-step N] [--api-url URL] [--env local|dev|prod]\n *\n * @module debug-workflow\n */\n\nimport fs from 'node:fs/promises';\nimport process from 'node:process';\nimport { createParser, type EventSourceMessage } from 'eventsource-parser';\nimport { resolveConfig, getArgValue, hasFlag } from './auth';\nimport { downloadStorageState } from './cli-helpers.js';\nimport { LocalBrowserHost } from './local-browser/host';\n\ninterface CreateSessionResponse {\n ok: boolean;\n sessionId: string;\n wsToken: string;\n wsUrl: string;\n expiresAt: string;\n error?: string;\n}\n\ninterface DebugSessionResponse {\n ok: boolean;\n sessionId?: string;\n workflowName?: string;\n nodes?: Array<{\n index: number;\n id: string;\n name: string;\n type: string;\n }>;\n isAuthenticated?: boolean;\n credential?: { credentialId: string; propertyId: string } | null;\n error?: string;\n}\n\ninterface StepStartedEvent {\n type: 'step-started';\n stepIndex: number;\n totalSteps: number;\n stepName: string;\n nodeType: string;\n nodeId: string;\n}\n\ninterface NodeCompletedEvent {\n type: 'completed';\n stepIndex?: number;\n result: { success: boolean; errorMessage?: string; source?: string };\n}\n\ninterface RunCompleteEvent {\n type: 'run-complete';\n stepsExecuted: number;\n totalSteps: number;\n stoppedEarly: boolean;\n reason?: string;\n}\n\ntype DebugEvent =\n | StepStartedEvent\n | NodeCompletedEvent\n | RunCompleteEvent\n | { type: string; [key: string]: unknown };\n\nexport async function runDebugWorkflow(argv: string[]): Promise<void> {\n const workflowId = argv.find((arg) => !arg.startsWith('-'));\n\n if (!workflowId) {\n console.error('Usage: canary debug <workflowId> [--to-step N] [--env local|dev|prod]');\n process.exit(1);\n }\n\n const toStepStr = getArgValue(argv, '--to-step');\n const toStep = toStepStr ? parseInt(toStepStr, 10) : undefined;\n const verbose = hasFlag(argv, '--verbose', '-v');\n\n if (toStep !== undefined && (isNaN(toStep) || toStep < 1)) {\n console.error('Error: --to-step must be a positive integer');\n process.exit(1);\n }\n\n const { apiUrl, token } = await resolveConfig(argv);\n\n // 1. Create local browser session\n console.log('Creating local browser session...');\n\n let localSession: CreateSessionResponse;\n try {\n const res = await fetch(`${apiUrl}/local-browser/sessions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({ browserMode: 'playwright' }),\n });\n\n if (!res.ok) {\n const text = await res.text();\n console.error(`Failed to create local browser session: ${res.status} ${text}`);\n process.exit(1);\n }\n\n localSession = (await res.json()) as CreateSessionResponse;\n if (!localSession.ok) {\n console.error(`Failed to create session: ${localSession.error}`);\n process.exit(1);\n }\n } catch (err) {\n console.error(`Failed to connect to API: ${err}`);\n process.exit(1);\n }\n\n // 2. Create debug session (before browser launch so we can download storage state)\n console.log('Loading workflow...');\n\n let debugSession: DebugSessionResponse;\n try {\n const res = await fetch(`${apiUrl}/cli/debug/sessions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${token}`,\n },\n body: JSON.stringify({\n workflowId,\n localBrowserSessionId: localSession.sessionId,\n }),\n });\n\n if (!res.ok) {\n const text = await res.text();\n console.error(`Failed to create debug session: ${res.status} ${text}`);\n process.exit(1);\n }\n\n debugSession = (await res.json()) as DebugSessionResponse;\n if (!debugSession.ok || !debugSession.sessionId) {\n console.error(`Failed to create debug session: ${debugSession.error}`);\n process.exit(1);\n }\n } catch (err) {\n console.error(`Failed to create debug session: ${err}`);\n process.exit(1);\n }\n\n const debugSessionId: string | null = debugSession.sessionId;\n\n // 3. Download storage state if the workflow has valid credentials\n let storageStatePath: string | undefined;\n if (debugSession.isAuthenticated && debugSession.credential) {\n console.log('Downloading storage state...');\n storageStatePath = await downloadStorageState({\n apiUrl,\n token,\n propertyId: debugSession.credential.propertyId,\n credentialId: debugSession.credential.credentialId,\n prefix: 'canary-debug-ss',\n });\n if (storageStatePath) {\n console.log('Storage state loaded.');\n } else {\n console.warn('Could not download storage state, continuing without it.');\n }\n }\n\n // 4. Launch headed browser with storage state\n console.log('Launching headed browser...');\n\n const host = new LocalBrowserHost({\n apiUrl,\n wsToken: localSession.wsToken,\n sessionId: localSession.sessionId,\n browserMode: 'playwright',\n headless: false,\n storageStatePath,\n onLog: (level, message, data) => {\n if (verbose) {\n const prefix = `[${level.toUpperCase()}]`;\n if (data) {\n console.log(prefix, message, data);\n } else {\n console.log(prefix, message);\n }\n }\n },\n });\n\n // Set up cleanup\n const cleanup = async () => {\n console.log('\\nShutting down...');\n\n if (debugSessionId) {\n try {\n await fetch(`${apiUrl}/cli/debug/sessions/${debugSessionId}`, {\n method: 'DELETE',\n headers: { Authorization: `Bearer ${token}` },\n });\n } catch {\n // Ignore cleanup errors\n }\n }\n\n if (storageStatePath) {\n await fs.unlink(storageStatePath).catch(() => {});\n }\n\n await host.stop();\n process.exit(0);\n };\n\n process.on('SIGINT', () => void cleanup());\n process.on('SIGTERM', () => void cleanup());\n\n try {\n await host.start();\n } catch (err) {\n console.error(`Failed to start local browser: ${err}`);\n await host.stop();\n process.exit(1);\n }\n\n console.log('Browser connected.');\n console.log();\n\n const nodes = debugSession.nodes ?? [];\n const totalSteps = nodes.length;\n const effectiveToStep = toStep ? Math.min(toStep, totalSteps) : totalSteps;\n\n console.log(`Debugging: \"${debugSession.workflowName}\" (${totalSteps} steps)`);\n if (toStep) {\n console.log(`Running steps 1-${effectiveToStep} of ${totalSteps}`);\n }\n console.log();\n\n // 5. Run steps via SSE\n let runRes: Response;\n try {\n runRes = await fetch(`${apiUrl}/cli/debug/sessions/${debugSessionId}/run`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${token}`,\n Accept: 'text/event-stream',\n },\n body: JSON.stringify({ toStep: effectiveToStep }),\n });\n } catch (err) {\n console.error(`Failed to start execution: ${err}`);\n await cleanup();\n return;\n }\n\n if (!runRes.ok || !runRes.body) {\n console.error(`Failed to start execution: ${runRes.status}`);\n await cleanup();\n return;\n }\n\n // Track execution state\n let currentStep = 0;\n let _hasError = false;\n const stepResults = new Map<number, { name: string; success: boolean; source?: string }>();\n\n const parser = createParser({\n onEvent: (event: EventSourceMessage) => {\n if (!event.data) return;\n\n try {\n const data = JSON.parse(event.data) as DebugEvent;\n\n if (verbose) {\n console.log(`[SSE] ${JSON.stringify(data)}`);\n }\n\n if (data.type === 'step-started') {\n const e = data as StepStartedEvent;\n currentStep = e.stepIndex;\n process.stdout.write(`[${e.stepIndex}/${e.totalSteps}] ${e.stepName} `);\n }\n\n if (data.type === 'completed') {\n const e = data as NodeCompletedEvent;\n const stepIndex = e.stepIndex ?? currentStep;\n const success = e.result.success;\n const source = e.result.source;\n\n if (success) {\n const sourceLabel =\n source === 'cache' ? '(cache)' : source === 'agent' ? '(agent)' : '';\n console.log(`\\u2713 ${sourceLabel}`);\n } else {\n console.log('\\u2717');\n if (e.result.errorMessage) {\n console.log(` Error: ${e.result.errorMessage.slice(0, 200)}`);\n }\n _hasError = true;\n }\n\n stepResults.set(stepIndex, {\n name: `Step ${stepIndex}`,\n success,\n source,\n });\n }\n\n if (data.type === 'run-complete') {\n const e = data as RunCompleteEvent;\n console.log();\n if (e.stoppedEarly) {\n console.log(\n `Stopped at step ${e.stepsExecuted} of ${e.totalSteps}. ${e.reason === 'step-failed' ? 'Step failed.' : 'Error occurred.'}`\n );\n } else {\n console.log(`All ${e.stepsExecuted} step(s) completed.`);\n }\n }\n\n if (data.type === 'run-error') {\n const errorData = data as { error?: string };\n console.log(`\\u2717`);\n if (errorData.error) {\n console.log(` Error: ${String(errorData.error).slice(0, 200)}`);\n }\n _hasError = true;\n }\n } catch {\n // Ignore parse errors\n }\n },\n });\n\n const reader = runRes.body.getReader();\n const decoder = new TextDecoder();\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n parser.feed(decoder.decode(value, { stream: true }));\n }\n } finally {\n reader.releaseLock();\n }\n\n // 6. Keep browser open for inspection\n console.log();\n console.log('Browser is open for inspection.');\n console.log('Press Ctrl+C to close.');\n\n // Keep the process running until Ctrl+C\n await new Promise(() => {});\n}\n"],"mappings":"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","names":[]}