pi-agent-browser-native 0.2.47 → 0.2.49

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 (185) hide show
  1. package/CHANGELOG.md +63 -19
  2. package/README.md +52 -19
  3. package/dist/extensions/agent-browser/index.js +785 -0
  4. package/dist/extensions/agent-browser/lib/argv-descriptor.js +71 -0
  5. package/dist/extensions/agent-browser/lib/argv-grammar.js +121 -0
  6. package/dist/extensions/agent-browser/lib/bash-guard.js +190 -0
  7. package/dist/extensions/agent-browser/lib/command-policy.js +85 -0
  8. package/dist/extensions/agent-browser/lib/command-taxonomy.js +302 -0
  9. package/dist/extensions/agent-browser/lib/config-policy.js +686 -0
  10. package/dist/extensions/agent-browser/lib/config.js +122 -0
  11. package/dist/extensions/agent-browser/lib/electron/cdp.js +51 -0
  12. package/dist/extensions/agent-browser/lib/electron/cleanup.js +212 -0
  13. package/dist/extensions/agent-browser/lib/electron/discovery.js +633 -0
  14. package/dist/extensions/agent-browser/lib/electron/launch.js +351 -0
  15. package/{extensions/agent-browser/lib/electron/text.ts → dist/extensions/agent-browser/lib/electron/text.js} +5 -5
  16. package/dist/extensions/agent-browser/lib/executable-path.js +20 -0
  17. package/dist/extensions/agent-browser/lib/fs-utils.js +18 -0
  18. package/dist/extensions/agent-browser/lib/input-modes/electron.js +165 -0
  19. package/dist/extensions/agent-browser/lib/input-modes/job.js +519 -0
  20. package/dist/extensions/agent-browser/lib/input-modes/lookups.js +440 -0
  21. package/dist/extensions/agent-browser/lib/input-modes/params.js +164 -0
  22. package/dist/extensions/agent-browser/lib/input-modes/semantic-action.js +119 -0
  23. package/dist/extensions/agent-browser/lib/input-modes/shared.js +42 -0
  24. package/dist/extensions/agent-browser/lib/input-modes/types.js +21 -0
  25. package/dist/extensions/agent-browser/lib/input-modes.js +10 -0
  26. package/dist/extensions/agent-browser/lib/json-schema.js +58 -0
  27. package/dist/extensions/agent-browser/lib/launch-scoped-flags.js +59 -0
  28. package/dist/extensions/agent-browser/lib/navigation-policy.js +83 -0
  29. package/dist/extensions/agent-browser/lib/orchestration/batch-stdin.js +62 -0
  30. package/dist/extensions/agent-browser/lib/orchestration/browser-run/artifact-paths.js +39 -0
  31. package/dist/extensions/agent-browser/lib/orchestration/browser-run/click-dispatch.js +276 -0
  32. package/dist/extensions/agent-browser/lib/orchestration/browser-run/diagnostics.js +909 -0
  33. package/dist/extensions/agent-browser/lib/orchestration/browser-run/final-result.js +443 -0
  34. package/dist/extensions/agent-browser/lib/orchestration/browser-run/index.js +47 -0
  35. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare/direct-anchor-download.js +141 -0
  36. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare/network-page-filter.js +108 -0
  37. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare/scroll-shims.js +112 -0
  38. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare/snapshot-filter.js +158 -0
  39. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare/wait-timeouts.js +54 -0
  40. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prepare.js +762 -0
  41. package/dist/extensions/agent-browser/lib/orchestration/browser-run/process-output.js +491 -0
  42. package/dist/extensions/agent-browser/lib/orchestration/browser-run/prompt-guards.js +40 -0
  43. package/dist/extensions/agent-browser/lib/orchestration/browser-run/session-artifacts.js +5 -0
  44. package/dist/extensions/agent-browser/lib/orchestration/browser-run/session-state.js +731 -0
  45. package/dist/extensions/agent-browser/lib/orchestration/browser-run/types.js +1 -0
  46. package/dist/extensions/agent-browser/lib/orchestration/electron-host/index.js +718 -0
  47. package/dist/extensions/agent-browser/lib/orchestration/input-plan.js +247 -0
  48. package/dist/extensions/agent-browser/lib/orchestration/output-file.js +68 -0
  49. package/{extensions/agent-browser/lib/parsing.ts → dist/extensions/agent-browser/lib/parsing.js} +12 -11
  50. package/dist/extensions/agent-browser/lib/pi-tool-rendering.js +241 -0
  51. package/dist/extensions/agent-browser/lib/playbook.js +121 -0
  52. package/dist/extensions/agent-browser/lib/process.js +448 -0
  53. package/dist/extensions/agent-browser/lib/prompt-policy.js +91 -0
  54. package/dist/extensions/agent-browser/lib/results/action-recommendations.js +220 -0
  55. package/dist/extensions/agent-browser/lib/results/artifact-manifest.js +111 -0
  56. package/{extensions/agent-browser/lib/results/artifact-state.ts → dist/extensions/agent-browser/lib/results/artifact-state.js} +4 -8
  57. package/dist/extensions/agent-browser/lib/results/categories.js +76 -0
  58. package/dist/extensions/agent-browser/lib/results/confirmation.js +63 -0
  59. package/dist/extensions/agent-browser/lib/results/contracts.js +8 -0
  60. package/dist/extensions/agent-browser/lib/results/editable-ref-evidence.js +74 -0
  61. package/dist/extensions/agent-browser/lib/results/envelope.js +166 -0
  62. package/dist/extensions/agent-browser/lib/results/network-routes.js +92 -0
  63. package/dist/extensions/agent-browser/lib/results/network.js +73 -0
  64. package/dist/extensions/agent-browser/lib/results/next-actions.js +72 -0
  65. package/dist/extensions/agent-browser/lib/results/presentation/artifacts.js +515 -0
  66. package/dist/extensions/agent-browser/lib/results/presentation/batch.js +397 -0
  67. package/dist/extensions/agent-browser/lib/results/presentation/browser-profile-recovery.js +55 -0
  68. package/dist/extensions/agent-browser/lib/results/presentation/common.js +46 -0
  69. package/dist/extensions/agent-browser/lib/results/presentation/content.js +24 -0
  70. package/dist/extensions/agent-browser/lib/results/presentation/diagnostics.js +960 -0
  71. package/dist/extensions/agent-browser/lib/results/presentation/errors.js +205 -0
  72. package/dist/extensions/agent-browser/lib/results/presentation/large-output.js +134 -0
  73. package/dist/extensions/agent-browser/lib/results/presentation/navigation.js +159 -0
  74. package/dist/extensions/agent-browser/lib/results/presentation/registry.js +216 -0
  75. package/dist/extensions/agent-browser/lib/results/presentation/semantic-action.js +104 -0
  76. package/dist/extensions/agent-browser/lib/results/presentation/skills.js +152 -0
  77. package/dist/extensions/agent-browser/lib/results/presentation.js +177 -0
  78. package/dist/extensions/agent-browser/lib/results/recovery-actions.js +107 -0
  79. package/dist/extensions/agent-browser/lib/results/recovery-next-actions.js +50 -0
  80. package/dist/extensions/agent-browser/lib/results/selector-recovery.js +225 -0
  81. package/{extensions/agent-browser/lib/results/shared.ts → dist/extensions/agent-browser/lib/results/shared.js} +0 -1
  82. package/dist/extensions/agent-browser/lib/results/snapshot-high-value-controls.js +208 -0
  83. package/dist/extensions/agent-browser/lib/results/snapshot-refs.js +78 -0
  84. package/dist/extensions/agent-browser/lib/results/snapshot-segments.js +331 -0
  85. package/dist/extensions/agent-browser/lib/results/snapshot-spill.js +40 -0
  86. package/dist/extensions/agent-browser/lib/results/snapshot.js +264 -0
  87. package/dist/extensions/agent-browser/lib/results/text.js +40 -0
  88. package/{extensions/agent-browser/lib/results.ts → dist/extensions/agent-browser/lib/results.js} +2 -32
  89. package/dist/extensions/agent-browser/lib/runtime.js +816 -0
  90. package/dist/extensions/agent-browser/lib/session-page-state.js +411 -0
  91. package/dist/extensions/agent-browser/lib/string-enum-schema.js +13 -0
  92. package/dist/extensions/agent-browser/lib/temp.js +498 -0
  93. package/dist/extensions/agent-browser/lib/web-search.js +562 -0
  94. package/docs/ARCHITECTURE.md +10 -10
  95. package/docs/COMMAND_REFERENCE.md +35 -21
  96. package/docs/ELECTRON.md +3 -3
  97. package/docs/RELEASE.md +46 -26
  98. package/docs/REQUIREMENTS.md +1 -1
  99. package/docs/SUPPORT_MATRIX.md +35 -106
  100. package/docs/TOOL_CONTRACT.md +23 -21
  101. package/package.json +12 -8
  102. package/scripts/agent-browser-capability-baseline.mjs +6 -3
  103. package/scripts/config.mjs +8 -2
  104. package/scripts/doctor.mjs +19 -17
  105. package/scripts/platform-smoke.mjs +1 -1
  106. package/extensions/agent-browser/index.ts +0 -952
  107. package/extensions/agent-browser/lib/argv-descriptor.ts +0 -90
  108. package/extensions/agent-browser/lib/argv-grammar.ts +0 -128
  109. package/extensions/agent-browser/lib/bash-guard.ts +0 -205
  110. package/extensions/agent-browser/lib/command-policy.ts +0 -71
  111. package/extensions/agent-browser/lib/command-taxonomy.ts +0 -336
  112. package/extensions/agent-browser/lib/config-policy.js +0 -690
  113. package/extensions/agent-browser/lib/config.ts +0 -209
  114. package/extensions/agent-browser/lib/electron/cdp.ts +0 -69
  115. package/extensions/agent-browser/lib/electron/cleanup.ts +0 -235
  116. package/extensions/agent-browser/lib/electron/discovery.ts +0 -710
  117. package/extensions/agent-browser/lib/electron/launch.ts +0 -499
  118. package/extensions/agent-browser/lib/executable-path.ts +0 -19
  119. package/extensions/agent-browser/lib/fs-utils.ts +0 -18
  120. package/extensions/agent-browser/lib/input-modes/electron.ts +0 -170
  121. package/extensions/agent-browser/lib/input-modes/job.ts +0 -451
  122. package/extensions/agent-browser/lib/input-modes/lookups.ts +0 -447
  123. package/extensions/agent-browser/lib/input-modes/params.ts +0 -205
  124. package/extensions/agent-browser/lib/input-modes/semantic-action.ts +0 -127
  125. package/extensions/agent-browser/lib/input-modes/shared.ts +0 -46
  126. package/extensions/agent-browser/lib/input-modes/types.ts +0 -225
  127. package/extensions/agent-browser/lib/input-modes.ts +0 -45
  128. package/extensions/agent-browser/lib/json-schema.ts +0 -73
  129. package/extensions/agent-browser/lib/launch-scoped-flags.ts +0 -67
  130. package/extensions/agent-browser/lib/navigation-policy.ts +0 -95
  131. package/extensions/agent-browser/lib/orchestration/batch-stdin.ts +0 -65
  132. package/extensions/agent-browser/lib/orchestration/browser-run/click-dispatch.ts +0 -257
  133. package/extensions/agent-browser/lib/orchestration/browser-run/diagnostics.ts +0 -912
  134. package/extensions/agent-browser/lib/orchestration/browser-run/final-result.ts +0 -512
  135. package/extensions/agent-browser/lib/orchestration/browser-run/index.ts +0 -53
  136. package/extensions/agent-browser/lib/orchestration/browser-run/prepare.ts +0 -1481
  137. package/extensions/agent-browser/lib/orchestration/browser-run/process-output.ts +0 -564
  138. package/extensions/agent-browser/lib/orchestration/browser-run/prompt-guards.ts +0 -47
  139. package/extensions/agent-browser/lib/orchestration/browser-run/session-state.ts +0 -868
  140. package/extensions/agent-browser/lib/orchestration/browser-run/types.ts +0 -564
  141. package/extensions/agent-browser/lib/orchestration/electron-host/index.ts +0 -855
  142. package/extensions/agent-browser/lib/orchestration/input-plan.ts +0 -375
  143. package/extensions/agent-browser/lib/orchestration/output-file.ts +0 -86
  144. package/extensions/agent-browser/lib/pi-tool-rendering.ts +0 -252
  145. package/extensions/agent-browser/lib/playbook.ts +0 -142
  146. package/extensions/agent-browser/lib/process.ts +0 -516
  147. package/extensions/agent-browser/lib/prompt-policy.ts +0 -105
  148. package/extensions/agent-browser/lib/results/action-recommendations.ts +0 -264
  149. package/extensions/agent-browser/lib/results/artifact-manifest.ts +0 -111
  150. package/extensions/agent-browser/lib/results/categories.ts +0 -106
  151. package/extensions/agent-browser/lib/results/confirmation.ts +0 -76
  152. package/extensions/agent-browser/lib/results/contracts.ts +0 -241
  153. package/extensions/agent-browser/lib/results/editable-ref-evidence.ts +0 -72
  154. package/extensions/agent-browser/lib/results/envelope.ts +0 -195
  155. package/extensions/agent-browser/lib/results/network-routes.ts +0 -83
  156. package/extensions/agent-browser/lib/results/network.ts +0 -78
  157. package/extensions/agent-browser/lib/results/next-actions.ts +0 -117
  158. package/extensions/agent-browser/lib/results/presentation/artifacts.ts +0 -588
  159. package/extensions/agent-browser/lib/results/presentation/batch.ts +0 -450
  160. package/extensions/agent-browser/lib/results/presentation/browser-profile-recovery.ts +0 -67
  161. package/extensions/agent-browser/lib/results/presentation/common.ts +0 -53
  162. package/extensions/agent-browser/lib/results/presentation/content.ts +0 -36
  163. package/extensions/agent-browser/lib/results/presentation/diagnostics.ts +0 -923
  164. package/extensions/agent-browser/lib/results/presentation/errors.ts +0 -227
  165. package/extensions/agent-browser/lib/results/presentation/large-output.ts +0 -182
  166. package/extensions/agent-browser/lib/results/presentation/navigation.ts +0 -184
  167. package/extensions/agent-browser/lib/results/presentation/registry.ts +0 -242
  168. package/extensions/agent-browser/lib/results/presentation/semantic-action.ts +0 -131
  169. package/extensions/agent-browser/lib/results/presentation/skills.ts +0 -143
  170. package/extensions/agent-browser/lib/results/presentation.ts +0 -257
  171. package/extensions/agent-browser/lib/results/recovery-actions.ts +0 -139
  172. package/extensions/agent-browser/lib/results/recovery-next-actions.ts +0 -71
  173. package/extensions/agent-browser/lib/results/selector-recovery.ts +0 -320
  174. package/extensions/agent-browser/lib/results/snapshot-high-value-controls.ts +0 -273
  175. package/extensions/agent-browser/lib/results/snapshot-refs.ts +0 -100
  176. package/extensions/agent-browser/lib/results/snapshot-segments.ts +0 -366
  177. package/extensions/agent-browser/lib/results/snapshot-spill.ts +0 -63
  178. package/extensions/agent-browser/lib/results/snapshot.ts +0 -329
  179. package/extensions/agent-browser/lib/results/text.ts +0 -40
  180. package/extensions/agent-browser/lib/runtime.ts +0 -988
  181. package/extensions/agent-browser/lib/session-page-state.ts +0 -512
  182. package/extensions/agent-browser/lib/string-enum-schema.ts +0 -20
  183. package/extensions/agent-browser/lib/temp.ts +0 -577
  184. package/extensions/agent-browser/lib/web-search.ts +0 -721
  185. /package/{extensions/agent-browser/lib/orchestration/browser-run.ts → dist/extensions/agent-browser/lib/orchestration/browser-run.js} +0 -0
@@ -0,0 +1,718 @@
1
+ /**
2
+ * Purpose: Own wrapper-side Electron host orchestration for agent_browser structured electron input.
3
+ * Responsibilities: Discover Electron apps, inspect/probe/cleanup wrapper-tracked Electron launches, and build Pi-facing Electron host results.
4
+ * Scope: Electron host actions that do not spawn the main upstream browser command; generic agent_browser execution stays in browser-run.
5
+ */
6
+ import { cleanupElectronLaunchResources, inspectElectronLaunchStatus } from "../../electron/cleanup.js";
7
+ import { discoverElectronApps } from "../../electron/discovery.js";
8
+ import { boundElectronProbeString } from "../../electron/text.js";
9
+ import { isRecord } from "../../parsing.js";
10
+ import { buildAgentBrowserNextActions, buildAgentBrowserResultCategoryDetails } from "../../results.js";
11
+ import { appendUniqueAgentBrowserNextActions } from "../../results/next-actions.js";
12
+ import { extractRefSnapshotFromData, isAboutBlankUrl, normalizeSessionTabTarget } from "../../session-page-state.js";
13
+ import { redactSensitiveText } from "../../runtime.js";
14
+ import { collectElectronManagedSessionTarget } from "../browser-run/diagnostics.js";
15
+ import { buildElectronHostFailureResult, formatElectronTargetLines, redactToolDetails } from "../browser-run/final-result.js";
16
+ import { buildElectronIdentifiers, buildElectronMismatchNextActions, buildElectronSessionMismatch, closeManagedSession, extractStringResultField, findElectronLaunchRecordForSession, formatElectronSessionMismatchText, getActiveElectronRecords, getLiveElectronRendererTargets, runSessionCommandData, } from "../browser-run/session-state.js";
17
+ const ELECTRON_PROFILE_ISOLATION_NOTE = "Profile note: electron.launch starts an isolated temporary profile; it does not reuse the app's normal signed-in profile or attach to an already-running authenticated app.";
18
+ const ELECTRON_EXISTING_AUTH_GUIDANCE = "For already-authenticated desktop app content, do not stop here: if host tools are allowed and the app is not running, launch the normal app with --remote-debugging-port=<port>, verify the port, then run agent_browser connect <port>; if it is already running without a debug port, ask before relaunching it.";
19
+ export const ELECTRON_PROFILE_ISOLATION_DETAILS = {
20
+ attachesToAlreadyRunningApp: false,
21
+ existingAuthenticatedAppGuidance: ELECTRON_EXISTING_AUTH_GUIDANCE,
22
+ hostDebugLaunchExample: "macOS: open -a <App Name> --args --remote-debugging-port=9222 --remote-allow-origins='*'; then agent_browser connect 9222 with sessionMode=fresh",
23
+ isolatedLaunch: true,
24
+ note: ELECTRON_PROFILE_ISOLATION_NOTE,
25
+ reusesExistingSignedInProfile: false,
26
+ };
27
+ export const ELECTRON_POST_COMMAND_STATUS_SETTLE_MS = 250;
28
+ const ELECTRON_PROBE_MAX_TABS = 6;
29
+ const ELECTRON_PROBE_MAX_REF_IDS = 20;
30
+ const ELECTRON_PROBE_MAX_SNAPSHOT_LINES = 12;
31
+ const ELECTRON_PROBE_MAX_SNAPSHOT_CHARS = 1_600;
32
+ function formatElectronListVisibleText(result) {
33
+ const visibleApps = result.apps.slice(0, 10);
34
+ const visibleOmittedCount = Math.max(0, result.apps.length - visibleApps.length);
35
+ const header = result.omittedCount > 0
36
+ ? `Electron apps (${result.apps.length} shown, ${result.omittedCount} omitted):`
37
+ : `Electron apps (${result.apps.length} found):`;
38
+ const lines = [header];
39
+ if (visibleApps.length === 0) {
40
+ lines.push(result.query ? `No Electron apps matched query "${result.query}".` : "No Electron apps found in the supported scan locations.");
41
+ }
42
+ else {
43
+ for (const app of visibleApps) {
44
+ const identifier = app.bundleId ?? app.desktopId;
45
+ const path = app.appPath ?? app.executablePath;
46
+ const sensitivity = app.sensitivity ? ` [likely sensitive: ${app.sensitivity.categories.join(", ")}]` : "";
47
+ lines.push(`- ${app.name}${identifier ? ` (${identifier})` : ""}${sensitivity} — ${path}`);
48
+ }
49
+ }
50
+ if (visibleOmittedCount > 0) {
51
+ lines.push(`${visibleOmittedCount} additional app(s) omitted from visible output; see details.electron.apps.`);
52
+ }
53
+ if (result.omittedCount > 0) {
54
+ lines.push(`${result.omittedCount} app(s) omitted by maxResults=${result.maxResults}.`);
55
+ }
56
+ if (result.apps.some((app) => app.sensitivity?.level === "likely-sensitive")) {
57
+ lines.push("Review likely-sensitive apps and use caller-owned allow/deny policy before launch.");
58
+ lines.push(ELECTRON_PROFILE_ISOLATION_NOTE);
59
+ lines.push(ELECTRON_EXISTING_AUTH_GUIDANCE);
60
+ }
61
+ return lines.join("\n");
62
+ }
63
+ function buildElectronListSuccessResult(compiledElectron, discovery) {
64
+ const text = redactSensitiveText(formatElectronListVisibleText(discovery));
65
+ const sensitiveAppCount = discovery.apps.filter((app) => app.sensitivity?.level === "likely-sensitive").length;
66
+ const details = {
67
+ args: [],
68
+ compiledElectron,
69
+ electron: {
70
+ action: "list",
71
+ apps: discovery.apps,
72
+ maxResults: discovery.maxResults,
73
+ omittedCount: discovery.omittedCount || undefined,
74
+ platform: discovery.platform,
75
+ profileIsolation: ELECTRON_PROFILE_ISOLATION_DETAILS,
76
+ query: discovery.query,
77
+ sensitiveAppCount: sensitiveAppCount || undefined,
78
+ skippedCount: discovery.skippedCount,
79
+ status: "succeeded",
80
+ },
81
+ ...buildAgentBrowserResultCategoryDetails({ args: [], succeeded: true }),
82
+ summary: discovery.omittedCount > 0
83
+ ? `Electron app discovery found ${discovery.apps.length} app(s) and omitted ${discovery.omittedCount}.`
84
+ : `Electron app discovery found ${discovery.apps.length} app(s).`,
85
+ };
86
+ return {
87
+ content: [{ type: "text", text }],
88
+ details: redactToolDetails(details, []),
89
+ isError: false,
90
+ };
91
+ }
92
+ function buildElectronListFailureResult(compiledElectron, error) {
93
+ const errorText = error instanceof Error ? error.message : String(error);
94
+ const text = redactSensitiveText(`Electron app discovery failed: ${errorText}`);
95
+ const details = {
96
+ args: [],
97
+ compiledElectron,
98
+ electron: {
99
+ action: "list",
100
+ error: errorText,
101
+ status: "failed",
102
+ },
103
+ ...buildAgentBrowserResultCategoryDetails({ args: [], errorText, succeeded: false }),
104
+ summary: "Electron app discovery failed.",
105
+ };
106
+ return {
107
+ content: [{ type: "text", text }],
108
+ details: redactToolDetails(details, []),
109
+ isError: true,
110
+ };
111
+ }
112
+ function isElectronLaunchRecord(value) {
113
+ if (!isRecord(value))
114
+ return false;
115
+ return value.version === 1 &&
116
+ value.launchedByWrapper === true &&
117
+ typeof value.launchId === "string" &&
118
+ typeof value.appName === "string" &&
119
+ typeof value.executablePath === "string" &&
120
+ typeof value.userDataDir === "string" &&
121
+ typeof value.port === "number" &&
122
+ typeof value.createdAtMs === "number";
123
+ }
124
+ export function restoreElectronLaunchRecordsFromBranch(branch) {
125
+ const records = new Map();
126
+ for (const entry of branch) {
127
+ if (!isRecord(entry) || entry.type !== "message")
128
+ continue;
129
+ const message = isRecord(entry.message) ? entry.message : undefined;
130
+ if (!message || message.toolName !== "agent_browser")
131
+ continue;
132
+ const details = isRecord(message.details) ? message.details : undefined;
133
+ const electron = isRecord(details?.electron) ? details.electron : undefined;
134
+ if (!electron)
135
+ continue;
136
+ const launch = isElectronLaunchRecord(electron.launch) ? electron.launch : undefined;
137
+ if (launch)
138
+ records.set(launch.launchId, launch);
139
+ const cleanupRecords = isRecord(electron.cleanup) && Array.isArray(electron.cleanup.records) ? electron.cleanup.records : [];
140
+ for (const cleanupRecord of cleanupRecords) {
141
+ if (isElectronLaunchRecord(cleanupRecord))
142
+ records.set(cleanupRecord.launchId, cleanupRecord);
143
+ }
144
+ }
145
+ return records;
146
+ }
147
+ function selectElectronRecords(compiledElectron, records) {
148
+ if (compiledElectron.launchId) {
149
+ const record = records.get(compiledElectron.launchId);
150
+ return record ? { records: [record] } : { error: `No wrapper-tracked Electron launch found for launchId ${compiledElectron.launchId}.` };
151
+ }
152
+ if (compiledElectron.all)
153
+ return { records: getActiveElectronRecords(records) };
154
+ const activeRecords = getActiveElectronRecords(records);
155
+ if (activeRecords.length === 0)
156
+ return { records: [] };
157
+ if (activeRecords.length > 1)
158
+ return { error: "Multiple wrapper-tracked Electron launches are active; pass electron.launchId or electron.all." };
159
+ return { records: activeRecords };
160
+ }
161
+ function extractTargetsFromStatus(statuses) {
162
+ return statuses.flatMap((status) => status.targets);
163
+ }
164
+ function formatElectronStatusVisibleText(statuses, records, mismatches = [], managedSessions = []) {
165
+ if (statuses.length === 0)
166
+ return "Electron status: no active wrapper-tracked launches.";
167
+ const recordsByLaunchId = new Map(records.map((record) => [record.launchId, record]));
168
+ const managedSessionsByName = new Map(managedSessions.map((managedSession) => [managedSession.sessionName, managedSession]));
169
+ const lines = [`Electron status: ${statuses.length} wrapper-tracked launch(es).`];
170
+ for (const status of statuses) {
171
+ const record = recordsByLaunchId.get(status.launchId);
172
+ const sessionName = record?.sessionName;
173
+ const appName = record?.appName ?? "Electron launch";
174
+ const sessionText = sessionName ? `, sessionName ${sessionName}` : "";
175
+ lines.push(`- ${status.launchId}: ${appName}${sessionText}; ${status.portAlive ? "debug port alive" : "debug port dead"}${status.pidAlive === undefined ? "" : status.pidAlive ? ", pid alive" : ", pid dead"} (port ${status.port})`);
176
+ lines.push(` Identifiers: launchId ${status.launchId}; sessionName ${sessionName ?? "not attached"}.`);
177
+ for (const targetLine of formatElectronTargetLines(status.targets, 4))
178
+ lines.push(` ${targetLine}`);
179
+ const managedSession = sessionName ? managedSessionsByName.get(sessionName) : undefined;
180
+ if (managedSession?.error)
181
+ lines.push(` Managed session warning: ${managedSession.error}`);
182
+ }
183
+ for (const mismatch of mismatches)
184
+ lines.push("", formatElectronSessionMismatchText(mismatch));
185
+ return lines.join("\n");
186
+ }
187
+ function buildElectronStatusResult(options) {
188
+ const baseNextActions = options.records.flatMap((record) => buildAgentBrowserNextActions({
189
+ electron: { launchId: record.launchId, sessionName: record.sessionName, status: record.cleanupState },
190
+ resultCategory: "success",
191
+ successCategory: "completed",
192
+ }) ?? []);
193
+ const mismatchNextActions = (options.mismatches ?? []).flatMap((mismatch) => {
194
+ const record = options.records.find((candidate) => candidate.launchId === mismatch.launchId);
195
+ return record ? buildElectronMismatchNextActions(record, mismatch.liveTarget) : [];
196
+ });
197
+ const nextActions = options.mismatches?.length
198
+ ? appendUniqueAgentBrowserNextActions([...mismatchNextActions], baseNextActions)
199
+ : appendUniqueAgentBrowserNextActions([...baseNextActions], mismatchNextActions);
200
+ const details = {
201
+ args: [],
202
+ compiledElectron: options.compiledElectron,
203
+ electron: {
204
+ action: "status",
205
+ identifierList: options.records.length > 1 ? options.records.map(buildElectronIdentifiers) : undefined,
206
+ identifiers: options.records.length === 1 && options.records[0] ? buildElectronIdentifiers(options.records[0]) : undefined,
207
+ launches: options.records,
208
+ managedSession: options.managedSessions?.length === 1 ? options.managedSessions[0] : undefined,
209
+ managedSessions: options.managedSessions && options.managedSessions.length > 0 ? options.managedSessions : undefined,
210
+ sessionMismatch: options.mismatches?.length === 1 ? options.mismatches[0] : undefined,
211
+ sessionMismatches: options.mismatches && options.mismatches.length > 1 ? options.mismatches : undefined,
212
+ status: "succeeded",
213
+ statuses: options.statuses,
214
+ targets: extractTargetsFromStatus(options.statuses),
215
+ },
216
+ nextActions: nextActions.length > 0 ? nextActions : undefined,
217
+ ...buildAgentBrowserResultCategoryDetails({ args: [], succeeded: true }),
218
+ summary: options.statuses.length === 0 ? "Electron status found no active wrapper-tracked launches." : `Electron status inspected ${options.statuses.length} launch(es).`,
219
+ };
220
+ return { content: [{ type: "text", text: redactSensitiveText(formatElectronStatusVisibleText(options.statuses, options.records, options.mismatches, options.managedSessions)) }], details: redactToolDetails(details, []), isError: false };
221
+ }
222
+ function formatElectronCleanupVisibleText(results) {
223
+ if (results.length === 0)
224
+ return "Electron cleanup: no active wrapper-tracked launches.";
225
+ const lines = [`Electron cleanup: ${results.filter((result) => !result.partial).length}/${results.length} launch(es) fully cleaned.`];
226
+ for (const result of results) {
227
+ lines.push(`- ${result.summary}`);
228
+ for (const step of result.steps)
229
+ lines.push(` - ${step.resource}: ${step.state}${step.error ? ` (${step.error})` : ""}`);
230
+ }
231
+ return lines.join("\n");
232
+ }
233
+ function buildElectronCleanupResult(compiledElectron, cleanupResults) {
234
+ const partial = cleanupResults.some((result) => result.partial);
235
+ const records = cleanupResults.map((result) => result.record);
236
+ const nextActions = cleanupResults.flatMap((result) => buildAgentBrowserNextActions({
237
+ electron: { launchId: result.launchId, sessionName: result.record.sessionName, status: result.record.cleanupState },
238
+ failureCategory: partial ? "cleanup-failed" : undefined,
239
+ resultCategory: partial ? "failure" : "success",
240
+ successCategory: partial ? undefined : "completed",
241
+ }) ?? []);
242
+ const errorText = partial ? cleanupResults.map((result) => result.summary).join("\n") : undefined;
243
+ const details = {
244
+ args: [],
245
+ compiledElectron,
246
+ electron: {
247
+ action: "cleanup",
248
+ cleanup: { partial, records, results: cleanupResults },
249
+ status: partial ? "partial" : "succeeded",
250
+ },
251
+ nextActions: nextActions.length > 0 ? nextActions : undefined,
252
+ ...buildAgentBrowserResultCategoryDetails({ args: [], errorText, failureCategory: partial ? "cleanup-failed" : undefined, succeeded: !partial }),
253
+ summary: partial ? "Electron cleanup was partial." : "Electron cleanup completed.",
254
+ };
255
+ return { content: [{ type: "text", text: redactSensitiveText(formatElectronCleanupVisibleText(cleanupResults)) }], details: redactToolDetails(details, []), isError: partial };
256
+ }
257
+ const ELECTRON_FOCUSED_ELEMENT_EVAL = `(() => {
258
+ const clean = (value, max = 80) => {
259
+ if (typeof value !== "string") return undefined;
260
+ const normalized = value.replace(/\\s+/g, " ").trim();
261
+ if (!normalized) return undefined;
262
+ return normalized.length > max ? normalized.slice(0, max - 3) + "..." : normalized;
263
+ };
264
+ const describeElement = (element) => {
265
+ if (!element || !(element instanceof Element)) return undefined;
266
+ const tagName = element.tagName.toLowerCase();
267
+ const inputLike = element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement || element instanceof HTMLSelectElement;
268
+ const contentEditable = element instanceof HTMLElement && element.isContentEditable;
269
+ const containerLike = tagName === "body" || tagName === "html";
270
+ const rawText = element.textContent || "";
271
+ const exposeText = !inputLike && !contentEditable && !containerLike;
272
+ const text = exposeText ? clean(rawText) : undefined;
273
+ return {
274
+ tagName: clean(tagName, 40),
275
+ role: clean(element.getAttribute("role") || "", 60),
276
+ name: clean(element.getAttribute("aria-label") || element.getAttribute("title") || text || "", 80),
277
+ id: clean(element.id || "", 80),
278
+ type: clean(element.getAttribute("type") || "", 40),
279
+ placeholder: clean(element.getAttribute("placeholder") || "", 80),
280
+ ariaLabel: clean(element.getAttribute("aria-label") || "", 80),
281
+ title: clean(element.getAttribute("title") || "", 80),
282
+ textLength: !exposeText && rawText ? rawText.length : undefined,
283
+ textPreview: text,
284
+ valueLength: inputLike && typeof element.value === "string" ? element.value.length : undefined,
285
+ isContentEditable: contentEditable || undefined,
286
+ };
287
+ };
288
+ return { focusedElement: describeElement(document.activeElement) };
289
+ })()`;
290
+ function getTrimmedString(value) {
291
+ return typeof value === "string" ? boundElectronProbeString(value) : undefined;
292
+ }
293
+ function getOptionalBoolean(value) {
294
+ return typeof value === "boolean" ? value : undefined;
295
+ }
296
+ function getOptionalNumber(value) {
297
+ return typeof value === "number" && Number.isFinite(value) ? value : undefined;
298
+ }
299
+ function extractElectronFocusedElement(data) {
300
+ const payload = isRecord(data) && isRecord(data.result) ? data.result : data;
301
+ const rawFocusedElement = isRecord(payload) && isRecord(payload.focusedElement) ? payload.focusedElement : isRecord(payload) ? payload : undefined;
302
+ if (!rawFocusedElement)
303
+ return undefined;
304
+ const focusedElement = {
305
+ ariaLabel: getTrimmedString(rawFocusedElement.ariaLabel),
306
+ id: getTrimmedString(rawFocusedElement.id),
307
+ isContentEditable: getOptionalBoolean(rawFocusedElement.isContentEditable),
308
+ name: getTrimmedString(rawFocusedElement.name),
309
+ placeholder: getTrimmedString(rawFocusedElement.placeholder),
310
+ role: getTrimmedString(rawFocusedElement.role),
311
+ tagName: getTrimmedString(rawFocusedElement.tagName),
312
+ textLength: getOptionalNumber(rawFocusedElement.textLength),
313
+ textPreview: getTrimmedString(rawFocusedElement.textPreview),
314
+ title: getTrimmedString(rawFocusedElement.title),
315
+ type: getTrimmedString(rawFocusedElement.type),
316
+ valueLength: getOptionalNumber(rawFocusedElement.valueLength),
317
+ };
318
+ return Object.values(focusedElement).some((value) => value !== undefined) ? focusedElement : undefined;
319
+ }
320
+ function extractElectronProbeTabs(data) {
321
+ const rawTabs = isRecord(data) && Array.isArray(data.tabs) ? data.tabs : Array.isArray(data) ? data : [];
322
+ const allTabs = rawTabs.filter(isRecord).map((tab, index) => ({
323
+ active: getOptionalBoolean(tab.active),
324
+ index: typeof tab.index === "number" && Number.isInteger(tab.index) ? tab.index : index,
325
+ tabId: getTrimmedString(tab.tabId) ?? getTrimmedString(tab.id),
326
+ title: getTrimmedString(tab.title) ?? getTrimmedString(tab.label),
327
+ type: getTrimmedString(tab.type),
328
+ url: getTrimmedString(tab.url),
329
+ }));
330
+ if (allTabs.length === 0)
331
+ return {};
332
+ const shown = allTabs.slice(0, ELECTRON_PROBE_MAX_TABS);
333
+ return {
334
+ activeTab: allTabs.find((tab) => tab.active) ?? allTabs[0],
335
+ tabs: {
336
+ omittedCount: allTabs.length > shown.length ? allTabs.length - shown.length : undefined,
337
+ shown,
338
+ total: allTabs.length,
339
+ },
340
+ };
341
+ }
342
+ function truncateElectronProbeSnapshotText(snapshotText) {
343
+ if (!snapshotText)
344
+ return { lineCount: 0 };
345
+ const lines = snapshotText.split(/\r?\n/);
346
+ const shownLines = [];
347
+ let usedChars = 0;
348
+ for (const line of lines) {
349
+ if (shownLines.length >= ELECTRON_PROBE_MAX_SNAPSHOT_LINES)
350
+ break;
351
+ const nextLength = usedChars + line.length + (shownLines.length > 0 ? 1 : 0);
352
+ if (nextLength > ELECTRON_PROBE_MAX_SNAPSHOT_CHARS) {
353
+ if (shownLines.length === 0)
354
+ shownLines.push(`${line.slice(0, ELECTRON_PROBE_MAX_SNAPSHOT_CHARS - 3)}...`);
355
+ break;
356
+ }
357
+ shownLines.push(line);
358
+ usedChars = nextLength;
359
+ }
360
+ return {
361
+ lineCount: lines.length,
362
+ omittedLineCount: lines.length > shownLines.length ? lines.length - shownLines.length : undefined,
363
+ text: shownLines.length > 0 ? shownLines.join("\n") : undefined,
364
+ };
365
+ }
366
+ function summarizeElectronProbeSnapshot(data) {
367
+ const refSnapshot = extractRefSnapshotFromData(data);
368
+ const rawSnapshotText = isRecord(data) ? getTrimmedString(data.snapshot) : undefined;
369
+ const truncatedText = truncateElectronProbeSnapshotText(rawSnapshotText);
370
+ const refIds = refSnapshot?.refIds ?? [];
371
+ const shownRefIds = refIds.slice(0, ELECTRON_PROBE_MAX_REF_IDS);
372
+ const snapshot = refSnapshot || truncatedText.text
373
+ ? {
374
+ lineCount: truncatedText.lineCount,
375
+ omittedLineCount: truncatedText.omittedLineCount,
376
+ omittedRefCount: refIds.length > shownRefIds.length ? refIds.length - shownRefIds.length : undefined,
377
+ refCount: refIds.length,
378
+ refIds: shownRefIds,
379
+ text: truncatedText.text,
380
+ }
381
+ : undefined;
382
+ return { refSnapshot, snapshot };
383
+ }
384
+ function getElectronProbeSummary(probe) {
385
+ const parts = [
386
+ probe.title ? `title "${probe.title}"` : undefined,
387
+ probe.url ? `url ${probe.url}` : undefined,
388
+ probe.focusedElement ? "focused element" : undefined,
389
+ probe.tabs ? `${probe.tabs.total} tab(s)` : undefined,
390
+ probe.snapshot ? `${probe.snapshot.refCount} ref(s)` : undefined,
391
+ ].filter((item) => item !== undefined);
392
+ return parts.length > 0 ? `Electron probe collected ${parts.join(", ")}.` : "Electron probe did not return current session state.";
393
+ }
394
+ async function runElectronProbeCommandData(options) {
395
+ try {
396
+ return { data: await runSessionCommandData(options) };
397
+ }
398
+ catch (error) {
399
+ return { error: error instanceof Error ? error.message : String(error) };
400
+ }
401
+ }
402
+ async function collectElectronProbe(options) {
403
+ const titleResult = await runElectronProbeCommandData({ args: ["get", "title"], cwd: options.cwd, sessionName: options.sessionName, signal: options.signal, timeoutMs: options.timeoutMs });
404
+ const urlResult = await runElectronProbeCommandData({ args: ["get", "url"], cwd: options.cwd, sessionName: options.sessionName, signal: options.signal, timeoutMs: options.timeoutMs });
405
+ const focusedResult = await runElectronProbeCommandData({ args: ["eval", "--stdin"], cwd: options.cwd, sessionName: options.sessionName, signal: options.signal, stdin: ELECTRON_FOCUSED_ELEMENT_EVAL, timeoutMs: options.timeoutMs });
406
+ const tabsResult = await runElectronProbeCommandData({ args: ["tab", "list"], cwd: options.cwd, sessionName: options.sessionName, signal: options.signal, timeoutMs: options.timeoutMs });
407
+ const snapshotResult = await runElectronProbeCommandData({ args: ["snapshot", "-i"], cwd: options.cwd, sessionName: options.sessionName, signal: options.signal, timeoutMs: options.timeoutMs });
408
+ const errors = [
409
+ titleResult.error ? `get title: ${titleResult.error}` : undefined,
410
+ urlResult.error ? `get url: ${urlResult.error}` : undefined,
411
+ focusedResult.error ? `focused element: ${focusedResult.error}` : undefined,
412
+ tabsResult.error ? `tab list: ${tabsResult.error}` : undefined,
413
+ snapshotResult.error ? `snapshot: ${snapshotResult.error}` : undefined,
414
+ ].filter((item) => item !== undefined).map((error) => boundElectronProbeString(error, 240) ?? "probe command failed");
415
+ const title = boundElectronProbeString(extractStringResultField(titleResult.data, "result") ?? extractStringResultField(titleResult.data, "title"), 160);
416
+ const url = boundElectronProbeString(extractStringResultField(urlResult.data, "result") ?? extractStringResultField(urlResult.data, "url"), 300);
417
+ const focusedElement = extractElectronFocusedElement(focusedResult.data);
418
+ const { activeTab, tabs } = extractElectronProbeTabs(tabsResult.data);
419
+ const { refSnapshot, snapshot } = summarizeElectronProbeSnapshot(snapshotResult.data);
420
+ const probeWithoutSummary = {
421
+ activeTab,
422
+ focusedElement,
423
+ errors: errors.length > 0 ? errors : undefined,
424
+ refSnapshot,
425
+ sessionName: options.sessionName,
426
+ snapshot,
427
+ status: errors.length === 0 && (title || url || focusedElement || tabs || snapshot) ? "succeeded" : "partial",
428
+ tabs,
429
+ title,
430
+ url,
431
+ };
432
+ return { ...probeWithoutSummary, summary: getElectronProbeSummary(probeWithoutSummary) };
433
+ }
434
+ function formatElectronProbeFocusedElement(focusedElement) {
435
+ if (!focusedElement)
436
+ return undefined;
437
+ const label = focusedElement.name ?? focusedElement.textPreview ?? focusedElement.placeholder ?? focusedElement.ariaLabel ?? focusedElement.title;
438
+ const descriptor = [focusedElement.role, focusedElement.tagName].filter(Boolean).join("/") || "element";
439
+ const suffix = [
440
+ focusedElement.id ? `#${focusedElement.id}` : undefined,
441
+ focusedElement.type ? `type=${focusedElement.type}` : undefined,
442
+ focusedElement.valueLength !== undefined ? `valueLength=${focusedElement.valueLength}` : undefined,
443
+ focusedElement.textLength !== undefined ? `textLength=${focusedElement.textLength}` : undefined,
444
+ ].filter((item) => item !== undefined).join(", ");
445
+ return `Focused: ${descriptor}${label ? ` "${label}"` : ""}${suffix ? ` (${suffix})` : ""}`;
446
+ }
447
+ function formatElectronProbeContextText(context) {
448
+ if (context.mode === "launchId") {
449
+ return `Probe context: wrapper launch ${context.launchId} session ${context.sessionName}.`;
450
+ }
451
+ if (context.note) {
452
+ return `Probe context: current managed session ${context.sessionName}; ${context.note}`;
453
+ }
454
+ if (context.launchId) {
455
+ return `Probe context: current managed session ${context.sessionName} maps to Electron launch ${context.launchId}.`;
456
+ }
457
+ return `Probe context: current managed session ${context.sessionName} only; pass electron.probe.launchId to compare wrapper-tracked launch status.`;
458
+ }
459
+ function formatElectronProbeLaunchStatusText(status, probe) {
460
+ if (!status)
461
+ return undefined;
462
+ const lines = [`Launch status: ${status.portAlive ? "debug port alive" : "debug port dead"}${status.pidAlive === undefined ? "" : status.pidAlive ? ", pid alive" : ", pid dead"}; ${status.targets.length} CDP target(s).`];
463
+ if (isAboutBlankUrl(probe.url) && (!status.portAlive || status.pidAlive === false || getLiveElectronRendererTargets(status.targets).length === 0)) {
464
+ lines.push("Electron lifecycle warning: the browser session is on about:blank and the wrapper launch has no live renderer target to reattach. Run electron.status, cleanup if dead, or relaunch the app.");
465
+ }
466
+ return lines.join("\n");
467
+ }
468
+ function formatElectronProbeVisibleText(options) {
469
+ const { context, mismatch, probe, status } = options;
470
+ const page = [probe.title, probe.url].filter(Boolean).join(" — ");
471
+ const lines = [`Electron probe: ${page || probe.sessionName}`];
472
+ if (context)
473
+ lines.push(formatElectronProbeContextText(context));
474
+ const launchStatusText = formatElectronProbeLaunchStatusText(status, probe);
475
+ if (launchStatusText)
476
+ lines.push(launchStatusText);
477
+ if (mismatch)
478
+ lines.push(formatElectronSessionMismatchText(mismatch));
479
+ const focusedLine = formatElectronProbeFocusedElement(probe.focusedElement);
480
+ if (focusedLine)
481
+ lines.push(focusedLine);
482
+ if (probe.tabs) {
483
+ const active = probe.activeTab;
484
+ lines.push(`Tabs: ${probe.tabs.total} total${probe.tabs.omittedCount ? ` (${probe.tabs.omittedCount} omitted)` : ""}${active ? `; active ${active.index ?? "?"}: ${[active.title, active.url].filter(Boolean).join(" — ") || active.tabId || "tab"}` : ""}`);
485
+ }
486
+ if (probe.snapshot) {
487
+ lines.push(`Snapshot: ${probe.snapshot.refCount} interactive ref(s)${probe.snapshot.omittedRefCount ? ` (${probe.snapshot.omittedRefCount} ref id(s) omitted)` : ""}.`);
488
+ if (probe.snapshot.text)
489
+ lines.push(probe.snapshot.text);
490
+ if (probe.snapshot.omittedLineCount)
491
+ lines.push(`... ${probe.snapshot.omittedLineCount} snapshot line(s) omitted`);
492
+ }
493
+ if (probe.status === "partial")
494
+ lines.push("Some probe commands did not return data; use raw agent_browser commands for deeper diagnostics.");
495
+ if (probe.errors && probe.errors.length > 0)
496
+ lines.push(`Probe warning: ${probe.errors.slice(0, 2).join("; ")}${probe.errors.length > 2 ? "; ..." : ""}`);
497
+ return lines.join("\n");
498
+ }
499
+ function buildElectronProbeResult(options) {
500
+ const { refSnapshot: _refSnapshot, ...boundedProbe } = options.probe;
501
+ const baseNextActions = options.record ? buildAgentBrowserNextActions({
502
+ electron: { launchId: options.record.launchId, sessionName: options.record.sessionName, status: options.record.cleanupState },
503
+ resultCategory: "success",
504
+ successCategory: "completed",
505
+ }) ?? [] : [];
506
+ const mismatchNextActions = options.mismatch && options.record ? buildElectronMismatchNextActions(options.record, options.mismatch.liveTarget) : [];
507
+ const nextActions = options.mismatch
508
+ ? appendUniqueAgentBrowserNextActions([...mismatchNextActions], baseNextActions)
509
+ : appendUniqueAgentBrowserNextActions([...baseNextActions], mismatchNextActions);
510
+ const details = {
511
+ args: [],
512
+ compiledElectron: options.compiledElectron,
513
+ electron: {
514
+ action: "probe",
515
+ identifiers: options.record ? buildElectronIdentifiers(options.record) : undefined,
516
+ probe: boundedProbe,
517
+ probeContext: options.probeContext,
518
+ sessionMismatch: options.mismatch,
519
+ status: options.probe.status,
520
+ statusTargets: options.status?.targets,
521
+ launchStatus: options.status,
522
+ },
523
+ nextActions: nextActions.length > 0 ? nextActions : undefined,
524
+ ...buildAgentBrowserResultCategoryDetails({ args: [], succeeded: true }),
525
+ sessionName: options.probe.sessionName,
526
+ sessionTabTarget: options.sessionTabTarget,
527
+ summary: options.mismatch?.summary ?? options.probe.summary,
528
+ usedImplicitSession: options.probeContext.mode === "current-managed-session",
529
+ };
530
+ return {
531
+ content: [{ type: "text", text: redactSensitiveText(formatElectronProbeVisibleText({ context: options.probeContext, mismatch: options.mismatch, probe: options.probe, status: options.status })) }],
532
+ details: redactToolDetails(details, []),
533
+ isError: false,
534
+ };
535
+ }
536
+ export async function cleanupTrackedElectronHostLaunches(options) {
537
+ const results = [];
538
+ for (const record of options.records) {
539
+ const managedSessionCloseError = record.sessionName
540
+ ? await closeManagedSession({ cwd: options.cwd, sessionName: record.sessionName, timeoutMs: options.timeoutMs })
541
+ : undefined;
542
+ const managedSessionStep = record.sessionName
543
+ ? managedSessionCloseError
544
+ ? { error: managedSessionCloseError, resource: "managed-session", sessionName: record.sessionName, state: "failed" }
545
+ : { resource: "managed-session", sessionName: record.sessionName, state: "removed" }
546
+ : undefined;
547
+ const cleanupResult = await cleanupElectronLaunchResources({
548
+ child: options.electronChildProcesses.get(record.launchId),
549
+ record,
550
+ timeoutMs: options.timeoutMs,
551
+ });
552
+ const cleanupRecord = record.sessionName && !managedSessionCloseError
553
+ ? { ...cleanupResult.record, sessionName: undefined }
554
+ : cleanupResult.record;
555
+ const result = managedSessionCloseError
556
+ ? {
557
+ ...cleanupResult,
558
+ partial: true,
559
+ record: { ...cleanupResult.record, cleanupState: "partial" },
560
+ remainingResources: [...new Set(["managed-session", ...cleanupResult.remainingResources])],
561
+ steps: [managedSessionStep, ...cleanupResult.steps].filter((step) => step !== undefined),
562
+ summary: `Electron cleanup for ${record.launchId} is partial; managed session close failed.`,
563
+ }
564
+ : {
565
+ ...cleanupResult,
566
+ record: cleanupRecord,
567
+ steps: [managedSessionStep, ...cleanupResult.steps].filter((step) => step !== undefined),
568
+ };
569
+ results.push(result);
570
+ options.electronLaunchRecords.set(record.launchId, result.record);
571
+ if (!result.partial)
572
+ options.electronChildProcesses.delete(record.launchId);
573
+ }
574
+ return results;
575
+ }
576
+ export async function cleanupActiveElectronHostLaunches(options) {
577
+ const activeRecords = getActiveElectronRecords(options.electronLaunchRecords);
578
+ return activeRecords.length > 0
579
+ ? cleanupTrackedElectronHostLaunches({ ...options, records: activeRecords })
580
+ : [];
581
+ }
582
+ export async function handleElectronHostInput(options) {
583
+ const { compiledElectron, cwd, electronChildProcesses, electronLaunchRecords, implicitSessionCloseTimeoutMs, managedSessionActive, managedSessionName, redactedCompiledElectron, sessionPageState, signal, } = options;
584
+ if (compiledElectron?.action === "list") {
585
+ try {
586
+ const discovery = await discoverElectronApps({ maxResults: compiledElectron.maxResults, query: compiledElectron.query });
587
+ return buildElectronListSuccessResult(redactedCompiledElectron ?? compiledElectron, discovery);
588
+ }
589
+ catch (error) {
590
+ return buildElectronListFailureResult(redactedCompiledElectron ?? compiledElectron, error);
591
+ }
592
+ }
593
+ if (compiledElectron?.action === "status") {
594
+ const selection = selectElectronRecords(compiledElectron, electronLaunchRecords);
595
+ if (selection.error)
596
+ return buildElectronHostFailureResult({ compiledElectron: redactedCompiledElectron ?? compiledElectron, errorText: selection.error, failureCategory: "validation-error" });
597
+ const records = selection.records ?? [];
598
+ const statuses = await Promise.all(records.map((record) => inspectElectronLaunchStatus(record)));
599
+ const managedSessions = (await Promise.all(records.map((record) => collectElectronManagedSessionTarget({
600
+ cwd,
601
+ sessionName: record.sessionName,
602
+ signal,
603
+ timeoutMs: compiledElectron.timeoutMs,
604
+ })))).filter((managedSession) => managedSession !== undefined);
605
+ const mismatches = managedSessions
606
+ .map((managedSession) => {
607
+ const record = records.find((candidate) => candidate.sessionName === managedSession.sessionName);
608
+ const status = record ? statuses.find((candidate) => candidate.launchId === record.launchId) : undefined;
609
+ return record && status ? buildElectronSessionMismatch({ managedSession, record, statusTargets: status.targets }) : undefined;
610
+ })
611
+ .filter((mismatch) => mismatch !== undefined);
612
+ return buildElectronStatusResult({
613
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
614
+ managedSessions,
615
+ mismatches,
616
+ records,
617
+ statuses,
618
+ });
619
+ }
620
+ if (compiledElectron?.action === "probe") {
621
+ const launchRecord = compiledElectron.launchId
622
+ ? electronLaunchRecords.get(compiledElectron.launchId)
623
+ : findElectronLaunchRecordForSession(managedSessionName, electronLaunchRecords);
624
+ if (compiledElectron.launchId && !launchRecord) {
625
+ return buildElectronHostFailureResult({
626
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
627
+ errorText: `No wrapper-tracked Electron launch found for launchId ${compiledElectron.launchId}.`,
628
+ failureCategory: "validation-error",
629
+ });
630
+ }
631
+ if (compiledElectron.launchId && !launchRecord?.sessionName) {
632
+ return buildElectronHostFailureResult({
633
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
634
+ errorText: `electron.probe launchId ${compiledElectron.launchId} has no attached managed sessionName; reattach with connect or run electron.launch again.`,
635
+ failureCategory: "validation-error",
636
+ });
637
+ }
638
+ if (!compiledElectron.launchId && !managedSessionActive) {
639
+ return buildElectronHostFailureResult({
640
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
641
+ errorText: "electron.probe requires an active attached session. Run electron.launch or connect to an Electron debug port first.",
642
+ failureCategory: "validation-error",
643
+ });
644
+ }
645
+ const probeSessionName = compiledElectron.launchId ? launchRecord?.sessionName : managedSessionName;
646
+ if (!probeSessionName) {
647
+ return buildElectronHostFailureResult({
648
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
649
+ errorText: "electron.probe could not resolve a managed session to inspect.",
650
+ failureCategory: "validation-error",
651
+ });
652
+ }
653
+ try {
654
+ const status = launchRecord ? await inspectElectronLaunchStatus(launchRecord) : undefined;
655
+ const probe = await collectElectronProbe({ cwd, sessionName: probeSessionName, signal, timeoutMs: compiledElectron.timeoutMs });
656
+ const managedSession = {
657
+ sessionName: probe.sessionName,
658
+ title: probe.title ?? probe.activeTab?.title,
659
+ url: probe.url ?? probe.activeTab?.url,
660
+ };
661
+ const sessionMismatch = launchRecord && status
662
+ ? buildElectronSessionMismatch({ managedSession, record: launchRecord, statusTargets: status.targets })
663
+ : undefined;
664
+ const probeContextNote = !launchRecord
665
+ ? "No wrapper-tracked Electron launch matched this current managed session."
666
+ : !compiledElectron.launchId && launchRecord.sessionName && launchRecord.sessionName !== probe.sessionName
667
+ ? `single active Electron launch ${launchRecord.launchId} uses wrapper session ${launchRecord.sessionName}; pass electron.probe.launchId to inspect that launch session directly.`
668
+ : undefined;
669
+ const probeContext = {
670
+ launchId: launchRecord?.launchId,
671
+ mode: compiledElectron.launchId ? "launchId" : "current-managed-session",
672
+ note: probeContextNote,
673
+ sessionName: probe.sessionName,
674
+ };
675
+ const sessionTabTarget = normalizeSessionTabTarget({
676
+ title: probe.title ?? probe.activeTab?.title ?? probe.refSnapshot?.target?.title,
677
+ url: probe.url ?? probe.activeTab?.url ?? probe.refSnapshot?.target?.url,
678
+ });
679
+ const pageStateUpdate = sessionPageState.beginUpdate();
680
+ if (sessionTabTarget) {
681
+ sessionPageState.applyTabTarget({ sessionName: probe.sessionName, target: sessionTabTarget, update: pageStateUpdate });
682
+ }
683
+ if (probe.refSnapshot) {
684
+ sessionPageState.applyRefSnapshot({
685
+ fallbackTarget: sessionTabTarget,
686
+ sessionName: probe.sessionName,
687
+ snapshot: probe.refSnapshot,
688
+ update: pageStateUpdate,
689
+ });
690
+ }
691
+ return buildElectronProbeResult({
692
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
693
+ mismatch: sessionMismatch,
694
+ probe,
695
+ probeContext,
696
+ record: launchRecord,
697
+ sessionTabTarget,
698
+ status,
699
+ });
700
+ }
701
+ catch (error) {
702
+ const errorText = error instanceof Error ? error.message : String(error);
703
+ return buildElectronHostFailureResult({
704
+ compiledElectron: redactedCompiledElectron ?? compiledElectron,
705
+ errorText: `Electron probe failed: ${errorText}`,
706
+ failureCategory: "upstream-error",
707
+ });
708
+ }
709
+ }
710
+ if (compiledElectron?.action === "cleanup") {
711
+ const selection = selectElectronRecords(compiledElectron, electronLaunchRecords);
712
+ if (selection.error)
713
+ return buildElectronHostFailureResult({ compiledElectron: redactedCompiledElectron ?? compiledElectron, errorText: selection.error, failureCategory: "validation-error" });
714
+ const cleanupResults = await cleanupTrackedElectronHostLaunches({ cwd, electronChildProcesses, electronLaunchRecords, records: selection.records ?? [], timeoutMs: compiledElectron.timeoutMs ?? implicitSessionCloseTimeoutMs });
715
+ return buildElectronCleanupResult(redactedCompiledElectron ?? compiledElectron, cleanupResults);
716
+ }
717
+ return undefined;
718
+ }