@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,852 @@
1
+ // Discord voice channel connection and audio stream handler.
2
+ // Manages joining/leaving voice channels, captures user audio, resamples to 16kHz,
3
+ // and routes audio to the GenAI worker for real-time voice assistant interactions.
4
+ import * as errore from 'errore'
5
+
6
+ import {
7
+ VoiceConnectionStatus,
8
+ EndBehaviorType,
9
+ joinVoiceChannel,
10
+ entersState,
11
+ type VoiceConnection,
12
+ } from '@discordjs/voice'
13
+ import { exec } from 'node:child_process'
14
+ import fs, { createWriteStream } from 'node:fs'
15
+ import { mkdir } from 'node:fs/promises'
16
+ import path from 'node:path'
17
+ import { promisify } from 'node:util'
18
+ import { Transform, type TransformCallback } from 'node:stream'
19
+ import * as prism from 'prism-media'
20
+ import dedent from 'string-dedent'
21
+ import {
22
+ Events,
23
+ ActionRowBuilder,
24
+ ButtonBuilder,
25
+ ButtonStyle,
26
+ type Client,
27
+ type Message,
28
+ type ThreadChannel,
29
+ type VoiceChannel,
30
+ type VoiceState,
31
+ } from 'discord.js'
32
+ import { createGenAIWorker, type GenAIWorker } from './genai-worker-wrapper.js'
33
+ import {
34
+ getVoiceChannelDirectory,
35
+ getGeminiApiKey,
36
+ getTranscriptionApiKey,
37
+ findTextChannelByVoiceChannel,
38
+ } from './database.js'
39
+ import {
40
+ sendThreadMessage,
41
+ escapeDiscordFormatting,
42
+ SILENT_MESSAGE_FLAGS,
43
+ hasKimakiBotPermission,
44
+ isGuildAllowed,
45
+ } from './discord-utils.js'
46
+ import { transcribeAudio, type TranscriptionResult } from './voice.js'
47
+ import { FetchError } from './errors.js'
48
+
49
+ import { createLogger, LogPrefix } from './logger.js'
50
+ import { notifyError } from './sentry.js'
51
+
52
+ const voiceLogger = createLogger(LogPrefix.VOICE)
53
+
54
+ export type VoiceConnectionData = {
55
+ connection: VoiceConnection
56
+ genAiWorker?: GenAIWorker
57
+ userAudioStream?: fs.WriteStream
58
+ }
59
+
60
+ export const voiceConnections = new Map<string, VoiceConnectionData>()
61
+
62
+
63
+
64
+ export function convertToMono16k(buffer: Buffer): Buffer {
65
+ const inputSampleRate = 48000
66
+ const outputSampleRate = 16000
67
+ const ratio = inputSampleRate / outputSampleRate
68
+ const inputChannels = 2
69
+ const bytesPerSample = 2
70
+
71
+ const inputSamples = buffer.length / (bytesPerSample * inputChannels)
72
+ const outputSamples = Math.floor(inputSamples / ratio)
73
+ const outputBuffer = Buffer.alloc(outputSamples * bytesPerSample)
74
+
75
+ for (let i = 0; i < outputSamples; i++) {
76
+ const inputIndex = Math.floor(i * ratio) * inputChannels * bytesPerSample
77
+
78
+ if (inputIndex + 3 < buffer.length) {
79
+ const leftSample = buffer.readInt16LE(inputIndex)
80
+ const rightSample = buffer.readInt16LE(inputIndex + 2)
81
+ const monoSample = Math.round((leftSample + rightSample) / 2)
82
+
83
+ outputBuffer.writeInt16LE(monoSample, i * bytesPerSample)
84
+ }
85
+ }
86
+
87
+ return outputBuffer
88
+ }
89
+
90
+ export async function createUserAudioLogStream(
91
+ guildId: string,
92
+ channelId: string,
93
+ ): Promise<fs.WriteStream | undefined> {
94
+ if (!process.env.DEBUG) return undefined
95
+
96
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-')
97
+ const audioDir = path.join(
98
+ process.cwd(),
99
+ 'discord-audio-logs',
100
+ guildId,
101
+ channelId,
102
+ )
103
+
104
+ try {
105
+ await mkdir(audioDir, { recursive: true })
106
+
107
+ const inputFileName = `user_${timestamp}.16.pcm`
108
+ const inputFilePath = path.join(audioDir, inputFileName)
109
+ const inputAudioStream = createWriteStream(inputFilePath)
110
+ voiceLogger.log(`Created user audio log: ${inputFilePath}`)
111
+
112
+ return inputAudioStream
113
+ } catch (error) {
114
+ voiceLogger.error('Failed to create audio log directory:', error)
115
+ return undefined
116
+ }
117
+ }
118
+
119
+ export function frameMono16khz(): Transform {
120
+ const FRAME_BYTES = (100 * 16_000 * 1 * 2) / 1000
121
+ let stash: Buffer = Buffer.alloc(0)
122
+ let offset = 0
123
+
124
+ return new Transform({
125
+ readableObjectMode: false,
126
+ writableObjectMode: false,
127
+
128
+ transform(chunk: Buffer, _enc: BufferEncoding, cb: TransformCallback) {
129
+ if (offset > 0) {
130
+ stash = stash.subarray(offset)
131
+ offset = 0
132
+ }
133
+
134
+ stash = stash.length ? Buffer.concat([stash, chunk]) : chunk
135
+
136
+ while (stash.length - offset >= FRAME_BYTES) {
137
+ this.push(stash.subarray(offset, offset + FRAME_BYTES))
138
+ offset += FRAME_BYTES
139
+ }
140
+
141
+ if (offset === stash.length) {
142
+ stash = Buffer.alloc(0)
143
+ offset = 0
144
+ }
145
+
146
+ cb()
147
+ },
148
+
149
+ flush(cb: TransformCallback) {
150
+ stash = Buffer.alloc(0)
151
+ offset = 0
152
+ cb()
153
+ },
154
+ })
155
+ }
156
+
157
+ export async function setupVoiceHandling({
158
+ connection,
159
+ guildId,
160
+ channelId,
161
+ appId,
162
+ discordClient,
163
+ }: {
164
+ connection: VoiceConnection
165
+ guildId: string
166
+ channelId: string
167
+ appId: string
168
+ discordClient: Client
169
+ }) {
170
+ voiceLogger.log(
171
+ `Setting up voice handling for guild ${guildId}, channel ${channelId}`,
172
+ )
173
+
174
+ const directory = await getVoiceChannelDirectory(channelId)
175
+
176
+ if (!directory) {
177
+ voiceLogger.log(
178
+ `Voice channel ${channelId} has no associated directory, skipping setup`,
179
+ )
180
+ return
181
+ }
182
+
183
+ voiceLogger.log(`Found directory for voice channel: ${directory}`)
184
+
185
+ const voiceData = voiceConnections.get(guildId)
186
+ if (!voiceData) {
187
+ voiceLogger.error(`No voice data found for guild ${guildId}`)
188
+ return
189
+ }
190
+
191
+ voiceData.userAudioStream = await createUserAudioLogStream(guildId, channelId)
192
+
193
+ const geminiApiKey = await getGeminiApiKey(appId)
194
+
195
+ const genAiWorker = await createGenAIWorker({
196
+ directory,
197
+ guildId,
198
+ channelId,
199
+ appId,
200
+ geminiApiKey,
201
+ systemMessage: dedent`
202
+ You are Kimaki, an AI similar to Jarvis: you help your user (an engineer) controlling his coding agent, just like Jarvis controls Ironman armor and machines. Speak fast.
203
+
204
+ You should talk like Jarvis, British accent, satirical, joking and calm. Be short and concise. Speak fast.
205
+
206
+ After tool calls give a super short summary of the assistant message, you should say what the assistant message writes.
207
+
208
+ Before starting a new session ask for confirmation if it is not clear if the user finished describing it. ask "message ready, send?"
209
+
210
+ NEVER repeat the whole tool call parameters or message.
211
+
212
+ Your job is to manage many opencode agent chat instances. Opencode is the agent used to write the code, it is similar to Claude Code.
213
+
214
+ For everything the user asks it is implicit that the user is asking for you to proxy the requests to opencode sessions.
215
+
216
+ You can
217
+ - start new chats on a given project
218
+ - read the chats to report progress to the user
219
+ - submit messages to the chat
220
+ - list files for a given projects, so you can translate imprecise user prompts to precise messages that mention filename paths using @
221
+
222
+ Common patterns
223
+ - to get the last session use the listChats tool
224
+ - when user asks you to do something you submit a new session to do it. it's implicit that you proxy requests to the agents chat!
225
+ - when you submit a session assume the session will take a minute or 2 to complete the task
226
+
227
+ Rules
228
+ - never spell files by mentioning dots, letters, etc. instead give a brief description of the filename
229
+ - NEVER spell hashes or IDs
230
+ - never read session ids or other ids
231
+
232
+ Your voice is calm and monotone, NEVER excited and goofy. But you speak without jargon or bs and do veiled short jokes.
233
+ You speak like you knew something other don't. You are cool and cold.
234
+ `,
235
+ onAssistantOpusPacket(packet) {
236
+ if (connection.state.status !== VoiceConnectionStatus.Ready) {
237
+ voiceLogger.log('Skipping packet: connection not ready')
238
+ return
239
+ }
240
+
241
+ try {
242
+ connection.setSpeaking(true)
243
+ connection.playOpusPacket(Buffer.from(packet))
244
+ } catch (error) {
245
+ voiceLogger.error('Error sending packet:', error)
246
+ }
247
+ },
248
+ onAssistantStartSpeaking() {
249
+ voiceLogger.log('Assistant started speaking')
250
+ connection.setSpeaking(true)
251
+ },
252
+ onAssistantStopSpeaking() {
253
+ voiceLogger.log('Assistant stopped speaking (natural finish)')
254
+ connection.setSpeaking(false)
255
+ },
256
+ onAssistantInterruptSpeaking() {
257
+ voiceLogger.log('Assistant interrupted while speaking')
258
+ genAiWorker.interrupt()
259
+ connection.setSpeaking(false)
260
+ },
261
+ onToolCallCompleted(params) {
262
+ const errorText: string | undefined = (() => {
263
+ if (!params.error) {
264
+ return undefined
265
+ }
266
+ if (params.error instanceof Error) {
267
+ return params.error.message
268
+ }
269
+ return String(params.error)
270
+ })()
271
+
272
+ const text = params.error
273
+ ? `<systemMessage>\nThe coding agent encountered an error while processing session ${params.sessionId}: ${errorText || 'Unknown error'}\n</systemMessage>`
274
+ : `<systemMessage>\nThe coding agent finished working on session ${params.sessionId}\n\nHere's what the assistant wrote:\n${params.markdown}\n</systemMessage>`
275
+
276
+ genAiWorker.sendTextInput(text)
277
+ },
278
+ async onError(error) {
279
+ voiceLogger.error('GenAI worker error:', error)
280
+ const textChannelId = await findTextChannelByVoiceChannel(channelId)
281
+
282
+ if (textChannelId) {
283
+ try {
284
+ const textChannel = await discordClient.channels.fetch(textChannelId)
285
+ if (textChannel?.isTextBased() && 'send' in textChannel) {
286
+ await textChannel.send({
287
+ content: `⚠️ Voice session error: ${String(error).slice(0, 1900)}`,
288
+ flags: SILENT_MESSAGE_FLAGS,
289
+ })
290
+ }
291
+ } catch (e) {
292
+ voiceLogger.error('Failed to send error to text channel:', e)
293
+ }
294
+ }
295
+ },
296
+ })
297
+
298
+ if (voiceData.genAiWorker) {
299
+ voiceLogger.log('Stopping existing GenAI worker before creating new one')
300
+ await voiceData.genAiWorker.stop()
301
+ }
302
+
303
+ genAiWorker.sendTextInput(
304
+ `<systemMessage>\nsay "Hello boss, how we doing today?"\n</systemMessage>`,
305
+ )
306
+
307
+ voiceData.genAiWorker = genAiWorker
308
+
309
+ const receiver = connection.receiver
310
+
311
+ receiver.speaking.removeAllListeners('start')
312
+
313
+ let speakingSessionCount = 0
314
+
315
+ receiver.speaking.on('start', (userId) => {
316
+ voiceLogger.log(`User ${userId} started speaking`)
317
+
318
+ speakingSessionCount++
319
+ const currentSessionCount = speakingSessionCount
320
+ voiceLogger.log(`Speaking session ${currentSessionCount} started`)
321
+
322
+ const audioStream = receiver.subscribe(userId, {
323
+ end: { behavior: EndBehaviorType.AfterSilence, duration: 500 },
324
+ })
325
+
326
+ const decoder = new prism.opus.Decoder({
327
+ rate: 48000,
328
+ channels: 2,
329
+ frameSize: 960,
330
+ })
331
+
332
+ decoder.on('error', (error) => {
333
+ voiceLogger.error(`Opus decoder error for user ${userId}:`, error)
334
+ void notifyError(error, `Opus decoder error for user ${userId}`)
335
+ })
336
+
337
+ const downsampleTransform = new Transform({
338
+ transform(chunk: Buffer, _encoding, callback) {
339
+ try {
340
+ const downsampled = convertToMono16k(chunk)
341
+ callback(null, downsampled)
342
+ } catch (error) {
343
+ callback(error as Error)
344
+ }
345
+ },
346
+ })
347
+
348
+ const framer = frameMono16khz()
349
+
350
+ const pipeline = audioStream
351
+ .pipe(decoder)
352
+ .pipe(downsampleTransform)
353
+ .pipe(framer)
354
+
355
+ pipeline
356
+ .on('data', (frame: Buffer) => {
357
+ if (currentSessionCount !== speakingSessionCount) {
358
+ return
359
+ }
360
+
361
+ if (!voiceData.genAiWorker) {
362
+ voiceLogger.warn(
363
+ `[VOICE] Received audio frame but no GenAI worker active for guild ${guildId}`,
364
+ )
365
+ return
366
+ }
367
+
368
+ voiceData.userAudioStream?.write(frame)
369
+
370
+ voiceData.genAiWorker.sendRealtimeInput({
371
+ audio: {
372
+ mimeType: 'audio/pcm;rate=16000',
373
+ data: frame.toString('base64'),
374
+ },
375
+ })
376
+ })
377
+ .on('end', () => {
378
+ if (currentSessionCount === speakingSessionCount) {
379
+ voiceLogger.log(
380
+ `User ${userId} stopped speaking (session ${currentSessionCount})`,
381
+ )
382
+ voiceData.genAiWorker?.sendRealtimeInput({
383
+ audioStreamEnd: true,
384
+ })
385
+ } else {
386
+ voiceLogger.log(
387
+ `User ${userId} stopped speaking (session ${currentSessionCount}), but skipping audioStreamEnd because newer session ${speakingSessionCount} exists`,
388
+ )
389
+ }
390
+ })
391
+ .on('error', (error) => {
392
+ voiceLogger.error(`Pipeline error for user ${userId}:`, error)
393
+ void notifyError(error, `Voice pipeline error for user ${userId}`)
394
+ })
395
+
396
+ audioStream.on('error', (error) => {
397
+ voiceLogger.error(`Audio stream error for user ${userId}:`, error)
398
+ void notifyError(error, `Audio stream error for user ${userId}`)
399
+ })
400
+
401
+ downsampleTransform.on('error', (error) => {
402
+ voiceLogger.error(`Downsample transform error for user ${userId}:`, error)
403
+ void notifyError(error, `Downsample transform error for user ${userId}`)
404
+ })
405
+
406
+ framer.on('error', (error) => {
407
+ voiceLogger.error(`Framer error for user ${userId}:`, error)
408
+ void notifyError(error, `Framer error for user ${userId}`)
409
+ })
410
+ })
411
+ }
412
+
413
+ export async function cleanupVoiceConnection(guildId: string) {
414
+ const voiceData = voiceConnections.get(guildId)
415
+ if (!voiceData) return
416
+
417
+ voiceLogger.log(`Starting cleanup for guild ${guildId}`)
418
+
419
+ try {
420
+ if (voiceData.genAiWorker) {
421
+ voiceLogger.log(`Stopping GenAI worker...`)
422
+ await voiceData.genAiWorker.stop()
423
+ voiceLogger.log(`GenAI worker stopped`)
424
+ }
425
+
426
+ if (voiceData.userAudioStream) {
427
+ voiceLogger.log(`Closing user audio stream...`)
428
+ await new Promise<void>((resolve) => {
429
+ voiceData.userAudioStream!.end(() => {
430
+ voiceLogger.log('User audio stream closed')
431
+ resolve()
432
+ })
433
+ setTimeout(resolve, 2000)
434
+ })
435
+ }
436
+
437
+ if (voiceData.connection.state.status !== VoiceConnectionStatus.Destroyed) {
438
+ voiceLogger.log(`Destroying voice connection...`)
439
+ voiceData.connection.destroy()
440
+ }
441
+
442
+ voiceConnections.delete(guildId)
443
+ voiceLogger.log(`Cleanup complete for guild ${guildId}`)
444
+ } catch (error) {
445
+ voiceLogger.error(`Error during cleanup for guild ${guildId}:`, error)
446
+ voiceConnections.delete(guildId)
447
+ }
448
+ }
449
+
450
+ type ProcessVoiceAttachmentArgs = {
451
+ message: Message
452
+ thread: ThreadChannel
453
+ projectDirectory?: string
454
+ isNewThread?: boolean
455
+ appId?: string
456
+ currentSessionContext?: string
457
+ lastSessionContext?: string
458
+ }
459
+
460
+ // Per-thread serialization is handled by threadMessageQueue in discord-bot.ts,
461
+ // so this function no longer needs its own queue.
462
+ export async function processVoiceAttachment({
463
+ message,
464
+ thread,
465
+ projectDirectory,
466
+ isNewThread = false,
467
+ appId,
468
+ currentSessionContext,
469
+ lastSessionContext,
470
+ }: ProcessVoiceAttachmentArgs): Promise<TranscriptionResult | null> {
471
+ const audioAttachment = Array.from(message.attachments.values()).find(
472
+ (attachment) => attachment.contentType?.startsWith('audio/'),
473
+ )
474
+
475
+ if (!audioAttachment) return null
476
+
477
+ voiceLogger.log(
478
+ `Detected audio attachment: ${audioAttachment.name} (${audioAttachment.contentType})`,
479
+ )
480
+
481
+ await sendThreadMessage(thread, '🎤 Transcribing voice message...')
482
+
483
+ const audioResponse = await errore.tryAsync({
484
+ try: () => fetch(audioAttachment.url),
485
+ catch: (e) => new FetchError({ url: audioAttachment.url, cause: e }),
486
+ })
487
+ if (audioResponse instanceof Error) {
488
+ voiceLogger.error(
489
+ `Failed to download audio attachment:`,
490
+ audioResponse.message,
491
+ )
492
+ await sendThreadMessage(
493
+ thread,
494
+ `⚠️ Failed to download audio: ${audioResponse.message}`,
495
+ )
496
+ return null
497
+ }
498
+ const audioBuffer = Buffer.from(await audioResponse.arrayBuffer())
499
+
500
+ voiceLogger.log(`Downloaded ${audioBuffer.length} bytes, transcribing...`)
501
+
502
+ let transcriptionPrompt = 'Discord voice message transcription'
503
+
504
+ if (projectDirectory) {
505
+ try {
506
+ voiceLogger.log(`Getting project file tree from ${projectDirectory}`)
507
+ const execAsync = promisify(exec)
508
+ const { stdout } = await execAsync('git ls-files | tree --fromfile -a', {
509
+ cwd: projectDirectory,
510
+ })
511
+
512
+ if (stdout) {
513
+ transcriptionPrompt = `Discord voice message transcription. Project file structure:\n${stdout}\n\nPlease transcribe file names and paths accurately based on this context.`
514
+ voiceLogger.log(`Added project context to transcription prompt`)
515
+ }
516
+ } catch (e) {
517
+ voiceLogger.log(`Could not get project tree:`, e)
518
+ }
519
+ }
520
+
521
+ // Resolve transcription API key: prefer OpenAI, fall back to Gemini, then env vars
522
+ let transcriptionApiKey: string | undefined
523
+ let transcriptionProvider: 'openai' | 'gemini' | undefined
524
+ if (appId) {
525
+ const stored = await getTranscriptionApiKey(appId)
526
+ if (stored) {
527
+ transcriptionApiKey = stored.apiKey
528
+ transcriptionProvider = stored.provider
529
+ }
530
+ }
531
+ if (!transcriptionApiKey) {
532
+ if (process.env.OPENAI_API_KEY) {
533
+ transcriptionApiKey = process.env.OPENAI_API_KEY
534
+ transcriptionProvider = 'openai'
535
+ } else if (process.env.GEMINI_API_KEY) {
536
+ transcriptionApiKey = process.env.GEMINI_API_KEY
537
+ transcriptionProvider = 'gemini'
538
+ }
539
+ }
540
+
541
+ if (!transcriptionApiKey) {
542
+ if (appId) {
543
+ const button = new ButtonBuilder()
544
+ .setCustomId(`transcription_apikey:${appId}`)
545
+ .setLabel('Set Transcription API Key')
546
+ .setStyle(ButtonStyle.Primary)
547
+
548
+ const row = new ActionRowBuilder<ButtonBuilder>().addComponents(button)
549
+
550
+ await thread.send({
551
+ content:
552
+ 'Voice transcription requires an API key (OpenAI or Gemini). Set one to enable voice message transcription.',
553
+ components: [row],
554
+ flags: SILENT_MESSAGE_FLAGS,
555
+ })
556
+ } else {
557
+ await sendThreadMessage(
558
+ thread,
559
+ 'Voice transcription requires an API key. Set OPENAI_API_KEY or GEMINI_API_KEY, or use /login in this channel.',
560
+ )
561
+ }
562
+ return null
563
+ }
564
+
565
+ const transcription = await transcribeAudio({
566
+ audio: audioBuffer,
567
+ prompt: transcriptionPrompt,
568
+ apiKey: transcriptionApiKey,
569
+ provider: transcriptionProvider,
570
+ mediaType: audioAttachment.contentType || undefined,
571
+ currentSessionContext,
572
+ lastSessionContext,
573
+ })
574
+
575
+ if (transcription instanceof Error) {
576
+ const errMsg = errore.matchError(transcription, {
577
+ ApiKeyMissingError: (e) => e.message,
578
+ InvalidAudioFormatError: (e) => e.message,
579
+ TranscriptionError: (e) => e.message,
580
+ EmptyTranscriptionError: (e) => e.message,
581
+ NoResponseContentError: (e) => e.message,
582
+ NoToolResponseError: (e) => e.message,
583
+ Error: (e) => e.message,
584
+ })
585
+ voiceLogger.error(`Transcription failed:`, transcription)
586
+ await sendThreadMessage(thread, `⚠️ Transcription failed: ${errMsg}`)
587
+ return null
588
+ }
589
+
590
+ const { transcription: text, queueMessage } = transcription
591
+
592
+ voiceLogger.log(
593
+ `Transcription successful: "${text.slice(0, 50)}${text.length > 50 ? '...' : ''}"${queueMessage ? ' [QUEUE]' : ''}`,
594
+ )
595
+
596
+ if (isNewThread) {
597
+ const threadName = text.replace(/\s+/g, ' ').trim().slice(0, 80)
598
+ if (threadName) {
599
+ const renamed = await Promise.race([
600
+ errore.tryAsync({
601
+ try: () => thread.setName(threadName),
602
+ catch: (e) => e as Error,
603
+ }),
604
+ new Promise<null>((resolve) => {
605
+ setTimeout(() => {
606
+ resolve(null)
607
+ }, 2000)
608
+ }),
609
+ ])
610
+ if (renamed === null) {
611
+ voiceLogger.log(`Thread name update timed out`)
612
+ } else if (renamed instanceof Error) {
613
+ voiceLogger.log(`Could not update thread name:`, renamed.message)
614
+ } else {
615
+ voiceLogger.log(`Updated thread name to: "${threadName}"`)
616
+ }
617
+ }
618
+ }
619
+
620
+ await sendThreadMessage(
621
+ thread,
622
+ `📝 **Transcribed message:** ${escapeDiscordFormatting(text)}`,
623
+ )
624
+ return transcription
625
+ }
626
+
627
+ export function registerVoiceStateHandler({
628
+ discordClient,
629
+ appId,
630
+ }: {
631
+ discordClient: Client
632
+ appId: string
633
+ }) {
634
+ discordClient.on(
635
+ Events.VoiceStateUpdate,
636
+ async (oldState: VoiceState, newState: VoiceState) => {
637
+ try {
638
+ const member = newState.member || oldState.member
639
+ if (!member) return
640
+
641
+ if (!hasKimakiBotPermission(member)) {
642
+ return
643
+ }
644
+
645
+ const guild = newState.guild || oldState.guild
646
+ if (!isGuildAllowed({ guildId: guild?.id })) {
647
+ return
648
+ }
649
+
650
+ if (oldState.channelId !== null && newState.channelId === null) {
651
+ voiceLogger.log(
652
+ `Permitted user ${member.user.tag} left voice channel: ${oldState.channel?.name}`,
653
+ )
654
+
655
+ const guildId = guild.id
656
+ const voiceData = voiceConnections.get(guildId)
657
+
658
+ if (
659
+ voiceData &&
660
+ voiceData.connection.joinConfig.channelId === oldState.channelId
661
+ ) {
662
+ const voiceChannel = oldState.channel as VoiceChannel
663
+ if (!voiceChannel) return
664
+
665
+ const hasOtherPermittedUsers = voiceChannel.members.some((m) => {
666
+ if (m.id === member.id || m.user.bot) {
667
+ return false
668
+ }
669
+ return hasKimakiBotPermission(m)
670
+ })
671
+
672
+ if (!hasOtherPermittedUsers) {
673
+ voiceLogger.log(
674
+ `No other permitted users in channel, bot leaving voice channel in guild: ${guild.name}`,
675
+ )
676
+
677
+ await cleanupVoiceConnection(guildId)
678
+ } else {
679
+ voiceLogger.log(
680
+ `Other permitted users still in channel, bot staying in voice channel`,
681
+ )
682
+ }
683
+ }
684
+ return
685
+ }
686
+
687
+ if (
688
+ oldState.channelId !== null &&
689
+ newState.channelId !== null &&
690
+ oldState.channelId !== newState.channelId
691
+ ) {
692
+ voiceLogger.log(
693
+ `Permitted user ${member.user.tag} moved from ${oldState.channel?.name} to ${newState.channel?.name}`,
694
+ )
695
+
696
+ const guildId = guild.id
697
+ const voiceData = voiceConnections.get(guildId)
698
+
699
+ if (
700
+ voiceData &&
701
+ voiceData.connection.joinConfig.channelId === oldState.channelId
702
+ ) {
703
+ const oldVoiceChannel = oldState.channel as VoiceChannel
704
+ if (oldVoiceChannel) {
705
+ const hasOtherPermittedUsers = oldVoiceChannel.members.some(
706
+ (m) => {
707
+ if (m.id === member.id || m.user.bot) {
708
+ return false
709
+ }
710
+ return hasKimakiBotPermission(m)
711
+ },
712
+ )
713
+
714
+ if (!hasOtherPermittedUsers) {
715
+ voiceLogger.log(
716
+ `Following admin to new channel: ${newState.channel?.name}`,
717
+ )
718
+ const voiceChannel = newState.channel as VoiceChannel
719
+ if (voiceChannel) {
720
+ voiceData.connection.rejoin({
721
+ channelId: voiceChannel.id,
722
+ selfDeaf: false,
723
+ selfMute: false,
724
+ })
725
+ }
726
+ } else {
727
+ voiceLogger.log(
728
+ `Other permitted users still in old channel, bot staying put`,
729
+ )
730
+ }
731
+ }
732
+ }
733
+ }
734
+
735
+ if (oldState.channelId === null && newState.channelId !== null) {
736
+ voiceLogger.log(
737
+ `Permitted user ${member.user.tag} joined voice channel: ${newState.channel?.name}`,
738
+ )
739
+ }
740
+
741
+ if (newState.channelId === null) return
742
+
743
+ const voiceChannel = newState.channel as VoiceChannel
744
+ if (!voiceChannel) return
745
+
746
+ const existingVoiceData = voiceConnections.get(newState.guild.id)
747
+ if (
748
+ existingVoiceData &&
749
+ existingVoiceData.connection.state.status !==
750
+ VoiceConnectionStatus.Destroyed
751
+ ) {
752
+ voiceLogger.log(
753
+ `Bot already connected to a voice channel in guild ${newState.guild.name}`,
754
+ )
755
+
756
+ if (
757
+ existingVoiceData.connection.joinConfig.channelId !==
758
+ voiceChannel.id
759
+ ) {
760
+ voiceLogger.log(
761
+ `Moving bot from channel ${existingVoiceData.connection.joinConfig.channelId} to ${voiceChannel.id}`,
762
+ )
763
+ existingVoiceData.connection.rejoin({
764
+ channelId: voiceChannel.id,
765
+ selfDeaf: false,
766
+ selfMute: false,
767
+ })
768
+ }
769
+ return
770
+ }
771
+
772
+ try {
773
+ voiceLogger.log(
774
+ `Attempting to join voice channel: ${voiceChannel.name} (${voiceChannel.id})`,
775
+ )
776
+
777
+ const connection = joinVoiceChannel({
778
+ channelId: voiceChannel.id,
779
+ guildId: newState.guild.id,
780
+ adapterCreator: newState.guild.voiceAdapterCreator,
781
+ selfDeaf: false,
782
+ debug: true,
783
+ daveEncryption: false,
784
+ selfMute: false,
785
+ })
786
+
787
+ voiceConnections.set(newState.guild.id, { connection })
788
+
789
+ await entersState(connection, VoiceConnectionStatus.Ready, 30_000)
790
+ voiceLogger.log(
791
+ `Successfully joined voice channel: ${voiceChannel.name} in guild: ${newState.guild.name}`,
792
+ )
793
+
794
+ await setupVoiceHandling({
795
+ connection,
796
+ guildId: newState.guild.id,
797
+ channelId: voiceChannel.id,
798
+ appId,
799
+ discordClient,
800
+ })
801
+
802
+ connection.on(VoiceConnectionStatus.Disconnected, async () => {
803
+ voiceLogger.log(
804
+ `Disconnected from voice channel in guild: ${newState.guild.name}`,
805
+ )
806
+ try {
807
+ await Promise.race([
808
+ entersState(
809
+ connection,
810
+ VoiceConnectionStatus.Signalling,
811
+ 5_000,
812
+ ),
813
+ entersState(
814
+ connection,
815
+ VoiceConnectionStatus.Connecting,
816
+ 5_000,
817
+ ),
818
+ ])
819
+ voiceLogger.log(`Reconnecting to voice channel`)
820
+ } catch (error) {
821
+ voiceLogger.log(`Failed to reconnect, destroying connection`)
822
+ connection.destroy()
823
+ voiceConnections.delete(newState.guild.id)
824
+ }
825
+ })
826
+
827
+ connection.on(VoiceConnectionStatus.Destroyed, async () => {
828
+ voiceLogger.log(
829
+ `Connection destroyed for guild: ${newState.guild.name}`,
830
+ )
831
+ await cleanupVoiceConnection(newState.guild.id)
832
+ })
833
+
834
+ connection.on('error', (error) => {
835
+ voiceLogger.error(
836
+ `Connection error in guild ${newState.guild.name}:`,
837
+ error,
838
+ )
839
+ void notifyError(error, `Voice connection error in guild ${newState.guild.name}`)
840
+ })
841
+ } catch (error) {
842
+ voiceLogger.error(`Failed to join voice channel:`, error)
843
+ void notifyError(error, 'Failed to join voice channel')
844
+ await cleanupVoiceConnection(newState.guild.id)
845
+ }
846
+ } catch (error) {
847
+ voiceLogger.error('Error in voice state update handler:', error)
848
+ void notifyError(error, 'Voice state update handler error')
849
+ }
850
+ },
851
+ )
852
+ }