@otto-assistant/bridge 0.4.92

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 (483) hide show
  1. package/bin.js +2 -0
  2. package/dist/agent-model.e2e.test.js +755 -0
  3. package/dist/ai-tool-to-genai.js +233 -0
  4. package/dist/ai-tool-to-genai.test.js +267 -0
  5. package/dist/ai-tool.js +6 -0
  6. package/dist/anthropic-auth-plugin.js +728 -0
  7. package/dist/anthropic-auth-plugin.test.js +125 -0
  8. package/dist/anthropic-auth-state.js +231 -0
  9. package/dist/bin.js +90 -0
  10. package/dist/channel-management.js +227 -0
  11. package/dist/cli-parsing.test.js +137 -0
  12. package/dist/cli-send-thread.e2e.test.js +356 -0
  13. package/dist/cli.js +3276 -0
  14. package/dist/commands/abort.js +65 -0
  15. package/dist/commands/action-buttons.js +245 -0
  16. package/dist/commands/add-project.js +113 -0
  17. package/dist/commands/agent.js +335 -0
  18. package/dist/commands/ask-question.js +274 -0
  19. package/dist/commands/btw.js +116 -0
  20. package/dist/commands/compact.js +120 -0
  21. package/dist/commands/context-usage.js +140 -0
  22. package/dist/commands/create-new-project.js +130 -0
  23. package/dist/commands/diff.js +63 -0
  24. package/dist/commands/file-upload.js +275 -0
  25. package/dist/commands/fork.js +220 -0
  26. package/dist/commands/gemini-apikey.js +70 -0
  27. package/dist/commands/login.js +885 -0
  28. package/dist/commands/mcp.js +239 -0
  29. package/dist/commands/memory-snapshot.js +24 -0
  30. package/dist/commands/mention-mode.js +44 -0
  31. package/dist/commands/merge-worktree.js +159 -0
  32. package/dist/commands/model-variant.js +364 -0
  33. package/dist/commands/model.js +776 -0
  34. package/dist/commands/new-worktree.js +366 -0
  35. package/dist/commands/paginated-select.js +57 -0
  36. package/dist/commands/permissions.js +274 -0
  37. package/dist/commands/queue.js +206 -0
  38. package/dist/commands/remove-project.js +115 -0
  39. package/dist/commands/restart-opencode-server.js +127 -0
  40. package/dist/commands/resume.js +149 -0
  41. package/dist/commands/run-command.js +79 -0
  42. package/dist/commands/screenshare.js +303 -0
  43. package/dist/commands/screenshare.test.js +20 -0
  44. package/dist/commands/session-id.js +78 -0
  45. package/dist/commands/session.js +176 -0
  46. package/dist/commands/share.js +80 -0
  47. package/dist/commands/tasks.js +205 -0
  48. package/dist/commands/types.js +2 -0
  49. package/dist/commands/undo-redo.js +305 -0
  50. package/dist/commands/unset-model.js +138 -0
  51. package/dist/commands/upgrade.js +42 -0
  52. package/dist/commands/user-command.js +155 -0
  53. package/dist/commands/verbosity.js +125 -0
  54. package/dist/commands/worktree-settings.js +43 -0
  55. package/dist/commands/worktrees.js +410 -0
  56. package/dist/condense-memory.js +33 -0
  57. package/dist/config.js +94 -0
  58. package/dist/context-awareness-plugin.js +363 -0
  59. package/dist/context-awareness-plugin.test.js +124 -0
  60. package/dist/critique-utils.js +95 -0
  61. package/dist/database.js +1310 -0
  62. package/dist/db.js +251 -0
  63. package/dist/db.test.js +138 -0
  64. package/dist/debounce-timeout.js +28 -0
  65. package/dist/debounced-process-flush.js +77 -0
  66. package/dist/discord-bot.js +1008 -0
  67. package/dist/discord-command-registration.js +524 -0
  68. package/dist/discord-urls.js +81 -0
  69. package/dist/discord-utils.js +591 -0
  70. package/dist/discord-utils.test.js +134 -0
  71. package/dist/errors.js +157 -0
  72. package/dist/escape-backticks.test.js +429 -0
  73. package/dist/event-stream-real-capture.e2e.test.js +533 -0
  74. package/dist/eventsource-parser.test.js +327 -0
  75. package/dist/exec-async.js +26 -0
  76. package/dist/external-opencode-sync.js +480 -0
  77. package/dist/format-tables.js +302 -0
  78. package/dist/format-tables.test.js +308 -0
  79. package/dist/forum-sync/config.js +79 -0
  80. package/dist/forum-sync/discord-operations.js +154 -0
  81. package/dist/forum-sync/index.js +5 -0
  82. package/dist/forum-sync/markdown.js +113 -0
  83. package/dist/forum-sync/sync-to-discord.js +417 -0
  84. package/dist/forum-sync/sync-to-files.js +190 -0
  85. package/dist/forum-sync/types.js +53 -0
  86. package/dist/forum-sync/watchers.js +307 -0
  87. package/dist/gateway-proxy-reconnect.e2e.test.js +394 -0
  88. package/dist/gateway-proxy.e2e.test.js +483 -0
  89. package/dist/genai-worker-wrapper.js +111 -0
  90. package/dist/genai-worker.js +311 -0
  91. package/dist/genai.js +232 -0
  92. package/dist/generated/browser.js +17 -0
  93. package/dist/generated/client.js +37 -0
  94. package/dist/generated/commonInputTypes.js +10 -0
  95. package/dist/generated/enums.js +52 -0
  96. package/dist/generated/internal/class.js +49 -0
  97. package/dist/generated/internal/prismaNamespace.js +253 -0
  98. package/dist/generated/internal/prismaNamespaceBrowser.js +223 -0
  99. package/dist/generated/models/bot_api_keys.js +1 -0
  100. package/dist/generated/models/bot_tokens.js +1 -0
  101. package/dist/generated/models/channel_agents.js +1 -0
  102. package/dist/generated/models/channel_directories.js +1 -0
  103. package/dist/generated/models/channel_mention_mode.js +1 -0
  104. package/dist/generated/models/channel_models.js +1 -0
  105. package/dist/generated/models/channel_verbosity.js +1 -0
  106. package/dist/generated/models/channel_worktrees.js +1 -0
  107. package/dist/generated/models/forum_sync_configs.js +1 -0
  108. package/dist/generated/models/global_models.js +1 -0
  109. package/dist/generated/models/ipc_requests.js +1 -0
  110. package/dist/generated/models/part_messages.js +1 -0
  111. package/dist/generated/models/scheduled_tasks.js +1 -0
  112. package/dist/generated/models/session_agents.js +1 -0
  113. package/dist/generated/models/session_events.js +1 -0
  114. package/dist/generated/models/session_models.js +1 -0
  115. package/dist/generated/models/session_start_sources.js +1 -0
  116. package/dist/generated/models/thread_sessions.js +1 -0
  117. package/dist/generated/models/thread_worktrees.js +1 -0
  118. package/dist/generated/models.js +1 -0
  119. package/dist/heap-monitor.js +122 -0
  120. package/dist/hrana-server.js +263 -0
  121. package/dist/hrana-server.test.js +370 -0
  122. package/dist/html-actions.js +123 -0
  123. package/dist/html-actions.test.js +70 -0
  124. package/dist/html-components.js +117 -0
  125. package/dist/html-components.test.js +34 -0
  126. package/dist/image-optimizer-plugin.js +153 -0
  127. package/dist/image-utils.js +112 -0
  128. package/dist/interaction-handler.js +397 -0
  129. package/dist/ipc-polling.js +252 -0
  130. package/dist/ipc-tools-plugin.js +193 -0
  131. package/dist/kimaki-digital-twin.e2e.test.js +161 -0
  132. package/dist/kimaki-opencode-plugin-loading.e2e.test.js +87 -0
  133. package/dist/kimaki-opencode-plugin.js +17 -0
  134. package/dist/kimaki-opencode-plugin.test.js +98 -0
  135. package/dist/limit-heading-depth.js +25 -0
  136. package/dist/limit-heading-depth.test.js +105 -0
  137. package/dist/logger.js +165 -0
  138. package/dist/markdown.js +342 -0
  139. package/dist/markdown.test.js +257 -0
  140. package/dist/message-finish-field.e2e.test.js +165 -0
  141. package/dist/message-formatting.js +413 -0
  142. package/dist/message-formatting.test.js +73 -0
  143. package/dist/message-preprocessing.js +330 -0
  144. package/dist/onboarding-tutorial.js +172 -0
  145. package/dist/onboarding-welcome.js +37 -0
  146. package/dist/openai-realtime.js +224 -0
  147. package/dist/opencode-command-detection.js +65 -0
  148. package/dist/opencode-command-detection.test.js +240 -0
  149. package/dist/opencode-command.js +129 -0
  150. package/dist/opencode-command.test.js +48 -0
  151. package/dist/opencode-interrupt-plugin.js +361 -0
  152. package/dist/opencode-interrupt-plugin.test.js +458 -0
  153. package/dist/opencode.js +861 -0
  154. package/dist/otto/branding.js +22 -0
  155. package/dist/otto/index.js +21 -0
  156. package/dist/parse-permission-rules.test.js +117 -0
  157. package/dist/patch-text-parser.js +97 -0
  158. package/dist/plugin-logger.js +59 -0
  159. package/dist/privacy-sanitizer.js +105 -0
  160. package/dist/queue-advanced-abort.e2e.test.js +293 -0
  161. package/dist/queue-advanced-action-buttons.e2e.test.js +206 -0
  162. package/dist/queue-advanced-e2e-setup.js +786 -0
  163. package/dist/queue-advanced-footer.e2e.test.js +472 -0
  164. package/dist/queue-advanced-model-switch.e2e.test.js +299 -0
  165. package/dist/queue-advanced-permissions-typing.e2e.test.js +180 -0
  166. package/dist/queue-advanced-question.e2e.test.js +261 -0
  167. package/dist/queue-advanced-typing-interrupt.e2e.test.js +114 -0
  168. package/dist/queue-advanced-typing.e2e.test.js +153 -0
  169. package/dist/queue-drain-after-interactive-ui.e2e.test.js +119 -0
  170. package/dist/queue-interrupt-drain.e2e.test.js +135 -0
  171. package/dist/queue-question-select-drain.e2e.test.js +120 -0
  172. package/dist/runtime-idle-sweeper.js +52 -0
  173. package/dist/runtime-lifecycle.e2e.test.js +508 -0
  174. package/dist/sentry.js +23 -0
  175. package/dist/session-handler/agent-utils.js +67 -0
  176. package/dist/session-handler/event-stream-state.js +420 -0
  177. package/dist/session-handler/event-stream-state.test.js +563 -0
  178. package/dist/session-handler/model-utils.js +124 -0
  179. package/dist/session-handler/opencode-session-event-log.js +94 -0
  180. package/dist/session-handler/thread-runtime-state.js +104 -0
  181. package/dist/session-handler/thread-session-runtime.js +3258 -0
  182. package/dist/session-handler.js +9 -0
  183. package/dist/session-search.js +100 -0
  184. package/dist/session-search.test.js +40 -0
  185. package/dist/session-title-rename.test.js +80 -0
  186. package/dist/startup-service.js +153 -0
  187. package/dist/startup-time.e2e.test.js +296 -0
  188. package/dist/store.js +17 -0
  189. package/dist/system-message.js +613 -0
  190. package/dist/system-message.test.js +602 -0
  191. package/dist/task-runner.js +295 -0
  192. package/dist/task-schedule.js +209 -0
  193. package/dist/task-schedule.test.js +71 -0
  194. package/dist/test-utils.js +299 -0
  195. package/dist/thinking-utils.js +35 -0
  196. package/dist/thread-message-queue.e2e.test.js +999 -0
  197. package/dist/tools.js +357 -0
  198. package/dist/undo-redo.e2e.test.js +161 -0
  199. package/dist/unnest-code-blocks.js +146 -0
  200. package/dist/unnest-code-blocks.test.js +673 -0
  201. package/dist/upgrade.js +114 -0
  202. package/dist/utils.js +144 -0
  203. package/dist/voice-attachment.js +34 -0
  204. package/dist/voice-handler.js +646 -0
  205. package/dist/voice-message.e2e.test.js +1021 -0
  206. package/dist/voice.js +447 -0
  207. package/dist/voice.test.js +235 -0
  208. package/dist/wait-session.js +94 -0
  209. package/dist/websockify.js +69 -0
  210. package/dist/worker-types.js +4 -0
  211. package/dist/worktree-lifecycle.e2e.test.js +308 -0
  212. package/dist/worktree-utils.js +3 -0
  213. package/dist/worktrees.js +929 -0
  214. package/dist/worktrees.test.js +189 -0
  215. package/dist/xml.js +92 -0
  216. package/dist/xml.test.js +32 -0
  217. package/package.json +98 -0
  218. package/schema.prisma +295 -0
  219. package/skills/batch/SKILL.md +87 -0
  220. package/skills/critique/SKILL.md +112 -0
  221. package/skills/egaki/SKILL.md +100 -0
  222. package/skills/errore/SKILL.md +647 -0
  223. package/skills/event-sourcing-state/SKILL.md +252 -0
  224. package/skills/gitchamber/SKILL.md +93 -0
  225. package/skills/goke/SKILL.md +644 -0
  226. package/skills/jitter/EDITOR.md +219 -0
  227. package/skills/jitter/EXPORT-INTERNALS.md +309 -0
  228. package/skills/jitter/SKILL.md +158 -0
  229. package/skills/jitter/jitter-clipboard.json +1042 -0
  230. package/skills/jitter/package.json +14 -0
  231. package/skills/jitter/tsconfig.json +15 -0
  232. package/skills/jitter/utils/actions.ts +212 -0
  233. package/skills/jitter/utils/export.ts +114 -0
  234. package/skills/jitter/utils/index.ts +141 -0
  235. package/skills/jitter/utils/snapshot.ts +154 -0
  236. package/skills/jitter/utils/traverse.ts +246 -0
  237. package/skills/jitter/utils/types.ts +279 -0
  238. package/skills/jitter/utils/wait.ts +133 -0
  239. package/skills/lintcn/SKILL.md +873 -0
  240. package/skills/new-skill/SKILL.md +211 -0
  241. package/skills/npm-package/SKILL.md +239 -0
  242. package/skills/playwriter/SKILL.md +35 -0
  243. package/skills/proxyman/SKILL.md +215 -0
  244. package/skills/security-review/SKILL.md +208 -0
  245. package/skills/simplify/SKILL.md +58 -0
  246. package/skills/spiceflow/SKILL.md +14 -0
  247. package/skills/termcast/SKILL.md +945 -0
  248. package/skills/tuistory/SKILL.md +250 -0
  249. package/skills/usecomputer/SKILL.md +264 -0
  250. package/skills/x-articles/SKILL.md +554 -0
  251. package/skills/zele/SKILL.md +112 -0
  252. package/skills/zustand-centralized-state/SKILL.md +1004 -0
  253. package/src/agent-model.e2e.test.ts +976 -0
  254. package/src/ai-tool-to-genai.test.ts +296 -0
  255. package/src/ai-tool-to-genai.ts +283 -0
  256. package/src/ai-tool.ts +39 -0
  257. package/src/anthropic-auth-plugin.test.ts +159 -0
  258. package/src/anthropic-auth-plugin.ts +861 -0
  259. package/src/anthropic-auth-state.ts +282 -0
  260. package/src/bin.ts +111 -0
  261. package/src/channel-management.ts +334 -0
  262. package/src/cli-parsing.test.ts +195 -0
  263. package/src/cli-send-thread.e2e.test.ts +464 -0
  264. package/src/cli.ts +4581 -0
  265. package/src/commands/abort.ts +89 -0
  266. package/src/commands/action-buttons.ts +364 -0
  267. package/src/commands/add-project.ts +149 -0
  268. package/src/commands/agent.ts +473 -0
  269. package/src/commands/ask-question.ts +390 -0
  270. package/src/commands/btw.ts +164 -0
  271. package/src/commands/compact.ts +157 -0
  272. package/src/commands/context-usage.ts +199 -0
  273. package/src/commands/create-new-project.ts +190 -0
  274. package/src/commands/diff.ts +91 -0
  275. package/src/commands/file-upload.ts +389 -0
  276. package/src/commands/fork.ts +321 -0
  277. package/src/commands/gemini-apikey.ts +104 -0
  278. package/src/commands/login.ts +1173 -0
  279. package/src/commands/mcp.ts +307 -0
  280. package/src/commands/memory-snapshot.ts +30 -0
  281. package/src/commands/mention-mode.ts +68 -0
  282. package/src/commands/merge-worktree.ts +223 -0
  283. package/src/commands/model-variant.ts +483 -0
  284. package/src/commands/model.ts +1053 -0
  285. package/src/commands/new-worktree.ts +510 -0
  286. package/src/commands/paginated-select.ts +81 -0
  287. package/src/commands/permissions.ts +397 -0
  288. package/src/commands/queue.ts +271 -0
  289. package/src/commands/remove-project.ts +155 -0
  290. package/src/commands/restart-opencode-server.ts +162 -0
  291. package/src/commands/resume.ts +230 -0
  292. package/src/commands/run-command.ts +123 -0
  293. package/src/commands/screenshare.test.ts +30 -0
  294. package/src/commands/screenshare.ts +366 -0
  295. package/src/commands/session-id.ts +109 -0
  296. package/src/commands/session.ts +227 -0
  297. package/src/commands/share.ts +106 -0
  298. package/src/commands/tasks.ts +293 -0
  299. package/src/commands/types.ts +25 -0
  300. package/src/commands/undo-redo.ts +386 -0
  301. package/src/commands/unset-model.ts +173 -0
  302. package/src/commands/upgrade.ts +52 -0
  303. package/src/commands/user-command.ts +198 -0
  304. package/src/commands/verbosity.ts +173 -0
  305. package/src/commands/worktree-settings.ts +70 -0
  306. package/src/commands/worktrees.ts +552 -0
  307. package/src/condense-memory.ts +36 -0
  308. package/src/config.ts +111 -0
  309. package/src/context-awareness-plugin.test.ts +142 -0
  310. package/src/context-awareness-plugin.ts +510 -0
  311. package/src/critique-utils.ts +139 -0
  312. package/src/database.ts +1876 -0
  313. package/src/db.test.ts +162 -0
  314. package/src/db.ts +286 -0
  315. package/src/debounce-timeout.ts +43 -0
  316. package/src/debounced-process-flush.ts +104 -0
  317. package/src/discord-bot.ts +1330 -0
  318. package/src/discord-command-registration.ts +693 -0
  319. package/src/discord-urls.ts +88 -0
  320. package/src/discord-utils.test.ts +153 -0
  321. package/src/discord-utils.ts +800 -0
  322. package/src/errors.ts +201 -0
  323. package/src/escape-backticks.test.ts +469 -0
  324. package/src/event-stream-real-capture.e2e.test.ts +692 -0
  325. package/src/eventsource-parser.test.ts +351 -0
  326. package/src/exec-async.ts +35 -0
  327. package/src/external-opencode-sync.ts +685 -0
  328. package/src/format-tables.test.ts +335 -0
  329. package/src/format-tables.ts +445 -0
  330. package/src/forum-sync/config.ts +92 -0
  331. package/src/forum-sync/discord-operations.ts +241 -0
  332. package/src/forum-sync/index.ts +9 -0
  333. package/src/forum-sync/markdown.ts +172 -0
  334. package/src/forum-sync/sync-to-discord.ts +595 -0
  335. package/src/forum-sync/sync-to-files.ts +294 -0
  336. package/src/forum-sync/types.ts +175 -0
  337. package/src/forum-sync/watchers.ts +454 -0
  338. package/src/gateway-proxy-reconnect.e2e.test.ts +523 -0
  339. package/src/gateway-proxy.e2e.test.ts +640 -0
  340. package/src/genai-worker-wrapper.ts +164 -0
  341. package/src/genai-worker.ts +386 -0
  342. package/src/genai.ts +321 -0
  343. package/src/generated/browser.ts +114 -0
  344. package/src/generated/client.ts +138 -0
  345. package/src/generated/commonInputTypes.ts +736 -0
  346. package/src/generated/enums.ts +88 -0
  347. package/src/generated/internal/class.ts +384 -0
  348. package/src/generated/internal/prismaNamespace.ts +2386 -0
  349. package/src/generated/internal/prismaNamespaceBrowser.ts +326 -0
  350. package/src/generated/models/bot_api_keys.ts +1288 -0
  351. package/src/generated/models/bot_tokens.ts +1656 -0
  352. package/src/generated/models/channel_agents.ts +1256 -0
  353. package/src/generated/models/channel_directories.ts +1859 -0
  354. package/src/generated/models/channel_mention_mode.ts +1300 -0
  355. package/src/generated/models/channel_models.ts +1288 -0
  356. package/src/generated/models/channel_verbosity.ts +1228 -0
  357. package/src/generated/models/channel_worktrees.ts +1300 -0
  358. package/src/generated/models/forum_sync_configs.ts +1452 -0
  359. package/src/generated/models/global_models.ts +1288 -0
  360. package/src/generated/models/ipc_requests.ts +1485 -0
  361. package/src/generated/models/part_messages.ts +1302 -0
  362. package/src/generated/models/scheduled_tasks.ts +2320 -0
  363. package/src/generated/models/session_agents.ts +1086 -0
  364. package/src/generated/models/session_events.ts +1439 -0
  365. package/src/generated/models/session_models.ts +1114 -0
  366. package/src/generated/models/session_start_sources.ts +1408 -0
  367. package/src/generated/models/thread_sessions.ts +1781 -0
  368. package/src/generated/models/thread_worktrees.ts +1356 -0
  369. package/src/generated/models.ts +30 -0
  370. package/src/heap-monitor.ts +152 -0
  371. package/src/hrana-server.test.ts +434 -0
  372. package/src/hrana-server.ts +314 -0
  373. package/src/html-actions.test.ts +87 -0
  374. package/src/html-actions.ts +174 -0
  375. package/src/html-components.test.ts +38 -0
  376. package/src/html-components.ts +181 -0
  377. package/src/image-optimizer-plugin.ts +194 -0
  378. package/src/image-utils.ts +149 -0
  379. package/src/interaction-handler.ts +576 -0
  380. package/src/ipc-polling.ts +326 -0
  381. package/src/ipc-tools-plugin.ts +236 -0
  382. package/src/kimaki-digital-twin.e2e.test.ts +199 -0
  383. package/src/kimaki-opencode-plugin-loading.e2e.test.ts +109 -0
  384. package/src/kimaki-opencode-plugin.test.ts +108 -0
  385. package/src/kimaki-opencode-plugin.ts +18 -0
  386. package/src/limit-heading-depth.test.ts +116 -0
  387. package/src/limit-heading-depth.ts +26 -0
  388. package/src/logger.ts +208 -0
  389. package/src/markdown.test.ts +308 -0
  390. package/src/markdown.ts +410 -0
  391. package/src/message-finish-field.e2e.test.ts +192 -0
  392. package/src/message-formatting.test.ts +81 -0
  393. package/src/message-formatting.ts +533 -0
  394. package/src/message-preprocessing.ts +455 -0
  395. package/src/onboarding-tutorial.ts +176 -0
  396. package/src/onboarding-welcome.ts +49 -0
  397. package/src/openai-realtime.ts +358 -0
  398. package/src/opencode-command-detection.test.ts +307 -0
  399. package/src/opencode-command-detection.ts +76 -0
  400. package/src/opencode-command.test.ts +70 -0
  401. package/src/opencode-command.ts +188 -0
  402. package/src/opencode-interrupt-plugin.test.ts +677 -0
  403. package/src/opencode-interrupt-plugin.ts +477 -0
  404. package/src/opencode.ts +1110 -0
  405. package/src/otto/branding.ts +23 -0
  406. package/src/otto/index.ts +22 -0
  407. package/src/parse-permission-rules.test.ts +127 -0
  408. package/src/patch-text-parser.ts +107 -0
  409. package/src/plugin-logger.ts +68 -0
  410. package/src/privacy-sanitizer.ts +142 -0
  411. package/src/queue-advanced-abort.e2e.test.ts +382 -0
  412. package/src/queue-advanced-action-buttons.e2e.test.ts +268 -0
  413. package/src/queue-advanced-e2e-setup.ts +873 -0
  414. package/src/queue-advanced-footer.e2e.test.ts +576 -0
  415. package/src/queue-advanced-model-switch.e2e.test.ts +383 -0
  416. package/src/queue-advanced-permissions-typing.e2e.test.ts +245 -0
  417. package/src/queue-advanced-question.e2e.test.ts +316 -0
  418. package/src/queue-advanced-typing-interrupt.e2e.test.ts +146 -0
  419. package/src/queue-advanced-typing.e2e.test.ts +199 -0
  420. package/src/queue-drain-after-interactive-ui.e2e.test.ts +151 -0
  421. package/src/queue-interrupt-drain.e2e.test.ts +166 -0
  422. package/src/queue-question-select-drain.e2e.test.ts +152 -0
  423. package/src/runtime-idle-sweeper.ts +76 -0
  424. package/src/runtime-lifecycle.e2e.test.ts +641 -0
  425. package/src/schema.sql +173 -0
  426. package/src/sentry.ts +26 -0
  427. package/src/session-handler/agent-utils.ts +97 -0
  428. package/src/session-handler/event-stream-fixtures/real-session-action-buttons.jsonl +45 -0
  429. package/src/session-handler/event-stream-fixtures/real-session-footer-suppressed-on-pre-idle-interrupt.jsonl +40 -0
  430. package/src/session-handler/event-stream-fixtures/real-session-permission-external-file.jsonl +23 -0
  431. package/src/session-handler/event-stream-fixtures/real-session-task-normal.jsonl +22 -0
  432. package/src/session-handler/event-stream-fixtures/real-session-task-three-parallel-sleeps.jsonl +277 -0
  433. package/src/session-handler/event-stream-fixtures/real-session-task-user-interruption.jsonl +46 -0
  434. package/src/session-handler/event-stream-fixtures/session-abort-after-idle-race.jsonl +21 -0
  435. package/src/session-handler/event-stream-fixtures/session-concurrent-messages-serialized.jsonl +56 -0
  436. package/src/session-handler/event-stream-fixtures/session-explicit-abort.jsonl +44 -0
  437. package/src/session-handler/event-stream-fixtures/session-normal-completion.jsonl +29 -0
  438. package/src/session-handler/event-stream-fixtures/session-tool-call-noisy-stream.jsonl +29 -0
  439. package/src/session-handler/event-stream-fixtures/session-two-completions-same-session.jsonl +50 -0
  440. package/src/session-handler/event-stream-fixtures/session-user-interruption.jsonl +59 -0
  441. package/src/session-handler/event-stream-fixtures/session-voice-queued-followup.jsonl +52 -0
  442. package/src/session-handler/event-stream-state.test.ts +645 -0
  443. package/src/session-handler/event-stream-state.ts +608 -0
  444. package/src/session-handler/model-utils.ts +183 -0
  445. package/src/session-handler/opencode-session-event-log.ts +130 -0
  446. package/src/session-handler/thread-runtime-state.ts +212 -0
  447. package/src/session-handler/thread-session-runtime.ts +4281 -0
  448. package/src/session-handler.ts +15 -0
  449. package/src/session-search.test.ts +50 -0
  450. package/src/session-search.ts +148 -0
  451. package/src/session-title-rename.test.ts +112 -0
  452. package/src/startup-service.ts +200 -0
  453. package/src/startup-time.e2e.test.ts +373 -0
  454. package/src/store.ts +122 -0
  455. package/src/system-message.test.ts +612 -0
  456. package/src/system-message.ts +723 -0
  457. package/src/task-runner.ts +421 -0
  458. package/src/task-schedule.test.ts +84 -0
  459. package/src/task-schedule.ts +311 -0
  460. package/src/test-utils.ts +435 -0
  461. package/src/thinking-utils.ts +61 -0
  462. package/src/thread-message-queue.e2e.test.ts +1219 -0
  463. package/src/tools.ts +430 -0
  464. package/src/undici.d.ts +12 -0
  465. package/src/undo-redo.e2e.test.ts +209 -0
  466. package/src/unnest-code-blocks.test.ts +713 -0
  467. package/src/unnest-code-blocks.ts +185 -0
  468. package/src/upgrade.ts +127 -0
  469. package/src/utils.ts +212 -0
  470. package/src/voice-attachment.ts +51 -0
  471. package/src/voice-handler.ts +908 -0
  472. package/src/voice-message.e2e.test.ts +1255 -0
  473. package/src/voice.test.ts +281 -0
  474. package/src/voice.ts +627 -0
  475. package/src/wait-session.ts +147 -0
  476. package/src/websockify.ts +101 -0
  477. package/src/worker-types.ts +64 -0
  478. package/src/worktree-lifecycle.e2e.test.ts +391 -0
  479. package/src/worktree-utils.ts +4 -0
  480. package/src/worktrees.test.ts +223 -0
  481. package/src/worktrees.ts +1294 -0
  482. package/src/xml.test.ts +38 -0
  483. package/src/xml.ts +121 -0
package/src/voice.ts ADDED
@@ -0,0 +1,627 @@
1
+ // Audio transcription service using AI SDK providers.
2
+ // Both providers use LanguageModelV3 (chat model) with audio file parts + tool calling,
3
+ // so we can pass full context (file tree, session info) for better word recognition.
4
+ // - OpenAI: gpt-4o-audio-preview via .chat() (Chat Completions API). MUST use .chat()
5
+ // because the default Responses API doesn't support audio file parts. The Chat
6
+ // Completions handler converts audio/mpeg file parts to input_audio format.
7
+ // - Gemini: gemini-2.5-flash natively accepts audio file parts in chat.
8
+ // Calls model.doGenerate() directly without the `ai` npm package.
9
+ // Uses errore for type-safe error handling.
10
+
11
+ import type {
12
+ LanguageModelV3,
13
+ LanguageModelV3CallOptions,
14
+ LanguageModelV3FunctionTool,
15
+ LanguageModelV3Content,
16
+ LanguageModelV3ToolCall,
17
+ } from '@ai-sdk/provider'
18
+ import { createGoogleGenerativeAI } from '@ai-sdk/google'
19
+ import { createOpenAI } from '@ai-sdk/openai'
20
+ import { Readable } from 'node:stream'
21
+ import prism from 'prism-media'
22
+ import * as errore from 'errore'
23
+ import { createLogger, LogPrefix } from './logger.js'
24
+ import {
25
+ ApiKeyMissingError,
26
+ InvalidAudioFormatError,
27
+ TranscriptionError,
28
+ EmptyTranscriptionError,
29
+ NoResponseContentError,
30
+ NoToolResponseError,
31
+ } from './errors.js'
32
+
33
+ const voiceLogger = createLogger(LogPrefix.VOICE)
34
+
35
+ // OpenAI input_audio only supports wav and mp3. Other formats (OGG Opus, etc)
36
+ // must be converted before sending.
37
+ const OPENAI_SUPPORTED_AUDIO_TYPES = new Set([
38
+ 'audio/mpeg',
39
+ 'audio/mp3',
40
+ 'audio/wav',
41
+ 'audio/x-wav',
42
+ ])
43
+
44
+ const OGG_AUDIO_TYPES = new Set([
45
+ 'audio/ogg',
46
+ 'audio/opus',
47
+ ])
48
+
49
+ const M4A_AUDIO_TYPES = new Set([
50
+ 'audio/mp4',
51
+ 'audio/m4a',
52
+ 'audio/x-m4a',
53
+ ])
54
+
55
+ export function normalizeAudioMediaType(mediaType: string): string {
56
+ const normalized = mediaType.trim().toLowerCase()
57
+ if (normalized === 'audio/x-m4a' || normalized === 'audio/m4a') {
58
+ return 'audio/mp4'
59
+ }
60
+ return normalized
61
+ }
62
+
63
+ type OpenAIAudioConversionStrategy =
64
+ | 'none'
65
+ | 'convert-ogg-to-wav'
66
+ | 'convert-m4a-to-wav'
67
+ | 'unsupported'
68
+
69
+ export function getOpenAIAudioConversionStrategy(
70
+ mediaType: string,
71
+ ): OpenAIAudioConversionStrategy {
72
+ if (OPENAI_SUPPORTED_AUDIO_TYPES.has(mediaType)) {
73
+ return 'none'
74
+ }
75
+ if (OGG_AUDIO_TYPES.has(mediaType)) {
76
+ return 'convert-ogg-to-wav'
77
+ }
78
+ if (M4A_AUDIO_TYPES.has(mediaType)) {
79
+ return 'convert-m4a-to-wav'
80
+ }
81
+ return 'unsupported'
82
+ }
83
+
84
+ /**
85
+ * Convert OGG Opus audio to WAV using prism-media (already installed for Discord voice).
86
+ * Pipeline: OGG buffer → OggDemuxer → Opus Decoder → PCM → WAV (with header).
87
+ * No ffmpeg needed — uses @discordjs/opus native bindings.
88
+ */
89
+ export function convertOggToWav(input: Buffer): Promise<TranscriptionError | Buffer> {
90
+ return new Promise((resolve) => {
91
+ const pcmChunks: Buffer[] = []
92
+
93
+ const demuxer = new prism.opus.OggDemuxer()
94
+ const decoder = new prism.opus.Decoder({
95
+ rate: 48000,
96
+ channels: 1,
97
+ frameSize: 960,
98
+ })
99
+
100
+ decoder.on('data', (chunk: Buffer) => {
101
+ pcmChunks.push(chunk)
102
+ })
103
+
104
+ decoder.on('end', () => {
105
+ const pcmData = Buffer.concat(pcmChunks)
106
+ const wavHeader = createWavHeader({
107
+ dataLength: pcmData.length,
108
+ sampleRate: 48000,
109
+ numChannels: 1,
110
+ bitsPerSample: 16,
111
+ })
112
+ resolve(Buffer.concat([wavHeader, pcmData]))
113
+ })
114
+
115
+ decoder.on('error', (err: Error) => {
116
+ resolve(
117
+ new TranscriptionError({
118
+ reason: `Opus decode failed: ${err.message}`,
119
+ cause: err,
120
+ }),
121
+ )
122
+ })
123
+
124
+ demuxer.on('error', (err: Error) => {
125
+ resolve(
126
+ new TranscriptionError({
127
+ reason: `OGG demux failed: ${err.message}`,
128
+ cause: err,
129
+ }),
130
+ )
131
+ })
132
+
133
+ Readable.from(input).pipe(demuxer).pipe(decoder)
134
+ })
135
+ }
136
+
137
+ /**
138
+ * Convert M4A/MP4 audio to WAV using prism-media FFmpeg wrapper.
139
+ * This depends on an ffmpeg binary available in PATH.
140
+ */
141
+ export function convertM4aToWav(input: Buffer): Promise<TranscriptionError | Buffer> {
142
+ return new Promise((resolve) => {
143
+ const pcmChunks: Buffer[] = []
144
+ const transcoder = new prism.FFmpeg({
145
+ args: [
146
+ '-analyzeduration',
147
+ '0',
148
+ '-loglevel',
149
+ '0',
150
+ '-f',
151
+ 'mp4',
152
+ '-i',
153
+ 'pipe:0',
154
+ '-f',
155
+ 's16le',
156
+ '-acodec',
157
+ 'pcm_s16le',
158
+ '-ac',
159
+ '1',
160
+ '-ar',
161
+ '48000',
162
+ 'pipe:1',
163
+ ],
164
+ })
165
+
166
+ transcoder.on('data', (chunk: Buffer) => {
167
+ pcmChunks.push(chunk)
168
+ })
169
+
170
+ transcoder.on('end', () => {
171
+ const pcmData = Buffer.concat(pcmChunks)
172
+ if (pcmData.length === 0) {
173
+ resolve(
174
+ new TranscriptionError({
175
+ reason: 'FFmpeg conversion produced empty audio output',
176
+ }),
177
+ )
178
+ return
179
+ }
180
+
181
+ const wavHeader = createWavHeader({
182
+ dataLength: pcmData.length,
183
+ sampleRate: 48000,
184
+ numChannels: 1,
185
+ bitsPerSample: 16,
186
+ })
187
+ resolve(Buffer.concat([wavHeader, pcmData]))
188
+ })
189
+
190
+ transcoder.on('error', (err: Error) => {
191
+ const lower = err.message.toLowerCase()
192
+ const isMissingFfmpeg =
193
+ lower.includes('ffmpeg') &&
194
+ (lower.includes('not found') ||
195
+ lower.includes('enoent') ||
196
+ lower.includes('spawn'))
197
+ if (isMissingFfmpeg) {
198
+ resolve(
199
+ new TranscriptionError({
200
+ reason:
201
+ 'M4A transcription with OpenAI requires ffmpeg to be installed and available in PATH',
202
+ cause: err,
203
+ }),
204
+ )
205
+ return
206
+ }
207
+
208
+ resolve(
209
+ new TranscriptionError({
210
+ reason: `M4A decode failed: ${err.message}`,
211
+ cause: err,
212
+ }),
213
+ )
214
+ })
215
+
216
+ Readable.from(input).pipe(transcoder)
217
+ })
218
+ }
219
+
220
+ function createWavHeader({
221
+ dataLength,
222
+ sampleRate,
223
+ numChannels,
224
+ bitsPerSample,
225
+ }: {
226
+ dataLength: number
227
+ sampleRate: number
228
+ numChannels: number
229
+ bitsPerSample: number
230
+ }): Buffer {
231
+ const byteRate = (sampleRate * numChannels * bitsPerSample) / 8
232
+ const blockAlign = (numChannels * bitsPerSample) / 8
233
+ const buffer = Buffer.alloc(44)
234
+ buffer.write('RIFF', 0)
235
+ buffer.writeUInt32LE(36 + dataLength, 4)
236
+ buffer.write('WAVE', 8)
237
+ buffer.write('fmt ', 12)
238
+ buffer.writeUInt32LE(16, 16)
239
+ buffer.writeUInt16LE(1, 20)
240
+ buffer.writeUInt16LE(numChannels, 22)
241
+ buffer.writeUInt32LE(sampleRate, 24)
242
+ buffer.writeUInt32LE(byteRate, 28)
243
+ buffer.writeUInt16LE(blockAlign, 32)
244
+ buffer.writeUInt16LE(bitsPerSample, 34)
245
+ buffer.write('data', 36)
246
+ buffer.writeUInt32LE(dataLength, 40)
247
+ return buffer
248
+ }
249
+
250
+ type TranscriptionLoopError =
251
+ | NoResponseContentError
252
+ | TranscriptionError
253
+ | EmptyTranscriptionError
254
+ | NoToolResponseError
255
+
256
+ // Build the transcription tool schema dynamically so the agent field can
257
+ // use an enum constrained to the actual available agent names.
258
+ function buildTranscriptionTool({
259
+ agentNames,
260
+ }: {
261
+ agentNames?: string[]
262
+ }): LanguageModelV3FunctionTool {
263
+ const properties: Record<string, Record<string, unknown>> = {
264
+ transcription: {
265
+ type: 'string',
266
+ description:
267
+ 'The final transcription of the audio. MUST be non-empty. If audio is unclear, transcribe your best interpretation. If silent, too short to understand, or completely incomprehensible, use "[inaudible audio]".',
268
+ },
269
+ queueMessage: {
270
+ type: 'boolean',
271
+ description:
272
+ 'Set to true ONLY if the user explicitly says "queue this message", "queue this", or similar phrasing indicating they want this message queued instead of sent immediately. If not mentioned, omit or set to false.',
273
+ },
274
+ }
275
+
276
+ if (agentNames && agentNames.length > 0) {
277
+ properties['agent'] = {
278
+ type: 'string',
279
+ enum: agentNames,
280
+ description:
281
+ 'The agent name ONLY if the user explicitly says "use the X agent", "switch to X agent", "with the X agent", or similar phrasing. Remove the agent instruction from the transcription text. Omit if no agent is mentioned.',
282
+ }
283
+ }
284
+
285
+ return {
286
+ type: 'function',
287
+ name: 'transcriptionResult',
288
+ description:
289
+ 'MANDATORY: You MUST call this tool to complete the task. This is the ONLY way to return results - text responses are ignored. Call this with your transcription, even if imperfect. An imperfect transcription is better than none.',
290
+ inputSchema: {
291
+ type: 'object',
292
+ properties,
293
+ required: ['transcription'],
294
+ },
295
+ }
296
+ }
297
+
298
+ export type TranscriptionResult = {
299
+ transcription: string
300
+ queueMessage: boolean
301
+ /** Agent name extracted from voice message, only set if user explicitly requested an agent. */
302
+ agent?: string
303
+ }
304
+
305
+ /**
306
+ * Extract transcription result from doGenerate content array.
307
+ * Looks for a tool-call named 'transcriptionResult', falls back to text content.
308
+ * Returns structured result with transcription text and queueMessage flag.
309
+ */
310
+ export function extractTranscription(
311
+ content: Array<LanguageModelV3Content>,
312
+ ): TranscriptionLoopError | TranscriptionResult {
313
+ const toolCall = content.find(
314
+ (c): c is LanguageModelV3ToolCall =>
315
+ c.type === 'tool-call' && c.toolName === 'transcriptionResult',
316
+ )
317
+
318
+ if (toolCall) {
319
+ // toolCall.input is a JSON string in LanguageModelV3
320
+ const args: Record<string, unknown> = (() => {
321
+ if (typeof toolCall.input === 'string') {
322
+ return JSON.parse(toolCall.input) as Record<string, unknown>
323
+ }
324
+ return {}
325
+ })()
326
+ const transcription = (typeof args.transcription === 'string' ? args.transcription : '').trim()
327
+ const queueMessage = args.queueMessage === true
328
+ const agent = typeof args.agent === 'string' ? args.agent : undefined
329
+ voiceLogger.log(
330
+ `Transcription result received: "${transcription.slice(0, 100)}..."${queueMessage ? ' [QUEUE]' : ''}${agent ? ` [AGENT:${agent}]` : ''}`,
331
+ )
332
+ if (!transcription) {
333
+ return new EmptyTranscriptionError()
334
+ }
335
+ return { transcription, queueMessage, agent }
336
+ }
337
+
338
+ // Fall back to text content if no tool call
339
+ const textPart = content.find((c) => c.type === 'text')
340
+ if (textPart && textPart.type === 'text' && textPart.text.trim()) {
341
+ voiceLogger.log(
342
+ `No tool call but got text: "${textPart.text.trim().slice(0, 100)}..."`,
343
+ )
344
+ return { transcription: textPart.text.trim(), queueMessage: false }
345
+ }
346
+
347
+ if (content.length === 0) {
348
+ return new NoResponseContentError()
349
+ }
350
+
351
+ return new TranscriptionError({
352
+ reason: 'Model did not produce a transcription',
353
+ })
354
+ }
355
+
356
+ async function runTranscriptionOnce({
357
+ model,
358
+ prompt,
359
+ audioBase64,
360
+ mediaType,
361
+ temperature,
362
+ agentNames,
363
+ }: {
364
+ model: LanguageModelV3
365
+ prompt: string
366
+ audioBase64: string
367
+ mediaType: string
368
+ temperature: number
369
+ agentNames?: string[]
370
+ }): Promise<TranscriptionLoopError | TranscriptionResult> {
371
+ const tool = buildTranscriptionTool({ agentNames })
372
+ const options: LanguageModelV3CallOptions = {
373
+ prompt: [
374
+ {
375
+ role: 'user',
376
+ content: [
377
+ { type: 'text', text: prompt },
378
+ {
379
+ type: 'file',
380
+ data: audioBase64,
381
+ mediaType,
382
+ },
383
+ ],
384
+ },
385
+ ],
386
+ temperature,
387
+ maxOutputTokens: 2048,
388
+ tools: [tool],
389
+ toolChoice: { type: 'tool', toolName: 'transcriptionResult' },
390
+ providerOptions: {
391
+ google: {
392
+ thinkingConfig: { thinkingBudget: 1024 },
393
+ },
394
+ },
395
+ }
396
+
397
+ // doGenerate returns PromiseLike, wrap in Promise.resolve for errore compatibility
398
+ const response = await errore.tryAsync({
399
+ try: () => Promise.resolve(model.doGenerate(options)),
400
+ catch: (e: Error) =>
401
+ new TranscriptionError({
402
+ reason: `API call failed: ${String(e)}`,
403
+ cause: e,
404
+ }),
405
+ })
406
+
407
+ if (response instanceof TranscriptionError) {
408
+ return response
409
+ }
410
+
411
+ return extractTranscription(response.content)
412
+ }
413
+
414
+ export type TranscribeAudioErrors =
415
+ | ApiKeyMissingError
416
+ | InvalidAudioFormatError
417
+ | TranscriptionLoopError
418
+
419
+ export type TranscriptionProvider = 'openai' | 'gemini'
420
+
421
+ /**
422
+ * Create a LanguageModelV3 for transcription.
423
+ * Both providers use chat models that accept audio file parts, so we get full
424
+ * context (prompt, session info, tool calling) for better word recognition.
425
+ *
426
+ * OpenAI: must use .chat() to get the Chat Completions API model, because the
427
+ * default callable (Responses API) doesn't support audio file parts.
428
+ * Gemini: language models natively accept audio in chat.
429
+ */
430
+ export function createTranscriptionModel({
431
+ apiKey,
432
+ provider,
433
+ }: {
434
+ apiKey: string
435
+ provider?: TranscriptionProvider
436
+ }): LanguageModelV3 {
437
+ const resolvedProvider: TranscriptionProvider =
438
+ provider || (apiKey.startsWith('sk-') ? 'openai' : 'gemini')
439
+
440
+ if (resolvedProvider === 'openai') {
441
+ const openai = createOpenAI({ apiKey })
442
+ return openai.chat('gpt-4o-audio-preview')
443
+ }
444
+
445
+ const google = createGoogleGenerativeAI({ apiKey })
446
+ return google('gemini-2.5-flash')
447
+ }
448
+
449
+ export async function transcribeAudio({
450
+ audio,
451
+ prompt,
452
+ language,
453
+ temperature,
454
+ apiKey: apiKeyParam,
455
+ model,
456
+ provider,
457
+ mediaType: mediaTypeParam,
458
+ currentSessionContext,
459
+ lastSessionContext,
460
+ agents,
461
+ }: {
462
+ audio: Buffer | Uint8Array | ArrayBuffer | string
463
+ prompt?: string
464
+ language?: string
465
+ temperature?: number
466
+ apiKey?: string
467
+ model?: LanguageModelV3
468
+ provider?: TranscriptionProvider
469
+ /** MIME type of the audio data (e.g. 'audio/ogg'). Defaults to 'audio/mpeg'. */
470
+ mediaType?: string
471
+ currentSessionContext?: string
472
+ lastSessionContext?: string
473
+ /** Available agents for agent selection via voice. Names used as enum values in the tool schema. */
474
+ agents?: Array<{ name: string; description?: string }>
475
+ }): Promise<TranscribeAudioErrors | TranscriptionResult> {
476
+ const apiKey =
477
+ apiKeyParam || process.env.OPENAI_API_KEY || process.env.GEMINI_API_KEY
478
+
479
+ if (!model && !apiKey) {
480
+ return Promise.resolve(new ApiKeyMissingError({ service: 'OpenAI or Gemini' }))
481
+ }
482
+
483
+ const resolvedProvider: TranscriptionProvider = (() => {
484
+ if (provider) {
485
+ return provider
486
+ }
487
+ if (apiKey) {
488
+ return apiKey.startsWith('sk-') ? 'openai' : 'gemini'
489
+ }
490
+ return 'gemini'
491
+ })()
492
+
493
+ const languageModel: LanguageModelV3 =
494
+ model || createTranscriptionModel({ apiKey: apiKey!, provider: resolvedProvider })
495
+
496
+ // Convert audio to Buffer for potential format conversion
497
+ const audioBuffer: Buffer = (() => {
498
+ if (typeof audio === 'string') {
499
+ return Buffer.from(audio, 'base64')
500
+ }
501
+ if (audio instanceof Buffer) {
502
+ return audio
503
+ }
504
+ if (audio instanceof ArrayBuffer) {
505
+ return Buffer.from(new Uint8Array(audio))
506
+ }
507
+ return Buffer.from(audio)
508
+ })()
509
+
510
+ if (audioBuffer.length === 0) {
511
+ return new InvalidAudioFormatError()
512
+ }
513
+
514
+ let mediaType = normalizeAudioMediaType(mediaTypeParam || 'audio/mpeg')
515
+ let finalAudioBase64 = audioBuffer.toString('base64')
516
+
517
+ // OpenAI input_audio supports only a subset of audio formats.
518
+ // Convert based on MIME so OGG conversion runs only for real OGG/Opus inputs.
519
+ if (resolvedProvider === 'openai') {
520
+ const conversionStrategy = getOpenAIAudioConversionStrategy(mediaType)
521
+ if (conversionStrategy === 'convert-ogg-to-wav') {
522
+ voiceLogger.log(`Converting ${mediaType} to WAV for OpenAI compatibility`)
523
+ const converted = await convertOggToWav(audioBuffer)
524
+ if (converted instanceof Error) {
525
+ return converted
526
+ }
527
+ finalAudioBase64 = converted.toString('base64')
528
+ mediaType = 'audio/wav'
529
+ } else if (conversionStrategy === 'convert-m4a-to-wav') {
530
+ voiceLogger.log(`Converting ${mediaType} to WAV for OpenAI compatibility`)
531
+ const converted = await convertM4aToWav(audioBuffer)
532
+ if (converted instanceof Error) {
533
+ return converted
534
+ }
535
+ finalAudioBase64 = converted.toString('base64')
536
+ mediaType = 'audio/wav'
537
+ } else if (conversionStrategy === 'unsupported') {
538
+ return new InvalidAudioFormatError()
539
+ }
540
+ }
541
+
542
+ const languageHint = language ? `The audio is in ${language}.\n\n` : ''
543
+
544
+ // build session context section
545
+ const sessionContextParts: string[] = []
546
+ if (lastSessionContext) {
547
+ sessionContextParts.push(`<last_session>
548
+ ${lastSessionContext}
549
+ </last_session>`)
550
+ }
551
+ if (currentSessionContext) {
552
+ sessionContextParts.push(`<current_session>
553
+ ${currentSessionContext}
554
+ </current_session>`)
555
+ }
556
+ const sessionContextSection =
557
+ sessionContextParts.length > 0
558
+ ? `\n<session_context>
559
+ ${sessionContextParts.join('\n\n')}
560
+ </session_context>`
561
+ : ''
562
+
563
+ const transcriptionPrompt = `${languageHint}Transcribe this audio for a coding agent (like Claude Code or OpenCode).
564
+
565
+ CRITICAL REQUIREMENT: You MUST call the "transcriptionResult" tool to complete this task.
566
+ - The transcriptionResult tool is the ONLY way to return results
567
+ - Text responses are completely ignored - only tool calls work
568
+ - You MUST call transcriptionResult even if you run out of tool calls
569
+ - Always call transcriptionResult with your best approximation of what was said
570
+ - DO NOT end without calling transcriptionResult
571
+
572
+ This is a software development environment. The speaker is giving instructions to an AI coding assistant. Expect:
573
+ - File paths, function names, CLI commands, package names, API endpoints
574
+
575
+ RULES:
576
+ - NEVER change the meaning or intent of the user's message. Your job is ONLY to transcribe, not to respond or answer.
577
+ - If the user asks a question, keep it as a question. Do NOT answer it. Do NOT rephrase it as a statement.
578
+ - Only fix grammar, punctuation, and markdown formatting. Preserve the original content faithfully.
579
+ - If audio is unclear, transcribe your best interpretation, even with strong accents. Always provide an approximation.
580
+ - If audio seems silent/empty, is too short to understand, or is completely incomprehensible, call transcriptionResult with "[inaudible audio]"
581
+ - The session context below is ONLY for understanding technical terms, file names, and function names. It may contain previous transcriptions — NEVER copy or reuse them. Always transcribe fresh from the current audio.
582
+
583
+ QUEUE DETECTION:
584
+ - If the user says "queue this message", "queue this", "add this to the queue", or similar phrasing indicating they want the message queued instead of sent immediately, set queueMessage to true.
585
+ - Remove the queue instruction from the transcription text itself — only include the actual message content.
586
+ - Example: "Queue this message. Fix the login bug in auth.ts" → transcription: "Fix the login bug in auth.ts", queueMessage: true
587
+ - If removing the queue phrase would leave empty content (user only said "queue this" with nothing else), keep the full spoken text as the transcription — never return an empty transcription.
588
+ - If no queue intent is detected, omit queueMessage or set it to false.
589
+ ${agents && agents.length > 0 ? `
590
+ AGENT SELECTION:
591
+ - If the user explicitly says "use the X agent", "switch to X agent", "with the X agent", or similar phrasing naming a specific agent, set the agent field to that agent name.
592
+ - Remove the agent instruction from the transcription text itself — only include the actual message content.
593
+ - Example: "Use the plan agent. Refactor the auth module" → transcription: "Refactor the auth module", agent: "plan"
594
+ - If removing the agent phrase would leave empty content, keep the full spoken text as the transcription.
595
+ - Only set agent if the user explicitly names one. Do not infer an agent from the task content.
596
+ - If no agent is mentioned, omit the agent field entirely.
597
+
598
+ Available agents:
599
+ ${agents.map((a) => { return `- ${a.name}${a.description ? `: ${a.description}` : ''}` }).join('\n')}
600
+ ` : ''}
601
+
602
+ Common corrections (apply without tool calls):
603
+ - "reacked" → "React", "jason" → "JSON", "get hub" → "GitHub", "no JS" → "Node.js", "dacker" → "Docker"
604
+
605
+ Project file structure:
606
+ <file_tree>
607
+ ${prompt}
608
+ </file_tree>
609
+ ${sessionContextSection}
610
+
611
+ REMEMBER: Call "transcriptionResult" tool with your transcription. This is mandatory.
612
+
613
+ Note: "critique" is a CLI tool for showing diffs in the browser.`
614
+
615
+ const agentNames = agents
616
+ ?.map((a) => { return a.name })
617
+ .filter((name) => { return name.length > 0 })
618
+
619
+ return runTranscriptionOnce({
620
+ model: languageModel,
621
+ prompt: transcriptionPrompt,
622
+ audioBase64: finalAudioBase64,
623
+ mediaType,
624
+ temperature: temperature ?? 0.3,
625
+ agentNames: agentNames && agentNames.length > 0 ? agentNames : undefined,
626
+ })
627
+ }