@brianli/kimaki 0.4.72-brianli.1

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 (328) hide show
  1. package/bin.js +2 -0
  2. package/dist/ai-tool-to-genai.js +233 -0
  3. package/dist/ai-tool-to-genai.test.js +267 -0
  4. package/dist/ai-tool.js +6 -0
  5. package/dist/bin.js +87 -0
  6. package/dist/bot-token.js +121 -0
  7. package/dist/bot-token.test.js +134 -0
  8. package/dist/channel-management.js +101 -0
  9. package/dist/cli-parsing.test.js +89 -0
  10. package/dist/cli.js +2529 -0
  11. package/dist/commands/abort.js +82 -0
  12. package/dist/commands/action-buttons.js +257 -0
  13. package/dist/commands/add-project.js +114 -0
  14. package/dist/commands/agent.js +291 -0
  15. package/dist/commands/ask-question.js +223 -0
  16. package/dist/commands/compact.js +120 -0
  17. package/dist/commands/context-usage.js +140 -0
  18. package/dist/commands/create-new-project.js +118 -0
  19. package/dist/commands/diff.js +128 -0
  20. package/dist/commands/file-upload.js +275 -0
  21. package/dist/commands/fork.js +217 -0
  22. package/dist/commands/gemini-apikey.js +70 -0
  23. package/dist/commands/login.js +490 -0
  24. package/dist/commands/mention-mode.js +51 -0
  25. package/dist/commands/merge-worktree.js +124 -0
  26. package/dist/commands/model.js +694 -0
  27. package/dist/commands/permissions.js +163 -0
  28. package/dist/commands/queue.js +217 -0
  29. package/dist/commands/remove-project.js +115 -0
  30. package/dist/commands/restart-opencode-server.js +116 -0
  31. package/dist/commands/resume.js +159 -0
  32. package/dist/commands/run-command.js +79 -0
  33. package/dist/commands/session-id.js +78 -0
  34. package/dist/commands/session.js +192 -0
  35. package/dist/commands/share.js +80 -0
  36. package/dist/commands/types.js +2 -0
  37. package/dist/commands/undo-redo.js +159 -0
  38. package/dist/commands/unset-model.js +152 -0
  39. package/dist/commands/upgrade.js +42 -0
  40. package/dist/commands/user-command.js +148 -0
  41. package/dist/commands/verbosity.js +60 -0
  42. package/dist/commands/worktree-settings.js +50 -0
  43. package/dist/commands/worktree.js +299 -0
  44. package/dist/condense-memory.js +33 -0
  45. package/dist/config.js +110 -0
  46. package/dist/database.js +1050 -0
  47. package/dist/db.js +159 -0
  48. package/dist/db.test.js +49 -0
  49. package/dist/discord-api.js +28 -0
  50. package/dist/discord-auth.js +231 -0
  51. package/dist/discord-auth.test.js +80 -0
  52. package/dist/discord-bot.js +997 -0
  53. package/dist/discord-utils.js +560 -0
  54. package/dist/discord-utils.test.js +115 -0
  55. package/dist/errors.js +167 -0
  56. package/dist/escape-backticks.test.js +429 -0
  57. package/dist/format-tables.js +122 -0
  58. package/dist/format-tables.test.js +199 -0
  59. package/dist/forum-sync/config.js +79 -0
  60. package/dist/forum-sync/discord-operations.js +154 -0
  61. package/dist/forum-sync/index.js +5 -0
  62. package/dist/forum-sync/markdown.js +117 -0
  63. package/dist/forum-sync/sync-to-discord.js +417 -0
  64. package/dist/forum-sync/sync-to-files.js +190 -0
  65. package/dist/forum-sync/types.js +53 -0
  66. package/dist/forum-sync/watchers.js +307 -0
  67. package/dist/gateway-consumer.js +232 -0
  68. package/dist/gateway-consumer.test.js +18 -0
  69. package/dist/genai-worker-wrapper.js +111 -0
  70. package/dist/genai-worker.js +311 -0
  71. package/dist/genai.js +232 -0
  72. package/dist/generated/browser.js +17 -0
  73. package/dist/generated/client.js +35 -0
  74. package/dist/generated/commonInputTypes.js +10 -0
  75. package/dist/generated/enums.js +30 -0
  76. package/dist/generated/internal/class.js +41 -0
  77. package/dist/generated/internal/prismaNamespace.js +239 -0
  78. package/dist/generated/internal/prismaNamespaceBrowser.js +209 -0
  79. package/dist/generated/models/bot_api_keys.js +1 -0
  80. package/dist/generated/models/bot_tokens.js +1 -0
  81. package/dist/generated/models/channel_agents.js +1 -0
  82. package/dist/generated/models/channel_directories.js +1 -0
  83. package/dist/generated/models/channel_mention_mode.js +1 -0
  84. package/dist/generated/models/channel_models.js +1 -0
  85. package/dist/generated/models/channel_verbosity.js +1 -0
  86. package/dist/generated/models/channel_worktrees.js +1 -0
  87. package/dist/generated/models/forum_sync_configs.js +1 -0
  88. package/dist/generated/models/global_models.js +1 -0
  89. package/dist/generated/models/ipc_requests.js +1 -0
  90. package/dist/generated/models/part_messages.js +1 -0
  91. package/dist/generated/models/scheduled_tasks.js +1 -0
  92. package/dist/generated/models/session_agents.js +1 -0
  93. package/dist/generated/models/session_models.js +1 -0
  94. package/dist/generated/models/session_start_sources.js +1 -0
  95. package/dist/generated/models/thread_sessions.js +1 -0
  96. package/dist/generated/models/thread_worktrees.js +1 -0
  97. package/dist/generated/models.js +1 -0
  98. package/dist/heap-monitor.js +95 -0
  99. package/dist/hrana-server.js +416 -0
  100. package/dist/hrana-server.test.js +368 -0
  101. package/dist/image-utils.js +112 -0
  102. package/dist/interaction-handler.js +327 -0
  103. package/dist/ipc-polling.js +251 -0
  104. package/dist/kimaki-digital-twin.e2e.test.js +165 -0
  105. package/dist/limit-heading-depth.js +25 -0
  106. package/dist/limit-heading-depth.test.js +105 -0
  107. package/dist/logger.js +160 -0
  108. package/dist/markdown.js +342 -0
  109. package/dist/markdown.test.js +253 -0
  110. package/dist/message-formatting.js +433 -0
  111. package/dist/message-formatting.test.js +73 -0
  112. package/dist/openai-realtime.js +228 -0
  113. package/dist/opencode-plugin-loading.e2e.test.js +91 -0
  114. package/dist/opencode-plugin.js +536 -0
  115. package/dist/opencode-plugin.test.js +98 -0
  116. package/dist/opencode.js +409 -0
  117. package/dist/privacy-sanitizer.js +105 -0
  118. package/dist/runtime-mode.js +51 -0
  119. package/dist/runtime-mode.test.js +115 -0
  120. package/dist/sentry.js +127 -0
  121. package/dist/session-handler/state.js +151 -0
  122. package/dist/session-handler.js +1874 -0
  123. package/dist/session-search.js +100 -0
  124. package/dist/session-search.test.js +40 -0
  125. package/dist/startup-service.js +153 -0
  126. package/dist/system-message.js +499 -0
  127. package/dist/task-runner.js +282 -0
  128. package/dist/task-schedule.js +191 -0
  129. package/dist/task-schedule.test.js +71 -0
  130. package/dist/thinking-utils.js +35 -0
  131. package/dist/thread-message-queue.e2e.test.js +781 -0
  132. package/dist/tools.js +359 -0
  133. package/dist/unnest-code-blocks.js +136 -0
  134. package/dist/unnest-code-blocks.test.js +641 -0
  135. package/dist/upgrade.js +114 -0
  136. package/dist/utils.js +109 -0
  137. package/dist/voice-handler.js +606 -0
  138. package/dist/voice.js +304 -0
  139. package/dist/voice.test.js +187 -0
  140. package/dist/wait-session.js +94 -0
  141. package/dist/worker-types.js +4 -0
  142. package/dist/worktree-utils.js +727 -0
  143. package/dist/xml.js +92 -0
  144. package/dist/xml.test.js +32 -0
  145. package/package.json +82 -0
  146. package/schema.prisma +246 -0
  147. package/skills/batch/SKILL.md +87 -0
  148. package/skills/critique/SKILL.md +129 -0
  149. package/skills/errore/SKILL.md +589 -0
  150. package/skills/goke/.prettierrc +5 -0
  151. package/skills/goke/CHANGELOG.md +40 -0
  152. package/skills/goke/LICENSE +21 -0
  153. package/skills/goke/README.md +666 -0
  154. package/skills/goke/SKILL.md +458 -0
  155. package/skills/goke/package.json +43 -0
  156. package/skills/goke/src/__test__/coerce.test.ts +411 -0
  157. package/skills/goke/src/__test__/index.test.ts +1798 -0
  158. package/skills/goke/src/__test__/types.test-d.ts +111 -0
  159. package/skills/goke/src/coerce.ts +547 -0
  160. package/skills/goke/src/goke.ts +1362 -0
  161. package/skills/goke/src/index.ts +16 -0
  162. package/skills/goke/src/mri.ts +164 -0
  163. package/skills/goke/tsconfig.json +15 -0
  164. package/skills/jitter/EDITOR.md +219 -0
  165. package/skills/jitter/EXPORT-INTERNALS.md +309 -0
  166. package/skills/jitter/SKILL.md +158 -0
  167. package/skills/jitter/jitter-clipboard.json +1042 -0
  168. package/skills/jitter/package.json +14 -0
  169. package/skills/jitter/tsconfig.json +15 -0
  170. package/skills/jitter/utils/actions.ts +212 -0
  171. package/skills/jitter/utils/export.ts +114 -0
  172. package/skills/jitter/utils/index.ts +141 -0
  173. package/skills/jitter/utils/snapshot.ts +154 -0
  174. package/skills/jitter/utils/traverse.ts +246 -0
  175. package/skills/jitter/utils/types.ts +279 -0
  176. package/skills/jitter/utils/wait.ts +133 -0
  177. package/skills/playwriter/SKILL.md +31 -0
  178. package/skills/security-review/SKILL.md +208 -0
  179. package/skills/simplify/SKILL.md +58 -0
  180. package/skills/termcast/SKILL.md +945 -0
  181. package/skills/tuistory/SKILL.md +250 -0
  182. package/skills/zustand-centralized-state/SKILL.md +582 -0
  183. package/src/__snapshots__/compact-session-context-no-system.md +35 -0
  184. package/src/__snapshots__/compact-session-context.md +41 -0
  185. package/src/__snapshots__/first-session-no-info.md +17 -0
  186. package/src/__snapshots__/first-session-with-info.md +23 -0
  187. package/src/__snapshots__/session-1.md +17 -0
  188. package/src/__snapshots__/session-2.md +5871 -0
  189. package/src/__snapshots__/session-3.md +17 -0
  190. package/src/__snapshots__/session-with-tools.md +5871 -0
  191. package/src/ai-tool-to-genai.test.ts +296 -0
  192. package/src/ai-tool-to-genai.ts +282 -0
  193. package/src/ai-tool.ts +39 -0
  194. package/src/bin.ts +108 -0
  195. package/src/bot-token.test.ts +171 -0
  196. package/src/bot-token.ts +159 -0
  197. package/src/channel-management.ts +172 -0
  198. package/src/cli-parsing.test.ts +132 -0
  199. package/src/cli.ts +3605 -0
  200. package/src/commands/abort.ts +112 -0
  201. package/src/commands/action-buttons.ts +376 -0
  202. package/src/commands/add-project.ts +152 -0
  203. package/src/commands/agent.ts +404 -0
  204. package/src/commands/ask-question.ts +330 -0
  205. package/src/commands/compact.ts +157 -0
  206. package/src/commands/context-usage.ts +199 -0
  207. package/src/commands/create-new-project.ts +179 -0
  208. package/src/commands/diff.ts +165 -0
  209. package/src/commands/file-upload.ts +389 -0
  210. package/src/commands/fork.ts +320 -0
  211. package/src/commands/gemini-apikey.ts +104 -0
  212. package/src/commands/login.ts +634 -0
  213. package/src/commands/mention-mode.ts +77 -0
  214. package/src/commands/merge-worktree.ts +177 -0
  215. package/src/commands/model.ts +961 -0
  216. package/src/commands/permissions.ts +261 -0
  217. package/src/commands/queue.ts +296 -0
  218. package/src/commands/remove-project.ts +155 -0
  219. package/src/commands/restart-opencode-server.ts +162 -0
  220. package/src/commands/resume.ts +242 -0
  221. package/src/commands/run-command.ts +123 -0
  222. package/src/commands/session-id.ts +109 -0
  223. package/src/commands/session.ts +250 -0
  224. package/src/commands/share.ts +106 -0
  225. package/src/commands/types.ts +25 -0
  226. package/src/commands/undo-redo.ts +221 -0
  227. package/src/commands/unset-model.ts +189 -0
  228. package/src/commands/upgrade.ts +52 -0
  229. package/src/commands/user-command.ts +193 -0
  230. package/src/commands/verbosity.ts +88 -0
  231. package/src/commands/worktree-settings.ts +79 -0
  232. package/src/commands/worktree.ts +431 -0
  233. package/src/condense-memory.ts +36 -0
  234. package/src/config.ts +148 -0
  235. package/src/database.ts +1530 -0
  236. package/src/db.test.ts +60 -0
  237. package/src/db.ts +190 -0
  238. package/src/discord-api.ts +35 -0
  239. package/src/discord-bot.ts +1316 -0
  240. package/src/discord-utils.test.ts +132 -0
  241. package/src/discord-utils.ts +767 -0
  242. package/src/errors.ts +213 -0
  243. package/src/escape-backticks.test.ts +469 -0
  244. package/src/format-tables.test.ts +223 -0
  245. package/src/format-tables.ts +145 -0
  246. package/src/forum-sync/config.ts +92 -0
  247. package/src/forum-sync/discord-operations.ts +241 -0
  248. package/src/forum-sync/index.ts +9 -0
  249. package/src/forum-sync/markdown.ts +176 -0
  250. package/src/forum-sync/sync-to-discord.ts +595 -0
  251. package/src/forum-sync/sync-to-files.ts +294 -0
  252. package/src/forum-sync/types.ts +175 -0
  253. package/src/forum-sync/watchers.ts +454 -0
  254. package/src/genai-worker-wrapper.ts +164 -0
  255. package/src/genai-worker.ts +386 -0
  256. package/src/genai.ts +321 -0
  257. package/src/generated/browser.ts +109 -0
  258. package/src/generated/client.ts +131 -0
  259. package/src/generated/commonInputTypes.ts +512 -0
  260. package/src/generated/enums.ts +46 -0
  261. package/src/generated/internal/class.ts +362 -0
  262. package/src/generated/internal/prismaNamespace.ts +2251 -0
  263. package/src/generated/internal/prismaNamespaceBrowser.ts +308 -0
  264. package/src/generated/models/bot_api_keys.ts +1288 -0
  265. package/src/generated/models/bot_tokens.ts +1577 -0
  266. package/src/generated/models/channel_agents.ts +1256 -0
  267. package/src/generated/models/channel_directories.ts +2104 -0
  268. package/src/generated/models/channel_mention_mode.ts +1300 -0
  269. package/src/generated/models/channel_models.ts +1288 -0
  270. package/src/generated/models/channel_verbosity.ts +1224 -0
  271. package/src/generated/models/channel_worktrees.ts +1308 -0
  272. package/src/generated/models/forum_sync_configs.ts +1452 -0
  273. package/src/generated/models/global_models.ts +1288 -0
  274. package/src/generated/models/ipc_requests.ts +1485 -0
  275. package/src/generated/models/part_messages.ts +1302 -0
  276. package/src/generated/models/scheduled_tasks.ts +2320 -0
  277. package/src/generated/models/session_agents.ts +1086 -0
  278. package/src/generated/models/session_models.ts +1114 -0
  279. package/src/generated/models/session_start_sources.ts +1408 -0
  280. package/src/generated/models/thread_sessions.ts +1599 -0
  281. package/src/generated/models/thread_worktrees.ts +1352 -0
  282. package/src/generated/models.ts +29 -0
  283. package/src/heap-monitor.ts +121 -0
  284. package/src/hrana-server.test.ts +428 -0
  285. package/src/hrana-server.ts +547 -0
  286. package/src/image-utils.ts +149 -0
  287. package/src/interaction-handler.ts +461 -0
  288. package/src/ipc-polling.ts +325 -0
  289. package/src/kimaki-digital-twin.e2e.test.ts +201 -0
  290. package/src/limit-heading-depth.test.ts +116 -0
  291. package/src/limit-heading-depth.ts +26 -0
  292. package/src/logger.ts +203 -0
  293. package/src/markdown.test.ts +360 -0
  294. package/src/markdown.ts +410 -0
  295. package/src/message-formatting.test.ts +81 -0
  296. package/src/message-formatting.ts +549 -0
  297. package/src/openai-realtime.ts +362 -0
  298. package/src/opencode-plugin-loading.e2e.test.ts +112 -0
  299. package/src/opencode-plugin.test.ts +108 -0
  300. package/src/opencode-plugin.ts +652 -0
  301. package/src/opencode.ts +554 -0
  302. package/src/privacy-sanitizer.ts +142 -0
  303. package/src/schema.sql +158 -0
  304. package/src/sentry.ts +137 -0
  305. package/src/session-handler/state.ts +232 -0
  306. package/src/session-handler.ts +2668 -0
  307. package/src/session-search.test.ts +50 -0
  308. package/src/session-search.ts +148 -0
  309. package/src/startup-service.ts +200 -0
  310. package/src/system-message.ts +568 -0
  311. package/src/task-runner.ts +425 -0
  312. package/src/task-schedule.test.ts +84 -0
  313. package/src/task-schedule.ts +287 -0
  314. package/src/thinking-utils.ts +61 -0
  315. package/src/thread-message-queue.e2e.test.ts +997 -0
  316. package/src/tools.ts +432 -0
  317. package/src/unnest-code-blocks.test.ts +679 -0
  318. package/src/unnest-code-blocks.ts +168 -0
  319. package/src/upgrade.ts +127 -0
  320. package/src/utils.ts +145 -0
  321. package/src/voice-handler.ts +852 -0
  322. package/src/voice.test.ts +219 -0
  323. package/src/voice.ts +444 -0
  324. package/src/wait-session.ts +147 -0
  325. package/src/worker-types.ts +64 -0
  326. package/src/worktree-utils.ts +988 -0
  327. package/src/xml.test.ts +38 -0
  328. package/src/xml.ts +121 -0
@@ -0,0 +1,219 @@
1
+ // Tests for voice transcription using AI SDK provider (LanguageModelV3).
2
+ // Uses the example audio files at scripts/example-audio.{mp3,ogg}.
3
+
4
+ import { describe, test, expect } from 'vitest'
5
+ import fs from 'node:fs'
6
+ import path from 'node:path'
7
+ import { transcribeAudio, convertOggToWav } from './voice.js'
8
+ import { extractTranscription } from './voice.js'
9
+
10
+ describe('extractTranscription', () => {
11
+ test('extracts transcription from tool call', () => {
12
+ const result = extractTranscription([
13
+ {
14
+ type: 'tool-call',
15
+ toolCallId: 'call_1',
16
+ toolName: 'transcriptionResult',
17
+ input: JSON.stringify({ transcription: 'hello world' }),
18
+ },
19
+ ])
20
+ expect(result).toMatchInlineSnapshot(`
21
+ {
22
+ "queueMessage": false,
23
+ "transcription": "hello world",
24
+ }
25
+ `)
26
+ })
27
+
28
+ test('extracts queueMessage: true from tool call', () => {
29
+ const result = extractTranscription([
30
+ {
31
+ type: 'tool-call',
32
+ toolCallId: 'call_1',
33
+ toolName: 'transcriptionResult',
34
+ input: JSON.stringify({
35
+ transcription: 'Fix the login bug in auth.ts',
36
+ queueMessage: true,
37
+ }),
38
+ },
39
+ ])
40
+ expect(result).toMatchInlineSnapshot(`
41
+ {
42
+ "queueMessage": true,
43
+ "transcription": "Fix the login bug in auth.ts",
44
+ }
45
+ `)
46
+ })
47
+
48
+ test('queueMessage defaults to false when omitted', () => {
49
+ const result = extractTranscription([
50
+ {
51
+ type: 'tool-call',
52
+ toolCallId: 'call_1',
53
+ toolName: 'transcriptionResult',
54
+ input: JSON.stringify({ transcription: 'regular message' }),
55
+ },
56
+ ])
57
+ expect(result).not.toBeInstanceOf(Error)
58
+ expect((result as { queueMessage: boolean }).queueMessage).toBe(false)
59
+ })
60
+
61
+ test('falls back to text when no tool call', () => {
62
+ const result = extractTranscription([
63
+ {
64
+ type: 'text',
65
+ text: 'fallback text response',
66
+ },
67
+ ])
68
+ expect(result).toMatchInlineSnapshot(`
69
+ {
70
+ "queueMessage": false,
71
+ "transcription": "fallback text response",
72
+ }
73
+ `)
74
+ })
75
+
76
+ test('returns NoResponseContentError for empty content', () => {
77
+ const result = extractTranscription([])
78
+ expect(result).toBeInstanceOf(Error)
79
+ expect((result as Error).message).toMatchInlineSnapshot(
80
+ `"No response content from model"`,
81
+ )
82
+ })
83
+
84
+ test('returns EmptyTranscriptionError for empty transcription string', () => {
85
+ const result = extractTranscription([
86
+ {
87
+ type: 'tool-call',
88
+ toolCallId: 'call_1',
89
+ toolName: 'transcriptionResult',
90
+ input: JSON.stringify({ transcription: ' ' }),
91
+ },
92
+ ])
93
+ expect(result).toBeInstanceOf(Error)
94
+ expect((result as Error).message).toMatchInlineSnapshot(
95
+ `"Model returned empty transcription"`,
96
+ )
97
+ })
98
+
99
+ test('returns TranscriptionError when content has no tool call or text', () => {
100
+ const result = extractTranscription([
101
+ {
102
+ type: 'reasoning',
103
+ text: 'thinking about it',
104
+ },
105
+ ])
106
+ expect(result).toBeInstanceOf(Error)
107
+ expect((result as Error).message).toMatchInlineSnapshot(
108
+ `"Transcription failed: Model did not produce a transcription"`,
109
+ )
110
+ })
111
+ })
112
+
113
+ describe('transcribeAudio with real API', () => {
114
+ const audioPath = path.join(
115
+ import.meta.dirname,
116
+ '..',
117
+ 'scripts',
118
+ 'example-audio.mp3',
119
+ )
120
+
121
+ test('transcribes with Gemini', { timeout: 30_000 }, async () => {
122
+ const apiKey = process.env.GEMINI_API_KEY
123
+ if (!apiKey) {
124
+ console.log('Skipping: GEMINI_API_KEY not set')
125
+ return
126
+ }
127
+ if (!fs.existsSync(audioPath)) {
128
+ console.log('Skipping: example-audio.mp3 not found')
129
+ return
130
+ }
131
+
132
+ const audio = fs.readFileSync(audioPath)
133
+ const result = await transcribeAudio({
134
+ audio,
135
+ prompt: 'test project',
136
+ apiKey,
137
+ provider: 'gemini',
138
+ })
139
+
140
+ expect(result).not.toBeInstanceOf(Error)
141
+ const { transcription } = result as { transcription: string }
142
+ expect(transcription.length).toBeGreaterThan(0)
143
+ console.log('Gemini transcription:', result)
144
+ })
145
+
146
+ test('transcribes with OpenAI', { timeout: 30_000 }, async () => {
147
+ const apiKey = process.env.OPENAI_API_KEY
148
+ if (!apiKey) {
149
+ console.log('Skipping: OPENAI_API_KEY not set')
150
+ return
151
+ }
152
+ if (!fs.existsSync(audioPath)) {
153
+ console.log('Skipping: example-audio.mp3 not found')
154
+ return
155
+ }
156
+
157
+ const audio = fs.readFileSync(audioPath)
158
+ const result = await transcribeAudio({
159
+ audio,
160
+ prompt: 'test project',
161
+ apiKey,
162
+ provider: 'openai',
163
+ })
164
+
165
+ expect(result).not.toBeInstanceOf(Error)
166
+ const { transcription } = result as { transcription: string }
167
+ expect(transcription.length).toBeGreaterThan(0)
168
+ console.log('OpenAI transcription:', result)
169
+ })
170
+
171
+ test('transcribes OGG with OpenAI (converts to WAV)', { timeout: 30_000 }, async () => {
172
+ const apiKey = process.env.OPENAI_API_KEY
173
+ const oggPath = path.join(import.meta.dirname, '..', 'scripts', 'example-audio.ogg')
174
+ if (!apiKey) {
175
+ console.log('Skipping: OPENAI_API_KEY not set')
176
+ return
177
+ }
178
+ if (!fs.existsSync(oggPath)) {
179
+ console.log('Skipping: example-audio.ogg not found')
180
+ return
181
+ }
182
+
183
+ const audio = fs.readFileSync(oggPath)
184
+ const result = await transcribeAudio({
185
+ audio,
186
+ prompt: 'test project',
187
+ apiKey,
188
+ provider: 'openai',
189
+ mediaType: 'audio/ogg',
190
+ })
191
+
192
+ expect(result).not.toBeInstanceOf(Error)
193
+ const { transcription } = result as { transcription: string }
194
+ expect(transcription.length).toBeGreaterThan(0)
195
+ console.log('OpenAI OGG transcription:', result)
196
+ })
197
+ })
198
+
199
+ describe('convertOggToWav', () => {
200
+ test('converts OGG Opus to valid WAV', async () => {
201
+ const oggPath = path.join(import.meta.dirname, '..', 'scripts', 'example-audio.ogg')
202
+ if (!fs.existsSync(oggPath)) {
203
+ console.log('Skipping: example-audio.ogg not found')
204
+ return
205
+ }
206
+
207
+ const ogg = fs.readFileSync(oggPath)
208
+ const result = await convertOggToWav(ogg)
209
+ expect(result).toBeInstanceOf(Buffer)
210
+
211
+ const wav = result as Buffer
212
+ // WAV header starts with RIFF
213
+ expect(wav.subarray(0, 4).toString()).toBe('RIFF')
214
+ expect(wav.subarray(8, 12).toString()).toBe('WAVE')
215
+ // Must be larger than just the header (44 bytes)
216
+ expect(wav.length).toBeGreaterThan(44)
217
+ console.log(`Converted OGG (${ogg.length} bytes) to WAV (${wav.length} bytes)`)
218
+ })
219
+ })
package/src/voice.ts ADDED
@@ -0,0 +1,444 @@
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
+ ])
42
+
43
+ /**
44
+ * Convert OGG Opus audio to WAV using prism-media (already installed for Discord voice).
45
+ * Pipeline: OGG buffer → OggDemuxer → Opus Decoder → PCM → WAV (with header).
46
+ * No ffmpeg needed — uses @discordjs/opus native bindings.
47
+ */
48
+ export function convertOggToWav(input: Buffer): Promise<TranscriptionError | Buffer> {
49
+ return new Promise((resolve) => {
50
+ const pcmChunks: Buffer[] = []
51
+
52
+ const demuxer = new prism.opus.OggDemuxer()
53
+ const decoder = new prism.opus.Decoder({
54
+ rate: 48000,
55
+ channels: 1,
56
+ frameSize: 960,
57
+ })
58
+
59
+ decoder.on('data', (chunk: Buffer) => {
60
+ pcmChunks.push(chunk)
61
+ })
62
+
63
+ decoder.on('end', () => {
64
+ const pcmData = Buffer.concat(pcmChunks)
65
+ const wavHeader = createWavHeader({
66
+ dataLength: pcmData.length,
67
+ sampleRate: 48000,
68
+ numChannels: 1,
69
+ bitsPerSample: 16,
70
+ })
71
+ resolve(Buffer.concat([wavHeader, pcmData]))
72
+ })
73
+
74
+ decoder.on('error', (err: Error) => {
75
+ resolve(
76
+ new TranscriptionError({
77
+ reason: `Opus decode failed: ${err.message}`,
78
+ cause: err,
79
+ }),
80
+ )
81
+ })
82
+
83
+ demuxer.on('error', (err: Error) => {
84
+ resolve(
85
+ new TranscriptionError({
86
+ reason: `OGG demux failed: ${err.message}`,
87
+ cause: err,
88
+ }),
89
+ )
90
+ })
91
+
92
+ Readable.from(input).pipe(demuxer).pipe(decoder)
93
+ })
94
+ }
95
+
96
+ function createWavHeader({
97
+ dataLength,
98
+ sampleRate,
99
+ numChannels,
100
+ bitsPerSample,
101
+ }: {
102
+ dataLength: number
103
+ sampleRate: number
104
+ numChannels: number
105
+ bitsPerSample: number
106
+ }): Buffer {
107
+ const byteRate = (sampleRate * numChannels * bitsPerSample) / 8
108
+ const blockAlign = (numChannels * bitsPerSample) / 8
109
+ const buffer = Buffer.alloc(44)
110
+ buffer.write('RIFF', 0)
111
+ buffer.writeUInt32LE(36 + dataLength, 4)
112
+ buffer.write('WAVE', 8)
113
+ buffer.write('fmt ', 12)
114
+ buffer.writeUInt32LE(16, 16)
115
+ buffer.writeUInt16LE(1, 20)
116
+ buffer.writeUInt16LE(numChannels, 22)
117
+ buffer.writeUInt32LE(sampleRate, 24)
118
+ buffer.writeUInt32LE(byteRate, 28)
119
+ buffer.writeUInt16LE(blockAlign, 32)
120
+ buffer.writeUInt16LE(bitsPerSample, 34)
121
+ buffer.write('data', 36)
122
+ buffer.writeUInt32LE(dataLength, 40)
123
+ return buffer
124
+ }
125
+
126
+ type TranscriptionLoopError =
127
+ | NoResponseContentError
128
+ | TranscriptionError
129
+ | EmptyTranscriptionError
130
+ | NoToolResponseError
131
+
132
+ const transcriptionTool: LanguageModelV3FunctionTool = {
133
+ type: 'function',
134
+ name: 'transcriptionResult',
135
+ description:
136
+ '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.',
137
+ inputSchema: {
138
+ type: 'object',
139
+ properties: {
140
+ transcription: {
141
+ type: 'string',
142
+ description:
143
+ 'The final transcription of the audio. MUST be non-empty. If audio is unclear, transcribe your best interpretation. If silent, use "[inaudible audio]".',
144
+ },
145
+ queueMessage: {
146
+ type: 'boolean',
147
+ description:
148
+ '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.',
149
+ },
150
+ },
151
+ required: ['transcription'],
152
+ },
153
+ }
154
+
155
+ export type TranscriptionResult = {
156
+ transcription: string
157
+ queueMessage: boolean
158
+ }
159
+
160
+ /**
161
+ * Extract transcription result from doGenerate content array.
162
+ * Looks for a tool-call named 'transcriptionResult', falls back to text content.
163
+ * Returns structured result with transcription text and queueMessage flag.
164
+ */
165
+ export function extractTranscription(
166
+ content: Array<LanguageModelV3Content>,
167
+ ): TranscriptionLoopError | TranscriptionResult {
168
+ const toolCall = content.find(
169
+ (c): c is LanguageModelV3ToolCall =>
170
+ c.type === 'tool-call' && c.toolName === 'transcriptionResult',
171
+ )
172
+
173
+ if (toolCall) {
174
+ // toolCall.input is a JSON string in LanguageModelV3
175
+ const args: Record<string, unknown> = (() => {
176
+ if (typeof toolCall.input === 'string') {
177
+ return JSON.parse(toolCall.input) as Record<string, unknown>
178
+ }
179
+ return {}
180
+ })()
181
+ const transcription = (typeof args.transcription === 'string' ? args.transcription : '').trim()
182
+ const queueMessage = args.queueMessage === true
183
+ voiceLogger.log(
184
+ `Transcription result received: "${transcription.slice(0, 100)}..."${queueMessage ? ' [QUEUE]' : ''}`,
185
+ )
186
+ if (!transcription) {
187
+ return new EmptyTranscriptionError()
188
+ }
189
+ return { transcription, queueMessage }
190
+ }
191
+
192
+ // Fall back to text content if no tool call
193
+ const textPart = content.find((c) => c.type === 'text')
194
+ if (textPart && textPart.type === 'text' && textPart.text.trim()) {
195
+ voiceLogger.log(
196
+ `No tool call but got text: "${textPart.text.trim().slice(0, 100)}..."`,
197
+ )
198
+ return { transcription: textPart.text.trim(), queueMessage: false }
199
+ }
200
+
201
+ if (content.length === 0) {
202
+ return new NoResponseContentError()
203
+ }
204
+
205
+ return new TranscriptionError({
206
+ reason: 'Model did not produce a transcription',
207
+ })
208
+ }
209
+
210
+ async function runTranscriptionOnce({
211
+ model,
212
+ prompt,
213
+ audioBase64,
214
+ mediaType,
215
+ temperature,
216
+ }: {
217
+ model: LanguageModelV3
218
+ prompt: string
219
+ audioBase64: string
220
+ mediaType: string
221
+ temperature: number
222
+ }): Promise<TranscriptionLoopError | TranscriptionResult> {
223
+ const options: LanguageModelV3CallOptions = {
224
+ prompt: [
225
+ {
226
+ role: 'user',
227
+ content: [
228
+ { type: 'text', text: prompt },
229
+ {
230
+ type: 'file',
231
+ data: audioBase64,
232
+ mediaType,
233
+ },
234
+ ],
235
+ },
236
+ ],
237
+ temperature,
238
+ maxOutputTokens: 2048,
239
+ tools: [transcriptionTool],
240
+ toolChoice: { type: 'tool', toolName: 'transcriptionResult' },
241
+ providerOptions: {
242
+ google: {
243
+ thinkingConfig: { thinkingBudget: 1024 },
244
+ },
245
+ },
246
+ }
247
+
248
+ // doGenerate returns PromiseLike, wrap in Promise.resolve for errore compatibility
249
+ const response = await errore.tryAsync({
250
+ try: () => Promise.resolve(model.doGenerate(options)),
251
+ catch: (e: Error) =>
252
+ new TranscriptionError({
253
+ reason: `API call failed: ${String(e)}`,
254
+ cause: e,
255
+ }),
256
+ })
257
+
258
+ if (response instanceof TranscriptionError) {
259
+ return response
260
+ }
261
+
262
+ return extractTranscription(response.content)
263
+ }
264
+
265
+ export type TranscribeAudioErrors =
266
+ | ApiKeyMissingError
267
+ | InvalidAudioFormatError
268
+ | TranscriptionLoopError
269
+
270
+ export type TranscriptionProvider = 'openai' | 'gemini'
271
+
272
+ /**
273
+ * Create a LanguageModelV3 for transcription.
274
+ * Both providers use chat models that accept audio file parts, so we get full
275
+ * context (prompt, session info, tool calling) for better word recognition.
276
+ *
277
+ * OpenAI: must use .chat() to get the Chat Completions API model, because the
278
+ * default callable (Responses API) doesn't support audio file parts.
279
+ * Gemini: language models natively accept audio in chat.
280
+ */
281
+ export function createTranscriptionModel({
282
+ apiKey,
283
+ provider,
284
+ }: {
285
+ apiKey: string
286
+ provider?: TranscriptionProvider
287
+ }): LanguageModelV3 {
288
+ const resolvedProvider: TranscriptionProvider =
289
+ provider || (apiKey.startsWith('sk-') ? 'openai' : 'gemini')
290
+
291
+ if (resolvedProvider === 'openai') {
292
+ const openai = createOpenAI({ apiKey })
293
+ return openai.chat('gpt-4o-audio-preview')
294
+ }
295
+
296
+ const google = createGoogleGenerativeAI({ apiKey })
297
+ return google('gemini-2.5-flash')
298
+ }
299
+
300
+ export async function transcribeAudio({
301
+ audio,
302
+ prompt,
303
+ language,
304
+ temperature,
305
+ apiKey: apiKeyParam,
306
+ model,
307
+ provider,
308
+ mediaType: mediaTypeParam,
309
+ currentSessionContext,
310
+ lastSessionContext,
311
+ }: {
312
+ audio: Buffer | Uint8Array | ArrayBuffer | string
313
+ prompt?: string
314
+ language?: string
315
+ temperature?: number
316
+ apiKey?: string
317
+ model?: LanguageModelV3
318
+ provider?: TranscriptionProvider
319
+ /** MIME type of the audio data (e.g. 'audio/ogg'). Defaults to 'audio/mpeg'. */
320
+ mediaType?: string
321
+ currentSessionContext?: string
322
+ lastSessionContext?: string
323
+ }): Promise<TranscribeAudioErrors | TranscriptionResult> {
324
+ const apiKey =
325
+ apiKeyParam || process.env.OPENAI_API_KEY || process.env.GEMINI_API_KEY
326
+
327
+ if (!model && !apiKey) {
328
+ return Promise.resolve(new ApiKeyMissingError({ service: 'OpenAI or Gemini' }))
329
+ }
330
+
331
+ const resolvedProvider: TranscriptionProvider = (() => {
332
+ if (provider) {
333
+ return provider
334
+ }
335
+ if (apiKey) {
336
+ return apiKey.startsWith('sk-') ? 'openai' : 'gemini'
337
+ }
338
+ return 'gemini'
339
+ })()
340
+
341
+ const languageModel: LanguageModelV3 =
342
+ model || createTranscriptionModel({ apiKey: apiKey!, provider: resolvedProvider })
343
+
344
+ // Convert audio to Buffer for potential format conversion
345
+ const audioBuffer: Buffer = (() => {
346
+ if (typeof audio === 'string') {
347
+ return Buffer.from(audio, 'base64')
348
+ }
349
+ if (audio instanceof Buffer) {
350
+ return audio
351
+ }
352
+ if (audio instanceof ArrayBuffer) {
353
+ return Buffer.from(new Uint8Array(audio))
354
+ }
355
+ return Buffer.from(audio)
356
+ })()
357
+
358
+ if (audioBuffer.length === 0) {
359
+ return new InvalidAudioFormatError()
360
+ }
361
+
362
+ let mediaType = mediaTypeParam || 'audio/mpeg'
363
+ let finalAudioBase64 = audioBuffer.toString('base64')
364
+
365
+ // OpenAI input_audio only supports mp3/wav. Convert OGG Opus (Discord voice) to WAV.
366
+ if (resolvedProvider === 'openai' && !OPENAI_SUPPORTED_AUDIO_TYPES.has(mediaType)) {
367
+ voiceLogger.log(`Converting ${mediaType} to WAV for OpenAI compatibility`)
368
+ const converted = await convertOggToWav(audioBuffer)
369
+ if (converted instanceof Error) {
370
+ return converted
371
+ }
372
+ finalAudioBase64 = converted.toString('base64')
373
+ mediaType = 'audio/wav'
374
+ }
375
+
376
+ const languageHint = language ? `The audio is in ${language}.\n\n` : ''
377
+
378
+ // build session context section
379
+ const sessionContextParts: string[] = []
380
+ if (lastSessionContext) {
381
+ sessionContextParts.push(`<last_session>
382
+ ${lastSessionContext}
383
+ </last_session>`)
384
+ }
385
+ if (currentSessionContext) {
386
+ sessionContextParts.push(`<current_session>
387
+ ${currentSessionContext}
388
+ </current_session>`)
389
+ }
390
+ const sessionContextSection =
391
+ sessionContextParts.length > 0
392
+ ? `\n<session_context>
393
+ ${sessionContextParts.join('\n\n')}
394
+ </session_context>`
395
+ : ''
396
+
397
+ const transcriptionPrompt = `${languageHint}Transcribe this audio for a coding agent (like Claude Code or OpenCode).
398
+
399
+ CRITICAL REQUIREMENT: You MUST call the "transcriptionResult" tool to complete this task.
400
+ - The transcriptionResult tool is the ONLY way to return results
401
+ - Text responses are completely ignored - only tool calls work
402
+ - You MUST call transcriptionResult even if you run out of tool calls
403
+ - Always call transcriptionResult with your best approximation of what was said
404
+ - DO NOT end without calling transcriptionResult
405
+
406
+ This is a software development environment. The speaker is giving instructions to an AI coding assistant. Expect:
407
+ - File paths, function names, CLI commands, package names, API endpoints
408
+
409
+ RULES:
410
+ - NEVER change the meaning or intent of the user's message. Your job is ONLY to transcribe, not to respond or answer.
411
+ - If the user asks a question, keep it as a question. Do NOT answer it. Do NOT rephrase it as a statement.
412
+ - Only fix grammar, punctuation, and markdown formatting. Preserve the original content faithfully.
413
+ - If audio is unclear, transcribe your best interpretation, even with strong accents. Always provide an approximation.
414
+ - If audio seems silent/empty, call transcriptionResult with "[inaudible audio]"
415
+ - 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.
416
+
417
+ QUEUE DETECTION:
418
+ - 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.
419
+ - Remove the queue instruction from the transcription text itself — only include the actual message content.
420
+ - Example: "Queue this message. Fix the login bug in auth.ts" → transcription: "Fix the login bug in auth.ts", queueMessage: true
421
+ - 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.
422
+ - If no queue intent is detected, omit queueMessage or set it to false.
423
+
424
+ Common corrections (apply without tool calls):
425
+ - "reacked" → "React", "jason" → "JSON", "get hub" → "GitHub", "no JS" → "Node.js", "dacker" → "Docker"
426
+
427
+ Project file structure:
428
+ <file_tree>
429
+ ${prompt}
430
+ </file_tree>
431
+ ${sessionContextSection}
432
+
433
+ REMEMBER: Call "transcriptionResult" tool with your transcription. This is mandatory.
434
+
435
+ Note: "critique" is a CLI tool for showing diffs in the browser.`
436
+
437
+ return runTranscriptionOnce({
438
+ model: languageModel,
439
+ prompt: transcriptionPrompt,
440
+ audioBase64: finalAudioBase64,
441
+ mediaType,
442
+ temperature: temperature ?? 0.3,
443
+ })
444
+ }