@bastani/atomic 0.7.17-1 → 0.8.0-0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1233) hide show
  1. package/CHANGELOG.md +4183 -0
  2. package/README.md +655 -0
  3. package/dist/builtin/intercom/CHANGELOG.md +195 -0
  4. package/dist/builtin/intercom/LICENSE +21 -0
  5. package/dist/builtin/intercom/README.md +484 -0
  6. package/dist/builtin/intercom/broker/broker.ts +346 -0
  7. package/dist/builtin/intercom/broker/client.ts +535 -0
  8. package/dist/builtin/intercom/broker/framing.ts +57 -0
  9. package/dist/builtin/intercom/broker/paths.ts +21 -0
  10. package/dist/builtin/intercom/broker/spawn.ts +308 -0
  11. package/dist/builtin/intercom/config.ts +109 -0
  12. package/dist/builtin/intercom/index.ts +1780 -0
  13. package/dist/builtin/intercom/package.json +59 -0
  14. package/dist/builtin/intercom/reply-tracker.ts +102 -0
  15. package/dist/builtin/intercom/skills/intercom/SKILL.md +513 -0
  16. package/dist/builtin/intercom/types.ts +46 -0
  17. package/dist/builtin/intercom/ui/compose.ts +139 -0
  18. package/dist/builtin/intercom/ui/inline-message.ts +76 -0
  19. package/dist/builtin/intercom/ui/session-list.ts +162 -0
  20. package/dist/builtin/mcp/CHANGELOG.md +346 -0
  21. package/dist/builtin/mcp/LICENSE +21 -0
  22. package/dist/builtin/mcp/OAUTH.md +324 -0
  23. package/dist/builtin/mcp/README.md +373 -0
  24. package/dist/builtin/mcp/agent-dir.ts +21 -0
  25. package/dist/builtin/mcp/app-bridge.bundle.js +67 -0
  26. package/dist/builtin/mcp/cli.js +186 -0
  27. package/dist/builtin/mcp/commands.ts +420 -0
  28. package/dist/builtin/mcp/config.ts +667 -0
  29. package/dist/builtin/mcp/consent-manager.ts +64 -0
  30. package/dist/builtin/mcp/direct-tools.ts +427 -0
  31. package/dist/builtin/mcp/errors.ts +219 -0
  32. package/dist/builtin/mcp/glimpse-ui.ts +80 -0
  33. package/dist/builtin/mcp/host-html-template.ts +427 -0
  34. package/dist/builtin/mcp/index.ts +334 -0
  35. package/dist/builtin/mcp/init.ts +336 -0
  36. package/dist/builtin/mcp/lifecycle.ts +93 -0
  37. package/dist/builtin/mcp/logger.ts +169 -0
  38. package/dist/builtin/mcp/mcp-auth-flow.ts +362 -0
  39. package/dist/builtin/mcp/mcp-auth.ts +297 -0
  40. package/dist/builtin/mcp/mcp-callback-server.ts +284 -0
  41. package/dist/builtin/mcp/mcp-oauth-provider.ts +302 -0
  42. package/dist/builtin/mcp/mcp-panel.ts +826 -0
  43. package/dist/builtin/mcp/mcp-setup-panel.ts +577 -0
  44. package/dist/builtin/mcp/metadata-cache.ts +201 -0
  45. package/dist/builtin/mcp/npx-resolver.ts +424 -0
  46. package/dist/builtin/mcp/oauth-handler.ts +60 -0
  47. package/dist/builtin/mcp/onboarding-state.ts +68 -0
  48. package/dist/builtin/mcp/package.json +61 -0
  49. package/dist/builtin/mcp/proxy-modes.ts +803 -0
  50. package/dist/builtin/mcp/resource-tools.ts +17 -0
  51. package/dist/builtin/mcp/sampling-handler.ts +268 -0
  52. package/dist/builtin/mcp/server-manager.ts +375 -0
  53. package/dist/builtin/mcp/state.ts +41 -0
  54. package/dist/builtin/mcp/tool-metadata.ts +152 -0
  55. package/dist/builtin/mcp/tool-registrar.ts +46 -0
  56. package/dist/builtin/mcp/tool-result-renderer.ts +65 -0
  57. package/dist/builtin/mcp/types.ts +441 -0
  58. package/dist/builtin/mcp/ui-resource-handler.ts +145 -0
  59. package/dist/builtin/mcp/ui-server.ts +623 -0
  60. package/dist/builtin/mcp/ui-session.ts +384 -0
  61. package/dist/builtin/mcp/ui-stream-types.ts +89 -0
  62. package/dist/builtin/mcp/utils.ts +129 -0
  63. package/dist/builtin/subagents/CHANGELOG.md +1019 -0
  64. package/dist/builtin/subagents/README.md +991 -0
  65. package/dist/builtin/subagents/agents/code-simplifier.md +84 -0
  66. package/dist/builtin/subagents/agents/codebase-analyzer.md +158 -0
  67. package/dist/builtin/subagents/agents/codebase-locator.md +113 -0
  68. package/dist/builtin/subagents/agents/codebase-online-researcher.md +317 -0
  69. package/dist/builtin/subagents/agents/codebase-pattern-finder.md +236 -0
  70. package/dist/builtin/subagents/agents/codebase-research-analyzer.md +181 -0
  71. package/dist/builtin/subagents/agents/codebase-research-locator.md +146 -0
  72. package/dist/builtin/subagents/agents/debugger.md +92 -0
  73. package/dist/builtin/subagents/package.json +67 -0
  74. package/dist/builtin/subagents/prompts/gather-context-and-clarify.md +20 -0
  75. package/dist/builtin/subagents/prompts/parallel-cleanup.md +60 -0
  76. package/dist/builtin/subagents/prompts/parallel-context-build.md +55 -0
  77. package/dist/builtin/subagents/prompts/parallel-handoff-plan.md +77 -0
  78. package/dist/builtin/subagents/prompts/parallel-research.md +58 -0
  79. package/dist/builtin/subagents/prompts/parallel-review.md +52 -0
  80. package/dist/builtin/subagents/prompts/review-loop.md +48 -0
  81. package/dist/builtin/subagents/skills/subagent/SKILL.md +734 -0
  82. package/dist/builtin/subagents/src/agents/agent-management.ts +644 -0
  83. package/dist/builtin/subagents/src/agents/agent-scope.ts +6 -0
  84. package/dist/builtin/subagents/src/agents/agent-selection.ts +23 -0
  85. package/dist/builtin/subagents/src/agents/agent-serializer.ts +84 -0
  86. package/dist/builtin/subagents/src/agents/agents.ts +809 -0
  87. package/dist/builtin/subagents/src/agents/chain-serializer.ts +137 -0
  88. package/dist/builtin/subagents/src/agents/frontmatter.ts +29 -0
  89. package/dist/builtin/subagents/src/agents/identity.ts +30 -0
  90. package/dist/builtin/subagents/src/agents/skills.ts +630 -0
  91. package/dist/builtin/subagents/src/extension/control-notices.ts +92 -0
  92. package/dist/builtin/subagents/src/extension/doctor.ts +199 -0
  93. package/dist/builtin/subagents/src/extension/index.ts +586 -0
  94. package/dist/builtin/subagents/src/extension/schemas.ts +168 -0
  95. package/dist/builtin/subagents/src/intercom/intercom-bridge.ts +378 -0
  96. package/dist/builtin/subagents/src/intercom/result-intercom.ts +269 -0
  97. package/dist/builtin/subagents/src/runs/background/async-execution.ts +612 -0
  98. package/dist/builtin/subagents/src/runs/background/async-job-tracker.ts +267 -0
  99. package/dist/builtin/subagents/src/runs/background/async-resume.ts +332 -0
  100. package/dist/builtin/subagents/src/runs/background/async-status.ts +295 -0
  101. package/dist/builtin/subagents/src/runs/background/completion-dedupe.ts +63 -0
  102. package/dist/builtin/subagents/src/runs/background/notify.ts +108 -0
  103. package/dist/builtin/subagents/src/runs/background/parallel-groups.ts +45 -0
  104. package/dist/builtin/subagents/src/runs/background/result-watcher.ts +250 -0
  105. package/dist/builtin/subagents/src/runs/background/run-status.ts +193 -0
  106. package/dist/builtin/subagents/src/runs/background/stale-run-reconciler.ts +291 -0
  107. package/dist/builtin/subagents/src/runs/background/subagent-runner.ts +1760 -0
  108. package/dist/builtin/subagents/src/runs/background/top-level-async.ts +13 -0
  109. package/dist/builtin/subagents/src/runs/foreground/chain-clarify.ts +1333 -0
  110. package/dist/builtin/subagents/src/runs/foreground/chain-execution.ts +932 -0
  111. package/dist/builtin/subagents/src/runs/foreground/execution.ts +902 -0
  112. package/dist/builtin/subagents/src/runs/foreground/subagent-executor.ts +2231 -0
  113. package/dist/builtin/subagents/src/runs/shared/completion-guard.ts +125 -0
  114. package/dist/builtin/subagents/src/runs/shared/long-running-guard.ts +175 -0
  115. package/dist/builtin/subagents/src/runs/shared/model-fallback.ts +103 -0
  116. package/dist/builtin/subagents/src/runs/shared/parallel-utils.ts +108 -0
  117. package/dist/builtin/subagents/src/runs/shared/pi-args.ts +163 -0
  118. package/dist/builtin/subagents/src/runs/shared/pi-spawn.ts +115 -0
  119. package/dist/builtin/subagents/src/runs/shared/run-history.ts +56 -0
  120. package/dist/builtin/subagents/src/runs/shared/single-output.ts +154 -0
  121. package/dist/builtin/subagents/src/runs/shared/subagent-control.ts +226 -0
  122. package/dist/builtin/subagents/src/runs/shared/subagent-prompt-runtime.ts +152 -0
  123. package/dist/builtin/subagents/src/runs/shared/worktree.ts +577 -0
  124. package/dist/builtin/subagents/src/shared/artifacts.ts +99 -0
  125. package/dist/builtin/subagents/src/shared/atomic-json.ts +16 -0
  126. package/dist/builtin/subagents/src/shared/file-coalescer.ts +40 -0
  127. package/dist/builtin/subagents/src/shared/fork-context.ts +76 -0
  128. package/dist/builtin/subagents/src/shared/formatters.ts +133 -0
  129. package/dist/builtin/subagents/src/shared/jsonl-writer.ts +81 -0
  130. package/dist/builtin/subagents/src/shared/model-info.ts +78 -0
  131. package/dist/builtin/subagents/src/shared/post-exit-stdio-guard.ts +85 -0
  132. package/dist/builtin/subagents/src/shared/session-identity.ts +10 -0
  133. package/dist/builtin/subagents/src/shared/session-tokens.ts +44 -0
  134. package/dist/builtin/subagents/src/shared/settings.ts +397 -0
  135. package/dist/builtin/subagents/src/shared/status-format.ts +49 -0
  136. package/dist/builtin/subagents/src/shared/types.ts +732 -0
  137. package/dist/builtin/subagents/src/shared/utils.ts +440 -0
  138. package/dist/builtin/subagents/src/slash/prompt-template-bridge.ts +397 -0
  139. package/dist/builtin/subagents/src/slash/slash-bridge.ts +174 -0
  140. package/dist/builtin/subagents/src/slash/slash-commands.ts +528 -0
  141. package/dist/builtin/subagents/src/slash/slash-live-state.ts +292 -0
  142. package/dist/builtin/subagents/src/tui/render-helpers.ts +80 -0
  143. package/dist/builtin/subagents/src/tui/render.ts +1257 -0
  144. package/dist/builtin/web-access/CHANGELOG.md +387 -0
  145. package/dist/builtin/web-access/LICENSE +21 -0
  146. package/dist/builtin/web-access/README.md +346 -0
  147. package/dist/builtin/web-access/activity.ts +101 -0
  148. package/dist/builtin/web-access/chrome-cookies.ts +322 -0
  149. package/dist/builtin/web-access/code-search.ts +107 -0
  150. package/dist/builtin/web-access/curator-page.ts +3359 -0
  151. package/dist/builtin/web-access/curator-server.ts +605 -0
  152. package/dist/builtin/web-access/exa.ts +521 -0
  153. package/dist/builtin/web-access/extract.ts +701 -0
  154. package/dist/builtin/web-access/gemini-api.ts +113 -0
  155. package/dist/builtin/web-access/gemini-search.ts +362 -0
  156. package/dist/builtin/web-access/gemini-url-context.ts +126 -0
  157. package/dist/builtin/web-access/gemini-web-config.ts +54 -0
  158. package/dist/builtin/web-access/gemini-web.ts +396 -0
  159. package/dist/builtin/web-access/github-api.ts +196 -0
  160. package/dist/builtin/web-access/github-extract.ts +635 -0
  161. package/dist/builtin/web-access/index.ts +2347 -0
  162. package/dist/builtin/web-access/package.json +54 -0
  163. package/dist/builtin/web-access/pdf-extract.ts +192 -0
  164. package/dist/builtin/web-access/perplexity.ts +196 -0
  165. package/dist/builtin/web-access/rsc-extract.ts +338 -0
  166. package/dist/builtin/web-access/storage.ts +72 -0
  167. package/dist/builtin/web-access/summary-review.ts +276 -0
  168. package/dist/builtin/web-access/utils.ts +44 -0
  169. package/dist/builtin/web-access/video-extract.ts +379 -0
  170. package/dist/builtin/web-access/youtube-extract.ts +311 -0
  171. package/dist/builtin/workflows/CHANGELOG.md +20 -0
  172. package/dist/builtin/workflows/README.md +323 -0
  173. package/dist/builtin/workflows/builtin/deep-research-codebase.ts +567 -0
  174. package/dist/builtin/workflows/builtin/index.ts +10 -0
  175. package/dist/builtin/workflows/builtin/open-claude-design.ts +985 -0
  176. package/dist/builtin/workflows/builtin/ralph.ts +613 -0
  177. package/dist/builtin/workflows/package.json +89 -0
  178. package/dist/builtin/workflows/skills/create-spec/SKILL.md +247 -0
  179. package/dist/builtin/workflows/skills/impeccable/SKILL.md +173 -0
  180. package/dist/builtin/workflows/skills/impeccable/reference/adapt.md +190 -0
  181. package/dist/builtin/workflows/skills/impeccable/reference/animate.md +175 -0
  182. package/dist/builtin/workflows/skills/impeccable/reference/audit.md +133 -0
  183. package/dist/builtin/workflows/skills/impeccable/reference/bolder.md +113 -0
  184. package/dist/builtin/workflows/skills/impeccable/reference/brand.md +118 -0
  185. package/dist/builtin/workflows/skills/impeccable/reference/clarify.md +174 -0
  186. package/dist/builtin/workflows/skills/impeccable/reference/codex.md +105 -0
  187. package/dist/builtin/workflows/skills/impeccable/reference/cognitive-load.md +106 -0
  188. package/dist/builtin/workflows/skills/impeccable/reference/color-and-contrast.md +105 -0
  189. package/dist/builtin/workflows/skills/impeccable/reference/colorize.md +154 -0
  190. package/dist/builtin/workflows/skills/impeccable/reference/craft.md +123 -0
  191. package/dist/builtin/workflows/skills/impeccable/reference/critique.md +261 -0
  192. package/dist/builtin/workflows/skills/impeccable/reference/delight.md +302 -0
  193. package/dist/builtin/workflows/skills/impeccable/reference/distill.md +111 -0
  194. package/dist/builtin/workflows/skills/impeccable/reference/document.md +427 -0
  195. package/dist/builtin/workflows/skills/impeccable/reference/extract.md +69 -0
  196. package/dist/builtin/workflows/skills/impeccable/reference/harden.md +347 -0
  197. package/dist/builtin/workflows/skills/impeccable/reference/heuristics-scoring.md +234 -0
  198. package/dist/builtin/workflows/skills/impeccable/reference/interaction-design.md +195 -0
  199. package/dist/builtin/workflows/skills/impeccable/reference/layout.md +141 -0
  200. package/dist/builtin/workflows/skills/impeccable/reference/live.md +622 -0
  201. package/dist/builtin/workflows/skills/impeccable/reference/motion-design.md +109 -0
  202. package/dist/builtin/workflows/skills/impeccable/reference/onboard.md +234 -0
  203. package/dist/builtin/workflows/skills/impeccable/reference/optimize.md +258 -0
  204. package/dist/builtin/workflows/skills/impeccable/reference/overdrive.md +130 -0
  205. package/dist/builtin/workflows/skills/impeccable/reference/personas.md +179 -0
  206. package/dist/builtin/workflows/skills/impeccable/reference/polish.md +242 -0
  207. package/dist/builtin/workflows/skills/impeccable/reference/product.md +62 -0
  208. package/dist/builtin/workflows/skills/impeccable/reference/quieter.md +99 -0
  209. package/dist/builtin/workflows/skills/impeccable/reference/responsive-design.md +114 -0
  210. package/dist/builtin/workflows/skills/impeccable/reference/shape.md +165 -0
  211. package/dist/builtin/workflows/skills/impeccable/reference/spatial-design.md +100 -0
  212. package/dist/builtin/workflows/skills/impeccable/reference/teach.md +156 -0
  213. package/dist/builtin/workflows/skills/impeccable/reference/typeset.md +124 -0
  214. package/dist/builtin/workflows/skills/impeccable/reference/typography.md +159 -0
  215. package/dist/builtin/workflows/skills/impeccable/reference/ux-writing.md +107 -0
  216. package/dist/builtin/workflows/skills/impeccable/scripts/cleanup-deprecated.mjs +284 -0
  217. package/dist/builtin/workflows/skills/impeccable/scripts/command-metadata.json +94 -0
  218. package/dist/builtin/workflows/skills/impeccable/scripts/critique-storage.mjs +226 -0
  219. package/dist/builtin/workflows/skills/impeccable/scripts/design-parser.mjs +820 -0
  220. package/dist/builtin/workflows/skills/impeccable/scripts/detect-csp.mjs +198 -0
  221. package/dist/builtin/workflows/skills/impeccable/scripts/impeccable-paths.mjs +110 -0
  222. package/dist/builtin/workflows/skills/impeccable/scripts/is-generated.mjs +69 -0
  223. package/dist/builtin/workflows/skills/impeccable/scripts/live-accept.mjs +646 -0
  224. package/dist/builtin/workflows/skills/impeccable/scripts/live-browser-session.js +123 -0
  225. package/dist/builtin/workflows/skills/impeccable/scripts/live-browser.js +4865 -0
  226. package/dist/builtin/workflows/skills/impeccable/scripts/live-complete.mjs +75 -0
  227. package/dist/builtin/workflows/skills/impeccable/scripts/live-completion.mjs +18 -0
  228. package/dist/builtin/workflows/skills/impeccable/scripts/live-inject.mjs +446 -0
  229. package/dist/builtin/workflows/skills/impeccable/scripts/live-poll.mjs +200 -0
  230. package/dist/builtin/workflows/skills/impeccable/scripts/live-resume.mjs +48 -0
  231. package/dist/builtin/workflows/skills/impeccable/scripts/live-server.mjs +847 -0
  232. package/dist/builtin/workflows/skills/impeccable/scripts/live-session-store.mjs +254 -0
  233. package/dist/builtin/workflows/skills/impeccable/scripts/live-status.mjs +47 -0
  234. package/dist/builtin/workflows/skills/impeccable/scripts/live-wrap.mjs +632 -0
  235. package/dist/builtin/workflows/skills/impeccable/scripts/live.mjs +247 -0
  236. package/dist/builtin/workflows/skills/impeccable/scripts/load-context.mjs +141 -0
  237. package/dist/builtin/workflows/skills/impeccable/scripts/modern-screenshot.umd.js +14 -0
  238. package/dist/builtin/workflows/skills/impeccable/scripts/pin.mjs +214 -0
  239. package/dist/builtin/workflows/skills/playwright-cli/SKILL.md +392 -0
  240. package/dist/builtin/workflows/skills/playwright-cli/references/element-attributes.md +23 -0
  241. package/dist/builtin/workflows/skills/playwright-cli/references/playwright-tests.md +39 -0
  242. package/dist/builtin/workflows/skills/playwright-cli/references/request-mocking.md +87 -0
  243. package/dist/builtin/workflows/skills/playwright-cli/references/running-code.md +241 -0
  244. package/dist/builtin/workflows/skills/playwright-cli/references/session-management.md +225 -0
  245. package/dist/builtin/workflows/skills/playwright-cli/references/spec-driven-testing.md +305 -0
  246. package/dist/builtin/workflows/skills/playwright-cli/references/storage-state.md +275 -0
  247. package/dist/builtin/workflows/skills/playwright-cli/references/test-generation.md +134 -0
  248. package/dist/builtin/workflows/skills/playwright-cli/references/tracing.md +139 -0
  249. package/dist/builtin/workflows/skills/playwright-cli/references/video-recording.md +143 -0
  250. package/dist/builtin/workflows/skills/prompt-engineer/SKILL.md +263 -0
  251. package/dist/builtin/workflows/skills/prompt-engineer/references/advanced_patterns.md +271 -0
  252. package/dist/builtin/workflows/skills/prompt-engineer/references/core_prompting.md +137 -0
  253. package/dist/builtin/workflows/skills/prompt-engineer/references/quality_improvement.md +193 -0
  254. package/dist/builtin/workflows/skills/research-codebase/SKILL.md +226 -0
  255. package/dist/builtin/workflows/skills/tdd/SKILL.md +109 -0
  256. package/dist/builtin/workflows/skills/tdd/deep-modules.md +33 -0
  257. package/dist/builtin/workflows/skills/tdd/interface-design.md +31 -0
  258. package/dist/builtin/workflows/skills/tdd/mocking.md +59 -0
  259. package/dist/builtin/workflows/skills/tdd/refactoring.md +10 -0
  260. package/dist/builtin/workflows/skills/tdd/tests.md +61 -0
  261. package/dist/builtin/workflows/skills/workflow/SKILL.md +255 -0
  262. package/dist/builtin/workflows/skills/workflow/references/context-engineering/advanced-evaluation.md +404 -0
  263. package/dist/builtin/workflows/skills/workflow/references/context-engineering/bdi-mental-states.md +313 -0
  264. package/dist/builtin/workflows/skills/workflow/references/context-engineering/context-compression.md +274 -0
  265. package/dist/builtin/workflows/skills/workflow/references/context-engineering/context-degradation.md +208 -0
  266. package/dist/builtin/workflows/skills/workflow/references/context-engineering/context-fundamentals.md +203 -0
  267. package/dist/builtin/workflows/skills/workflow/references/context-engineering/context-optimization.md +197 -0
  268. package/dist/builtin/workflows/skills/workflow/references/context-engineering/evaluation.md +253 -0
  269. package/dist/builtin/workflows/skills/workflow/references/context-engineering/filesystem-context.md +289 -0
  270. package/dist/builtin/workflows/skills/workflow/references/context-engineering/hosted-agents.md +262 -0
  271. package/dist/builtin/workflows/skills/workflow/references/context-engineering/memory-systems.md +221 -0
  272. package/dist/builtin/workflows/skills/workflow/references/context-engineering/multi-agent-patterns.md +259 -0
  273. package/dist/builtin/workflows/skills/workflow/references/context-engineering/project-development.md +293 -0
  274. package/dist/builtin/workflows/skills/workflow/references/context-engineering/tool-design.md +273 -0
  275. package/dist/builtin/workflows/skills/workflow/references/context-engineering.md +23 -0
  276. package/dist/builtin/workflows/skills/workflow/references/design-checklist.md +79 -0
  277. package/dist/builtin/workflows/skills/workflow/references/running-workflows.md +107 -0
  278. package/dist/builtin/workflows/skills/workflow/references/sdk-authoring.md +140 -0
  279. package/dist/builtin/workflows/src/extension/background-ui-adapter.ts +168 -0
  280. package/dist/builtin/workflows/src/extension/companions.ts +210 -0
  281. package/dist/builtin/workflows/src/extension/config-loader.ts +493 -0
  282. package/dist/builtin/workflows/src/extension/discovery.ts +501 -0
  283. package/dist/builtin/workflows/src/extension/dispatcher.ts +173 -0
  284. package/dist/builtin/workflows/src/extension/index.ts +2143 -0
  285. package/dist/builtin/workflows/src/extension/mcp.ts +110 -0
  286. package/dist/builtin/workflows/src/extension/render-call.ts +39 -0
  287. package/dist/builtin/workflows/src/extension/render-result.ts +214 -0
  288. package/dist/builtin/workflows/src/extension/renderers.ts +87 -0
  289. package/dist/builtin/workflows/src/extension/runtime.ts +360 -0
  290. package/dist/builtin/workflows/src/extension/status-writer.ts +167 -0
  291. package/dist/builtin/workflows/src/extension/wiring.ts +555 -0
  292. package/dist/builtin/workflows/src/extension/workflow-schema.ts +102 -0
  293. package/dist/builtin/workflows/src/index.ts +25 -0
  294. package/dist/builtin/workflows/src/intercom/intercom-bridge.ts +93 -0
  295. package/dist/builtin/workflows/src/intercom/intercom-routing.ts +125 -0
  296. package/dist/builtin/workflows/src/intercom/result-intercom.ts +240 -0
  297. package/dist/builtin/workflows/src/runs/background/cancellation-registry.ts +113 -0
  298. package/dist/builtin/workflows/src/runs/background/job-tracker.ts +81 -0
  299. package/dist/builtin/workflows/src/runs/background/runner.ts +152 -0
  300. package/dist/builtin/workflows/src/runs/background/status.ts +354 -0
  301. package/dist/builtin/workflows/src/runs/foreground/executor.ts +1522 -0
  302. package/dist/builtin/workflows/src/runs/foreground/stage-control-registry.ts +233 -0
  303. package/dist/builtin/workflows/src/runs/foreground/stage-runner.ts +712 -0
  304. package/dist/builtin/workflows/src/runs/shared/concurrency.ts +76 -0
  305. package/dist/builtin/workflows/src/runs/shared/graph-inference.ts +69 -0
  306. package/dist/builtin/workflows/src/runs/shared/model-fallback.ts +293 -0
  307. package/dist/builtin/workflows/src/runs/shared/validate-inputs.ts +83 -0
  308. package/dist/builtin/workflows/src/runs/shared/workflow-runner.ts +170 -0
  309. package/dist/builtin/workflows/src/runs/shared/worktree.ts +577 -0
  310. package/dist/builtin/workflows/src/shared/persistence-compaction-policy.ts +72 -0
  311. package/dist/builtin/workflows/src/shared/persistence-restore.ts +257 -0
  312. package/dist/builtin/workflows/src/shared/persistence-session-entries.ts +145 -0
  313. package/dist/builtin/workflows/src/shared/render-inputs-schema.ts +196 -0
  314. package/dist/builtin/workflows/src/shared/store-types.ts +160 -0
  315. package/dist/builtin/workflows/src/shared/store.ts +579 -0
  316. package/dist/builtin/workflows/src/shared/types.ts +566 -0
  317. package/dist/builtin/workflows/src/tui/chat-surface-message.ts +224 -0
  318. package/dist/builtin/workflows/src/tui/chat-surface.ts +511 -0
  319. package/dist/builtin/workflows/src/tui/color-utils.ts +64 -0
  320. package/dist/builtin/workflows/src/tui/connectors.ts +88 -0
  321. package/dist/builtin/workflows/src/tui/dispatch-confirm.ts +307 -0
  322. package/dist/builtin/workflows/src/tui/edge.ts +24 -0
  323. package/dist/builtin/workflows/src/tui/graph-canvas.ts +108 -0
  324. package/dist/builtin/workflows/src/tui/graph-theme.ts +283 -0
  325. package/dist/builtin/workflows/src/tui/graph-view.ts +1217 -0
  326. package/dist/builtin/workflows/src/tui/header.ts +172 -0
  327. package/dist/builtin/workflows/src/tui/inline-form-card.ts +421 -0
  328. package/dist/builtin/workflows/src/tui/inline-form-editor.ts +638 -0
  329. package/dist/builtin/workflows/src/tui/inline-form-overlay.ts +326 -0
  330. package/dist/builtin/workflows/src/tui/inline-form-store.ts +78 -0
  331. package/dist/builtin/workflows/src/tui/inputs-overlay.ts +163 -0
  332. package/dist/builtin/workflows/src/tui/inputs-picker.ts +888 -0
  333. package/dist/builtin/workflows/src/tui/keybindings-adapter.ts +154 -0
  334. package/dist/builtin/workflows/src/tui/layout.ts +153 -0
  335. package/dist/builtin/workflows/src/tui/node-card.ts +274 -0
  336. package/dist/builtin/workflows/src/tui/overlay-adapter.ts +277 -0
  337. package/dist/builtin/workflows/src/tui/prompt-card.ts +501 -0
  338. package/dist/builtin/workflows/src/tui/renderers.ts +15 -0
  339. package/dist/builtin/workflows/src/tui/run-detail.ts +339 -0
  340. package/dist/builtin/workflows/src/tui/session-confirm.ts +202 -0
  341. package/dist/builtin/workflows/src/tui/session-list.ts +32 -0
  342. package/dist/builtin/workflows/src/tui/session-overlays.ts +239 -0
  343. package/dist/builtin/workflows/src/tui/session-picker.ts +399 -0
  344. package/dist/builtin/workflows/src/tui/stage-chat-view.ts +1873 -0
  345. package/dist/builtin/workflows/src/tui/status-helpers.ts +73 -0
  346. package/dist/builtin/workflows/src/tui/status-list.ts +361 -0
  347. package/dist/builtin/workflows/src/tui/store-widget-installer.ts +206 -0
  348. package/dist/builtin/workflows/src/tui/switcher.ts +121 -0
  349. package/dist/builtin/workflows/src/tui/text-helpers.ts +31 -0
  350. package/dist/builtin/workflows/src/tui/toast.ts +106 -0
  351. package/dist/builtin/workflows/src/tui/widget.ts +348 -0
  352. package/dist/builtin/workflows/src/tui/workflow-attach-pane.ts +285 -0
  353. package/dist/builtin/workflows/src/tui/workflow-list.ts +224 -0
  354. package/dist/builtin/workflows/src/workflows/define-workflow.ts +150 -0
  355. package/dist/builtin/workflows/src/workflows/identity.ts +39 -0
  356. package/dist/builtin/workflows/src/workflows/registry.ts +113 -0
  357. package/dist/bun/cli.d.ts +3 -0
  358. package/dist/bun/cli.d.ts.map +1 -0
  359. package/dist/bun/cli.js +9 -0
  360. package/dist/bun/cli.js.map +1 -0
  361. package/dist/bun/register-bedrock.d.ts +2 -0
  362. package/dist/bun/register-bedrock.d.ts.map +1 -0
  363. package/dist/bun/register-bedrock.js +4 -0
  364. package/dist/bun/register-bedrock.js.map +1 -0
  365. package/dist/bun/restore-sandbox-env.d.ts +13 -0
  366. package/dist/bun/restore-sandbox-env.d.ts.map +1 -0
  367. package/dist/bun/restore-sandbox-env.js +32 -0
  368. package/dist/bun/restore-sandbox-env.js.map +1 -0
  369. package/dist/cli/args.d.ts +53 -0
  370. package/dist/cli/args.d.ts.map +1 -0
  371. package/dist/cli/args.js +341 -0
  372. package/dist/cli/args.js.map +1 -0
  373. package/dist/cli/config-selector.d.ts +14 -0
  374. package/dist/cli/config-selector.d.ts.map +1 -0
  375. package/dist/cli/config-selector.js +31 -0
  376. package/dist/cli/config-selector.js.map +1 -0
  377. package/dist/cli/file-processor.d.ts +15 -0
  378. package/dist/cli/file-processor.d.ts.map +1 -0
  379. package/dist/cli/file-processor.js +83 -0
  380. package/dist/cli/file-processor.js.map +1 -0
  381. package/dist/cli/initial-message.d.ts +18 -0
  382. package/dist/cli/initial-message.d.ts.map +1 -0
  383. package/dist/cli/initial-message.js +22 -0
  384. package/dist/cli/initial-message.js.map +1 -0
  385. package/dist/cli/list-models.d.ts +9 -0
  386. package/dist/cli/list-models.d.ts.map +1 -0
  387. package/dist/cli/list-models.js +98 -0
  388. package/dist/cli/list-models.js.map +1 -0
  389. package/dist/cli/session-picker.d.ts +9 -0
  390. package/dist/cli/session-picker.d.ts.map +1 -0
  391. package/dist/cli/session-picker.js +35 -0
  392. package/dist/cli/session-picker.js.map +1 -0
  393. package/dist/cli.d.ts +3 -0
  394. package/dist/cli.d.ts.map +1 -0
  395. package/dist/cli.js +20 -0
  396. package/dist/cli.js.map +1 -0
  397. package/dist/config.d.ts +102 -0
  398. package/dist/config.d.ts.map +1 -0
  399. package/dist/config.js +411 -0
  400. package/dist/config.js.map +1 -0
  401. package/dist/core/agent-session-runtime.d.ts +117 -0
  402. package/dist/core/agent-session-runtime.d.ts.map +1 -0
  403. package/dist/core/agent-session-runtime.js +292 -0
  404. package/dist/core/agent-session-runtime.js.map +1 -0
  405. package/dist/core/agent-session-services.d.ts +86 -0
  406. package/dist/core/agent-session-services.d.ts.map +1 -0
  407. package/dist/core/agent-session-services.js +117 -0
  408. package/dist/core/agent-session-services.js.map +1 -0
  409. package/dist/core/agent-session.d.ts +595 -0
  410. package/dist/core/agent-session.d.ts.map +1 -0
  411. package/dist/core/agent-session.js +2518 -0
  412. package/dist/core/agent-session.js.map +1 -0
  413. package/dist/core/auth-guidance.d.ts +5 -0
  414. package/dist/core/auth-guidance.d.ts.map +1 -0
  415. package/dist/core/auth-guidance.js +21 -0
  416. package/dist/core/auth-guidance.js.map +1 -0
  417. package/dist/core/auth-storage.d.ts +141 -0
  418. package/dist/core/auth-storage.d.ts.map +1 -0
  419. package/dist/core/auth-storage.js +437 -0
  420. package/dist/core/auth-storage.js.map +1 -0
  421. package/dist/core/bash-executor.d.ts +32 -0
  422. package/dist/core/bash-executor.d.ts.map +1 -0
  423. package/dist/core/bash-executor.js +111 -0
  424. package/dist/core/bash-executor.js.map +1 -0
  425. package/dist/core/builtin-packages.d.ts +14 -0
  426. package/dist/core/builtin-packages.d.ts.map +1 -0
  427. package/dist/core/builtin-packages.js +113 -0
  428. package/dist/core/builtin-packages.js.map +1 -0
  429. package/dist/core/compaction/branch-summarization.d.ts +88 -0
  430. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  431. package/dist/core/compaction/branch-summarization.js +243 -0
  432. package/dist/core/compaction/branch-summarization.js.map +1 -0
  433. package/dist/core/compaction/compaction.d.ts +121 -0
  434. package/dist/core/compaction/compaction.d.ts.map +1 -0
  435. package/dist/core/compaction/compaction.js +615 -0
  436. package/dist/core/compaction/compaction.js.map +1 -0
  437. package/dist/core/compaction/index.d.ts +7 -0
  438. package/dist/core/compaction/index.d.ts.map +1 -0
  439. package/dist/core/compaction/index.js +7 -0
  440. package/dist/core/compaction/index.js.map +1 -0
  441. package/dist/core/compaction/utils.d.ts +38 -0
  442. package/dist/core/compaction/utils.d.ts.map +1 -0
  443. package/dist/core/compaction/utils.js +153 -0
  444. package/dist/core/compaction/utils.js.map +1 -0
  445. package/dist/core/defaults.d.ts +3 -0
  446. package/dist/core/defaults.d.ts.map +1 -0
  447. package/dist/core/defaults.js +2 -0
  448. package/dist/core/defaults.js.map +1 -0
  449. package/dist/core/diagnostics.d.ts +15 -0
  450. package/dist/core/diagnostics.d.ts.map +1 -0
  451. package/dist/core/diagnostics.js +2 -0
  452. package/dist/core/diagnostics.js.map +1 -0
  453. package/dist/core/event-bus.d.ts +9 -0
  454. package/dist/core/event-bus.d.ts.map +1 -0
  455. package/dist/core/event-bus.js +25 -0
  456. package/dist/core/event-bus.js.map +1 -0
  457. package/dist/core/exec.d.ts +29 -0
  458. package/dist/core/exec.d.ts.map +1 -0
  459. package/dist/core/exec.js +75 -0
  460. package/dist/core/exec.js.map +1 -0
  461. package/dist/core/export-html/ansi-to-html.d.ts +22 -0
  462. package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
  463. package/dist/core/export-html/ansi-to-html.js +249 -0
  464. package/dist/core/export-html/ansi-to-html.js.map +1 -0
  465. package/dist/core/export-html/index.d.ts +37 -0
  466. package/dist/core/export-html/index.d.ts.map +1 -0
  467. package/dist/core/export-html/index.js +224 -0
  468. package/dist/core/export-html/index.js.map +1 -0
  469. package/dist/core/export-html/template.css +1066 -0
  470. package/dist/core/export-html/template.html +55 -0
  471. package/dist/core/export-html/template.js +1834 -0
  472. package/dist/core/export-html/tool-renderer.d.ts +34 -0
  473. package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
  474. package/dist/core/export-html/tool-renderer.js +108 -0
  475. package/dist/core/export-html/tool-renderer.js.map +1 -0
  476. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  477. package/dist/core/export-html/vendor/marked.min.js +6 -0
  478. package/dist/core/extensions/index.d.ts +12 -0
  479. package/dist/core/extensions/index.d.ts.map +1 -0
  480. package/dist/core/extensions/index.js +9 -0
  481. package/dist/core/extensions/index.js.map +1 -0
  482. package/dist/core/extensions/loader.d.ts +24 -0
  483. package/dist/core/extensions/loader.d.ts.map +1 -0
  484. package/dist/core/extensions/loader.js +501 -0
  485. package/dist/core/extensions/loader.js.map +1 -0
  486. package/dist/core/extensions/runner.d.ts +159 -0
  487. package/dist/core/extensions/runner.d.ts.map +1 -0
  488. package/dist/core/extensions/runner.js +817 -0
  489. package/dist/core/extensions/runner.js.map +1 -0
  490. package/dist/core/extensions/types.d.ts +1173 -0
  491. package/dist/core/extensions/types.d.ts.map +1 -0
  492. package/dist/core/extensions/types.js +45 -0
  493. package/dist/core/extensions/types.js.map +1 -0
  494. package/dist/core/extensions/wrapper.d.ts +20 -0
  495. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  496. package/dist/core/extensions/wrapper.js +22 -0
  497. package/dist/core/extensions/wrapper.js.map +1 -0
  498. package/dist/core/footer-data-provider.d.ts +52 -0
  499. package/dist/core/footer-data-provider.d.ts.map +1 -0
  500. package/dist/core/footer-data-provider.js +309 -0
  501. package/dist/core/footer-data-provider.js.map +1 -0
  502. package/dist/core/index.d.ts +12 -0
  503. package/dist/core/index.d.ts.map +1 -0
  504. package/dist/core/index.js +12 -0
  505. package/dist/core/index.js.map +1 -0
  506. package/dist/core/keybindings.d.ts +353 -0
  507. package/dist/core/keybindings.d.ts.map +1 -0
  508. package/dist/core/keybindings.js +294 -0
  509. package/dist/core/keybindings.js.map +1 -0
  510. package/dist/core/messages.d.ts +77 -0
  511. package/dist/core/messages.d.ts.map +1 -0
  512. package/dist/core/messages.js +123 -0
  513. package/dist/core/messages.js.map +1 -0
  514. package/dist/core/model-registry.d.ts +150 -0
  515. package/dist/core/model-registry.d.ts.map +1 -0
  516. package/dist/core/model-registry.js +726 -0
  517. package/dist/core/model-registry.js.map +1 -0
  518. package/dist/core/model-resolver.d.ts +110 -0
  519. package/dist/core/model-resolver.d.ts.map +1 -0
  520. package/dist/core/model-resolver.js +493 -0
  521. package/dist/core/model-resolver.js.map +1 -0
  522. package/dist/core/output-guard.d.ts +6 -0
  523. package/dist/core/output-guard.d.ts.map +1 -0
  524. package/dist/core/output-guard.js +59 -0
  525. package/dist/core/output-guard.js.map +1 -0
  526. package/dist/core/package-manager.d.ts +198 -0
  527. package/dist/core/package-manager.d.ts.map +1 -0
  528. package/dist/core/package-manager.js +1970 -0
  529. package/dist/core/package-manager.js.map +1 -0
  530. package/dist/core/prompt-templates.d.ts +52 -0
  531. package/dist/core/prompt-templates.d.ts.map +1 -0
  532. package/dist/core/prompt-templates.js +250 -0
  533. package/dist/core/prompt-templates.js.map +1 -0
  534. package/dist/core/provider-display-names.d.ts +2 -0
  535. package/dist/core/provider-display-names.d.ts.map +1 -0
  536. package/dist/core/provider-display-names.js +33 -0
  537. package/dist/core/provider-display-names.js.map +1 -0
  538. package/dist/core/resolve-config-value.d.ts +23 -0
  539. package/dist/core/resolve-config-value.d.ts.map +1 -0
  540. package/dist/core/resolve-config-value.js +126 -0
  541. package/dist/core/resolve-config-value.js.map +1 -0
  542. package/dist/core/resource-loader.d.ts +196 -0
  543. package/dist/core/resource-loader.d.ts.map +1 -0
  544. package/dist/core/resource-loader.js +698 -0
  545. package/dist/core/resource-loader.js.map +1 -0
  546. package/dist/core/sdk.d.ts +107 -0
  547. package/dist/core/sdk.d.ts.map +1 -0
  548. package/dist/core/sdk.js +291 -0
  549. package/dist/core/sdk.js.map +1 -0
  550. package/dist/core/session-cwd.d.ts +19 -0
  551. package/dist/core/session-cwd.d.ts.map +1 -0
  552. package/dist/core/session-cwd.js +37 -0
  553. package/dist/core/session-cwd.js.map +1 -0
  554. package/dist/core/session-manager.d.ts +333 -0
  555. package/dist/core/session-manager.d.ts.map +1 -0
  556. package/dist/core/session-manager.js +1118 -0
  557. package/dist/core/session-manager.js.map +1 -0
  558. package/dist/core/settings-manager.d.ts +261 -0
  559. package/dist/core/settings-manager.d.ts.map +1 -0
  560. package/dist/core/settings-manager.js +773 -0
  561. package/dist/core/settings-manager.js.map +1 -0
  562. package/dist/core/skills.d.ts +60 -0
  563. package/dist/core/skills.d.ts.map +1 -0
  564. package/dist/core/skills.js +404 -0
  565. package/dist/core/skills.js.map +1 -0
  566. package/dist/core/slash-commands.d.ts +14 -0
  567. package/dist/core/slash-commands.d.ts.map +1 -0
  568. package/dist/core/slash-commands.js +25 -0
  569. package/dist/core/slash-commands.js.map +1 -0
  570. package/dist/core/source-info.d.ts +18 -0
  571. package/dist/core/source-info.d.ts.map +1 -0
  572. package/dist/core/source-info.js +19 -0
  573. package/dist/core/source-info.js.map +1 -0
  574. package/dist/core/system-prompt.d.ts +28 -0
  575. package/dist/core/system-prompt.d.ts.map +1 -0
  576. package/dist/core/system-prompt.js +120 -0
  577. package/dist/core/system-prompt.js.map +1 -0
  578. package/dist/core/telemetry.d.ts +3 -0
  579. package/dist/core/telemetry.d.ts.map +1 -0
  580. package/dist/core/telemetry.js +10 -0
  581. package/dist/core/telemetry.js.map +1 -0
  582. package/dist/core/timings.d.ts +8 -0
  583. package/dist/core/timings.d.ts.map +1 -0
  584. package/dist/core/timings.js +32 -0
  585. package/dist/core/timings.js.map +1 -0
  586. package/dist/core/tools/ask-user-question/ask-user-question.d.ts +10 -0
  587. package/dist/core/tools/ask-user-question/ask-user-question.d.ts.map +1 -0
  588. package/dist/core/tools/ask-user-question/ask-user-question.js +82 -0
  589. package/dist/core/tools/ask-user-question/ask-user-question.js.map +1 -0
  590. package/dist/core/tools/ask-user-question/config.d.ts +11 -0
  591. package/dist/core/tools/ask-user-question/config.d.ts.map +1 -0
  592. package/dist/core/tools/ask-user-question/config.js +34 -0
  593. package/dist/core/tools/ask-user-question/config.js.map +1 -0
  594. package/dist/core/tools/ask-user-question/index.d.ts +19 -0
  595. package/dist/core/tools/ask-user-question/index.d.ts.map +1 -0
  596. package/dist/core/tools/ask-user-question/index.js +19 -0
  597. package/dist/core/tools/ask-user-question/index.js.map +1 -0
  598. package/dist/core/tools/ask-user-question/state/build-questionnaire.d.ts +36 -0
  599. package/dist/core/tools/ask-user-question/state/build-questionnaire.d.ts.map +1 -0
  600. package/dist/core/tools/ask-user-question/state/build-questionnaire.js +213 -0
  601. package/dist/core/tools/ask-user-question/state/build-questionnaire.js.map +1 -0
  602. package/dist/core/tools/ask-user-question/state/key-router.d.ts +53 -0
  603. package/dist/core/tools/ask-user-question/state/key-router.d.ts.map +1 -0
  604. package/dist/core/tools/ask-user-question/state/key-router.js +240 -0
  605. package/dist/core/tools/ask-user-question/state/key-router.js.map +1 -0
  606. package/dist/core/tools/ask-user-question/state/questionnaire-session.d.ts +61 -0
  607. package/dist/core/tools/ask-user-question/state/questionnaire-session.d.ts.map +1 -0
  608. package/dist/core/tools/ask-user-question/state/questionnaire-session.js +147 -0
  609. package/dist/core/tools/ask-user-question/state/questionnaire-session.js.map +1 -0
  610. package/dist/core/tools/ask-user-question/state/row-intent.d.ts +91 -0
  611. package/dist/core/tools/ask-user-question/state/row-intent.d.ts.map +1 -0
  612. package/dist/core/tools/ask-user-question/state/row-intent.js +91 -0
  613. package/dist/core/tools/ask-user-question/state/row-intent.js.map +1 -0
  614. package/dist/core/tools/ask-user-question/state/selectors/contract.d.ts +21 -0
  615. package/dist/core/tools/ask-user-question/state/selectors/contract.d.ts.map +1 -0
  616. package/dist/core/tools/ask-user-question/state/selectors/contract.js +2 -0
  617. package/dist/core/tools/ask-user-question/state/selectors/contract.js.map +1 -0
  618. package/dist/core/tools/ask-user-question/state/selectors/derivations.d.ts +44 -0
  619. package/dist/core/tools/ask-user-question/state/selectors/derivations.d.ts.map +1 -0
  620. package/dist/core/tools/ask-user-question/state/selectors/derivations.js +76 -0
  621. package/dist/core/tools/ask-user-question/state/selectors/derivations.js.map +1 -0
  622. package/dist/core/tools/ask-user-question/state/selectors/focus.d.ts +15 -0
  623. package/dist/core/tools/ask-user-question/state/selectors/focus.d.ts.map +1 -0
  624. package/dist/core/tools/ask-user-question/state/selectors/focus.js +18 -0
  625. package/dist/core/tools/ask-user-question/state/selectors/focus.js.map +1 -0
  626. package/dist/core/tools/ask-user-question/state/selectors/projections.d.ts +16 -0
  627. package/dist/core/tools/ask-user-question/state/selectors/projections.d.ts.map +1 -0
  628. package/dist/core/tools/ask-user-question/state/selectors/projections.js +71 -0
  629. package/dist/core/tools/ask-user-question/state/selectors/projections.js.map +1 -0
  630. package/dist/core/tools/ask-user-question/state/state-reducer.d.ts +44 -0
  631. package/dist/core/tools/ask-user-question/state/state-reducer.d.ts.map +1 -0
  632. package/dist/core/tools/ask-user-question/state/state-reducer.js +239 -0
  633. package/dist/core/tools/ask-user-question/state/state-reducer.js.map +1 -0
  634. package/dist/core/tools/ask-user-question/state/state.d.ts +42 -0
  635. package/dist/core/tools/ask-user-question/state/state.d.ts.map +1 -0
  636. package/dist/core/tools/ask-user-question/state/state.js +2 -0
  637. package/dist/core/tools/ask-user-question/state/state.js.map +1 -0
  638. package/dist/core/tools/ask-user-question/tool/format-answer.d.ts +30 -0
  639. package/dist/core/tools/ask-user-question/tool/format-answer.d.ts.map +1 -0
  640. package/dist/core/tools/ask-user-question/tool/format-answer.js +39 -0
  641. package/dist/core/tools/ask-user-question/tool/format-answer.js.map +1 -0
  642. package/dist/core/tools/ask-user-question/tool/response-envelope.d.ts +29 -0
  643. package/dist/core/tools/ask-user-question/tool/response-envelope.d.ts.map +1 -0
  644. package/dist/core/tools/ask-user-question/tool/response-envelope.js +46 -0
  645. package/dist/core/tools/ask-user-question/tool/response-envelope.js.map +1 -0
  646. package/dist/core/tools/ask-user-question/tool/types.d.ts +113 -0
  647. package/dist/core/tools/ask-user-question/tool/types.d.ts.map +1 -0
  648. package/dist/core/tools/ask-user-question/tool/types.js +81 -0
  649. package/dist/core/tools/ask-user-question/tool/types.js.map +1 -0
  650. package/dist/core/tools/ask-user-question/tool/validate-questionnaire.d.ts +21 -0
  651. package/dist/core/tools/ask-user-question/tool/validate-questionnaire.d.ts.map +1 -0
  652. package/dist/core/tools/ask-user-question/tool/validate-questionnaire.js +49 -0
  653. package/dist/core/tools/ask-user-question/tool/validate-questionnaire.js.map +1 -0
  654. package/dist/core/tools/ask-user-question/view/body-residual-spacer.d.ts +18 -0
  655. package/dist/core/tools/ask-user-question/view/body-residual-spacer.d.ts.map +1 -0
  656. package/dist/core/tools/ask-user-question/view/body-residual-spacer.js +21 -0
  657. package/dist/core/tools/ask-user-question/view/body-residual-spacer.js.map +1 -0
  658. package/dist/core/tools/ask-user-question/view/component-binding.d.ts +23 -0
  659. package/dist/core/tools/ask-user-question/view/component-binding.d.ts.map +1 -0
  660. package/dist/core/tools/ask-user-question/view/component-binding.js +16 -0
  661. package/dist/core/tools/ask-user-question/view/component-binding.js.map +1 -0
  662. package/dist/core/tools/ask-user-question/view/components/chat-row-view.d.ts +40 -0
  663. package/dist/core/tools/ask-user-question/view/components/chat-row-view.d.ts.map +1 -0
  664. package/dist/core/tools/ask-user-question/view/components/chat-row-view.js +31 -0
  665. package/dist/core/tools/ask-user-question/view/components/chat-row-view.js.map +1 -0
  666. package/dist/core/tools/ask-user-question/view/components/multi-select-view.d.ts +35 -0
  667. package/dist/core/tools/ask-user-question/view/components/multi-select-view.d.ts.map +1 -0
  668. package/dist/core/tools/ask-user-question/view/components/multi-select-view.js +91 -0
  669. package/dist/core/tools/ask-user-question/view/components/multi-select-view.js.map +1 -0
  670. package/dist/core/tools/ask-user-question/view/components/option-list-view.d.ts +43 -0
  671. package/dist/core/tools/ask-user-question/view/components/option-list-view.d.ts.map +1 -0
  672. package/dist/core/tools/ask-user-question/view/components/option-list-view.js +37 -0
  673. package/dist/core/tools/ask-user-question/view/components/option-list-view.js.map +1 -0
  674. package/dist/core/tools/ask-user-question/view/components/preview/markdown-content-cache.d.ts +36 -0
  675. package/dist/core/tools/ask-user-question/view/components/preview/markdown-content-cache.d.ts.map +1 -0
  676. package/dist/core/tools/ask-user-question/view/components/preview/markdown-content-cache.js +66 -0
  677. package/dist/core/tools/ask-user-question/view/components/preview/markdown-content-cache.js.map +1 -0
  678. package/dist/core/tools/ask-user-question/view/components/preview/preview-block-renderer.d.ts +46 -0
  679. package/dist/core/tools/ask-user-question/view/components/preview/preview-block-renderer.d.ts.map +1 -0
  680. package/dist/core/tools/ask-user-question/view/components/preview/preview-block-renderer.js +69 -0
  681. package/dist/core/tools/ask-user-question/view/components/preview/preview-block-renderer.js.map +1 -0
  682. package/dist/core/tools/ask-user-question/view/components/preview/preview-box-renderer.d.ts +39 -0
  683. package/dist/core/tools/ask-user-question/view/components/preview/preview-box-renderer.d.ts.map +1 -0
  684. package/dist/core/tools/ask-user-question/view/components/preview/preview-box-renderer.js +76 -0
  685. package/dist/core/tools/ask-user-question/view/components/preview/preview-box-renderer.js.map +1 -0
  686. package/dist/core/tools/ask-user-question/view/components/preview/preview-layout-decider.d.ts +116 -0
  687. package/dist/core/tools/ask-user-question/view/components/preview/preview-layout-decider.d.ts.map +1 -0
  688. package/dist/core/tools/ask-user-question/view/components/preview/preview-layout-decider.js +173 -0
  689. package/dist/core/tools/ask-user-question/view/components/preview/preview-layout-decider.js.map +1 -0
  690. package/dist/core/tools/ask-user-question/view/components/preview/preview-pane.d.ts +66 -0
  691. package/dist/core/tools/ask-user-question/view/components/preview/preview-pane.d.ts.map +1 -0
  692. package/dist/core/tools/ask-user-question/view/components/preview/preview-pane.js +124 -0
  693. package/dist/core/tools/ask-user-question/view/components/preview/preview-pane.js.map +1 -0
  694. package/dist/core/tools/ask-user-question/view/components/submit-picker.d.ts +37 -0
  695. package/dist/core/tools/ask-user-question/view/components/submit-picker.d.ts.map +1 -0
  696. package/dist/core/tools/ask-user-question/view/components/submit-picker.js +44 -0
  697. package/dist/core/tools/ask-user-question/view/components/submit-picker.js.map +1 -0
  698. package/dist/core/tools/ask-user-question/view/components/tab-bar.d.ts +32 -0
  699. package/dist/core/tools/ask-user-question/view/components/tab-bar.d.ts.map +1 -0
  700. package/dist/core/tools/ask-user-question/view/components/tab-bar.js +33 -0
  701. package/dist/core/tools/ask-user-question/view/components/tab-bar.js.map +1 -0
  702. package/dist/core/tools/ask-user-question/view/components/wrapping-select.d.ts +122 -0
  703. package/dist/core/tools/ask-user-question/view/components/wrapping-select.d.ts.map +1 -0
  704. package/dist/core/tools/ask-user-question/view/components/wrapping-select.js +161 -0
  705. package/dist/core/tools/ask-user-question/view/components/wrapping-select.js.map +1 -0
  706. package/dist/core/tools/ask-user-question/view/dialog-builder.d.ts +66 -0
  707. package/dist/core/tools/ask-user-question/view/dialog-builder.d.ts.map +1 -0
  708. package/dist/core/tools/ask-user-question/view/dialog-builder.js +85 -0
  709. package/dist/core/tools/ask-user-question/view/dialog-builder.js.map +1 -0
  710. package/dist/core/tools/ask-user-question/view/props-adapter.d.ts +58 -0
  711. package/dist/core/tools/ask-user-question/view/props-adapter.d.ts.map +1 -0
  712. package/dist/core/tools/ask-user-question/view/props-adapter.js +67 -0
  713. package/dist/core/tools/ask-user-question/view/props-adapter.js.map +1 -0
  714. package/dist/core/tools/ask-user-question/view/stateful-view.d.ts +24 -0
  715. package/dist/core/tools/ask-user-question/view/stateful-view.d.ts.map +1 -0
  716. package/dist/core/tools/ask-user-question/view/stateful-view.js +2 -0
  717. package/dist/core/tools/ask-user-question/view/stateful-view.js.map +1 -0
  718. package/dist/core/tools/ask-user-question/view/tab-components.d.ts +15 -0
  719. package/dist/core/tools/ask-user-question/view/tab-components.d.ts.map +1 -0
  720. package/dist/core/tools/ask-user-question/view/tab-components.js +2 -0
  721. package/dist/core/tools/ask-user-question/view/tab-components.js.map +1 -0
  722. package/dist/core/tools/ask-user-question/view/tab-content-strategy.d.ts +71 -0
  723. package/dist/core/tools/ask-user-question/view/tab-content-strategy.d.ts.map +1 -0
  724. package/dist/core/tools/ask-user-question/view/tab-content-strategy.js +129 -0
  725. package/dist/core/tools/ask-user-question/view/tab-content-strategy.js.map +1 -0
  726. package/dist/core/tools/bash.d.ts +68 -0
  727. package/dist/core/tools/bash.d.ts.map +1 -0
  728. package/dist/core/tools/bash.js +338 -0
  729. package/dist/core/tools/bash.js.map +1 -0
  730. package/dist/core/tools/edit-diff.d.ts +85 -0
  731. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  732. package/dist/core/tools/edit-diff.js +338 -0
  733. package/dist/core/tools/edit-diff.js.map +1 -0
  734. package/dist/core/tools/edit.d.ts +49 -0
  735. package/dist/core/tools/edit.d.ts.map +1 -0
  736. package/dist/core/tools/edit.js +324 -0
  737. package/dist/core/tools/edit.js.map +1 -0
  738. package/dist/core/tools/file-mutation-queue.d.ts +6 -0
  739. package/dist/core/tools/file-mutation-queue.d.ts.map +1 -0
  740. package/dist/core/tools/file-mutation-queue.js +37 -0
  741. package/dist/core/tools/file-mutation-queue.js.map +1 -0
  742. package/dist/core/tools/find.d.ts +35 -0
  743. package/dist/core/tools/find.d.ts.map +1 -0
  744. package/dist/core/tools/find.js +298 -0
  745. package/dist/core/tools/find.js.map +1 -0
  746. package/dist/core/tools/grep.d.ts +37 -0
  747. package/dist/core/tools/grep.d.ts.map +1 -0
  748. package/dist/core/tools/grep.js +304 -0
  749. package/dist/core/tools/grep.js.map +1 -0
  750. package/dist/core/tools/index.d.ts +42 -0
  751. package/dist/core/tools/index.d.ts.map +1 -0
  752. package/dist/core/tools/index.js +139 -0
  753. package/dist/core/tools/index.js.map +1 -0
  754. package/dist/core/tools/ls.d.ts +37 -0
  755. package/dist/core/tools/ls.d.ts.map +1 -0
  756. package/dist/core/tools/ls.js +169 -0
  757. package/dist/core/tools/ls.js.map +1 -0
  758. package/dist/core/tools/output-accumulator.d.ts +50 -0
  759. package/dist/core/tools/output-accumulator.d.ts.map +1 -0
  760. package/dist/core/tools/output-accumulator.js +172 -0
  761. package/dist/core/tools/output-accumulator.js.map +1 -0
  762. package/dist/core/tools/path-utils.d.ts +8 -0
  763. package/dist/core/tools/path-utils.d.ts.map +1 -0
  764. package/dist/core/tools/path-utils.js +81 -0
  765. package/dist/core/tools/path-utils.js.map +1 -0
  766. package/dist/core/tools/read.d.ts +35 -0
  767. package/dist/core/tools/read.d.ts.map +1 -0
  768. package/dist/core/tools/read.js +289 -0
  769. package/dist/core/tools/read.js.map +1 -0
  770. package/dist/core/tools/render-utils.d.ts +21 -0
  771. package/dist/core/tools/render-utils.d.ts.map +1 -0
  772. package/dist/core/tools/render-utils.js +49 -0
  773. package/dist/core/tools/render-utils.js.map +1 -0
  774. package/dist/core/tools/todos.d.ts +35 -0
  775. package/dist/core/tools/todos.d.ts.map +1 -0
  776. package/dist/core/tools/todos.js +906 -0
  777. package/dist/core/tools/todos.js.map +1 -0
  778. package/dist/core/tools/tool-definition-wrapper.d.ts +14 -0
  779. package/dist/core/tools/tool-definition-wrapper.d.ts.map +1 -0
  780. package/dist/core/tools/tool-definition-wrapper.js +34 -0
  781. package/dist/core/tools/tool-definition-wrapper.js.map +1 -0
  782. package/dist/core/tools/truncate.d.ts +70 -0
  783. package/dist/core/tools/truncate.d.ts.map +1 -0
  784. package/dist/core/tools/truncate.js +205 -0
  785. package/dist/core/tools/truncate.js.map +1 -0
  786. package/dist/core/tools/write.d.ts +26 -0
  787. package/dist/core/tools/write.d.ts.map +1 -0
  788. package/dist/core/tools/write.js +212 -0
  789. package/dist/core/tools/write.js.map +1 -0
  790. package/dist/index.d.ts +28 -0
  791. package/dist/index.d.ts.map +1 -0
  792. package/dist/index.js +41 -0
  793. package/dist/index.js.map +1 -0
  794. package/dist/main.d.ts +13 -0
  795. package/dist/main.d.ts.map +1 -0
  796. package/dist/main.js +586 -0
  797. package/dist/main.js.map +1 -0
  798. package/dist/migrations.d.ts +33 -0
  799. package/dist/migrations.d.ts.map +1 -0
  800. package/dist/migrations.js +281 -0
  801. package/dist/migrations.js.map +1 -0
  802. package/dist/modes/index.d.ts +9 -0
  803. package/dist/modes/index.d.ts.map +1 -0
  804. package/dist/modes/index.js +8 -0
  805. package/dist/modes/index.js.map +1 -0
  806. package/dist/modes/interactive/assets/clankolas.png +3 -0
  807. package/dist/modes/interactive/components/armin.d.ts +34 -0
  808. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  809. package/dist/modes/interactive/components/armin.js +329 -0
  810. package/dist/modes/interactive/components/armin.js.map +1 -0
  811. package/dist/modes/interactive/components/assistant-message.d.ts +20 -0
  812. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  813. package/dist/modes/interactive/components/assistant-message.js +116 -0
  814. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  815. package/dist/modes/interactive/components/bash-execution.d.ts +34 -0
  816. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  817. package/dist/modes/interactive/components/bash-execution.js +170 -0
  818. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  819. package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
  820. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  821. package/dist/modes/interactive/components/bordered-loader.js +51 -0
  822. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  823. package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
  824. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  825. package/dist/modes/interactive/components/branch-summary-message.js +42 -0
  826. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  827. package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
  828. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  829. package/dist/modes/interactive/components/compaction-summary-message.js +43 -0
  830. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  831. package/dist/modes/interactive/components/config-selector.d.ts +71 -0
  832. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  833. package/dist/modes/interactive/components/config-selector.js +496 -0
  834. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  835. package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
  836. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  837. package/dist/modes/interactive/components/countdown-timer.js +28 -0
  838. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  839. package/dist/modes/interactive/components/custom-editor.d.ts +29 -0
  840. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  841. package/dist/modes/interactive/components/custom-editor.js +113 -0
  842. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  843. package/dist/modes/interactive/components/custom-message.d.ts +20 -0
  844. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  845. package/dist/modes/interactive/components/custom-message.js +74 -0
  846. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  847. package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
  848. package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
  849. package/dist/modes/interactive/components/daxnuts.js +138 -0
  850. package/dist/modes/interactive/components/daxnuts.js.map +1 -0
  851. package/dist/modes/interactive/components/diff.d.ts +12 -0
  852. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  853. package/dist/modes/interactive/components/diff.js +151 -0
  854. package/dist/modes/interactive/components/diff.js.map +1 -0
  855. package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
  856. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  857. package/dist/modes/interactive/components/dynamic-border.js +20 -0
  858. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  859. package/dist/modes/interactive/components/earendil-announcement.d.ts +5 -0
  860. package/dist/modes/interactive/components/earendil-announcement.d.ts.map +1 -0
  861. package/dist/modes/interactive/components/earendil-announcement.js +40 -0
  862. package/dist/modes/interactive/components/earendil-announcement.js.map +1 -0
  863. package/dist/modes/interactive/components/extension-editor.d.ts +20 -0
  864. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  865. package/dist/modes/interactive/components/extension-editor.js +106 -0
  866. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  867. package/dist/modes/interactive/components/extension-input.d.ts +23 -0
  868. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  869. package/dist/modes/interactive/components/extension-input.js +55 -0
  870. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  871. package/dist/modes/interactive/components/extension-selector.d.ts +26 -0
  872. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  873. package/dist/modes/interactive/components/extension-selector.js +75 -0
  874. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  875. package/dist/modes/interactive/components/footer.d.ts +27 -0
  876. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  877. package/dist/modes/interactive/components/footer.js +199 -0
  878. package/dist/modes/interactive/components/footer.js.map +1 -0
  879. package/dist/modes/interactive/components/index.d.ts +32 -0
  880. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  881. package/dist/modes/interactive/components/index.js +33 -0
  882. package/dist/modes/interactive/components/index.js.map +1 -0
  883. package/dist/modes/interactive/components/keybinding-hints.d.ts +13 -0
  884. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
  885. package/dist/modes/interactive/components/keybinding-hints.js +36 -0
  886. package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
  887. package/dist/modes/interactive/components/login-dialog.d.ts +46 -0
  888. package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
  889. package/dist/modes/interactive/components/login-dialog.js +158 -0
  890. package/dist/modes/interactive/components/login-dialog.js.map +1 -0
  891. package/dist/modes/interactive/components/model-selector.d.ts +47 -0
  892. package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
  893. package/dist/modes/interactive/components/model-selector.js +266 -0
  894. package/dist/modes/interactive/components/model-selector.js.map +1 -0
  895. package/dist/modes/interactive/components/oauth-selector.d.ts +31 -0
  896. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
  897. package/dist/modes/interactive/components/oauth-selector.js +156 -0
  898. package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
  899. package/dist/modes/interactive/components/scoped-models-selector.d.ts +42 -0
  900. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
  901. package/dist/modes/interactive/components/scoped-models-selector.js +286 -0
  902. package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
  903. package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
  904. package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
  905. package/dist/modes/interactive/components/session-selector-search.js +155 -0
  906. package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
  907. package/dist/modes/interactive/components/session-selector.d.ts +96 -0
  908. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  909. package/dist/modes/interactive/components/session-selector.js +836 -0
  910. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  911. package/dist/modes/interactive/components/settings-selector.d.ts +67 -0
  912. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  913. package/dist/modes/interactive/components/settings-selector.js +371 -0
  914. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  915. package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
  916. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  917. package/dist/modes/interactive/components/show-images-selector.js +38 -0
  918. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  919. package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
  920. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  921. package/dist/modes/interactive/components/skill-invocation-message.js +45 -0
  922. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  923. package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
  924. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  925. package/dist/modes/interactive/components/theme-selector.js +48 -0
  926. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  927. package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
  928. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  929. package/dist/modes/interactive/components/thinking-selector.js +50 -0
  930. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  931. package/dist/modes/interactive/components/tool-execution.d.ts +63 -0
  932. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  933. package/dist/modes/interactive/components/tool-execution.js +280 -0
  934. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  935. package/dist/modes/interactive/components/tree-selector.d.ts +89 -0
  936. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  937. package/dist/modes/interactive/components/tree-selector.js +1079 -0
  938. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  939. package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
  940. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  941. package/dist/modes/interactive/components/user-message-selector.js +111 -0
  942. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  943. package/dist/modes/interactive/components/user-message.d.ts +10 -0
  944. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  945. package/dist/modes/interactive/components/user-message.js +28 -0
  946. package/dist/modes/interactive/components/user-message.js.map +1 -0
  947. package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
  948. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  949. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  950. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  951. package/dist/modes/interactive/interactive-mode.d.ts +369 -0
  952. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  953. package/dist/modes/interactive/interactive-mode.js +4709 -0
  954. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  955. package/dist/modes/interactive/theme/catppuccin-frappe.json +90 -0
  956. package/dist/modes/interactive/theme/catppuccin-latte.json +90 -0
  957. package/dist/modes/interactive/theme/catppuccin-macchiato.json +90 -0
  958. package/dist/modes/interactive/theme/catppuccin-mocha.json +90 -0
  959. package/dist/modes/interactive/theme/dark.json +85 -0
  960. package/dist/modes/interactive/theme/light.json +84 -0
  961. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  962. package/dist/modes/interactive/theme/theme.d.ts +81 -0
  963. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  964. package/dist/modes/interactive/theme/theme.js +970 -0
  965. package/dist/modes/interactive/theme/theme.js.map +1 -0
  966. package/dist/modes/interactive/whimsical-messages.d.ts +5 -0
  967. package/dist/modes/interactive/whimsical-messages.d.ts.map +1 -0
  968. package/dist/modes/interactive/whimsical-messages.js +464 -0
  969. package/dist/modes/interactive/whimsical-messages.js.map +1 -0
  970. package/dist/modes/print-mode.d.ts +28 -0
  971. package/dist/modes/print-mode.d.ts.map +1 -0
  972. package/dist/modes/print-mode.js +131 -0
  973. package/dist/modes/print-mode.js.map +1 -0
  974. package/dist/modes/rpc/jsonl.d.ts +17 -0
  975. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  976. package/dist/modes/rpc/jsonl.js +49 -0
  977. package/dist/modes/rpc/jsonl.js.map +1 -0
  978. package/dist/modes/rpc/rpc-client.d.ts +224 -0
  979. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  980. package/dist/modes/rpc/rpc-client.js +409 -0
  981. package/dist/modes/rpc/rpc-client.js.map +1 -0
  982. package/dist/modes/rpc/rpc-mode.d.ts +20 -0
  983. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  984. package/dist/modes/rpc/rpc-mode.js +601 -0
  985. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  986. package/dist/modes/rpc/rpc-types.d.ts +419 -0
  987. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  988. package/dist/modes/rpc/rpc-types.js +8 -0
  989. package/dist/modes/rpc/rpc-types.js.map +1 -0
  990. package/dist/package-manager-cli.d.ts +4 -0
  991. package/dist/package-manager-cli.d.ts.map +1 -0
  992. package/dist/package-manager-cli.js +460 -0
  993. package/dist/package-manager-cli.js.map +1 -0
  994. package/dist/utils/ansi.d.ts +2 -0
  995. package/dist/utils/ansi.d.ts.map +1 -0
  996. package/dist/utils/ansi.js +52 -0
  997. package/dist/utils/ansi.js.map +1 -0
  998. package/dist/utils/changelog.d.ts +21 -0
  999. package/dist/utils/changelog.d.ts.map +1 -0
  1000. package/dist/utils/changelog.js +87 -0
  1001. package/dist/utils/changelog.js.map +1 -0
  1002. package/dist/utils/child-process.d.ts +12 -0
  1003. package/dist/utils/child-process.d.ts.map +1 -0
  1004. package/dist/utils/child-process.js +86 -0
  1005. package/dist/utils/child-process.js.map +1 -0
  1006. package/dist/utils/clipboard-image.d.ts +11 -0
  1007. package/dist/utils/clipboard-image.d.ts.map +1 -0
  1008. package/dist/utils/clipboard-image.js +245 -0
  1009. package/dist/utils/clipboard-image.js.map +1 -0
  1010. package/dist/utils/clipboard-native.d.ts +8 -0
  1011. package/dist/utils/clipboard-native.d.ts.map +1 -0
  1012. package/dist/utils/clipboard-native.js +14 -0
  1013. package/dist/utils/clipboard-native.js.map +1 -0
  1014. package/dist/utils/clipboard.d.ts +2 -0
  1015. package/dist/utils/clipboard.d.ts.map +1 -0
  1016. package/dist/utils/clipboard.js +117 -0
  1017. package/dist/utils/clipboard.js.map +1 -0
  1018. package/dist/utils/exif-orientation.d.ts +5 -0
  1019. package/dist/utils/exif-orientation.d.ts.map +1 -0
  1020. package/dist/utils/exif-orientation.js +158 -0
  1021. package/dist/utils/exif-orientation.js.map +1 -0
  1022. package/dist/utils/frontmatter.d.ts +8 -0
  1023. package/dist/utils/frontmatter.d.ts.map +1 -0
  1024. package/dist/utils/frontmatter.js +26 -0
  1025. package/dist/utils/frontmatter.js.map +1 -0
  1026. package/dist/utils/fs-watch.d.ts +5 -0
  1027. package/dist/utils/fs-watch.d.ts.map +1 -0
  1028. package/dist/utils/fs-watch.js +25 -0
  1029. package/dist/utils/fs-watch.js.map +1 -0
  1030. package/dist/utils/git.d.ts +26 -0
  1031. package/dist/utils/git.d.ts.map +1 -0
  1032. package/dist/utils/git.js +163 -0
  1033. package/dist/utils/git.js.map +1 -0
  1034. package/dist/utils/html.d.ts +7 -0
  1035. package/dist/utils/html.d.ts.map +1 -0
  1036. package/dist/utils/html.js +40 -0
  1037. package/dist/utils/html.js.map +1 -0
  1038. package/dist/utils/image-convert.d.ts +9 -0
  1039. package/dist/utils/image-convert.d.ts.map +1 -0
  1040. package/dist/utils/image-convert.js +39 -0
  1041. package/dist/utils/image-convert.js.map +1 -0
  1042. package/dist/utils/image-resize.d.ts +36 -0
  1043. package/dist/utils/image-resize.d.ts.map +1 -0
  1044. package/dist/utils/image-resize.js +137 -0
  1045. package/dist/utils/image-resize.js.map +1 -0
  1046. package/dist/utils/mime.d.ts +3 -0
  1047. package/dist/utils/mime.d.ts.map +1 -0
  1048. package/dist/utils/mime.js +69 -0
  1049. package/dist/utils/mime.js.map +1 -0
  1050. package/dist/utils/paths.d.ts +16 -0
  1051. package/dist/utils/paths.d.ts.map +1 -0
  1052. package/dist/utils/paths.js +50 -0
  1053. package/dist/utils/paths.js.map +1 -0
  1054. package/dist/utils/photon.d.ts +21 -0
  1055. package/dist/utils/photon.d.ts.map +1 -0
  1056. package/dist/utils/photon.js +121 -0
  1057. package/dist/utils/photon.js.map +1 -0
  1058. package/dist/utils/pi-user-agent.d.ts +2 -0
  1059. package/dist/utils/pi-user-agent.d.ts.map +1 -0
  1060. package/dist/utils/pi-user-agent.js +5 -0
  1061. package/dist/utils/pi-user-agent.js.map +1 -0
  1062. package/dist/utils/shell.d.ts +30 -0
  1063. package/dist/utils/shell.d.ts.map +1 -0
  1064. package/dist/utils/shell.js +190 -0
  1065. package/dist/utils/shell.js.map +1 -0
  1066. package/dist/utils/sleep.d.ts +5 -0
  1067. package/dist/utils/sleep.d.ts.map +1 -0
  1068. package/dist/utils/sleep.js +17 -0
  1069. package/dist/utils/sleep.js.map +1 -0
  1070. package/dist/utils/syntax-highlight.d.ts +12 -0
  1071. package/dist/utils/syntax-highlight.d.ts.map +1 -0
  1072. package/dist/utils/syntax-highlight.js +118 -0
  1073. package/dist/utils/syntax-highlight.js.map +1 -0
  1074. package/dist/utils/tools-manager.d.ts +3 -0
  1075. package/dist/utils/tools-manager.d.ts.map +1 -0
  1076. package/dist/utils/tools-manager.js +325 -0
  1077. package/dist/utils/tools-manager.js.map +1 -0
  1078. package/dist/utils/version-check.d.ts +14 -0
  1079. package/dist/utils/version-check.d.ts.map +1 -0
  1080. package/dist/utils/version-check.js +76 -0
  1081. package/dist/utils/version-check.js.map +1 -0
  1082. package/docs/compaction.md +394 -0
  1083. package/docs/custom-provider.md +646 -0
  1084. package/docs/development.md +71 -0
  1085. package/docs/docs.json +148 -0
  1086. package/docs/extensions.md +2596 -0
  1087. package/docs/images/doom-extension.png +0 -0
  1088. package/docs/images/exy.png +3 -0
  1089. package/docs/images/interactive-mode.png +0 -0
  1090. package/docs/images/tree-view.png +0 -0
  1091. package/docs/index.md +70 -0
  1092. package/docs/json.md +82 -0
  1093. package/docs/keybindings.md +197 -0
  1094. package/docs/models.md +474 -0
  1095. package/docs/packages.md +223 -0
  1096. package/docs/prompt-templates.md +88 -0
  1097. package/docs/providers.md +243 -0
  1098. package/docs/quickstart.md +142 -0
  1099. package/docs/rpc.md +1407 -0
  1100. package/docs/sdk.md +1129 -0
  1101. package/docs/session-format.md +412 -0
  1102. package/docs/sessions.md +137 -0
  1103. package/docs/settings.md +279 -0
  1104. package/docs/shell-aliases.md +13 -0
  1105. package/docs/skills.md +232 -0
  1106. package/docs/terminal-setup.md +106 -0
  1107. package/docs/termux.md +127 -0
  1108. package/docs/themes.md +299 -0
  1109. package/docs/tmux.md +61 -0
  1110. package/docs/tui.md +918 -0
  1111. package/docs/usage.md +277 -0
  1112. package/docs/windows.md +17 -0
  1113. package/examples/README.md +25 -0
  1114. package/examples/extensions/README.md +208 -0
  1115. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  1116. package/examples/extensions/bash-spawn-hook.ts +30 -0
  1117. package/examples/extensions/bookmark.ts +50 -0
  1118. package/examples/extensions/border-status-editor.ts +150 -0
  1119. package/examples/extensions/built-in-tool-renderer.ts +249 -0
  1120. package/examples/extensions/claude-rules.ts +86 -0
  1121. package/examples/extensions/commands.ts +72 -0
  1122. package/examples/extensions/confirm-destructive.ts +59 -0
  1123. package/examples/extensions/custom-compaction.ts +127 -0
  1124. package/examples/extensions/custom-footer.ts +64 -0
  1125. package/examples/extensions/custom-header.ts +73 -0
  1126. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  1127. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  1128. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  1129. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  1130. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  1131. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  1132. package/examples/extensions/dirty-repo-guard.ts +56 -0
  1133. package/examples/extensions/doom-overlay/README.md +46 -0
  1134. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  1135. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  1136. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  1137. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  1138. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  1139. package/examples/extensions/doom-overlay/index.ts +74 -0
  1140. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  1141. package/examples/extensions/dynamic-resources/SKILL.md +8 -0
  1142. package/examples/extensions/dynamic-resources/dynamic.json +79 -0
  1143. package/examples/extensions/dynamic-resources/dynamic.md +5 -0
  1144. package/examples/extensions/dynamic-resources/index.ts +15 -0
  1145. package/examples/extensions/dynamic-tools.ts +74 -0
  1146. package/examples/extensions/event-bus.ts +43 -0
  1147. package/examples/extensions/file-trigger.ts +41 -0
  1148. package/examples/extensions/git-checkpoint.ts +53 -0
  1149. package/examples/extensions/github-issue-autocomplete.ts +185 -0
  1150. package/examples/extensions/handoff.ts +191 -0
  1151. package/examples/extensions/hello.ts +26 -0
  1152. package/examples/extensions/hidden-thinking-label.ts +53 -0
  1153. package/examples/extensions/inline-bash.ts +94 -0
  1154. package/examples/extensions/input-transform.ts +43 -0
  1155. package/examples/extensions/interactive-shell.ts +196 -0
  1156. package/examples/extensions/mac-system-theme.ts +47 -0
  1157. package/examples/extensions/message-renderer.ts +59 -0
  1158. package/examples/extensions/minimal-mode.ts +426 -0
  1159. package/examples/extensions/modal-editor.ts +85 -0
  1160. package/examples/extensions/model-status.ts +31 -0
  1161. package/examples/extensions/notify.ts +55 -0
  1162. package/examples/extensions/overlay-qa-tests.ts +1348 -0
  1163. package/examples/extensions/overlay-test.ts +150 -0
  1164. package/examples/extensions/permission-gate.ts +34 -0
  1165. package/examples/extensions/pirate.ts +47 -0
  1166. package/examples/extensions/plan-mode/README.md +65 -0
  1167. package/examples/extensions/plan-mode/index.ts +340 -0
  1168. package/examples/extensions/plan-mode/utils.ts +168 -0
  1169. package/examples/extensions/preset.ts +430 -0
  1170. package/examples/extensions/prompt-customizer.ts +97 -0
  1171. package/examples/extensions/protected-paths.ts +30 -0
  1172. package/examples/extensions/provider-payload.ts +18 -0
  1173. package/examples/extensions/qna.ts +122 -0
  1174. package/examples/extensions/question.ts +264 -0
  1175. package/examples/extensions/questionnaire.ts +427 -0
  1176. package/examples/extensions/rainbow-editor.ts +88 -0
  1177. package/examples/extensions/reload-runtime.ts +37 -0
  1178. package/examples/extensions/rpc-demo.ts +118 -0
  1179. package/examples/extensions/sandbox/index.ts +321 -0
  1180. package/examples/extensions/sandbox/package-lock.json +92 -0
  1181. package/examples/extensions/sandbox/package.json +19 -0
  1182. package/examples/extensions/send-user-message.ts +97 -0
  1183. package/examples/extensions/session-name.ts +27 -0
  1184. package/examples/extensions/shutdown-command.ts +63 -0
  1185. package/examples/extensions/snake.ts +343 -0
  1186. package/examples/extensions/space-invaders.ts +560 -0
  1187. package/examples/extensions/ssh.ts +220 -0
  1188. package/examples/extensions/status-line.ts +32 -0
  1189. package/examples/extensions/structured-output.ts +65 -0
  1190. package/examples/extensions/subagent/README.md +172 -0
  1191. package/examples/extensions/subagent/agents/planner.md +37 -0
  1192. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  1193. package/examples/extensions/subagent/agents/scout.md +50 -0
  1194. package/examples/extensions/subagent/agents/worker.md +24 -0
  1195. package/examples/extensions/subagent/agents.ts +126 -0
  1196. package/examples/extensions/subagent/index.ts +987 -0
  1197. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  1198. package/examples/extensions/subagent/prompts/implement.md +10 -0
  1199. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  1200. package/examples/extensions/summarize.ts +206 -0
  1201. package/examples/extensions/system-prompt-header.ts +17 -0
  1202. package/examples/extensions/tic-tac-toe.ts +1008 -0
  1203. package/examples/extensions/timed-confirm.ts +70 -0
  1204. package/examples/extensions/titlebar-spinner.ts +58 -0
  1205. package/examples/extensions/todo.ts +297 -0
  1206. package/examples/extensions/tool-override.ts +144 -0
  1207. package/examples/extensions/tools.ts +141 -0
  1208. package/examples/extensions/trigger-compact.ts +50 -0
  1209. package/examples/extensions/truncated-tool.ts +195 -0
  1210. package/examples/extensions/widget-placement.ts +9 -0
  1211. package/examples/extensions/with-deps/index.ts +32 -0
  1212. package/examples/extensions/with-deps/package-lock.json +31 -0
  1213. package/examples/extensions/with-deps/package.json +22 -0
  1214. package/examples/extensions/working-indicator.ts +123 -0
  1215. package/examples/extensions/working-message-test.ts +25 -0
  1216. package/examples/rpc-extension-ui.ts +632 -0
  1217. package/examples/sdk/01-minimal.ts +26 -0
  1218. package/examples/sdk/02-custom-model.ts +53 -0
  1219. package/examples/sdk/03-custom-prompt.ts +75 -0
  1220. package/examples/sdk/04-skills.ts +55 -0
  1221. package/examples/sdk/05-tools.ts +48 -0
  1222. package/examples/sdk/06-extensions.ts +99 -0
  1223. package/examples/sdk/07-context-files.ts +47 -0
  1224. package/examples/sdk/08-prompt-templates.ts +51 -0
  1225. package/examples/sdk/09-api-keys-and-oauth.ts +52 -0
  1226. package/examples/sdk/10-settings.ts +53 -0
  1227. package/examples/sdk/11-sessions.ts +52 -0
  1228. package/examples/sdk/12-full-control.ts +77 -0
  1229. package/examples/sdk/13-session-runtime.ts +67 -0
  1230. package/examples/sdk/README.md +144 -0
  1231. package/package.json +94 -19
  1232. package/bin/atomic +0 -82
  1233. /package/{LICENSE → dist/builtin/workflows/LICENSE} +0 -0
@@ -0,0 +1,2347 @@
1
+ import type { ExtensionAPI, ExtensionContext } from "@mariozechner/pi-coding-agent";
2
+ import { Box, Text, truncateToWidth } from "@mariozechner/pi-tui";
3
+ import { Type } from "typebox";
4
+ import { StringEnum, complete, getModel, type Model } from "@mariozechner/pi-ai";
5
+ import { fetchAllContent, type ExtractedContent } from "./extract.js";
6
+ import { clearCloneCache } from "./github-extract.js";
7
+ import { search, type SearchProvider, type ResolvedSearchProvider } from "./gemini-search.js";
8
+ import { executeCodeSearch } from "./code-search.js";
9
+ import type { SearchResult } from "./perplexity.js";
10
+ import { formatSeconds } from "./utils.js";
11
+ import {
12
+ clearResults,
13
+ deleteResult,
14
+ generateId,
15
+ getAllResults,
16
+ getResult,
17
+ restoreFromSession,
18
+ storeResult,
19
+ type QueryResultData,
20
+ type StoredSearchData,
21
+ } from "./storage.js";
22
+ import { activityMonitor, type ActivityEntry } from "./activity.js";
23
+ import { startCuratorServer, type CuratorServerHandle } from "./curator-server.js";
24
+ import {
25
+ buildDeterministicSummary,
26
+ generateSummaryDraft,
27
+ type SummaryGenerationContext,
28
+ type SummaryMeta,
29
+ } from "./summary-review.js";
30
+ import { randomUUID } from "node:crypto";
31
+ import { execFileSync } from "node:child_process";
32
+ import { createRequire } from "node:module";
33
+ import { platform, homedir } from "node:os";
34
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
35
+ import { join } from "node:path";
36
+ import { CONFIG_DIR_NAME } from "@bastani/atomic";
37
+ import { isPerplexityAvailable } from "./perplexity.js";
38
+ import { isExaAvailable } from "./exa.js";
39
+ import { isGeminiApiAvailable } from "./gemini-api.js";
40
+ import { getActiveGoogleEmail, isGeminiWebAvailable } from "./gemini-web.js";
41
+ import { isBrowserCookieAccessAllowed } from "./gemini-web-config.ts";
42
+
43
+ const WEB_SEARCH_CONFIG_PATH = join(homedir(), CONFIG_DIR_NAME, "web-search.json");
44
+
45
+ interface WebSearchConfig {
46
+ provider?: string;
47
+ workflow?: string;
48
+ curatorTimeoutSeconds?: unknown;
49
+ summaryModel?: string;
50
+ shortcuts?: {
51
+ curate?: string;
52
+ activity?: string;
53
+ };
54
+ }
55
+
56
+ interface ProviderAvailability {
57
+ perplexity: boolean;
58
+ exa: boolean;
59
+ gemini: boolean;
60
+ }
61
+
62
+ type WebSearchWorkflow = "none" | "summary-review";
63
+ type CuratorWorkflow = "summary-review";
64
+
65
+ interface CuratorBootstrap {
66
+ availableProviders: ProviderAvailability;
67
+ defaultProvider: ResolvedSearchProvider;
68
+ timeoutSeconds: number;
69
+ }
70
+
71
+ function loadConfig(): WebSearchConfig {
72
+ if (!existsSync(WEB_SEARCH_CONFIG_PATH)) return {};
73
+ const raw = readFileSync(WEB_SEARCH_CONFIG_PATH, "utf-8");
74
+ try {
75
+ return JSON.parse(raw) as WebSearchConfig;
76
+ } catch (err) {
77
+ const message = err instanceof Error ? err.message : String(err);
78
+ throw new Error(`Failed to parse ${WEB_SEARCH_CONFIG_PATH}: ${message}`);
79
+ }
80
+ }
81
+
82
+ function saveConfig(updates: Partial<WebSearchConfig>): void {
83
+ let config: Record<string, unknown> = {};
84
+ if (existsSync(WEB_SEARCH_CONFIG_PATH)) {
85
+ const raw = readFileSync(WEB_SEARCH_CONFIG_PATH, "utf-8");
86
+ try {
87
+ config = JSON.parse(raw) as Record<string, unknown>;
88
+ } catch (err) {
89
+ const message = err instanceof Error ? err.message : String(err);
90
+ throw new Error(`Failed to parse ${WEB_SEARCH_CONFIG_PATH}: ${message}`);
91
+ }
92
+ }
93
+
94
+ Object.assign(config, updates);
95
+ const dir = join(homedir(), CONFIG_DIR_NAME);
96
+ if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
97
+ writeFileSync(WEB_SEARCH_CONFIG_PATH, JSON.stringify(config, null, 2) + "\n");
98
+ }
99
+
100
+ const DEFAULT_SHORTCUTS = { curate: "ctrl+shift+s", activity: "ctrl+shift+w" };
101
+ const DEFAULT_CURATOR_TIMEOUT_SECONDS = 20;
102
+ const MAX_CURATOR_TIMEOUT_SECONDS = 600;
103
+
104
+ function loadConfigForExtensionInit(): WebSearchConfig {
105
+ try {
106
+ return loadConfig();
107
+ } catch (err) {
108
+ const message = err instanceof Error ? err.message : String(err);
109
+ console.error(`[pi-web-access] ${message}`);
110
+ return {};
111
+ }
112
+ }
113
+
114
+ function normalizeProviderInput(value: unknown): SearchProvider | undefined {
115
+ if (value === undefined) return undefined;
116
+ if (typeof value !== "string") return "auto";
117
+ const normalized = value.trim().toLowerCase();
118
+ if (normalized === "auto" || normalized === "exa" || normalized === "perplexity" || normalized === "gemini") {
119
+ return normalized;
120
+ }
121
+ return "auto";
122
+ }
123
+
124
+ function normalizeCuratorTimeoutSeconds(value: unknown): number | undefined {
125
+ if (typeof value !== "number" || !Number.isFinite(value)) return undefined;
126
+ const normalized = Math.floor(value);
127
+ if (normalized < 1) return undefined;
128
+ return Math.min(normalized, MAX_CURATOR_TIMEOUT_SECONDS);
129
+ }
130
+
131
+ function resolveWorkflow(input: unknown, hasUI: boolean): WebSearchWorkflow {
132
+ if (!hasUI) return "none";
133
+ if (typeof input === "string" && input.trim().toLowerCase() === "none") return "none";
134
+ return "summary-review";
135
+ }
136
+
137
+ function normalizeQueryList(queryList: unknown[]): string[] {
138
+ const normalized: string[] = [];
139
+ for (const query of queryList) {
140
+ if (typeof query !== "string") continue;
141
+ const trimmed = query.trim();
142
+ if (trimmed.length > 0) normalized.push(trimmed);
143
+ }
144
+ return normalized;
145
+ }
146
+
147
+ function getCuratorTimeoutSeconds(): number {
148
+ const source = loadConfig();
149
+ return normalizeCuratorTimeoutSeconds(source.curatorTimeoutSeconds) ?? DEFAULT_CURATOR_TIMEOUT_SECONDS;
150
+ }
151
+
152
+ async function getProviderAvailability(): Promise<ProviderAvailability> {
153
+ const geminiWebAvail = await isGeminiWebAvailable();
154
+ return {
155
+ perplexity: isPerplexityAvailable(),
156
+ exa: isExaAvailable(),
157
+ gemini: isGeminiApiAvailable() || !!geminiWebAvail,
158
+ };
159
+ }
160
+
161
+ async function loadCuratorBootstrap(requestedProvider: unknown): Promise<CuratorBootstrap> {
162
+ const availableProviders = await getProviderAvailability();
163
+ return {
164
+ availableProviders,
165
+ defaultProvider: resolveProvider(requestedProvider, availableProviders),
166
+ timeoutSeconds: getCuratorTimeoutSeconds(),
167
+ };
168
+ }
169
+
170
+ function resolveProvider(
171
+ requested: unknown,
172
+ available: ProviderAvailability,
173
+ ): ResolvedSearchProvider {
174
+ const provider = normalizeProviderInput(requested ?? loadConfig().provider ?? "auto") ?? "auto";
175
+
176
+ if (provider === "auto") {
177
+ if (available.exa) return "exa";
178
+ if (available.perplexity) return "perplexity";
179
+ if (available.gemini) return "gemini";
180
+ return "exa";
181
+ }
182
+ if (provider === "exa" && !available.exa) {
183
+ if (available.perplexity) return "perplexity";
184
+ return available.gemini ? "gemini" : "exa";
185
+ }
186
+ if (provider === "perplexity" && !available.perplexity) {
187
+ if (available.exa) return "exa";
188
+ return available.gemini ? "gemini" : "perplexity";
189
+ }
190
+ if (provider === "gemini" && !available.gemini) {
191
+ if (available.exa) return "exa";
192
+ return available.perplexity ? "perplexity" : "gemini";
193
+ }
194
+ return provider;
195
+ }
196
+
197
+ const pendingFetches = new Map<string, AbortController>();
198
+ let sessionActive = false;
199
+ let widgetVisible = false;
200
+ let widgetUnsubscribe: (() => void) | null = null;
201
+ let activeCurator: CuratorServerHandle | null = null;
202
+ let glimpseWin: GlimpseWindow | null = null;
203
+
204
+ interface PendingCurate {
205
+ phase: "searching" | "curating";
206
+ workflow: CuratorWorkflow;
207
+ summaryContext: SummaryGenerationContext;
208
+ searchResults: Map<number, QueryResultData>;
209
+ allInlineContent: ExtractedContent[];
210
+ queryList: string[];
211
+ includeContent: boolean;
212
+ numResults?: number;
213
+ recencyFilter?: "day" | "week" | "month" | "year";
214
+ domainFilter?: string[];
215
+ availableProviders: ProviderAvailability;
216
+ defaultProvider: ResolvedSearchProvider;
217
+ summaryModels: Array<{ value: string; label: string }>;
218
+ defaultSummaryModel: string | null;
219
+ timeoutSeconds: number;
220
+ onUpdate: ((update: { content: Array<{ type: string; text: string }>; details?: Record<string, unknown> }) => void) | undefined;
221
+ signal: AbortSignal | undefined;
222
+ abortSearches: () => void;
223
+ finish: (value: unknown) => void;
224
+ cancel: (reason?: "user" | "stale") => void;
225
+ browserPromise?: Promise<void>;
226
+ }
227
+
228
+ let pendingCurate: PendingCurate | null = null;
229
+
230
+ function cancelPendingCurate(reason: "user" | "stale" = "stale"): void {
231
+ pendingCurate?.cancel(reason);
232
+ }
233
+
234
+ const MAX_INLINE_CONTENT = 30000; // Content returned directly to agent
235
+
236
+ function stripThumbnails(results: ExtractedContent[]): ExtractedContent[] {
237
+ return results.map(({ thumbnail, frames, ...rest }) => rest);
238
+ }
239
+
240
+ function formatSearchSummary(results: SearchResult[], answer: string): string {
241
+ let output = answer ? `${answer}\n\n---\n\n**Sources:**\n` : "";
242
+ output += results.map((r, i) => `${i + 1}. ${r.title}\n ${r.url}`).join("\n\n");
243
+ return output;
244
+ }
245
+
246
+ function duplicateQuerySet(results: QueryResultData[]): Set<string> {
247
+ const counts = new Map<string, number>();
248
+ for (const result of results) {
249
+ counts.set(result.query, (counts.get(result.query) ?? 0) + 1);
250
+ }
251
+ const duplicates = new Set<string>();
252
+ for (const [query, count] of counts) {
253
+ if (count > 1) duplicates.add(query);
254
+ }
255
+ return duplicates;
256
+ }
257
+
258
+ function formatQueryHeader(query: string, provider: string | undefined, duplicateQueries: Set<string>): string {
259
+ const suffix = duplicateQueries.has(query) && provider ? ` (${provider})` : "";
260
+ return `## Query: "${query}"${suffix}\n\n`;
261
+ }
262
+
263
+ function hasFullInlineCoverage(urls: string[], inlineContent: ExtractedContent[] | undefined): boolean {
264
+ if (!inlineContent || inlineContent.length === 0) return false;
265
+ const coveredUrls = new Set(inlineContent.map(c => c.url));
266
+ return urls.every(url => coveredUrls.has(url));
267
+ }
268
+
269
+ function formatFullResults(queryData: QueryResultData): string {
270
+ let output = `## Results for: "${queryData.query}"\n\n`;
271
+ if (queryData.answer) {
272
+ output += `${queryData.answer}\n\n---\n\n`;
273
+ }
274
+ for (const r of queryData.results) {
275
+ output += `### ${r.title}\n${r.url}\n\n`;
276
+ }
277
+ return output;
278
+ }
279
+
280
+ function abortPendingFetches(): void {
281
+ for (const controller of pendingFetches.values()) {
282
+ controller.abort();
283
+ }
284
+ pendingFetches.clear();
285
+ }
286
+
287
+ function closeCurator(): void {
288
+ const win = glimpseWin;
289
+ glimpseWin = null;
290
+ try { win?.close(); } catch {}
291
+ cancelPendingCurate();
292
+ if (activeCurator) {
293
+ activeCurator.close();
294
+ activeCurator = null;
295
+ }
296
+ }
297
+
298
+ async function openInBrowser(pi: ExtensionAPI, url: string): Promise<void> {
299
+ const plat = platform();
300
+ const result = plat === "darwin"
301
+ ? await pi.exec("open", [url])
302
+ : plat === "win32"
303
+ ? await pi.exec("cmd", ["/c", "start", "", url])
304
+ : await pi.exec("xdg-open", [url]);
305
+ if (result.code !== 0) {
306
+ throw new Error(result.stderr || `Failed to open browser (exit code ${result.code})`);
307
+ }
308
+ }
309
+
310
+ interface GlimpseWindow {
311
+ on(event: "closed", handler: () => void): void;
312
+ on(event: "message", handler: (data: unknown) => void): void;
313
+ on(event: "ready", handler: (info: { screen?: { visibleHeight?: number } }) => void): void;
314
+ close(): void;
315
+ _write(obj: Record<string, unknown>): void;
316
+ }
317
+
318
+ let glimpseOpen: ((html: string, opts: Record<string, unknown>) => GlimpseWindow) | null | undefined;
319
+
320
+ function findGlimpseMjs(): string | null {
321
+ try {
322
+ const req = createRequire(import.meta.url);
323
+ return req.resolve("glimpseui");
324
+ } catch {
325
+ // Optional dependency.
326
+ }
327
+ try {
328
+ const globalRoot = execFileSync("npm", ["root", "-g"], { encoding: "utf-8" }).trim();
329
+ const entry = join(globalRoot, "glimpseui", "src", "glimpse.mjs");
330
+ if (existsSync(entry)) return entry;
331
+ } catch {
332
+ // npm may be unavailable.
333
+ }
334
+ return null;
335
+ }
336
+
337
+ async function getGlimpseOpen() {
338
+ if (glimpseOpen !== undefined) return glimpseOpen;
339
+ const resolved = findGlimpseMjs();
340
+ if (resolved) {
341
+ try {
342
+ glimpseOpen = (await import(resolved)).open;
343
+ return glimpseOpen;
344
+ } catch {}
345
+ }
346
+ glimpseOpen = null;
347
+ return glimpseOpen;
348
+ }
349
+
350
+ function openInGlimpse(
351
+ open: (html: string, opts: Record<string, unknown>) => GlimpseWindow,
352
+ url: string,
353
+ title: string,
354
+ ): GlimpseWindow {
355
+ const shellHTML = `<!DOCTYPE html>
356
+ <html>
357
+ <head><meta charset="UTF-8"><title>${title}</title></head>
358
+ <body style="margin:0; background:#1a1a2e;">
359
+ <script>window.location.replace(${JSON.stringify(url)});</script>
360
+ </body>
361
+ </html>`;
362
+ const win = open(shellHTML, {
363
+ width: 800,
364
+ height: 900,
365
+ title,
366
+ });
367
+
368
+ let maxHeight = 1200;
369
+ win.on("ready", (info) => {
370
+ const visibleHeight = info?.screen?.visibleHeight;
371
+ if (typeof visibleHeight === "number" && visibleHeight > 0) {
372
+ maxHeight = Math.floor(visibleHeight * 0.85);
373
+ }
374
+ });
375
+ win.on("message", (data) => {
376
+ if (!data || typeof data !== "object") return;
377
+ const msg = data as Record<string, unknown>;
378
+ if (msg.type !== "resize" || typeof msg.height !== "number") return;
379
+ const clamped = Math.max(400, Math.min(Math.round(msg.height), maxHeight));
380
+ win._write({ type: "resize", width: 800, height: clamped });
381
+ });
382
+
383
+ return win;
384
+ }
385
+
386
+ function extractDomain(url: string): string {
387
+ try { return new URL(url).hostname; }
388
+ catch { return url; }
389
+ }
390
+
391
+ function updateWidget(ctx: ExtensionContext): void {
392
+ const theme = ctx.ui.theme;
393
+ const entries = activityMonitor.getEntries();
394
+ const lines: string[] = [];
395
+
396
+ lines.push(theme.fg("accent", "─── Web Search Activity " + "─".repeat(36)));
397
+
398
+ if (entries.length === 0) {
399
+ lines.push(theme.fg("muted", " No activity yet"));
400
+ } else {
401
+ for (const e of entries) {
402
+ lines.push(" " + formatEntryLine(e, theme));
403
+ }
404
+ }
405
+
406
+ lines.push(theme.fg("accent", "─".repeat(60)));
407
+
408
+ const rateInfo = activityMonitor.getRateLimitInfo();
409
+ const resetMs = rateInfo.oldestTimestamp ? Math.max(0, rateInfo.oldestTimestamp + rateInfo.windowMs - Date.now()) : 0;
410
+ const resetSec = Math.ceil(resetMs / 1000);
411
+ lines.push(
412
+ theme.fg("muted", `Rate: ${rateInfo.used}/${rateInfo.max}`) +
413
+ (resetMs > 0 ? theme.fg("dim", ` (resets in ${resetSec}s)`) : ""),
414
+ );
415
+
416
+ ctx.ui.setWidget("web-activity", new Text(lines.join("\n"), 0, 0));
417
+ }
418
+
419
+ function formatEntryLine(
420
+ entry: ActivityEntry,
421
+ theme: { fg: (color: string, text: string) => string },
422
+ ): string {
423
+ const typeStr = entry.type === "api" ? "API" : "GET";
424
+ const target =
425
+ entry.type === "api"
426
+ ? `"${truncateToWidth(entry.query || "", 28, "")}"`
427
+ : truncateToWidth(entry.url?.replace(/^https?:\/\//, "") || "", 30, "");
428
+
429
+ const duration = entry.endTime
430
+ ? `${((entry.endTime - entry.startTime) / 1000).toFixed(1)}s`
431
+ : `${((Date.now() - entry.startTime) / 1000).toFixed(1)}s`;
432
+
433
+ let statusStr: string;
434
+ let indicator: string;
435
+ if (entry.error) {
436
+ statusStr = "err";
437
+ indicator = theme.fg("error", "✗");
438
+ } else if (entry.status === null) {
439
+ statusStr = "...";
440
+ indicator = theme.fg("warning", "⋯");
441
+ } else if (entry.status === 0) {
442
+ statusStr = "abort";
443
+ indicator = theme.fg("muted", "○");
444
+ } else {
445
+ statusStr = String(entry.status);
446
+ indicator = entry.status >= 200 && entry.status < 300 ? theme.fg("success", "✓") : theme.fg("error", "✗");
447
+ }
448
+
449
+ return `${typeStr.padEnd(4)} ${target.padEnd(32)} ${statusStr.padStart(5)} ${duration.padStart(5)} ${indicator}`;
450
+ }
451
+
452
+ function handleSessionChange(ctx: ExtensionContext): void {
453
+ abortPendingFetches();
454
+ closeCurator();
455
+ clearCloneCache();
456
+ sessionActive = true;
457
+ restoreFromSession(ctx);
458
+ // Unsubscribe before clear() to avoid callback with stale ctx
459
+ widgetUnsubscribe?.();
460
+ widgetUnsubscribe = null;
461
+ activityMonitor.clear();
462
+ if (widgetVisible) {
463
+ // Re-subscribe with new ctx
464
+ widgetUnsubscribe = activityMonitor.onUpdate(() => updateWidget(ctx));
465
+ updateWidget(ctx);
466
+ }
467
+ }
468
+
469
+ export default function (pi: ExtensionAPI) {
470
+ const initConfig = loadConfigForExtensionInit();
471
+ const curateKey = initConfig.shortcuts?.curate || DEFAULT_SHORTCUTS.curate;
472
+ const activityKey = initConfig.shortcuts?.activity || DEFAULT_SHORTCUTS.activity;
473
+
474
+ function startBackgroundFetch(urls: string[]): string | null {
475
+ if (urls.length === 0) return null;
476
+ const fetchId = generateId();
477
+ const controller = new AbortController();
478
+ pendingFetches.set(fetchId, controller);
479
+ fetchAllContent(urls, controller.signal)
480
+ .then((fetched) => {
481
+ if (!sessionActive || !pendingFetches.has(fetchId)) return;
482
+ const data: StoredSearchData = {
483
+ id: fetchId,
484
+ type: "fetch",
485
+ timestamp: Date.now(),
486
+ urls: stripThumbnails(fetched),
487
+ };
488
+ storeResult(fetchId, data);
489
+ pi.appendEntry("web-search-results", data);
490
+ const ok = fetched.filter(f => !f.error).length;
491
+ pi.sendMessage(
492
+ {
493
+ customType: "web-search-content-ready",
494
+ content: `Content fetched for ${ok}/${fetched.length} URLs [${fetchId}]. Full page content now available.`,
495
+ display: true,
496
+ },
497
+ { triggerTurn: true },
498
+ );
499
+ })
500
+ .catch((err) => {
501
+ if (!sessionActive || !pendingFetches.has(fetchId)) return;
502
+ const message = err instanceof Error ? err.message : String(err);
503
+ const isAbort = (err instanceof Error && err.name === "AbortError") || message.toLowerCase().includes("abort");
504
+ if (!isAbort) {
505
+ pi.sendMessage(
506
+ {
507
+ customType: "web-search-error",
508
+ content: `Content fetch failed [${fetchId}]: ${message}`,
509
+ display: true,
510
+ },
511
+ { triggerTurn: false },
512
+ );
513
+ }
514
+ })
515
+ .finally(() => { pendingFetches.delete(fetchId); });
516
+ return fetchId;
517
+ }
518
+
519
+ function storeAndPublishSearch(results: QueryResultData[]): string {
520
+ const id = generateId();
521
+ const data: StoredSearchData = {
522
+ id, type: "search", timestamp: Date.now(), queries: results,
523
+ };
524
+ storeResult(id, data);
525
+ pi.appendEntry("web-search-results", data);
526
+ return id;
527
+ }
528
+
529
+ interface SearchReturnOptions {
530
+ queryList: string[];
531
+ results: QueryResultData[];
532
+ urls: string[];
533
+ includeContent: boolean;
534
+ inlineContent?: ExtractedContent[];
535
+ curated?: boolean;
536
+ curatedFrom?: number;
537
+ workflow?: CuratorWorkflow;
538
+ approvedSummary?: string;
539
+ summaryMeta?: SummaryMeta;
540
+ }
541
+
542
+ function normalizeSummaryMeta(meta: SummaryMeta | undefined, summaryText: string): SummaryMeta {
543
+ const normalizedText = summaryText.trim();
544
+ if (!meta) {
545
+ return {
546
+ model: null,
547
+ durationMs: 0,
548
+ tokenEstimate: normalizedText.length > 0 ? Math.max(1, Math.ceil(normalizedText.length / 4)) : 0,
549
+ fallbackUsed: false,
550
+ edited: false,
551
+ };
552
+ }
553
+
554
+ return {
555
+ model: meta.model,
556
+ durationMs: Number.isFinite(meta.durationMs) && meta.durationMs >= 0 ? meta.durationMs : 0,
557
+ tokenEstimate: Number.isFinite(meta.tokenEstimate) && meta.tokenEstimate >= 0
558
+ ? meta.tokenEstimate
559
+ : (normalizedText.length > 0 ? Math.max(1, Math.ceil(normalizedText.length / 4)) : 0),
560
+ fallbackUsed: meta.fallbackUsed === true,
561
+ fallbackReason: meta.fallbackReason,
562
+ edited: meta.edited === true,
563
+ };
564
+ }
565
+
566
+ function buildCurationCancelledReturn(reason: "user" | "stale") {
567
+ const message = `Search curation cancelled (${reason}).`;
568
+ return {
569
+ content: [{ type: "text", text: message }],
570
+ details: {
571
+ error: message,
572
+ cancelled: true,
573
+ cancelReason: reason,
574
+ },
575
+ };
576
+ }
577
+
578
+ async function resolveFirstAvailableModel(
579
+ ctx: SummaryGenerationContext,
580
+ candidates: Array<{ provider: string; id: string }>,
581
+ ): Promise<{ model: Model; apiKey: string; headers?: Record<string, string> }> {
582
+ for (const { provider, id } of candidates) {
583
+ const model = getModel(provider, id);
584
+ if (!model) continue;
585
+ const auth = await ctx.modelRegistry.getApiKeyAndHeaders(model);
586
+ if (auth.ok && auth.apiKey) return { model, apiKey: auth.apiKey, headers: auth.headers };
587
+ }
588
+ throw new Error(`No model available: ${candidates.map(c => `${c.provider}/${c.id}`).join(", ")}`);
589
+ }
590
+
591
+ async function rewriteSearchQuery(query: string, ctx: SummaryGenerationContext, signal: AbortSignal): Promise<string> {
592
+ const { model, apiKey, headers } = await resolveFirstAvailableModel(ctx, [
593
+ { provider: "anthropic", id: "claude-haiku-4-5" },
594
+ { provider: "google", id: "gemini-2.5-flash" },
595
+ { provider: "openai", id: "gpt-4.1-mini" },
596
+ ]);
597
+ const response = await complete(
598
+ model,
599
+ {
600
+ messages: [{
601
+ role: "user",
602
+ content: [{ type: "text", text: `Rewrite this web search query to get better, more specific results. Add relevant year qualifiers, precise technical terms, and specificity. Return ONLY the improved query text, nothing else.\n\nQuery: ${query}` }],
603
+ timestamp: Date.now(),
604
+ }],
605
+ },
606
+ { apiKey, headers, signal },
607
+ );
608
+ if (response.stopReason === "aborted") throw new Error("Aborted");
609
+ const contentParts = Array.isArray(response.content) ? response.content : [];
610
+ const text = contentParts
611
+ .map(p => {
612
+ if (!p || typeof p !== "object") return "";
613
+ const part = p as Record<string, unknown>;
614
+ return typeof part.text === "string" ? part.text : "";
615
+ })
616
+ .join("")
617
+ .trim();
618
+ if (!text) throw new Error("Rewrite returned empty response");
619
+ return text;
620
+ }
621
+
622
+ async function generateSummaryForSelectedIndices(
623
+ selectedQueryIndices: number[],
624
+ resultsByIndex: Map<number, QueryResultData>,
625
+ summaryContext: SummaryGenerationContext,
626
+ signal?: AbortSignal,
627
+ modelOverride?: string,
628
+ feedback?: string,
629
+ ): Promise<{ summary: string; meta: SummaryMeta }> {
630
+ const selectedResults: QueryResultData[] = [];
631
+ for (const qi of selectedQueryIndices) {
632
+ const result = resultsByIndex.get(qi);
633
+ if (result) selectedResults.push(result);
634
+ }
635
+ if (selectedResults.length === 0) {
636
+ throw new Error("No selected results available for summary generation");
637
+ }
638
+ try {
639
+ return await generateSummaryDraft(selectedResults, summaryContext, signal, modelOverride, feedback);
640
+ } catch (err) {
641
+ const isEmptyResponse = err instanceof Error && err.message.includes("Summary model returned empty response");
642
+ if (!isEmptyResponse) throw err;
643
+ const deterministic = buildDeterministicSummary(selectedResults);
644
+ return {
645
+ summary: deterministic.summary,
646
+ meta: {
647
+ ...deterministic.meta,
648
+ fallbackReason: "summary-model-empty-response",
649
+ },
650
+ };
651
+ }
652
+ }
653
+
654
+ async function loadSummaryModelChoices(
655
+ summaryContext: SummaryGenerationContext,
656
+ ): Promise<{ summaryModels: Array<{ value: string; label: string }>; defaultSummaryModel: string | null }> {
657
+ const summaryModels: Array<{ value: string; label: string }> = [];
658
+ const seen = new Set<string>();
659
+ const availableValues = new Set<string>();
660
+
661
+ const addModel = (provider: string, id: string) => {
662
+ const value = `${provider}/${id}`;
663
+ if (seen.has(value)) return;
664
+ seen.add(value);
665
+ summaryModels.push({ value, label: value });
666
+ };
667
+
668
+ try {
669
+ const availableModels = summaryContext.modelRegistry.getAvailable();
670
+ for (const model of availableModels) {
671
+ const value = `${model.provider}/${model.id}`;
672
+ availableValues.add(value);
673
+ addModel(model.provider, model.id);
674
+ }
675
+ } catch (err) {
676
+ const message = err instanceof Error ? err.message : String(err);
677
+ console.error(`Failed to load summary models: ${message}`);
678
+ }
679
+
680
+ const currentModelValue = summaryContext.model
681
+ ? `${summaryContext.model.provider}/${summaryContext.model.id}`
682
+ : null;
683
+ if (summaryContext.model && currentModelValue && !seen.has(currentModelValue)) {
684
+ addModel(summaryContext.model.provider, summaryContext.model.id);
685
+ }
686
+
687
+ const config = loadConfig();
688
+ const configuredSummaryModel = typeof config.summaryModel === "string" ? config.summaryModel.trim() : "";
689
+ const preferredDefaults = [
690
+ "anthropic/claude-haiku-4-5",
691
+ "openai-codex/gpt-5.3-codex-spark",
692
+ ];
693
+
694
+ let defaultSummaryModel: string | null = null;
695
+ if (configuredSummaryModel.length > 0 && availableValues.has(configuredSummaryModel)) {
696
+ defaultSummaryModel = configuredSummaryModel;
697
+ }
698
+ if (!defaultSummaryModel) {
699
+ for (const preferred of preferredDefaults) {
700
+ if (availableValues.has(preferred)) {
701
+ defaultSummaryModel = preferred;
702
+ break;
703
+ }
704
+ }
705
+ }
706
+ if (!defaultSummaryModel && summaryModels.length > 0) {
707
+ defaultSummaryModel = summaryModels[0].value;
708
+ }
709
+
710
+ return { summaryModels, defaultSummaryModel };
711
+ }
712
+
713
+ function resolveSummaryForSubmit(
714
+ payload: { selectedQueryIndices: number[]; summary?: string; summaryMeta?: SummaryMeta },
715
+ resultsByIndex: Map<number, QueryResultData>,
716
+ ): { approvedSummary: string; summaryMeta: SummaryMeta } {
717
+ const submittedSummary = typeof payload.summary === "string" ? payload.summary.trim() : "";
718
+ if (submittedSummary.length > 0) {
719
+ return {
720
+ approvedSummary: submittedSummary,
721
+ summaryMeta: normalizeSummaryMeta(payload.summaryMeta, submittedSummary),
722
+ };
723
+ }
724
+
725
+ const selected = filterByQueryIndices(payload.selectedQueryIndices, resultsByIndex).results;
726
+ const fallbackResults = selected.length > 0 ? selected : [...resultsByIndex.values()];
727
+ const deterministic = buildDeterministicSummary(fallbackResults);
728
+ return {
729
+ approvedSummary: deterministic.summary,
730
+ summaryMeta: deterministic.meta,
731
+ };
732
+ }
733
+
734
+ function buildSearchReturn(opts: SearchReturnOptions) {
735
+ const sc = opts.results.filter(r => !r.error).length;
736
+ const tr = opts.results.reduce((sum, r) => sum + r.results.length, 0);
737
+
738
+ const hasApprovedSummary = typeof opts.approvedSummary === "string" && opts.approvedSummary.trim().length > 0;
739
+ let output = "";
740
+ if (hasApprovedSummary) {
741
+ output = opts.approvedSummary!.trim();
742
+ } else {
743
+ if (opts.curated) {
744
+ output += "[These results were manually curated by the user in the browser. Use them as-is — do not re-search or discard.]\n\n";
745
+ }
746
+ const duplicateQueries = opts.curated ? duplicateQuerySet(opts.results) : new Set<string>();
747
+ for (const { query, answer, results, error, provider } of opts.results) {
748
+ if (opts.queryList.length > 1) {
749
+ output += opts.curated
750
+ ? formatQueryHeader(query, provider, duplicateQueries)
751
+ : `## Query: "${query}"\n\n`;
752
+ }
753
+ if (error) output += `Error: ${error}\n\n`;
754
+ else if (results.length === 0) output += "No results found.\n\n";
755
+ else output += formatSearchSummary(results, answer) + "\n\n";
756
+ }
757
+ }
758
+
759
+ const hasInlineReady = hasFullInlineCoverage(opts.urls, opts.inlineContent);
760
+ let fetchId: string | null = null;
761
+ if (hasInlineReady && opts.inlineContent) {
762
+ fetchId = generateId();
763
+ const data: StoredSearchData = {
764
+ id: fetchId,
765
+ type: "fetch",
766
+ timestamp: Date.now(),
767
+ urls: opts.inlineContent,
768
+ };
769
+ storeResult(fetchId, data);
770
+ pi.appendEntry("web-search-results", data);
771
+ if (!hasApprovedSummary) {
772
+ output += `---\nFull content for ${opts.inlineContent.length} sources available [${fetchId}].`;
773
+ }
774
+ } else if (opts.includeContent) {
775
+ fetchId = startBackgroundFetch(opts.urls);
776
+ if (fetchId && !hasApprovedSummary) {
777
+ output += `---\nContent fetching in background [${fetchId}]. Will notify when ready.`;
778
+ }
779
+ }
780
+
781
+ const searchId = storeAndPublishSearch(opts.results);
782
+ const isBackgroundFetch = fetchId !== null && !hasInlineReady;
783
+
784
+ return {
785
+ content: [{ type: "text", text: output.trim() }],
786
+ details: {
787
+ queries: opts.queryList,
788
+ queryCount: opts.queryList.length,
789
+ successfulQueries: sc,
790
+ totalResults: tr,
791
+ includeContent: opts.includeContent,
792
+ fetchId,
793
+ fetchUrls: isBackgroundFetch ? opts.urls : undefined,
794
+ searchId,
795
+ ...(opts.curated ? {
796
+ curated: true,
797
+ curatedFrom: opts.curatedFrom,
798
+ curatedQueries: opts.results.map(r => ({
799
+ query: r.query,
800
+ provider: r.provider || null,
801
+ answer: r.answer || null,
802
+ sources: r.results.map(s => ({ title: s.title, url: s.url })),
803
+ error: r.error,
804
+ })),
805
+ } : {}),
806
+ ...((opts.workflow && hasApprovedSummary)
807
+ ? {
808
+ summary: {
809
+ text: opts.approvedSummary!.trim(),
810
+ workflow: opts.workflow,
811
+ model: opts.summaryMeta?.model ?? null,
812
+ durationMs: opts.summaryMeta?.durationMs ?? 0,
813
+ tokenEstimate: opts.summaryMeta?.tokenEstimate ?? 0,
814
+ fallbackUsed: opts.summaryMeta?.fallbackUsed === true,
815
+ fallbackReason: opts.summaryMeta?.fallbackReason,
816
+ edited: opts.summaryMeta?.edited === true,
817
+ },
818
+ }
819
+ : {}),
820
+ },
821
+ };
822
+ }
823
+
824
+ function filterByQueryIndices(selectedQueryIndices: number[], results: Map<number, QueryResultData>) {
825
+ const filteredResults: QueryResultData[] = [];
826
+ const filteredUrls: string[] = [];
827
+ for (const qi of selectedQueryIndices) {
828
+ const r = results.get(qi);
829
+ if (r) {
830
+ filteredResults.push(r);
831
+ for (const res of r.results) {
832
+ if (!filteredUrls.includes(res.url)) filteredUrls.push(res.url);
833
+ }
834
+ }
835
+ }
836
+ return { results: filteredResults, urls: filteredUrls };
837
+ }
838
+
839
+ function collectAllResultsAndUrls(resultsByIndex: Map<number, QueryResultData>) {
840
+ const results = [...resultsByIndex.values()];
841
+ const urls: string[] = [];
842
+ for (const result of results) {
843
+ for (const source of result.results) {
844
+ if (!urls.includes(source.url)) urls.push(source.url);
845
+ }
846
+ }
847
+ return { results, urls };
848
+ }
849
+
850
+ async function openCuratorBrowser(pc: PendingCurate, searchesComplete = true): Promise<void> {
851
+ let handle: CuratorServerHandle | null = null;
852
+ try {
853
+ pc.phase = "curating";
854
+
855
+ const searchAbort = new AbortController();
856
+ const addSearchSignal = pc.signal
857
+ ? AbortSignal.any([pc.signal, searchAbort.signal])
858
+ : searchAbort.signal;
859
+
860
+ const sessionToken = randomUUID();
861
+ handle = await startCuratorServer(
862
+ {
863
+ queries: pc.queryList,
864
+ sessionToken,
865
+ timeout: pc.timeoutSeconds,
866
+ availableProviders: pc.availableProviders,
867
+ defaultProvider: pc.defaultProvider,
868
+ summaryModels: pc.summaryModels,
869
+ defaultSummaryModel: pc.defaultSummaryModel,
870
+ },
871
+ {
872
+ async onSummarize(selectedQueryIndices, summarizeSignal, model, feedback) {
873
+ if (pendingCurate !== pc) throw new Error("Curator session is no longer active.");
874
+ pc.onUpdate?.({
875
+ content: [{ type: "text", text: "Generating summary draft..." }],
876
+ details: { phase: "generating-summary", progress: 0.9 },
877
+ });
878
+ const draft = await generateSummaryForSelectedIndices(
879
+ selectedQueryIndices,
880
+ pc.searchResults,
881
+ pc.summaryContext,
882
+ summarizeSignal,
883
+ model,
884
+ feedback,
885
+ );
886
+ if (pendingCurate !== pc) throw new Error("Curator session is no longer active.");
887
+ pc.onUpdate?.({
888
+ content: [{ type: "text", text: "Summary draft ready — waiting for approval..." }],
889
+ details: { phase: "waiting-for-approval", progress: 1 },
890
+ });
891
+ return draft;
892
+ },
893
+ onSubmit(payload) {
894
+ if (pendingCurate !== pc) return;
895
+ searchAbort.abort();
896
+ const filtered = payload.selectedQueryIndices.length > 0
897
+ ? filterByQueryIndices(payload.selectedQueryIndices, pc.searchResults)
898
+ : collectAllResultsAndUrls(pc.searchResults);
899
+ const filteredInline = pc.allInlineContent.filter(c => filtered.urls.includes(c.url));
900
+ const base: SearchReturnOptions = {
901
+ queryList: filtered.results.map(r => r.query),
902
+ results: filtered.results,
903
+ urls: filtered.urls,
904
+ includeContent: pc.includeContent,
905
+ inlineContent: filteredInline.length > 0 ? filteredInline : undefined,
906
+ curated: true,
907
+ curatedFrom: pc.searchResults.size,
908
+ };
909
+ if (!payload.rawResults) {
910
+ const resolvedSummary = resolveSummaryForSubmit(payload, pc.searchResults);
911
+ base.workflow = pc.workflow;
912
+ base.approvedSummary = resolvedSummary.approvedSummary;
913
+ base.summaryMeta = resolvedSummary.summaryMeta;
914
+ }
915
+ pc.finish(buildSearchReturn(base));
916
+ closeCurator();
917
+ },
918
+ onCancel(reason) {
919
+ if (pendingCurate !== pc) return;
920
+ searchAbort.abort();
921
+ if (reason === "timeout") {
922
+ const resolvedSummary = resolveSummaryForSubmit({ selectedQueryIndices: [], summary: undefined, summaryMeta: undefined }, pc.searchResults);
923
+ const all = collectAllResultsAndUrls(pc.searchResults);
924
+ const filteredInline = pc.allInlineContent.filter(c => all.urls.includes(c.url));
925
+ pc.finish(buildSearchReturn({
926
+ queryList: all.results.map(r => r.query),
927
+ results: all.results,
928
+ urls: all.urls,
929
+ includeContent: pc.includeContent,
930
+ inlineContent: filteredInline.length > 0 ? filteredInline : undefined,
931
+ curated: true,
932
+ curatedFrom: pc.searchResults.size,
933
+ workflow: pc.workflow,
934
+ approvedSummary: resolvedSummary.approvedSummary,
935
+ summaryMeta: resolvedSummary.summaryMeta,
936
+ }));
937
+ } else {
938
+ pc.finish(buildCurationCancelledReturn(reason));
939
+ }
940
+ closeCurator();
941
+ },
942
+ onProviderChange(provider) {
943
+ if (pendingCurate !== pc) return;
944
+ const normalized = normalizeProviderInput(provider);
945
+ if (!normalized || normalized === "auto") return;
946
+ pc.defaultProvider = normalized;
947
+ try {
948
+ saveConfig({ provider: normalized });
949
+ } catch (err) {
950
+ const message = err instanceof Error ? err.message : String(err);
951
+ console.error(`Failed to persist default provider: ${message}`);
952
+ }
953
+ },
954
+ async onAddSearch(query, queryIndex, provider) {
955
+ if (pendingCurate !== pc) throw new Error("Curator session is no longer active.");
956
+ const normalizedProvider = normalizeProviderInput(provider);
957
+ const requestedProvider = !normalizedProvider || normalizedProvider === "auto"
958
+ ? pc.defaultProvider
959
+ : normalizedProvider;
960
+ try {
961
+ const { answer, results, inlineContent, provider: actualProvider } = await search(query, {
962
+ provider: requestedProvider,
963
+ numResults: pc.numResults,
964
+ recencyFilter: pc.recencyFilter,
965
+ domainFilter: pc.domainFilter,
966
+ includeContent: pc.includeContent,
967
+ signal: addSearchSignal,
968
+ });
969
+ if (pendingCurate !== pc) throw new Error("Curator session is no longer active.");
970
+ pc.searchResults.set(queryIndex, { query, answer, results, error: null, provider: actualProvider });
971
+ if (inlineContent) pc.allInlineContent.push(...inlineContent);
972
+ return {
973
+ answer,
974
+ results: results.map(r => ({ title: r.title, url: r.url, domain: extractDomain(r.url) })),
975
+ provider: actualProvider,
976
+ };
977
+ } catch (err) {
978
+ const message = err instanceof Error ? err.message : String(err);
979
+ if (pendingCurate === pc) {
980
+ pc.searchResults.set(queryIndex, { query, answer: "", results: [], error: message, provider: requestedProvider });
981
+ }
982
+ throw err;
983
+ }
984
+ },
985
+ async onRewriteQuery(query, rewriteSignal) {
986
+ if (pendingCurate !== pc) throw new Error("Curator session is no longer active.");
987
+ return rewriteSearchQuery(query, pc.summaryContext, rewriteSignal);
988
+ },
989
+ },
990
+ );
991
+
992
+ if (pendingCurate !== pc) {
993
+ handle.close();
994
+ return;
995
+ }
996
+
997
+ activeCurator = handle;
998
+
999
+ for (const [qi, data] of pc.searchResults) {
1000
+ if (data.error) {
1001
+ handle.pushError(qi, data.error, data.provider);
1002
+ } else {
1003
+ handle.pushResult(qi, {
1004
+ answer: data.answer,
1005
+ results: data.results.map(r => ({ title: r.title, url: r.url, domain: extractDomain(r.url) })),
1006
+ provider: data.provider || pc.defaultProvider,
1007
+ });
1008
+ }
1009
+ }
1010
+ if (searchesComplete) handle.searchesDone();
1011
+
1012
+ pc.onUpdate?.({
1013
+ content: [{ type: "text", text: searchesComplete ? "Waiting for summary approval in browser..." : "Searches streaming to browser..." }],
1014
+ details: { phase: "curating", progress: searchesComplete ? 1 : 0.5 },
1015
+ });
1016
+
1017
+ const open = platform() === "darwin" ? await getGlimpseOpen() : null;
1018
+ if (open) {
1019
+ try {
1020
+ const win = openInGlimpse(open, handle.url, "Search Curator");
1021
+ glimpseWin = win;
1022
+ win.on("closed", () => {
1023
+ if (glimpseWin === win) {
1024
+ glimpseWin = null;
1025
+ closeCurator();
1026
+ }
1027
+ });
1028
+ return;
1029
+ } catch (err) {
1030
+ const message = err instanceof Error ? err.message : String(err);
1031
+ console.error(`Failed to open Glimpse curator window: ${message}`);
1032
+ glimpseWin = null;
1033
+ }
1034
+ }
1035
+ await openInBrowser(pi, handle.url);
1036
+ } catch (err) {
1037
+ const message = err instanceof Error ? err.message : String(err);
1038
+ console.error(`Failed to open curator UI: ${message}`);
1039
+ if (pendingCurate === pc || (handle && activeCurator === handle)) {
1040
+ closeCurator();
1041
+ }
1042
+ }
1043
+ }
1044
+
1045
+ pi.registerShortcut(curateKey, {
1046
+ description: "Review search results",
1047
+ handler: async (ctx) => {
1048
+ if (!pendingCurate) return;
1049
+
1050
+ if (pendingCurate.phase === "searching") {
1051
+ pendingCurate.browserPromise = openCuratorBrowser(pendingCurate, false);
1052
+ ctx.ui.notify("Opening curator — remaining searches will stream in", "info");
1053
+ return;
1054
+ }
1055
+ },
1056
+ });
1057
+
1058
+ pi.registerShortcut(activityKey, {
1059
+ description: "Toggle web search activity",
1060
+ handler: async (ctx) => {
1061
+ widgetVisible = !widgetVisible;
1062
+ if (widgetVisible) {
1063
+ widgetUnsubscribe = activityMonitor.onUpdate(() => updateWidget(ctx));
1064
+ updateWidget(ctx);
1065
+ } else {
1066
+ widgetUnsubscribe?.();
1067
+ widgetUnsubscribe = null;
1068
+ ctx.ui.setWidget("web-activity", null);
1069
+ }
1070
+ },
1071
+ });
1072
+
1073
+ pi.on("session_start", async (_event, ctx) => handleSessionChange(ctx));
1074
+ pi.on("session_tree", async (_event, ctx) => handleSessionChange(ctx));
1075
+
1076
+ pi.on("session_shutdown", () => {
1077
+ sessionActive = false;
1078
+ abortPendingFetches();
1079
+ closeCurator();
1080
+ clearCloneCache();
1081
+ clearResults();
1082
+ // Unsubscribe before clear() to avoid callback with stale ctx
1083
+ widgetUnsubscribe?.();
1084
+ widgetUnsubscribe = null;
1085
+ activityMonitor.clear();
1086
+ widgetVisible = false;
1087
+ });
1088
+
1089
+ pi.registerTool({
1090
+ name: "web_search",
1091
+ label: "Web Search",
1092
+ description:
1093
+ `Search the web using Perplexity AI, Exa, or Gemini. Returns an AI-synthesized answer with source citations. For comprehensive research, prefer queries (plural) with 2-4 varied angles over a single query — each query gets its own synthesized answer, so varying phrasing and scope gives much broader coverage. When includeContent is true, full page content is fetched in the background. Searches auto-open the interactive browser curator and stream results live; set workflow to "none" to skip curation. Provider auto-selects: Exa (direct API with key, MCP fallback without), else Perplexity (needs key), else Gemini API (needs key), else Gemini Web (needs a supported Chromium-based browser login).`,
1094
+ promptSnippet:
1095
+ "Use for web research questions. Prefer {queries:[...]} with 2-4 varied angles over a single query for broader coverage.",
1096
+ parameters: Type.Object({
1097
+ query: Type.Optional(Type.String({ description: "Single search query. For research tasks, prefer 'queries' with multiple varied angles instead." })),
1098
+ queries: Type.Optional(Type.Array(Type.String(), { description: "Multiple queries searched in sequence, each returning its own synthesized answer. Prefer this for research — vary phrasing, scope, and angle across 2-4 queries to maximize coverage. Good: ['React vs Vue performance benchmarks 2026', 'React vs Vue developer experience comparison', 'React ecosystem size vs Vue ecosystem']. Bad: ['React vs Vue', 'React vs Vue comparison', 'React vs Vue review'] (too similar, redundant results)." })),
1099
+ numResults: Type.Optional(Type.Number({ description: "Results per query (default: 5, max: 20)" })),
1100
+ includeContent: Type.Optional(Type.Boolean({ description: "Fetch full page content (async)" })),
1101
+ recencyFilter: Type.Optional(
1102
+ StringEnum(["day", "week", "month", "year"], { description: "Filter by recency" }),
1103
+ ),
1104
+ domainFilter: Type.Optional(Type.Array(Type.String(), { description: "Limit to domains (prefix with - to exclude)" })),
1105
+ provider: Type.Optional(
1106
+ StringEnum(["auto", "perplexity", "gemini", "exa"], { description: "Search provider (default: auto)" }),
1107
+ ),
1108
+ workflow: Type.Optional(
1109
+ StringEnum(["none", "summary-review"], {
1110
+ description: "Search workflow mode: none = no curator, summary-review = open curator with auto summary draft (default)",
1111
+ }),
1112
+ ),
1113
+ }),
1114
+
1115
+ async execute(_toolCallId, params, signal, onUpdate, ctx) {
1116
+ const rawQueryList: unknown[] = Array.isArray(params.queries)
1117
+ ? params.queries
1118
+ : (params.query !== undefined ? [params.query] : []);
1119
+ const queryList = normalizeQueryList(rawQueryList);
1120
+ const configWorkflow = loadConfigForExtensionInit().workflow;
1121
+ const workflow = resolveWorkflow(params.workflow ?? configWorkflow, ctx?.hasUI !== false);
1122
+ const shouldCurate = workflow !== "none";
1123
+
1124
+ if (queryList.length === 0) {
1125
+ return {
1126
+ content: [{ type: "text", text: "Error: No query provided. Use 'query' or 'queries' parameter." }],
1127
+ details: { error: "No query provided" },
1128
+ };
1129
+ }
1130
+
1131
+ if (shouldCurate && !ctx) {
1132
+ return {
1133
+ content: [{ type: "text", text: "Error: Curation requires an active extension context." }],
1134
+ details: { error: "Missing extension context" },
1135
+ };
1136
+ }
1137
+
1138
+ if (shouldCurate) {
1139
+ closeCurator();
1140
+
1141
+ let resolvePromise: (value: unknown) => void = () => {};
1142
+ const promise = new Promise<unknown>((resolve) => {
1143
+ resolvePromise = resolve;
1144
+ });
1145
+ const includeContent = params.includeContent ?? false;
1146
+ const searchResults = new Map<number, QueryResultData>();
1147
+ const allInlineContent: ExtractedContent[] = [];
1148
+ const searchAbort = new AbortController();
1149
+ const searchSignal = signal
1150
+ ? AbortSignal.any([signal, searchAbort.signal])
1151
+ : searchAbort.signal;
1152
+ let cancelled = false;
1153
+
1154
+ const bootstrap = await loadCuratorBootstrap(params.provider);
1155
+ const availableProviders = bootstrap.availableProviders;
1156
+ const defaultProvider = bootstrap.defaultProvider;
1157
+ const curatorTimeoutSeconds = bootstrap.timeoutSeconds;
1158
+ const curatorWorkflow: CuratorWorkflow = "summary-review";
1159
+
1160
+ const summaryContext: SummaryGenerationContext = {
1161
+ model: ctx.model,
1162
+ modelRegistry: ctx.modelRegistry,
1163
+ };
1164
+ const summaryModelChoices = await loadSummaryModelChoices(summaryContext);
1165
+
1166
+ const pc: PendingCurate = {
1167
+ phase: "searching",
1168
+ workflow: curatorWorkflow,
1169
+ summaryContext,
1170
+ searchResults,
1171
+ allInlineContent,
1172
+ queryList,
1173
+ includeContent,
1174
+ numResults: params.numResults,
1175
+ recencyFilter: params.recencyFilter,
1176
+ domainFilter: params.domainFilter,
1177
+ availableProviders,
1178
+ defaultProvider,
1179
+ summaryModels: summaryModelChoices.summaryModels,
1180
+ defaultSummaryModel: summaryModelChoices.defaultSummaryModel,
1181
+ timeoutSeconds: curatorTimeoutSeconds,
1182
+ onUpdate: onUpdate as PendingCurate["onUpdate"],
1183
+ signal,
1184
+ abortSearches: () => {
1185
+ if (!searchAbort.signal.aborted) searchAbort.abort();
1186
+ },
1187
+ finish: () => {},
1188
+ cancel: () => {},
1189
+ };
1190
+
1191
+ const finish = (value: unknown) => {
1192
+ if (cancelled) return;
1193
+ cancelled = true;
1194
+ pc.abortSearches();
1195
+ signal?.removeEventListener("abort", onAbort);
1196
+ pendingCurate = null;
1197
+ resolvePromise(value);
1198
+ };
1199
+
1200
+ const cancel = (reason: "user" | "stale" = "stale") => {
1201
+ if (cancelled) return;
1202
+ finish(buildCurationCancelledReturn(reason));
1203
+ };
1204
+
1205
+ pc.finish = finish;
1206
+ pc.cancel = cancel;
1207
+
1208
+ const onAbort = () => closeCurator();
1209
+ pendingCurate = pc;
1210
+ signal?.addEventListener("abort", onAbort, { once: true });
1211
+ pc.browserPromise = openCuratorBrowser(pc, false);
1212
+
1213
+ for (let qi = 0; qi < queryList.length; qi++) {
1214
+ if (signal?.aborted || cancelled || searchAbort.signal.aborted) break;
1215
+ onUpdate?.({
1216
+ content: [{ type: "text", text: `Searching ${qi + 1}/${queryList.length}: "${queryList[qi]}"...` }],
1217
+ details: { phase: "searching", progress: qi / queryList.length, currentQuery: queryList[qi] },
1218
+ });
1219
+ const requestedProvider = pc.defaultProvider;
1220
+ try {
1221
+ const { answer, results, inlineContent, provider } = await search(queryList[qi], {
1222
+ provider: requestedProvider,
1223
+ numResults: params.numResults,
1224
+ recencyFilter: params.recencyFilter,
1225
+ domainFilter: params.domainFilter,
1226
+ includeContent: params.includeContent,
1227
+ signal: searchSignal,
1228
+ });
1229
+ if (signal?.aborted || cancelled || searchAbort.signal.aborted) break;
1230
+ searchResults.set(qi, { query: queryList[qi], answer, results, error: null, provider });
1231
+ if (inlineContent) allInlineContent.push(...inlineContent);
1232
+ if (activeCurator) {
1233
+ activeCurator.pushResult(qi, {
1234
+ answer,
1235
+ results: results.map(r => ({ title: r.title, url: r.url, domain: extractDomain(r.url) })),
1236
+ provider,
1237
+ });
1238
+ }
1239
+ } catch (err) {
1240
+ if (signal?.aborted || cancelled || searchAbort.signal.aborted) break;
1241
+ const message = err instanceof Error ? err.message : String(err);
1242
+ searchResults.set(qi, { query: queryList[qi], answer: "", results: [], error: message, provider: requestedProvider });
1243
+ if (activeCurator) {
1244
+ activeCurator.pushError(qi, message, requestedProvider);
1245
+ }
1246
+ }
1247
+ }
1248
+
1249
+ if (signal?.aborted || cancelled || searchAbort.signal.aborted) {
1250
+ cancel();
1251
+ return promise;
1252
+ }
1253
+
1254
+ await pc.browserPromise;
1255
+ if (activeCurator && !cancelled) {
1256
+ activeCurator.searchesDone();
1257
+ pc.onUpdate?.({
1258
+ content: [{ type: "text", text: "All searches complete — waiting for summary approval in browser..." }],
1259
+ details: { phase: "curating", progress: 1 },
1260
+ });
1261
+ }
1262
+
1263
+ return promise;
1264
+ }
1265
+
1266
+ const searchResults: QueryResultData[] = [];
1267
+ const allUrls: string[] = [];
1268
+ const allInlineContent: ExtractedContent[] = [];
1269
+ const resolvedProvider = normalizeProviderInput(params.provider ?? loadConfig().provider);
1270
+
1271
+ for (let i = 0; i < queryList.length; i++) {
1272
+ const query = queryList[i];
1273
+
1274
+ onUpdate?.({
1275
+ content: [{ type: "text", text: `Searching ${i + 1}/${queryList.length}: "${query}"...` }],
1276
+ details: { phase: "search", progress: i / queryList.length, currentQuery: query },
1277
+ });
1278
+
1279
+ try {
1280
+ const { answer, results, inlineContent, provider } = await search(query, {
1281
+ provider: resolvedProvider,
1282
+ numResults: params.numResults,
1283
+ recencyFilter: params.recencyFilter,
1284
+ domainFilter: params.domainFilter,
1285
+ includeContent: params.includeContent,
1286
+ signal,
1287
+ });
1288
+
1289
+ searchResults.push({ query, answer, results, error: null, provider });
1290
+ for (const r of results) {
1291
+ if (!allUrls.includes(r.url)) {
1292
+ allUrls.push(r.url);
1293
+ }
1294
+ }
1295
+ if (inlineContent) allInlineContent.push(...inlineContent);
1296
+ } catch (err) {
1297
+ const message = err instanceof Error ? err.message : String(err);
1298
+ const requestedProvider = typeof resolvedProvider === "string" && resolvedProvider !== "auto"
1299
+ ? resolvedProvider
1300
+ : undefined;
1301
+ searchResults.push({ query, answer: "", results: [], error: message, provider: requestedProvider });
1302
+ }
1303
+ }
1304
+
1305
+ return buildSearchReturn({
1306
+ queryList,
1307
+ results: searchResults,
1308
+ urls: allUrls,
1309
+ includeContent: params.includeContent ?? false,
1310
+ inlineContent: allInlineContent.length > 0 ? allInlineContent : undefined,
1311
+ });
1312
+ },
1313
+
1314
+ renderCall(args, theme) {
1315
+ const input = args as { query?: unknown; queries?: unknown };
1316
+ const rawQueryList: unknown[] = Array.isArray(input.queries)
1317
+ ? input.queries
1318
+ : (input.query !== undefined ? [input.query] : []);
1319
+ const queryList = normalizeQueryList(rawQueryList);
1320
+ if (queryList.length === 0) {
1321
+ return new Text(theme.fg("toolTitle", theme.bold("search ")) + theme.fg("error", "(no query)"), 0, 0);
1322
+ }
1323
+ if (queryList.length === 1) {
1324
+ const q = queryList[0];
1325
+ const display = q.length > 60 ? q.slice(0, 57) + "..." : q;
1326
+ return new Text(theme.fg("toolTitle", theme.bold("search ")) + theme.fg("accent", `"${display}"`), 0, 0);
1327
+ }
1328
+ const lines = [theme.fg("toolTitle", theme.bold("search ")) + theme.fg("accent", `${queryList.length} queries`)];
1329
+ for (const q of queryList.slice(0, 5)) {
1330
+ const display = q.length > 50 ? q.slice(0, 47) + "..." : q;
1331
+ lines.push(theme.fg("muted", ` "${display}"`));
1332
+ }
1333
+ if (queryList.length > 5) {
1334
+ lines.push(theme.fg("muted", ` ... and ${queryList.length - 5} more`));
1335
+ }
1336
+ return new Text(lines.join("\n"), 0, 0);
1337
+ },
1338
+
1339
+ renderResult(result, { expanded, isPartial }, theme) {
1340
+ type QueryDetail = {
1341
+ query: string;
1342
+ provider: string | null;
1343
+ answer: string | null;
1344
+ sources: Array<{ title: string; url: string }>;
1345
+ error: string | null;
1346
+ };
1347
+ const details = result.details as {
1348
+ queryCount?: number;
1349
+ successfulQueries?: number;
1350
+ totalResults?: number;
1351
+ error?: string;
1352
+ fetchId?: string;
1353
+ fetchUrls?: string[];
1354
+ phase?: string;
1355
+ progress?: number;
1356
+ currentQuery?: string;
1357
+ curated?: boolean;
1358
+ curatedFrom?: number;
1359
+ curatedQueries?: QueryDetail[];
1360
+ cancelled?: boolean;
1361
+ cancelReason?: string;
1362
+ summary?: {
1363
+ text: string;
1364
+ workflow: CuratorWorkflow;
1365
+ model: string | null;
1366
+ durationMs: number;
1367
+ tokenEstimate: number;
1368
+ fallbackUsed: boolean;
1369
+ fallbackReason?: string;
1370
+ edited?: boolean;
1371
+ };
1372
+ };
1373
+
1374
+ if (isPartial) {
1375
+ if (details?.phase === "curating") {
1376
+ return new Text(theme.fg("accent", "waiting for summary approval..."), 0, 0);
1377
+ }
1378
+ if (details?.phase === "searching") {
1379
+ const progress = details?.progress ?? 0;
1380
+ const bar = "\u2588".repeat(Math.floor(progress * 10)) + "\u2591".repeat(10 - Math.floor(progress * 10));
1381
+ const query = details?.currentQuery || "";
1382
+ const display = query.length > 40 ? query.slice(0, 37) + "..." : query;
1383
+ return new Text(theme.fg("accent", `[${bar}] ${display}`), 0, 0);
1384
+ }
1385
+ const progress = details?.progress ?? 0;
1386
+ const bar = "\u2588".repeat(Math.floor(progress * 10)) + "\u2591".repeat(10 - Math.floor(progress * 10));
1387
+ return new Text(theme.fg("accent", `[${bar}] ${details?.phase || "searching"}`), 0, 0);
1388
+ }
1389
+
1390
+ if (details?.error) {
1391
+ return new Text(theme.fg("error", `Error: ${details.error}`), 0, 0);
1392
+ }
1393
+
1394
+ let statusLine: string;
1395
+ const queryInfo = details?.queryCount === 1 ? "" : `${details?.successfulQueries}/${details?.queryCount} queries, `;
1396
+ statusLine = theme.fg("success", `${queryInfo}${details?.totalResults ?? 0} sources`);
1397
+ if (details?.curated && details?.curatedFrom) {
1398
+ statusLine += theme.fg("muted", ` (${details.queryCount}/${details.curatedFrom} queries curated)`);
1399
+ }
1400
+ if (details?.fetchId && details?.fetchUrls) {
1401
+ statusLine += theme.fg("muted", ` (fetching ${details.fetchUrls.length} URLs)`);
1402
+ } else if (details?.fetchId) {
1403
+ statusLine += theme.fg("muted", " (content ready)");
1404
+ }
1405
+
1406
+ // Build expanded lines first so collapsed view can reference total count
1407
+ const lines = [statusLine];
1408
+ if (details?.summary?.text) {
1409
+ lines.push("");
1410
+ lines.push(theme.fg("accent", `── Summary (${details.summary.workflow}) ` + "─".repeat(32)));
1411
+ lines.push("");
1412
+ for (const line of details.summary.text.split("\n")) {
1413
+ lines.push(` ${line}`);
1414
+ }
1415
+ lines.push("");
1416
+ const metaParts = [
1417
+ details.summary.model ? `model=${details.summary.model}` : "model=deterministic",
1418
+ `duration=${details.summary.durationMs}ms`,
1419
+ `tokens~${details.summary.tokenEstimate}`,
1420
+ details.summary.fallbackUsed ? "fallback=true" : "fallback=false",
1421
+ details.summary.edited ? "edited=true" : "edited=false",
1422
+ ];
1423
+ if (details.summary.fallbackReason) {
1424
+ metaParts.push(`reason=${details.summary.fallbackReason}`);
1425
+ }
1426
+ lines.push(theme.fg("dim", " " + metaParts.join(" · ")));
1427
+ }
1428
+
1429
+ const queryDetails = details?.curatedQueries;
1430
+ if (queryDetails?.length) {
1431
+ const kept = queryDetails.length;
1432
+ const from = details?.curatedFrom ?? kept;
1433
+ lines.push("");
1434
+ lines.push(theme.fg("accent", `\u2500\u2500 Curated Results (${kept} of ${from} queries kept) ` + "\u2500".repeat(24)));
1435
+
1436
+ for (const cq of queryDetails) {
1437
+ lines.push("");
1438
+ const dq = cq.query.length > 65 ? cq.query.slice(0, 62) + "..." : cq.query;
1439
+ const providerLabel = cq.provider ? ` (${cq.provider})` : "";
1440
+ lines.push(theme.fg("accent", ` "${dq}"${providerLabel}`));
1441
+
1442
+ if (cq.error) {
1443
+ lines.push(theme.fg("error", ` ${cq.error}`));
1444
+ } else if (cq.answer) {
1445
+ lines.push("");
1446
+ for (const line of cq.answer.split("\n")) {
1447
+ lines.push(` ${line}`);
1448
+ }
1449
+ }
1450
+
1451
+ if (cq.sources.length > 0) {
1452
+ lines.push("");
1453
+ for (const s of cq.sources) {
1454
+ const domain = s.url.replace(/^https?:\/\//, "").replace(/\/.*$/, "");
1455
+ const title = s.title.length > 50 ? s.title.slice(0, 47) + "..." : s.title;
1456
+ lines.push(theme.fg("muted", ` \u25b8 ${title}`) + theme.fg("dim", ` \u00b7 ${domain}`));
1457
+ }
1458
+ }
1459
+ }
1460
+ lines.push("");
1461
+ } else {
1462
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1463
+ const preview = textContent.length > 500 ? textContent.slice(0, 500) + "..." : textContent;
1464
+ for (const line of preview.split("\n")) {
1465
+ lines.push(theme.fg("dim", line));
1466
+ }
1467
+ }
1468
+
1469
+ if (details?.fetchUrls && details.fetchUrls.length > 0) {
1470
+ if (details.curated) {
1471
+ lines.push(theme.fg("muted", `Fetching ${details.fetchUrls.length} URLs in background`));
1472
+ } else {
1473
+ lines.push(theme.fg("muted", "Fetching:"));
1474
+ for (const u of details.fetchUrls.slice(0, 5)) {
1475
+ const display = u.length > 60 ? u.slice(0, 57) + "..." : u;
1476
+ lines.push(theme.fg("dim", " " + display));
1477
+ }
1478
+ if (details.fetchUrls.length > 5) {
1479
+ lines.push(theme.fg("dim", ` ... and ${details.fetchUrls.length - 5} more`));
1480
+ }
1481
+ }
1482
+ }
1483
+
1484
+ const totalLines = lines.length;
1485
+
1486
+ if (!expanded) {
1487
+ const box = new Box(1, 0, (t) => theme.bg("toolSuccessBg", t));
1488
+ box.addChild(new Text(statusLine, 0, 0));
1489
+
1490
+ let collapsedLines = 1; // statusLine
1491
+ const summaryPreview = details?.summary?.text?.trim() || "";
1492
+ if (summaryPreview) {
1493
+ const preview = summaryPreview.length > 120 ? summaryPreview.slice(0, 117) + "..." : summaryPreview;
1494
+ box.addChild(new Text(theme.fg("dim", preview), 0, 0));
1495
+ collapsedLines++;
1496
+ } else if (details?.curatedQueries?.length) {
1497
+ for (const cq of details.curatedQueries.slice(0, 3)) {
1498
+ const dq = cq.query.length > 55 ? cq.query.slice(0, 52) + "..." : cq.query;
1499
+ const srcCount = cq.sources?.length ?? 0;
1500
+ const suffix = cq.error ? theme.fg("error", " (error)") : theme.fg("dim", ` · ${srcCount} sources`);
1501
+ box.addChild(new Text(theme.fg("accent", ` "${dq}"`) + suffix, 0, 0));
1502
+ collapsedLines++;
1503
+ }
1504
+ if (details.curatedQueries.length > 3) {
1505
+ box.addChild(new Text(theme.fg("dim", ` ... and ${details.curatedQueries.length - 3} more`), 0, 0));
1506
+ collapsedLines++;
1507
+ }
1508
+ } else {
1509
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1510
+ const firstContentLine = textContent.split("\n").find(l => {
1511
+ const t = l.trim();
1512
+ return t && !t.startsWith("[") && !t.startsWith("#") && !t.startsWith("---");
1513
+ });
1514
+ const fallbackLine = (firstContentLine?.trim() || "").replace(/\*\*/g, "");
1515
+ if (fallbackLine) {
1516
+ const preview = fallbackLine.length > 120 ? fallbackLine.slice(0, 117) + "..." : fallbackLine;
1517
+ box.addChild(new Text(theme.fg("dim", preview), 0, 0));
1518
+ collapsedLines++;
1519
+ }
1520
+ }
1521
+ const moreLines = Math.max(0, totalLines - collapsedLines);
1522
+ if (moreLines > 0) {
1523
+ box.addChild(new Text(theme.fg("muted", `\n... (${moreLines} more lines, ${totalLines} total, ctrl+o to expand)`), 0, 0));
1524
+ }
1525
+ return box;
1526
+ }
1527
+
1528
+ return new Text(lines.join("\n"), 0, 0);
1529
+ },
1530
+ });
1531
+
1532
+ pi.registerTool({
1533
+ name: "code_search",
1534
+ label: "Code Search",
1535
+ description: "Search for code examples, documentation, and API references. Returns relevant code snippets and docs from GitHub, Stack Overflow, and official documentation. Use for any programming question — API usage, library examples, debugging help.",
1536
+ promptSnippet:
1537
+ "Use for programming/API/library questions to retrieve concrete examples and docs before implementing or debugging code.",
1538
+ parameters: Type.Object({
1539
+ query: Type.String({ description: "Programming question, API, library, or debugging topic to search for" }),
1540
+ maxTokens: Type.Optional(Type.Integer({
1541
+ minimum: 1000,
1542
+ maximum: 50000,
1543
+ description: "Maximum tokens of code/documentation context to return (default: 5000)",
1544
+ })),
1545
+ }),
1546
+
1547
+ async execute(toolCallId, params, signal) {
1548
+ return executeCodeSearch(toolCallId, params, signal);
1549
+ },
1550
+
1551
+ renderCall(args, theme) {
1552
+ const { query } = args as { query?: string };
1553
+ const display = !query
1554
+ ? "(no query)"
1555
+ : query.length > 70 ? query.slice(0, 67) + "..." : query;
1556
+ return new Text(theme.fg("toolTitle", theme.bold("code_search ")) + theme.fg("accent", display), 0, 0);
1557
+ },
1558
+
1559
+ renderResult(result, { expanded }, theme) {
1560
+ const details = result.details as { query?: string; maxTokens?: number; error?: string };
1561
+ if (details?.error) {
1562
+ return new Text(theme.fg("error", `Error: ${details.error}`), 0, 0);
1563
+ }
1564
+
1565
+ const summary = theme.fg("success", "code context returned") +
1566
+ theme.fg("muted", ` (${details?.maxTokens ?? 5000} tokens max)`);
1567
+ if (!expanded) return new Text(summary, 0, 0);
1568
+
1569
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1570
+ const preview = textContent.length > 500 ? textContent.slice(0, 500) + "..." : textContent;
1571
+ return new Text(summary + "\n" + theme.fg("dim", preview), 0, 0);
1572
+ },
1573
+ });
1574
+
1575
+ pi.registerTool({
1576
+ name: "fetch_content",
1577
+ label: "Fetch Content",
1578
+ description: "Fetch URL(s) and extract readable content as markdown. Supports YouTube video transcripts (with thumbnail), GitHub repository contents, and local video files (with frame thumbnail). Video frames can be extracted via timestamp/range or sampled across the entire video with frames alone. Falls back to Gemini for pages that block bots or fail Readability extraction. For YouTube and video files: ALWAYS pass the user's specific question via the prompt parameter — this directs the AI to focus on that aspect of the video, producing much better results than a generic extraction. Content is always stored and can be retrieved with get_search_content.",
1579
+ promptSnippet:
1580
+ "Use to extract readable content from URL(s), YouTube, GitHub repos, or local videos. For video questions, pass the user's exact question in prompt.",
1581
+ parameters: Type.Object({
1582
+ url: Type.Optional(Type.String({ description: "Single URL to fetch" })),
1583
+ urls: Type.Optional(Type.Array(Type.String(), { description: "Multiple URLs (parallel)" })),
1584
+ forceClone: Type.Optional(Type.Boolean({
1585
+ description: "Force cloning large GitHub repositories that exceed the size threshold",
1586
+ })),
1587
+ prompt: Type.Optional(Type.String({
1588
+ description: "Question or instruction for video analysis (YouTube and video files). Pass the user's specific question here — e.g. 'describe the book shown at the advice for beginners section'. Without this, a generic transcript extraction is used which may miss what the user is asking about.",
1589
+ })),
1590
+ timestamp: Type.Optional(Type.String({
1591
+ description: "Extract video frame(s) at a timestamp or time range. Single: '1:23:45', '23:45', or '85' (seconds). Range: '23:41-25:00' extracts evenly-spaced frames across that span (default 6). Use frames with ranges to control density; single+frames uses a fixed 5s interval. YouTube requires yt-dlp + ffmpeg; local videos require ffmpeg. Use a range when you know the approximate area but not the exact moment — you'll get a contact sheet to visually identify the right frame.",
1592
+ })),
1593
+ frames: Type.Optional(Type.Integer({
1594
+ minimum: 1,
1595
+ maximum: 12,
1596
+ description: "Number of frames to extract. Use with timestamp range for custom density, with single timestamp to get N frames at 5s intervals, or alone to sample across the entire video. Requires yt-dlp + ffmpeg for YouTube, ffmpeg for local video.",
1597
+ })),
1598
+ model: Type.Optional(Type.String({
1599
+ description: "Override the Gemini model for video/YouTube analysis (e.g. 'gemini-2.5-flash', 'gemini-3-flash-preview'). Defaults to config or gemini-3-flash-preview.",
1600
+ })),
1601
+ }),
1602
+
1603
+ async execute(_toolCallId, params, signal, onUpdate) {
1604
+ const urlList = params.urls ?? (params.url ? [params.url] : []);
1605
+ if (urlList.length === 0) {
1606
+ return {
1607
+ content: [{ type: "text", text: "Error: No URL provided." }],
1608
+ details: { error: "No URL provided" },
1609
+ };
1610
+ }
1611
+
1612
+ onUpdate?.({
1613
+ content: [{ type: "text", text: `Fetching ${urlList.length} URL(s)...` }],
1614
+ details: { phase: "fetch", progress: 0 },
1615
+ });
1616
+
1617
+ const fetchResults = await fetchAllContent(urlList, signal, {
1618
+ forceClone: params.forceClone,
1619
+ prompt: params.prompt,
1620
+ timestamp: params.timestamp,
1621
+ frames: params.frames,
1622
+ model: params.model,
1623
+ });
1624
+ const successful = fetchResults.filter((r) => !r.error).length;
1625
+ const totalChars = fetchResults.reduce((sum, r) => sum + r.content.length, 0);
1626
+
1627
+ // ALWAYS store results (even for single URL)
1628
+ const responseId = generateId();
1629
+ const data: StoredSearchData = {
1630
+ id: responseId,
1631
+ type: "fetch",
1632
+ timestamp: Date.now(),
1633
+ urls: stripThumbnails(fetchResults),
1634
+ };
1635
+ storeResult(responseId, data);
1636
+ pi.appendEntry("web-search-results", data);
1637
+
1638
+ // Single URL: return content directly (possibly truncated) with responseId
1639
+ if (urlList.length === 1) {
1640
+ const result = fetchResults[0];
1641
+ if (result.error) {
1642
+ return {
1643
+ content: [{ type: "text", text: `Error: ${result.error}` }],
1644
+ details: { urls: urlList, urlCount: 1, successful: 0, error: result.error, responseId, prompt: params.prompt, timestamp: params.timestamp, frames: params.frames },
1645
+ };
1646
+ }
1647
+
1648
+ const fullLength = result.content.length;
1649
+ const truncated = fullLength > MAX_INLINE_CONTENT;
1650
+ let output = truncated
1651
+ ? result.content.slice(0, MAX_INLINE_CONTENT) + "\n\n[Content truncated...]"
1652
+ : result.content;
1653
+
1654
+ if (truncated) {
1655
+ output += `\n\n---\nShowing ${MAX_INLINE_CONTENT} of ${fullLength} chars. ` +
1656
+ `Use get_search_content({ responseId: "${responseId}", urlIndex: 0 }) for full content.`;
1657
+ }
1658
+
1659
+ const content: Array<{ type: string; text?: string; data?: string; mimeType?: string }> = [];
1660
+ if (result.frames?.length) {
1661
+ for (const frame of result.frames) {
1662
+ content.push({ type: "image", data: frame.data, mimeType: frame.mimeType });
1663
+ content.push({ type: "text", text: `Frame at ${frame.timestamp}` });
1664
+ }
1665
+ } else if (result.thumbnail) {
1666
+ content.push({ type: "image", data: result.thumbnail.data, mimeType: result.thumbnail.mimeType });
1667
+ }
1668
+ content.push({ type: "text", text: output });
1669
+
1670
+ const imageCount = (result.frames?.length ?? 0) + (result.thumbnail ? 1 : 0);
1671
+ return {
1672
+ content,
1673
+ details: {
1674
+ urls: urlList,
1675
+ urlCount: 1,
1676
+ successful: 1,
1677
+ totalChars: fullLength,
1678
+ title: result.title,
1679
+ responseId,
1680
+ truncated,
1681
+ hasImage: imageCount > 0,
1682
+ imageCount,
1683
+ prompt: params.prompt,
1684
+ timestamp: params.timestamp,
1685
+ frames: params.frames,
1686
+ duration: result.duration,
1687
+ },
1688
+ };
1689
+ }
1690
+
1691
+ // Multi-URL: existing behavior (summary + responseId)
1692
+ let output = "## Fetched URLs\n\n";
1693
+ for (const { url, title, content, error } of fetchResults) {
1694
+ if (error) {
1695
+ output += `- ${url}: Error - ${error}\n`;
1696
+ } else {
1697
+ output += `- ${title || url} (${content.length} chars)\n`;
1698
+ }
1699
+ }
1700
+ output += `\n---\nUse get_search_content({ responseId: "${responseId}", urlIndex: 0 }) to retrieve full content.`;
1701
+
1702
+ return {
1703
+ content: [{ type: "text", text: output }],
1704
+ details: { urls: urlList, urlCount: urlList.length, successful, totalChars, responseId },
1705
+ };
1706
+ },
1707
+
1708
+ renderCall(args, theme) {
1709
+ const { url, urls, prompt, timestamp, frames, model } = args as { url?: string; urls?: string[]; prompt?: string; timestamp?: string; frames?: number; model?: string };
1710
+ const urlList = urls ?? (url ? [url] : []);
1711
+ if (urlList.length === 0) {
1712
+ return new Text(theme.fg("toolTitle", theme.bold("fetch ")) + theme.fg("error", "(no URL)"), 0, 0);
1713
+ }
1714
+ const lines: string[] = [];
1715
+ if (urlList.length === 1) {
1716
+ const display = urlList[0].length > 60 ? urlList[0].slice(0, 57) + "..." : urlList[0];
1717
+ lines.push(theme.fg("toolTitle", theme.bold("fetch ")) + theme.fg("accent", display));
1718
+ } else {
1719
+ lines.push(theme.fg("toolTitle", theme.bold("fetch ")) + theme.fg("accent", `${urlList.length} URLs`));
1720
+ for (const u of urlList.slice(0, 5)) {
1721
+ const display = u.length > 60 ? u.slice(0, 57) + "..." : u;
1722
+ lines.push(theme.fg("muted", " " + display));
1723
+ }
1724
+ if (urlList.length > 5) {
1725
+ lines.push(theme.fg("muted", ` ... and ${urlList.length - 5} more`));
1726
+ }
1727
+ }
1728
+ if (timestamp) {
1729
+ lines.push(theme.fg("dim", " timestamp: ") + theme.fg("warning", timestamp));
1730
+ }
1731
+ if (typeof frames === "number") {
1732
+ lines.push(theme.fg("dim", " frames: ") + theme.fg("warning", String(frames)));
1733
+ }
1734
+ if (prompt) {
1735
+ const display = prompt.length > 250 ? prompt.slice(0, 247) + "..." : prompt;
1736
+ lines.push(theme.fg("dim", " prompt: ") + theme.fg("muted", `"${display}"`));
1737
+ }
1738
+ if (model) {
1739
+ lines.push(theme.fg("dim", " model: ") + theme.fg("warning", model));
1740
+ }
1741
+ return new Text(lines.join("\n"), 0, 0);
1742
+ },
1743
+
1744
+ renderResult(result, { expanded, isPartial }, theme) {
1745
+ const details = result.details as {
1746
+ urlCount?: number;
1747
+ successful?: number;
1748
+ totalChars?: number;
1749
+ error?: string;
1750
+ title?: string;
1751
+ truncated?: boolean;
1752
+ responseId?: string;
1753
+ phase?: string;
1754
+ progress?: number;
1755
+ hasImage?: boolean;
1756
+ imageCount?: number;
1757
+ prompt?: string;
1758
+ timestamp?: string;
1759
+ frames?: number;
1760
+ duration?: number;
1761
+ };
1762
+
1763
+ if (isPartial) {
1764
+ const progress = details?.progress ?? 0;
1765
+ const bar = "\u2588".repeat(Math.floor(progress * 10)) + "\u2591".repeat(10 - Math.floor(progress * 10));
1766
+ return new Text(theme.fg("accent", `[${bar}] ${details?.phase || "fetching"}`), 0, 0);
1767
+ }
1768
+
1769
+ if (details?.error) {
1770
+ return new Text(theme.fg("error", `Error: ${details.error}`), 0, 0);
1771
+ }
1772
+
1773
+ if (details?.urlCount === 1) {
1774
+ const title = details?.title || "Untitled";
1775
+ const imgCount = details?.imageCount ?? (details?.hasImage ? 1 : 0);
1776
+ const imageBadge = imgCount > 1
1777
+ ? theme.fg("accent", ` [${imgCount} images]`)
1778
+ : imgCount === 1
1779
+ ? theme.fg("accent", " [image]")
1780
+ : "";
1781
+ let statusLine = theme.fg("success", title) + theme.fg("muted", ` (${details?.totalChars ?? 0} chars)`) + imageBadge;
1782
+ if (details?.truncated) {
1783
+ statusLine += theme.fg("warning", " [truncated]");
1784
+ }
1785
+ if (typeof details?.duration === "number") {
1786
+ statusLine += theme.fg("muted", ` | ${formatSeconds(Math.floor(details.duration))} total`);
1787
+ }
1788
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1789
+ if (!expanded) {
1790
+ const brief = textContent.length > 200 ? textContent.slice(0, 200) + "..." : textContent;
1791
+ return new Text(statusLine + "\n" + theme.fg("dim", brief), 0, 0);
1792
+ }
1793
+ const lines = [statusLine];
1794
+ if (details?.prompt) {
1795
+ const display = details.prompt.length > 250 ? details.prompt.slice(0, 247) + "..." : details.prompt;
1796
+ lines.push(theme.fg("dim", ` prompt: "${display}"`));
1797
+ }
1798
+ if (details?.timestamp) {
1799
+ lines.push(theme.fg("dim", ` timestamp: ${details.timestamp}`));
1800
+ }
1801
+ if (typeof details?.frames === "number") {
1802
+ lines.push(theme.fg("dim", ` frames: ${details.frames}`));
1803
+ }
1804
+ const preview = textContent.length > 500 ? textContent.slice(0, 500) + "..." : textContent;
1805
+ lines.push(theme.fg("dim", preview));
1806
+ return new Text(lines.join("\n"), 0, 0);
1807
+ }
1808
+
1809
+ const countColor = (details?.successful ?? 0) > 0 ? "success" : "error";
1810
+ const statusLine = theme.fg(countColor, `${details?.successful}/${details?.urlCount} URLs`) + theme.fg("muted", " (content stored)");
1811
+ if (!expanded) {
1812
+ return new Text(statusLine, 0, 0);
1813
+ }
1814
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1815
+ const preview = textContent.length > 500 ? textContent.slice(0, 500) + "..." : textContent;
1816
+ return new Text(statusLine + "\n" + theme.fg("dim", preview), 0, 0);
1817
+ },
1818
+ });
1819
+
1820
+ pi.registerTool({
1821
+ name: "get_search_content",
1822
+ label: "Get Search Content",
1823
+ description: "Retrieve full content from a previous web_search or fetch_content call.",
1824
+ promptSnippet:
1825
+ "Use after web_search/fetch_content when full stored content is needed via responseId plus query/url selectors.",
1826
+ parameters: Type.Object({
1827
+ responseId: Type.String({ description: "The responseId from web_search or fetch_content" }),
1828
+ query: Type.Optional(Type.String({ description: "Get content for this query (web_search)" })),
1829
+ queryIndex: Type.Optional(Type.Number({ description: "Get content for query at index" })),
1830
+ url: Type.Optional(Type.String({ description: "Get content for this URL" })),
1831
+ urlIndex: Type.Optional(Type.Number({ description: "Get content for URL at index" })),
1832
+ }),
1833
+
1834
+ async execute(_toolCallId, params) {
1835
+ const data = getResult(params.responseId);
1836
+ if (!data) {
1837
+ return {
1838
+ content: [{ type: "text", text: `Error: No stored results for "${params.responseId}"` }],
1839
+ details: { error: "Not found", responseId: params.responseId },
1840
+ };
1841
+ }
1842
+
1843
+ if (data.type === "search" && data.queries) {
1844
+ let queryData: QueryResultData | undefined;
1845
+
1846
+ if (params.query !== undefined) {
1847
+ queryData = data.queries.find((q) => q.query === params.query);
1848
+ if (!queryData) {
1849
+ const available = data.queries.map((q) => `"${q.query}"`).join(", ");
1850
+ return {
1851
+ content: [{ type: "text", text: `Query "${params.query}" not found. Available: ${available}` }],
1852
+ details: { error: "Query not found" },
1853
+ };
1854
+ }
1855
+ } else if (params.queryIndex !== undefined) {
1856
+ queryData = data.queries[params.queryIndex];
1857
+ if (!queryData) {
1858
+ return {
1859
+ content: [{ type: "text", text: `Index ${params.queryIndex} out of range (0-${data.queries.length - 1})` }],
1860
+ details: { error: "Index out of range" },
1861
+ };
1862
+ }
1863
+ } else {
1864
+ const available = data.queries.map((q, i) => `${i}: "${q.query}"`).join(", ");
1865
+ return {
1866
+ content: [{ type: "text", text: `Specify query or queryIndex. Available: ${available}` }],
1867
+ details: { error: "No query specified" },
1868
+ };
1869
+ }
1870
+
1871
+ if (queryData.error) {
1872
+ return {
1873
+ content: [{ type: "text", text: `Error for "${queryData.query}": ${queryData.error}` }],
1874
+ details: { error: queryData.error, query: queryData.query },
1875
+ };
1876
+ }
1877
+
1878
+ return {
1879
+ content: [{ type: "text", text: formatFullResults(queryData) }],
1880
+ details: { query: queryData.query, resultCount: queryData.results.length },
1881
+ };
1882
+ }
1883
+
1884
+ if (data.type === "fetch" && data.urls) {
1885
+ let urlData: ExtractedContent | undefined;
1886
+
1887
+ if (params.url !== undefined) {
1888
+ urlData = data.urls.find((u) => u.url === params.url);
1889
+ if (!urlData) {
1890
+ const available = data.urls.map((u) => u.url).join("\n ");
1891
+ return {
1892
+ content: [{ type: "text", text: `URL not found. Available:\n ${available}` }],
1893
+ details: { error: "URL not found" },
1894
+ };
1895
+ }
1896
+ } else if (params.urlIndex !== undefined) {
1897
+ urlData = data.urls[params.urlIndex];
1898
+ if (!urlData) {
1899
+ return {
1900
+ content: [{ type: "text", text: `Index ${params.urlIndex} out of range (0-${data.urls.length - 1})` }],
1901
+ details: { error: "Index out of range" },
1902
+ };
1903
+ }
1904
+ } else {
1905
+ const available = data.urls.map((u, i) => `${i}: ${u.url}`).join("\n ");
1906
+ return {
1907
+ content: [{ type: "text", text: `Specify url or urlIndex. Available:\n ${available}` }],
1908
+ details: { error: "No URL specified" },
1909
+ };
1910
+ }
1911
+
1912
+ if (urlData.error) {
1913
+ return {
1914
+ content: [{ type: "text", text: `Error for ${urlData.url}: ${urlData.error}` }],
1915
+ details: { error: urlData.error, url: urlData.url },
1916
+ };
1917
+ }
1918
+
1919
+ return {
1920
+ content: [{ type: "text", text: `# ${urlData.title}\n\n${urlData.content}` }],
1921
+ details: { url: urlData.url, title: urlData.title, contentLength: urlData.content.length },
1922
+ };
1923
+ }
1924
+
1925
+ return {
1926
+ content: [{ type: "text", text: "Invalid stored data format" }],
1927
+ details: { error: "Invalid data" },
1928
+ };
1929
+ },
1930
+
1931
+ renderCall(args, theme) {
1932
+ const { responseId, query, queryIndex, url, urlIndex } = args as {
1933
+ responseId: string;
1934
+ query?: string;
1935
+ queryIndex?: number;
1936
+ url?: string;
1937
+ urlIndex?: number;
1938
+ };
1939
+ let target = "";
1940
+ if (query) target = `query="${query}"`;
1941
+ else if (queryIndex !== undefined) target = `queryIndex=${queryIndex}`;
1942
+ else if (url) target = url.length > 30 ? url.slice(0, 27) + "..." : url;
1943
+ else if (urlIndex !== undefined) target = `urlIndex=${urlIndex}`;
1944
+ return new Text(theme.fg("toolTitle", theme.bold("get_content ")) + theme.fg("accent", target || responseId.slice(0, 8)), 0, 0);
1945
+ },
1946
+
1947
+ renderResult(result, { expanded }, theme) {
1948
+ const details = result.details as {
1949
+ error?: string;
1950
+ query?: string;
1951
+ url?: string;
1952
+ title?: string;
1953
+ resultCount?: number;
1954
+ contentLength?: number;
1955
+ };
1956
+
1957
+ if (details?.error) {
1958
+ return new Text(theme.fg("error", `Error: ${details.error}`), 0, 0);
1959
+ }
1960
+
1961
+ let statusLine: string;
1962
+ if (details?.query) {
1963
+ statusLine = theme.fg("success", `"${details.query}"`) + theme.fg("muted", ` (${details.resultCount} results)`);
1964
+ } else {
1965
+ statusLine = theme.fg("success", details?.title || "Content") + theme.fg("muted", ` (${details?.contentLength ?? 0} chars)`);
1966
+ }
1967
+
1968
+ if (!expanded) {
1969
+ return new Text(statusLine, 0, 0);
1970
+ }
1971
+
1972
+ const textContent = result.content.find((c) => c.type === "text")?.text || "";
1973
+ const preview = textContent.length > 500 ? textContent.slice(0, 500) + "..." : textContent;
1974
+ return new Text(statusLine + "\n" + theme.fg("dim", preview), 0, 0);
1975
+ },
1976
+ });
1977
+
1978
+ pi.registerCommand("websearch", {
1979
+ description: "Open web search curator",
1980
+ handler: async (args, ctx) => {
1981
+ closeCurator();
1982
+ const sessionToken = randomUUID();
1983
+
1984
+ const raw = args.trim();
1985
+ const queries = raw.length > 0
1986
+ ? normalizeQueryList(raw.split(","))
1987
+ : [];
1988
+
1989
+ let bootstrap: CuratorBootstrap;
1990
+ try {
1991
+ bootstrap = await loadCuratorBootstrap(undefined);
1992
+ } catch (err) {
1993
+ const message = err instanceof Error ? err.message : String(err);
1994
+ ctx.ui.notify(`Failed to load web search config: ${message}`, "error");
1995
+ return;
1996
+ }
1997
+ const availableProviders = bootstrap.availableProviders;
1998
+ const initialProvider = bootstrap.defaultProvider;
1999
+ const curatorTimeoutSeconds = bootstrap.timeoutSeconds;
2000
+ let currentProvider = initialProvider;
2001
+ const summaryContext: SummaryGenerationContext = {
2002
+ model: ctx.model,
2003
+ modelRegistry: ctx.modelRegistry,
2004
+ };
2005
+ const summaryModelChoices = await loadSummaryModelChoices(summaryContext);
2006
+
2007
+ ctx.ui.notify("Opening web search curator...", "info");
2008
+
2009
+ const collected = new Map<number, QueryResultData>();
2010
+ const searchAbort = new AbortController();
2011
+ let aborted = false;
2012
+ let commandHandle: CuratorServerHandle | null = null;
2013
+
2014
+ function sendFollowUpFromReturn(payload: ReturnType<typeof buildSearchReturn>) {
2015
+ pi.sendMessage({
2016
+ customType: "web-search-results",
2017
+ content: payload.content,
2018
+ display: "tool",
2019
+ details: payload.details,
2020
+ }, { triggerTurn: true, deliverAs: "followUp" });
2021
+ }
2022
+
2023
+ try {
2024
+ const handle = await startCuratorServer(
2025
+ {
2026
+ queries,
2027
+ sessionToken,
2028
+ timeout: curatorTimeoutSeconds,
2029
+ availableProviders,
2030
+ defaultProvider: initialProvider,
2031
+ summaryModels: summaryModelChoices.summaryModels,
2032
+ defaultSummaryModel: summaryModelChoices.defaultSummaryModel,
2033
+ },
2034
+ {
2035
+ async onSummarize(selectedQueryIndices, summarizeSignal, model, feedback) {
2036
+ if (commandHandle && activeCurator !== commandHandle) {
2037
+ throw new Error("Curator session is no longer active.");
2038
+ }
2039
+ return generateSummaryForSelectedIndices(
2040
+ selectedQueryIndices,
2041
+ collected,
2042
+ summaryContext,
2043
+ summarizeSignal,
2044
+ model,
2045
+ feedback,
2046
+ );
2047
+ },
2048
+ onSubmit(payload) {
2049
+ if (commandHandle && activeCurator !== commandHandle) return;
2050
+ aborted = true;
2051
+ searchAbort.abort();
2052
+ const filtered = payload.selectedQueryIndices.length > 0
2053
+ ? filterByQueryIndices(payload.selectedQueryIndices, collected)
2054
+ : collectAllResultsAndUrls(collected);
2055
+ const base: SearchReturnOptions = {
2056
+ queryList: filtered.results.map(r => r.query),
2057
+ results: filtered.results,
2058
+ urls: filtered.urls,
2059
+ includeContent: false,
2060
+ curated: true,
2061
+ curatedFrom: collected.size,
2062
+ };
2063
+ if (!payload.rawResults) {
2064
+ const resolvedSummary = resolveSummaryForSubmit(payload, collected);
2065
+ base.workflow = "summary-review";
2066
+ base.approvedSummary = resolvedSummary.approvedSummary;
2067
+ base.summaryMeta = resolvedSummary.summaryMeta;
2068
+ }
2069
+ sendFollowUpFromReturn(buildSearchReturn(base));
2070
+ closeCurator();
2071
+ },
2072
+ onCancel(reason) {
2073
+ if (commandHandle && activeCurator !== commandHandle) return;
2074
+ aborted = true;
2075
+ searchAbort.abort();
2076
+ if (reason === "timeout") {
2077
+ const all = collectAllResultsAndUrls(collected);
2078
+ const resolvedSummary = resolveSummaryForSubmit({ selectedQueryIndices: [], summary: undefined, summaryMeta: undefined }, collected);
2079
+ sendFollowUpFromReturn(buildSearchReturn({
2080
+ queryList: all.results.map(r => r.query),
2081
+ results: all.results,
2082
+ urls: all.urls,
2083
+ includeContent: false,
2084
+ curated: true,
2085
+ curatedFrom: collected.size,
2086
+ workflow: "summary-review",
2087
+ approvedSummary: resolvedSummary.approvedSummary,
2088
+ summaryMeta: resolvedSummary.summaryMeta,
2089
+ }));
2090
+ }
2091
+ closeCurator();
2092
+ },
2093
+ onProviderChange(provider) {
2094
+ if (commandHandle && activeCurator !== commandHandle) return;
2095
+ const normalized = normalizeProviderInput(provider);
2096
+ if (!normalized || normalized === "auto") return;
2097
+ currentProvider = normalized;
2098
+ try {
2099
+ saveConfig({ provider: normalized });
2100
+ } catch (err) {
2101
+ const message = err instanceof Error ? err.message : String(err);
2102
+ console.error(`Failed to persist default provider: ${message}`);
2103
+ }
2104
+ },
2105
+ async onAddSearch(query, queryIndex, provider) {
2106
+ if (commandHandle && activeCurator !== commandHandle) {
2107
+ throw new Error("Curator session is no longer active.");
2108
+ }
2109
+ const normalizedProvider = normalizeProviderInput(provider);
2110
+ const requestedProvider = !normalizedProvider || normalizedProvider === "auto"
2111
+ ? currentProvider
2112
+ : normalizedProvider;
2113
+ try {
2114
+ const { answer, results, provider: actualProvider } = await search(query, {
2115
+ provider: requestedProvider,
2116
+ signal: searchAbort.signal,
2117
+ });
2118
+ if (commandHandle && activeCurator !== commandHandle) {
2119
+ throw new Error("Curator session is no longer active.");
2120
+ }
2121
+ collected.set(queryIndex, { query, answer, results, error: null, provider: actualProvider });
2122
+ return {
2123
+ answer,
2124
+ results: results.map(r => ({ title: r.title, url: r.url, domain: extractDomain(r.url) })),
2125
+ provider: actualProvider,
2126
+ };
2127
+ } catch (err) {
2128
+ const message = err instanceof Error ? err.message : String(err);
2129
+ if (!commandHandle || activeCurator === commandHandle) {
2130
+ collected.set(queryIndex, { query, answer: "", results: [], error: message, provider: requestedProvider });
2131
+ }
2132
+ throw err;
2133
+ }
2134
+ },
2135
+ async onRewriteQuery(query, rewriteSignal) {
2136
+ if (commandHandle && activeCurator !== commandHandle) {
2137
+ throw new Error("Curator session is no longer active.");
2138
+ }
2139
+ return rewriteSearchQuery(query, summaryContext, rewriteSignal);
2140
+ },
2141
+ },
2142
+ );
2143
+
2144
+ commandHandle = handle;
2145
+ activeCurator = handle;
2146
+ const open = platform() === "darwin" ? await getGlimpseOpen() : null;
2147
+ if (open) {
2148
+ try {
2149
+ const win = openInGlimpse(open, handle.url, "Search Curator");
2150
+ glimpseWin = win;
2151
+ win.on("closed", () => {
2152
+ if (glimpseWin === win) {
2153
+ glimpseWin = null;
2154
+ closeCurator();
2155
+ }
2156
+ });
2157
+ } catch (err) {
2158
+ const message = err instanceof Error ? err.message : String(err);
2159
+ console.error(`Failed to open Glimpse curator window: ${message}`);
2160
+ glimpseWin = null;
2161
+ await openInBrowser(pi, handle.url);
2162
+ }
2163
+ } else {
2164
+ await openInBrowser(pi, handle.url);
2165
+ }
2166
+
2167
+ if (queries.length > 0) {
2168
+ (async () => {
2169
+ for (let qi = 0; qi < queries.length; qi++) {
2170
+ if (aborted || activeCurator !== handle) break;
2171
+ const requestedProvider = currentProvider;
2172
+ try {
2173
+ const { answer, results, provider } = await search(queries[qi], {
2174
+ provider: requestedProvider,
2175
+ signal: searchAbort.signal,
2176
+ });
2177
+ if (aborted || activeCurator !== handle) break;
2178
+ handle.pushResult(qi, {
2179
+ answer,
2180
+ results: results.map(r => ({ title: r.title, url: r.url, domain: extractDomain(r.url) })),
2181
+ provider,
2182
+ });
2183
+ collected.set(qi, { query: queries[qi], answer, results, error: null, provider });
2184
+ } catch (err) {
2185
+ if (aborted || activeCurator !== handle) break;
2186
+ const message = err instanceof Error ? err.message : String(err);
2187
+ handle.pushError(qi, message, requestedProvider);
2188
+ collected.set(qi, { query: queries[qi], answer: "", results: [], error: message, provider: requestedProvider });
2189
+ }
2190
+ }
2191
+ if (!aborted && activeCurator === handle) handle.searchesDone();
2192
+ })();
2193
+ } else {
2194
+ if (activeCurator === handle) handle.searchesDone();
2195
+ }
2196
+ } catch (err) {
2197
+ closeCurator();
2198
+ const message = err instanceof Error ? err.message : String(err);
2199
+ ctx.ui.notify(`Failed to open curator: ${message}`, "error");
2200
+ }
2201
+ },
2202
+ });
2203
+
2204
+ pi.registerCommand("curator", {
2205
+ description: "Toggle or configure the search curator workflow",
2206
+ handler: async (args, ctx) => {
2207
+ const arg = args.trim().toLowerCase();
2208
+
2209
+ let newWorkflow: WebSearchWorkflow;
2210
+ if (arg.length === 0) {
2211
+ const current = resolveWorkflow(loadConfigForExtensionInit().workflow, true);
2212
+ newWorkflow = current === "none" ? "summary-review" : "none";
2213
+ } else if (arg === "on") {
2214
+ newWorkflow = "summary-review";
2215
+ } else if (arg === "off") {
2216
+ newWorkflow = "none";
2217
+ } else if (arg === "none" || arg === "summary-review") {
2218
+ newWorkflow = arg;
2219
+ } else {
2220
+ ctx.ui.notify(`Unknown option: ${arg}. Use on, off, or summary-review.`, "error");
2221
+ return;
2222
+ }
2223
+
2224
+ try {
2225
+ saveConfig({ workflow: newWorkflow });
2226
+ } catch (err) {
2227
+ const message = err instanceof Error ? err.message : String(err);
2228
+ ctx.ui.notify(`Failed to save config: ${message}`, "error");
2229
+ return;
2230
+ }
2231
+
2232
+ const label = newWorkflow === "none"
2233
+ ? "Curator disabled — web_search will return raw results"
2234
+ : "Curator enabled — web_search will open curator and auto-generate a summary draft";
2235
+ pi.sendMessage({
2236
+ customType: "curator-config",
2237
+ content: [{ type: "text", text: label }],
2238
+ display: "tool",
2239
+ details: { workflow: newWorkflow },
2240
+ }, { triggerTurn: false, deliverAs: "followUp" });
2241
+ },
2242
+ });
2243
+
2244
+ pi.registerCommand("google-account", {
2245
+ description: "Show the active Google account for Gemini Web",
2246
+ handler: async () => {
2247
+ if (!isBrowserCookieAccessAllowed()) {
2248
+ pi.sendMessage({
2249
+ customType: "google-account",
2250
+ content: [{ type: "text", text: `Gemini Web browser cookie access is disabled. Set allowBrowserCookies: true in ~/${CONFIG_DIR_NAME}/web-search.json to enable it.` }],
2251
+ display: "tool",
2252
+ details: { available: false, cookieAccessAllowed: false },
2253
+ }, { triggerTurn: true, deliverAs: "followUp" });
2254
+ return;
2255
+ }
2256
+
2257
+ const cookies = await isGeminiWebAvailable();
2258
+ if (!cookies) {
2259
+ pi.sendMessage({
2260
+ customType: "google-account",
2261
+ content: [{ type: "text", text: "Gemini Web is unavailable. Sign into gemini.google.com in a supported Chromium-based browser." }],
2262
+ display: "tool",
2263
+ details: { available: false, cookieAccessAllowed: true },
2264
+ }, { triggerTurn: true, deliverAs: "followUp" });
2265
+ return;
2266
+ }
2267
+
2268
+ const email = await getActiveGoogleEmail(cookies);
2269
+ const text = email
2270
+ ? `Active Google account: ${email}`
2271
+ : "Gemini Web is available, but the active Google account could not be determined.";
2272
+
2273
+ pi.sendMessage({
2274
+ customType: "google-account",
2275
+ content: [{ type: "text", text }],
2276
+ display: "tool",
2277
+ details: { available: true, email: email ?? null },
2278
+ }, { triggerTurn: true, deliverAs: "followUp" });
2279
+ },
2280
+ });
2281
+
2282
+ pi.registerCommand("search", {
2283
+ description: "Browse stored web search results",
2284
+ handler: async (_args, ctx) => {
2285
+ const results = getAllResults();
2286
+
2287
+ if (results.length === 0) {
2288
+ ctx.ui.notify("No stored search results", "info");
2289
+ return;
2290
+ }
2291
+
2292
+ const options = results.map((r) => {
2293
+ const age = Math.floor((Date.now() - r.timestamp) / 60000);
2294
+ const ageStr = age < 60 ? `${age}m ago` : `${Math.floor(age / 60)}h ago`;
2295
+ if (r.type === "search" && r.queries) {
2296
+ const query = r.queries[0]?.query || "unknown";
2297
+ return `[${r.id.slice(0, 6)}] "${query}" (${r.queries.length} queries) - ${ageStr}`;
2298
+ }
2299
+ if (r.type === "fetch" && r.urls) {
2300
+ return `[${r.id.slice(0, 6)}] ${r.urls.length} URLs fetched - ${ageStr}`;
2301
+ }
2302
+ return `[${r.id.slice(0, 6)}] ${r.type} - ${ageStr}`;
2303
+ });
2304
+
2305
+ const choice = await ctx.ui.select("Stored Search Results", options);
2306
+ if (!choice) return;
2307
+
2308
+ const match = choice.match(/^\[([a-z0-9]+)\]/);
2309
+ if (!match) return;
2310
+
2311
+ const selected = results.find((r) => r.id.startsWith(match[1]));
2312
+ if (!selected) return;
2313
+
2314
+ const actions = ["View details", "Delete"];
2315
+ const action = await ctx.ui.select(`Result ${selected.id.slice(0, 6)}`, actions);
2316
+
2317
+ if (action === "Delete") {
2318
+ deleteResult(selected.id);
2319
+ ctx.ui.notify(`Deleted ${selected.id.slice(0, 6)}`, "info");
2320
+ } else if (action === "View details") {
2321
+ let info = `ID: ${selected.id}\nType: ${selected.type}\nAge: ${Math.floor((Date.now() - selected.timestamp) / 60000)}m\n\n`;
2322
+ if (selected.type === "search" && selected.queries) {
2323
+ info += "Queries:\n";
2324
+ const queries = selected.queries.slice(0, 10);
2325
+ for (const q of queries) {
2326
+ info += `- "${q.query}" (${q.results.length} results)\n`;
2327
+ }
2328
+ if (selected.queries.length > 10) {
2329
+ info += `... and ${selected.queries.length - 10} more\n`;
2330
+ }
2331
+ }
2332
+ if (selected.type === "fetch" && selected.urls) {
2333
+ info += "URLs:\n";
2334
+ const urls = selected.urls.slice(0, 10);
2335
+ for (const u of urls) {
2336
+ const urlDisplay = u.url.length > 50 ? u.url.slice(0, 47) + "..." : u.url;
2337
+ info += `- ${urlDisplay} (${u.error || `${u.content.length} chars`})\n`;
2338
+ }
2339
+ if (selected.urls.length > 10) {
2340
+ info += `... and ${selected.urls.length - 10} more\n`;
2341
+ }
2342
+ }
2343
+ ctx.ui.notify(info, "info");
2344
+ }
2345
+ },
2346
+ });
2347
+ }