@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.
- package/bin.js +2 -0
- package/dist/agent-model.e2e.test.js +755 -0
- package/dist/ai-tool-to-genai.js +233 -0
- package/dist/ai-tool-to-genai.test.js +267 -0
- package/dist/ai-tool.js +6 -0
- package/dist/anthropic-auth-plugin.js +728 -0
- package/dist/anthropic-auth-plugin.test.js +125 -0
- package/dist/anthropic-auth-state.js +231 -0
- package/dist/bin.js +90 -0
- package/dist/channel-management.js +227 -0
- package/dist/cli-parsing.test.js +137 -0
- package/dist/cli-send-thread.e2e.test.js +356 -0
- package/dist/cli.js +3276 -0
- package/dist/commands/abort.js +65 -0
- package/dist/commands/action-buttons.js +245 -0
- package/dist/commands/add-project.js +113 -0
- package/dist/commands/agent.js +335 -0
- package/dist/commands/ask-question.js +274 -0
- package/dist/commands/btw.js +116 -0
- package/dist/commands/compact.js +120 -0
- package/dist/commands/context-usage.js +140 -0
- package/dist/commands/create-new-project.js +130 -0
- package/dist/commands/diff.js +63 -0
- package/dist/commands/file-upload.js +275 -0
- package/dist/commands/fork.js +220 -0
- package/dist/commands/gemini-apikey.js +70 -0
- package/dist/commands/login.js +885 -0
- package/dist/commands/mcp.js +239 -0
- package/dist/commands/memory-snapshot.js +24 -0
- package/dist/commands/mention-mode.js +44 -0
- package/dist/commands/merge-worktree.js +159 -0
- package/dist/commands/model-variant.js +364 -0
- package/dist/commands/model.js +776 -0
- package/dist/commands/new-worktree.js +366 -0
- package/dist/commands/paginated-select.js +57 -0
- package/dist/commands/permissions.js +274 -0
- package/dist/commands/queue.js +206 -0
- package/dist/commands/remove-project.js +115 -0
- package/dist/commands/restart-opencode-server.js +127 -0
- package/dist/commands/resume.js +149 -0
- package/dist/commands/run-command.js +79 -0
- package/dist/commands/screenshare.js +303 -0
- package/dist/commands/screenshare.test.js +20 -0
- package/dist/commands/session-id.js +78 -0
- package/dist/commands/session.js +176 -0
- package/dist/commands/share.js +80 -0
- package/dist/commands/tasks.js +205 -0
- package/dist/commands/types.js +2 -0
- package/dist/commands/undo-redo.js +305 -0
- package/dist/commands/unset-model.js +138 -0
- package/dist/commands/upgrade.js +42 -0
- package/dist/commands/user-command.js +155 -0
- package/dist/commands/verbosity.js +125 -0
- package/dist/commands/worktree-settings.js +43 -0
- package/dist/commands/worktrees.js +410 -0
- package/dist/condense-memory.js +33 -0
- package/dist/config.js +94 -0
- package/dist/context-awareness-plugin.js +363 -0
- package/dist/context-awareness-plugin.test.js +124 -0
- package/dist/critique-utils.js +95 -0
- package/dist/database.js +1310 -0
- package/dist/db.js +251 -0
- package/dist/db.test.js +138 -0
- package/dist/debounce-timeout.js +28 -0
- package/dist/debounced-process-flush.js +77 -0
- package/dist/discord-bot.js +1008 -0
- package/dist/discord-command-registration.js +524 -0
- package/dist/discord-urls.js +81 -0
- package/dist/discord-utils.js +591 -0
- package/dist/discord-utils.test.js +134 -0
- package/dist/errors.js +157 -0
- package/dist/escape-backticks.test.js +429 -0
- package/dist/event-stream-real-capture.e2e.test.js +533 -0
- package/dist/eventsource-parser.test.js +327 -0
- package/dist/exec-async.js +26 -0
- package/dist/external-opencode-sync.js +480 -0
- package/dist/format-tables.js +302 -0
- package/dist/format-tables.test.js +308 -0
- package/dist/forum-sync/config.js +79 -0
- package/dist/forum-sync/discord-operations.js +154 -0
- package/dist/forum-sync/index.js +5 -0
- package/dist/forum-sync/markdown.js +113 -0
- package/dist/forum-sync/sync-to-discord.js +417 -0
- package/dist/forum-sync/sync-to-files.js +190 -0
- package/dist/forum-sync/types.js +53 -0
- package/dist/forum-sync/watchers.js +307 -0
- package/dist/gateway-proxy-reconnect.e2e.test.js +394 -0
- package/dist/gateway-proxy.e2e.test.js +483 -0
- package/dist/genai-worker-wrapper.js +111 -0
- package/dist/genai-worker.js +311 -0
- package/dist/genai.js +232 -0
- package/dist/generated/browser.js +17 -0
- package/dist/generated/client.js +37 -0
- package/dist/generated/commonInputTypes.js +10 -0
- package/dist/generated/enums.js +52 -0
- package/dist/generated/internal/class.js +49 -0
- package/dist/generated/internal/prismaNamespace.js +253 -0
- package/dist/generated/internal/prismaNamespaceBrowser.js +223 -0
- package/dist/generated/models/bot_api_keys.js +1 -0
- package/dist/generated/models/bot_tokens.js +1 -0
- package/dist/generated/models/channel_agents.js +1 -0
- package/dist/generated/models/channel_directories.js +1 -0
- package/dist/generated/models/channel_mention_mode.js +1 -0
- package/dist/generated/models/channel_models.js +1 -0
- package/dist/generated/models/channel_verbosity.js +1 -0
- package/dist/generated/models/channel_worktrees.js +1 -0
- package/dist/generated/models/forum_sync_configs.js +1 -0
- package/dist/generated/models/global_models.js +1 -0
- package/dist/generated/models/ipc_requests.js +1 -0
- package/dist/generated/models/part_messages.js +1 -0
- package/dist/generated/models/scheduled_tasks.js +1 -0
- package/dist/generated/models/session_agents.js +1 -0
- package/dist/generated/models/session_events.js +1 -0
- package/dist/generated/models/session_models.js +1 -0
- package/dist/generated/models/session_start_sources.js +1 -0
- package/dist/generated/models/thread_sessions.js +1 -0
- package/dist/generated/models/thread_worktrees.js +1 -0
- package/dist/generated/models.js +1 -0
- package/dist/heap-monitor.js +122 -0
- package/dist/hrana-server.js +263 -0
- package/dist/hrana-server.test.js +370 -0
- package/dist/html-actions.js +123 -0
- package/dist/html-actions.test.js +70 -0
- package/dist/html-components.js +117 -0
- package/dist/html-components.test.js +34 -0
- package/dist/image-optimizer-plugin.js +153 -0
- package/dist/image-utils.js +112 -0
- package/dist/interaction-handler.js +397 -0
- package/dist/ipc-polling.js +252 -0
- package/dist/ipc-tools-plugin.js +193 -0
- package/dist/kimaki-digital-twin.e2e.test.js +161 -0
- package/dist/kimaki-opencode-plugin-loading.e2e.test.js +87 -0
- package/dist/kimaki-opencode-plugin.js +17 -0
- package/dist/kimaki-opencode-plugin.test.js +98 -0
- package/dist/limit-heading-depth.js +25 -0
- package/dist/limit-heading-depth.test.js +105 -0
- package/dist/logger.js +165 -0
- package/dist/markdown.js +342 -0
- package/dist/markdown.test.js +257 -0
- package/dist/message-finish-field.e2e.test.js +165 -0
- package/dist/message-formatting.js +413 -0
- package/dist/message-formatting.test.js +73 -0
- package/dist/message-preprocessing.js +330 -0
- package/dist/onboarding-tutorial.js +172 -0
- package/dist/onboarding-welcome.js +37 -0
- package/dist/openai-realtime.js +224 -0
- package/dist/opencode-command-detection.js +65 -0
- package/dist/opencode-command-detection.test.js +240 -0
- package/dist/opencode-command.js +129 -0
- package/dist/opencode-command.test.js +48 -0
- package/dist/opencode-interrupt-plugin.js +361 -0
- package/dist/opencode-interrupt-plugin.test.js +458 -0
- package/dist/opencode.js +861 -0
- package/dist/otto/branding.js +22 -0
- package/dist/otto/index.js +21 -0
- package/dist/parse-permission-rules.test.js +117 -0
- package/dist/patch-text-parser.js +97 -0
- package/dist/plugin-logger.js +59 -0
- package/dist/privacy-sanitizer.js +105 -0
- package/dist/queue-advanced-abort.e2e.test.js +293 -0
- package/dist/queue-advanced-action-buttons.e2e.test.js +206 -0
- package/dist/queue-advanced-e2e-setup.js +786 -0
- package/dist/queue-advanced-footer.e2e.test.js +472 -0
- package/dist/queue-advanced-model-switch.e2e.test.js +299 -0
- package/dist/queue-advanced-permissions-typing.e2e.test.js +180 -0
- package/dist/queue-advanced-question.e2e.test.js +261 -0
- package/dist/queue-advanced-typing-interrupt.e2e.test.js +114 -0
- package/dist/queue-advanced-typing.e2e.test.js +153 -0
- package/dist/queue-drain-after-interactive-ui.e2e.test.js +119 -0
- package/dist/queue-interrupt-drain.e2e.test.js +135 -0
- package/dist/queue-question-select-drain.e2e.test.js +120 -0
- package/dist/runtime-idle-sweeper.js +52 -0
- package/dist/runtime-lifecycle.e2e.test.js +508 -0
- package/dist/sentry.js +23 -0
- package/dist/session-handler/agent-utils.js +67 -0
- package/dist/session-handler/event-stream-state.js +420 -0
- package/dist/session-handler/event-stream-state.test.js +563 -0
- package/dist/session-handler/model-utils.js +124 -0
- package/dist/session-handler/opencode-session-event-log.js +94 -0
- package/dist/session-handler/thread-runtime-state.js +104 -0
- package/dist/session-handler/thread-session-runtime.js +3258 -0
- package/dist/session-handler.js +9 -0
- package/dist/session-search.js +100 -0
- package/dist/session-search.test.js +40 -0
- package/dist/session-title-rename.test.js +80 -0
- package/dist/startup-service.js +153 -0
- package/dist/startup-time.e2e.test.js +296 -0
- package/dist/store.js +17 -0
- package/dist/system-message.js +613 -0
- package/dist/system-message.test.js +602 -0
- package/dist/task-runner.js +295 -0
- package/dist/task-schedule.js +209 -0
- package/dist/task-schedule.test.js +71 -0
- package/dist/test-utils.js +299 -0
- package/dist/thinking-utils.js +35 -0
- package/dist/thread-message-queue.e2e.test.js +999 -0
- package/dist/tools.js +357 -0
- package/dist/undo-redo.e2e.test.js +161 -0
- package/dist/unnest-code-blocks.js +146 -0
- package/dist/unnest-code-blocks.test.js +673 -0
- package/dist/upgrade.js +114 -0
- package/dist/utils.js +144 -0
- package/dist/voice-attachment.js +34 -0
- package/dist/voice-handler.js +646 -0
- package/dist/voice-message.e2e.test.js +1021 -0
- package/dist/voice.js +447 -0
- package/dist/voice.test.js +235 -0
- package/dist/wait-session.js +94 -0
- package/dist/websockify.js +69 -0
- package/dist/worker-types.js +4 -0
- package/dist/worktree-lifecycle.e2e.test.js +308 -0
- package/dist/worktree-utils.js +3 -0
- package/dist/worktrees.js +929 -0
- package/dist/worktrees.test.js +189 -0
- package/dist/xml.js +92 -0
- package/dist/xml.test.js +32 -0
- package/package.json +98 -0
- package/schema.prisma +295 -0
- package/skills/batch/SKILL.md +87 -0
- package/skills/critique/SKILL.md +112 -0
- package/skills/egaki/SKILL.md +100 -0
- package/skills/errore/SKILL.md +647 -0
- package/skills/event-sourcing-state/SKILL.md +252 -0
- package/skills/gitchamber/SKILL.md +93 -0
- package/skills/goke/SKILL.md +644 -0
- package/skills/jitter/EDITOR.md +219 -0
- package/skills/jitter/EXPORT-INTERNALS.md +309 -0
- package/skills/jitter/SKILL.md +158 -0
- package/skills/jitter/jitter-clipboard.json +1042 -0
- package/skills/jitter/package.json +14 -0
- package/skills/jitter/tsconfig.json +15 -0
- package/skills/jitter/utils/actions.ts +212 -0
- package/skills/jitter/utils/export.ts +114 -0
- package/skills/jitter/utils/index.ts +141 -0
- package/skills/jitter/utils/snapshot.ts +154 -0
- package/skills/jitter/utils/traverse.ts +246 -0
- package/skills/jitter/utils/types.ts +279 -0
- package/skills/jitter/utils/wait.ts +133 -0
- package/skills/lintcn/SKILL.md +873 -0
- package/skills/new-skill/SKILL.md +211 -0
- package/skills/npm-package/SKILL.md +239 -0
- package/skills/playwriter/SKILL.md +35 -0
- package/skills/proxyman/SKILL.md +215 -0
- package/skills/security-review/SKILL.md +208 -0
- package/skills/simplify/SKILL.md +58 -0
- package/skills/spiceflow/SKILL.md +14 -0
- package/skills/termcast/SKILL.md +945 -0
- package/skills/tuistory/SKILL.md +250 -0
- package/skills/usecomputer/SKILL.md +264 -0
- package/skills/x-articles/SKILL.md +554 -0
- package/skills/zele/SKILL.md +112 -0
- package/skills/zustand-centralized-state/SKILL.md +1004 -0
- package/src/agent-model.e2e.test.ts +976 -0
- package/src/ai-tool-to-genai.test.ts +296 -0
- package/src/ai-tool-to-genai.ts +283 -0
- package/src/ai-tool.ts +39 -0
- package/src/anthropic-auth-plugin.test.ts +159 -0
- package/src/anthropic-auth-plugin.ts +861 -0
- package/src/anthropic-auth-state.ts +282 -0
- package/src/bin.ts +111 -0
- package/src/channel-management.ts +334 -0
- package/src/cli-parsing.test.ts +195 -0
- package/src/cli-send-thread.e2e.test.ts +464 -0
- package/src/cli.ts +4581 -0
- package/src/commands/abort.ts +89 -0
- package/src/commands/action-buttons.ts +364 -0
- package/src/commands/add-project.ts +149 -0
- package/src/commands/agent.ts +473 -0
- package/src/commands/ask-question.ts +390 -0
- package/src/commands/btw.ts +164 -0
- package/src/commands/compact.ts +157 -0
- package/src/commands/context-usage.ts +199 -0
- package/src/commands/create-new-project.ts +190 -0
- package/src/commands/diff.ts +91 -0
- package/src/commands/file-upload.ts +389 -0
- package/src/commands/fork.ts +321 -0
- package/src/commands/gemini-apikey.ts +104 -0
- package/src/commands/login.ts +1173 -0
- package/src/commands/mcp.ts +307 -0
- package/src/commands/memory-snapshot.ts +30 -0
- package/src/commands/mention-mode.ts +68 -0
- package/src/commands/merge-worktree.ts +223 -0
- package/src/commands/model-variant.ts +483 -0
- package/src/commands/model.ts +1053 -0
- package/src/commands/new-worktree.ts +510 -0
- package/src/commands/paginated-select.ts +81 -0
- package/src/commands/permissions.ts +397 -0
- package/src/commands/queue.ts +271 -0
- package/src/commands/remove-project.ts +155 -0
- package/src/commands/restart-opencode-server.ts +162 -0
- package/src/commands/resume.ts +230 -0
- package/src/commands/run-command.ts +123 -0
- package/src/commands/screenshare.test.ts +30 -0
- package/src/commands/screenshare.ts +366 -0
- package/src/commands/session-id.ts +109 -0
- package/src/commands/session.ts +227 -0
- package/src/commands/share.ts +106 -0
- package/src/commands/tasks.ts +293 -0
- package/src/commands/types.ts +25 -0
- package/src/commands/undo-redo.ts +386 -0
- package/src/commands/unset-model.ts +173 -0
- package/src/commands/upgrade.ts +52 -0
- package/src/commands/user-command.ts +198 -0
- package/src/commands/verbosity.ts +173 -0
- package/src/commands/worktree-settings.ts +70 -0
- package/src/commands/worktrees.ts +552 -0
- package/src/condense-memory.ts +36 -0
- package/src/config.ts +111 -0
- package/src/context-awareness-plugin.test.ts +142 -0
- package/src/context-awareness-plugin.ts +510 -0
- package/src/critique-utils.ts +139 -0
- package/src/database.ts +1876 -0
- package/src/db.test.ts +162 -0
- package/src/db.ts +286 -0
- package/src/debounce-timeout.ts +43 -0
- package/src/debounced-process-flush.ts +104 -0
- package/src/discord-bot.ts +1330 -0
- package/src/discord-command-registration.ts +693 -0
- package/src/discord-urls.ts +88 -0
- package/src/discord-utils.test.ts +153 -0
- package/src/discord-utils.ts +800 -0
- package/src/errors.ts +201 -0
- package/src/escape-backticks.test.ts +469 -0
- package/src/event-stream-real-capture.e2e.test.ts +692 -0
- package/src/eventsource-parser.test.ts +351 -0
- package/src/exec-async.ts +35 -0
- package/src/external-opencode-sync.ts +685 -0
- package/src/format-tables.test.ts +335 -0
- package/src/format-tables.ts +445 -0
- package/src/forum-sync/config.ts +92 -0
- package/src/forum-sync/discord-operations.ts +241 -0
- package/src/forum-sync/index.ts +9 -0
- package/src/forum-sync/markdown.ts +172 -0
- package/src/forum-sync/sync-to-discord.ts +595 -0
- package/src/forum-sync/sync-to-files.ts +294 -0
- package/src/forum-sync/types.ts +175 -0
- package/src/forum-sync/watchers.ts +454 -0
- package/src/gateway-proxy-reconnect.e2e.test.ts +523 -0
- package/src/gateway-proxy.e2e.test.ts +640 -0
- package/src/genai-worker-wrapper.ts +164 -0
- package/src/genai-worker.ts +386 -0
- package/src/genai.ts +321 -0
- package/src/generated/browser.ts +114 -0
- package/src/generated/client.ts +138 -0
- package/src/generated/commonInputTypes.ts +736 -0
- package/src/generated/enums.ts +88 -0
- package/src/generated/internal/class.ts +384 -0
- package/src/generated/internal/prismaNamespace.ts +2386 -0
- package/src/generated/internal/prismaNamespaceBrowser.ts +326 -0
- package/src/generated/models/bot_api_keys.ts +1288 -0
- package/src/generated/models/bot_tokens.ts +1656 -0
- package/src/generated/models/channel_agents.ts +1256 -0
- package/src/generated/models/channel_directories.ts +1859 -0
- package/src/generated/models/channel_mention_mode.ts +1300 -0
- package/src/generated/models/channel_models.ts +1288 -0
- package/src/generated/models/channel_verbosity.ts +1228 -0
- package/src/generated/models/channel_worktrees.ts +1300 -0
- package/src/generated/models/forum_sync_configs.ts +1452 -0
- package/src/generated/models/global_models.ts +1288 -0
- package/src/generated/models/ipc_requests.ts +1485 -0
- package/src/generated/models/part_messages.ts +1302 -0
- package/src/generated/models/scheduled_tasks.ts +2320 -0
- package/src/generated/models/session_agents.ts +1086 -0
- package/src/generated/models/session_events.ts +1439 -0
- package/src/generated/models/session_models.ts +1114 -0
- package/src/generated/models/session_start_sources.ts +1408 -0
- package/src/generated/models/thread_sessions.ts +1781 -0
- package/src/generated/models/thread_worktrees.ts +1356 -0
- package/src/generated/models.ts +30 -0
- package/src/heap-monitor.ts +152 -0
- package/src/hrana-server.test.ts +434 -0
- package/src/hrana-server.ts +314 -0
- package/src/html-actions.test.ts +87 -0
- package/src/html-actions.ts +174 -0
- package/src/html-components.test.ts +38 -0
- package/src/html-components.ts +181 -0
- package/src/image-optimizer-plugin.ts +194 -0
- package/src/image-utils.ts +149 -0
- package/src/interaction-handler.ts +576 -0
- package/src/ipc-polling.ts +326 -0
- package/src/ipc-tools-plugin.ts +236 -0
- package/src/kimaki-digital-twin.e2e.test.ts +199 -0
- package/src/kimaki-opencode-plugin-loading.e2e.test.ts +109 -0
- package/src/kimaki-opencode-plugin.test.ts +108 -0
- package/src/kimaki-opencode-plugin.ts +18 -0
- package/src/limit-heading-depth.test.ts +116 -0
- package/src/limit-heading-depth.ts +26 -0
- package/src/logger.ts +208 -0
- package/src/markdown.test.ts +308 -0
- package/src/markdown.ts +410 -0
- package/src/message-finish-field.e2e.test.ts +192 -0
- package/src/message-formatting.test.ts +81 -0
- package/src/message-formatting.ts +533 -0
- package/src/message-preprocessing.ts +455 -0
- package/src/onboarding-tutorial.ts +176 -0
- package/src/onboarding-welcome.ts +49 -0
- package/src/openai-realtime.ts +358 -0
- package/src/opencode-command-detection.test.ts +307 -0
- package/src/opencode-command-detection.ts +76 -0
- package/src/opencode-command.test.ts +70 -0
- package/src/opencode-command.ts +188 -0
- package/src/opencode-interrupt-plugin.test.ts +677 -0
- package/src/opencode-interrupt-plugin.ts +477 -0
- package/src/opencode.ts +1110 -0
- package/src/otto/branding.ts +23 -0
- package/src/otto/index.ts +22 -0
- package/src/parse-permission-rules.test.ts +127 -0
- package/src/patch-text-parser.ts +107 -0
- package/src/plugin-logger.ts +68 -0
- package/src/privacy-sanitizer.ts +142 -0
- package/src/queue-advanced-abort.e2e.test.ts +382 -0
- package/src/queue-advanced-action-buttons.e2e.test.ts +268 -0
- package/src/queue-advanced-e2e-setup.ts +873 -0
- package/src/queue-advanced-footer.e2e.test.ts +576 -0
- package/src/queue-advanced-model-switch.e2e.test.ts +383 -0
- package/src/queue-advanced-permissions-typing.e2e.test.ts +245 -0
- package/src/queue-advanced-question.e2e.test.ts +316 -0
- package/src/queue-advanced-typing-interrupt.e2e.test.ts +146 -0
- package/src/queue-advanced-typing.e2e.test.ts +199 -0
- package/src/queue-drain-after-interactive-ui.e2e.test.ts +151 -0
- package/src/queue-interrupt-drain.e2e.test.ts +166 -0
- package/src/queue-question-select-drain.e2e.test.ts +152 -0
- package/src/runtime-idle-sweeper.ts +76 -0
- package/src/runtime-lifecycle.e2e.test.ts +641 -0
- package/src/schema.sql +173 -0
- package/src/sentry.ts +26 -0
- package/src/session-handler/agent-utils.ts +97 -0
- package/src/session-handler/event-stream-fixtures/real-session-action-buttons.jsonl +45 -0
- package/src/session-handler/event-stream-fixtures/real-session-footer-suppressed-on-pre-idle-interrupt.jsonl +40 -0
- package/src/session-handler/event-stream-fixtures/real-session-permission-external-file.jsonl +23 -0
- package/src/session-handler/event-stream-fixtures/real-session-task-normal.jsonl +22 -0
- package/src/session-handler/event-stream-fixtures/real-session-task-three-parallel-sleeps.jsonl +277 -0
- package/src/session-handler/event-stream-fixtures/real-session-task-user-interruption.jsonl +46 -0
- package/src/session-handler/event-stream-fixtures/session-abort-after-idle-race.jsonl +21 -0
- package/src/session-handler/event-stream-fixtures/session-concurrent-messages-serialized.jsonl +56 -0
- package/src/session-handler/event-stream-fixtures/session-explicit-abort.jsonl +44 -0
- package/src/session-handler/event-stream-fixtures/session-normal-completion.jsonl +29 -0
- package/src/session-handler/event-stream-fixtures/session-tool-call-noisy-stream.jsonl +29 -0
- package/src/session-handler/event-stream-fixtures/session-two-completions-same-session.jsonl +50 -0
- package/src/session-handler/event-stream-fixtures/session-user-interruption.jsonl +59 -0
- package/src/session-handler/event-stream-fixtures/session-voice-queued-followup.jsonl +52 -0
- package/src/session-handler/event-stream-state.test.ts +645 -0
- package/src/session-handler/event-stream-state.ts +608 -0
- package/src/session-handler/model-utils.ts +183 -0
- package/src/session-handler/opencode-session-event-log.ts +130 -0
- package/src/session-handler/thread-runtime-state.ts +212 -0
- package/src/session-handler/thread-session-runtime.ts +4281 -0
- package/src/session-handler.ts +15 -0
- package/src/session-search.test.ts +50 -0
- package/src/session-search.ts +148 -0
- package/src/session-title-rename.test.ts +112 -0
- package/src/startup-service.ts +200 -0
- package/src/startup-time.e2e.test.ts +373 -0
- package/src/store.ts +122 -0
- package/src/system-message.test.ts +612 -0
- package/src/system-message.ts +723 -0
- package/src/task-runner.ts +421 -0
- package/src/task-schedule.test.ts +84 -0
- package/src/task-schedule.ts +311 -0
- package/src/test-utils.ts +435 -0
- package/src/thinking-utils.ts +61 -0
- package/src/thread-message-queue.e2e.test.ts +1219 -0
- package/src/tools.ts +430 -0
- package/src/undici.d.ts +12 -0
- package/src/undo-redo.e2e.test.ts +209 -0
- package/src/unnest-code-blocks.test.ts +713 -0
- package/src/unnest-code-blocks.ts +185 -0
- package/src/upgrade.ts +127 -0
- package/src/utils.ts +212 -0
- package/src/voice-attachment.ts +51 -0
- package/src/voice-handler.ts +908 -0
- package/src/voice-message.e2e.test.ts +1255 -0
- package/src/voice.test.ts +281 -0
- package/src/voice.ts +627 -0
- package/src/wait-session.ts +147 -0
- package/src/websockify.ts +101 -0
- package/src/worker-types.ts +64 -0
- package/src/worktree-lifecycle.e2e.test.ts +391 -0
- package/src/worktree-utils.ts +4 -0
- package/src/worktrees.test.ts +223 -0
- package/src/worktrees.ts +1294 -0
- package/src/xml.test.ts +38 -0
- package/src/xml.ts +121 -0
|
@@ -0,0 +1,473 @@
|
|
|
1
|
+
// /agent command - Set the preferred agent for this channel or session.
|
|
2
|
+
// Also provides quick agent commands like /plan-agent, /build-agent that switch instantly.
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
ChatInputCommandInteraction,
|
|
6
|
+
StringSelectMenuInteraction,
|
|
7
|
+
StringSelectMenuBuilder,
|
|
8
|
+
ActionRowBuilder,
|
|
9
|
+
ChannelType,
|
|
10
|
+
type ThreadChannel,
|
|
11
|
+
type TextChannel,
|
|
12
|
+
MessageFlags,
|
|
13
|
+
} from 'discord.js'
|
|
14
|
+
import crypto from 'node:crypto'
|
|
15
|
+
import {
|
|
16
|
+
setChannelAgent,
|
|
17
|
+
setSessionAgent,
|
|
18
|
+
clearSessionModel,
|
|
19
|
+
getThreadSession,
|
|
20
|
+
getSessionAgent,
|
|
21
|
+
getChannelAgent,
|
|
22
|
+
} from '../database.js'
|
|
23
|
+
import { initializeOpencodeForDirectory } from '../opencode.js'
|
|
24
|
+
import { resolveTextChannel, getKimakiMetadata } from '../discord-utils.js'
|
|
25
|
+
import { createLogger, LogPrefix } from '../logger.js'
|
|
26
|
+
|
|
27
|
+
const agentLogger = createLogger(LogPrefix.AGENT)
|
|
28
|
+
|
|
29
|
+
const AGENT_CONTEXT_TTL_MS = 10 * 60 * 1000
|
|
30
|
+
const pendingAgentContexts = new Map<
|
|
31
|
+
string,
|
|
32
|
+
{
|
|
33
|
+
dir: string
|
|
34
|
+
channelId: string
|
|
35
|
+
sessionId?: string
|
|
36
|
+
isThread: boolean
|
|
37
|
+
}
|
|
38
|
+
>()
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Context for agent commands, containing channel/session info.
|
|
42
|
+
*/
|
|
43
|
+
export type AgentCommandContext = {
|
|
44
|
+
dir: string
|
|
45
|
+
channelId: string
|
|
46
|
+
sessionId?: string
|
|
47
|
+
isThread: boolean
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export type CurrentAgentInfo =
|
|
51
|
+
| { type: 'session'; agent: string }
|
|
52
|
+
| { type: 'channel'; agent: string }
|
|
53
|
+
| { type: 'none' }
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Get the current agent info for a channel/session, including where it comes from.
|
|
57
|
+
* Priority: session > channel > none
|
|
58
|
+
*/
|
|
59
|
+
export async function getCurrentAgentInfo({
|
|
60
|
+
sessionId,
|
|
61
|
+
channelId,
|
|
62
|
+
}: {
|
|
63
|
+
sessionId?: string
|
|
64
|
+
channelId?: string
|
|
65
|
+
}): Promise<CurrentAgentInfo> {
|
|
66
|
+
if (sessionId) {
|
|
67
|
+
const sessionAgent = await getSessionAgent(sessionId)
|
|
68
|
+
if (sessionAgent) {
|
|
69
|
+
return { type: 'session', agent: sessionAgent }
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
if (channelId) {
|
|
73
|
+
const channelAgent = await getChannelAgent(channelId)
|
|
74
|
+
if (channelAgent) {
|
|
75
|
+
return { type: 'channel', agent: channelAgent }
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
return { type: 'none' }
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Sanitize an agent name to be a valid Discord command name component.
|
|
83
|
+
* Lowercase, alphanumeric and hyphens only.
|
|
84
|
+
*/
|
|
85
|
+
export function sanitizeAgentName(name: string): string {
|
|
86
|
+
return name
|
|
87
|
+
.toLowerCase()
|
|
88
|
+
.replace(/[^a-z0-9-]/g, '-')
|
|
89
|
+
.replace(/-+/g, '-')
|
|
90
|
+
.replace(/^-|-$/g, '')
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const QUICK_AGENT_DESCRIPTION_PATTERN = /^\[agent:([^\]]+)\]/
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Build quick-agent command description with an embedded original agent name.
|
|
97
|
+
* Metadata format: [agent:<original-name>] <visible description>
|
|
98
|
+
*/
|
|
99
|
+
export function buildQuickAgentCommandDescription({
|
|
100
|
+
agentName,
|
|
101
|
+
description,
|
|
102
|
+
}: {
|
|
103
|
+
agentName: string
|
|
104
|
+
description?: string
|
|
105
|
+
}): string {
|
|
106
|
+
const metadataPrefix = `[agent:${agentName}]`
|
|
107
|
+
if (metadataPrefix.length > 100) {
|
|
108
|
+
return metadataPrefix.slice(0, 100)
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
const visibleDescription = description || `Switch to ${agentName} agent`
|
|
112
|
+
const maxVisibleLength = 100 - metadataPrefix.length - 1
|
|
113
|
+
|
|
114
|
+
if (maxVisibleLength <= 0) {
|
|
115
|
+
return metadataPrefix
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
const trimmedVisible = visibleDescription.slice(0, maxVisibleLength).trim()
|
|
119
|
+
if (!trimmedVisible) {
|
|
120
|
+
return metadataPrefix
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
return `${metadataPrefix} ${trimmedVisible}`
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
function parseQuickAgentNameFromDescription(
|
|
127
|
+
description: string | undefined,
|
|
128
|
+
): string | undefined {
|
|
129
|
+
if (!description) {
|
|
130
|
+
return undefined
|
|
131
|
+
}
|
|
132
|
+
const match = QUICK_AGENT_DESCRIPTION_PATTERN.exec(description)
|
|
133
|
+
if (!match) {
|
|
134
|
+
return undefined
|
|
135
|
+
}
|
|
136
|
+
const agentName = match[1]?.trim()
|
|
137
|
+
if (!agentName) {
|
|
138
|
+
return undefined
|
|
139
|
+
}
|
|
140
|
+
return agentName
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
async function resolveQuickAgentNameFromInteraction({
|
|
144
|
+
command,
|
|
145
|
+
}: {
|
|
146
|
+
command: ChatInputCommandInteraction
|
|
147
|
+
}): Promise<string | undefined> {
|
|
148
|
+
const fromCommandObject = parseQuickAgentNameFromDescription(
|
|
149
|
+
command.command?.description,
|
|
150
|
+
)
|
|
151
|
+
if (fromCommandObject) {
|
|
152
|
+
return fromCommandObject
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
if (!command.guild) {
|
|
156
|
+
return undefined
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
const fetchedCommand = await command.guild.commands.fetch(command.commandId)
|
|
160
|
+
if (!fetchedCommand) {
|
|
161
|
+
return undefined
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
return parseQuickAgentNameFromDescription(fetchedCommand.description)
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Resolve the context for an agent command (directory, channel, session).
|
|
169
|
+
* Returns null if the command cannot be executed in this context.
|
|
170
|
+
*/
|
|
171
|
+
export async function resolveAgentCommandContext({
|
|
172
|
+
interaction,
|
|
173
|
+
}: {
|
|
174
|
+
interaction: ChatInputCommandInteraction
|
|
175
|
+
appId: string
|
|
176
|
+
}): Promise<AgentCommandContext | null> {
|
|
177
|
+
const channel = interaction.channel
|
|
178
|
+
|
|
179
|
+
if (!channel) {
|
|
180
|
+
await interaction.editReply({
|
|
181
|
+
content: 'This command can only be used in a channel',
|
|
182
|
+
})
|
|
183
|
+
return null
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
const isThread = [
|
|
187
|
+
ChannelType.PublicThread,
|
|
188
|
+
ChannelType.PrivateThread,
|
|
189
|
+
ChannelType.AnnouncementThread,
|
|
190
|
+
].includes(channel.type)
|
|
191
|
+
|
|
192
|
+
let projectDirectory: string | undefined
|
|
193
|
+
let targetChannelId: string
|
|
194
|
+
let sessionId: string | undefined
|
|
195
|
+
|
|
196
|
+
if (isThread) {
|
|
197
|
+
const thread = channel as ThreadChannel
|
|
198
|
+
const textChannel = await resolveTextChannel(thread)
|
|
199
|
+
const metadata = await getKimakiMetadata(textChannel)
|
|
200
|
+
projectDirectory = metadata.projectDirectory
|
|
201
|
+
targetChannelId = textChannel?.id || channel.id
|
|
202
|
+
|
|
203
|
+
sessionId = await getThreadSession(thread.id)
|
|
204
|
+
} else if (channel.type === ChannelType.GuildText) {
|
|
205
|
+
const textChannel = channel as TextChannel
|
|
206
|
+
const metadata = await getKimakiMetadata(textChannel)
|
|
207
|
+
projectDirectory = metadata.projectDirectory
|
|
208
|
+
targetChannelId = channel.id
|
|
209
|
+
} else {
|
|
210
|
+
await interaction.editReply({
|
|
211
|
+
content: 'This command can only be used in text channels or threads',
|
|
212
|
+
})
|
|
213
|
+
return null
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
if (!projectDirectory) {
|
|
217
|
+
await interaction.editReply({
|
|
218
|
+
content: 'This channel is not configured with a project directory',
|
|
219
|
+
})
|
|
220
|
+
return null
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
return {
|
|
224
|
+
dir: projectDirectory,
|
|
225
|
+
channelId: targetChannelId,
|
|
226
|
+
sessionId,
|
|
227
|
+
isThread,
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
/**
|
|
232
|
+
* Set the agent preference for a context (session or channel).
|
|
233
|
+
* When switching agents for a session, clears session model preference
|
|
234
|
+
* so the new agent's model takes effect (agent model > channel model).
|
|
235
|
+
*/
|
|
236
|
+
export async function setAgentForContext({
|
|
237
|
+
context,
|
|
238
|
+
agentName,
|
|
239
|
+
}: {
|
|
240
|
+
context: AgentCommandContext
|
|
241
|
+
agentName: string
|
|
242
|
+
}): Promise<void> {
|
|
243
|
+
if (context.isThread && context.sessionId) {
|
|
244
|
+
await setSessionAgent(context.sessionId, agentName)
|
|
245
|
+
// Clear session model so the new agent's model takes effect
|
|
246
|
+
await clearSessionModel(context.sessionId)
|
|
247
|
+
agentLogger.log(
|
|
248
|
+
`Set agent ${agentName} for session ${context.sessionId} (cleared session model)`,
|
|
249
|
+
)
|
|
250
|
+
} else {
|
|
251
|
+
await setChannelAgent(context.channelId, agentName)
|
|
252
|
+
agentLogger.log(`Set agent ${agentName} for channel ${context.channelId}`)
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
export async function handleAgentCommand({
|
|
257
|
+
interaction,
|
|
258
|
+
appId,
|
|
259
|
+
}: {
|
|
260
|
+
interaction: ChatInputCommandInteraction
|
|
261
|
+
appId: string
|
|
262
|
+
}): Promise<void> {
|
|
263
|
+
await interaction.deferReply({ flags: MessageFlags.Ephemeral })
|
|
264
|
+
|
|
265
|
+
const context = await resolveAgentCommandContext({ interaction, appId })
|
|
266
|
+
if (!context) {
|
|
267
|
+
return
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
try {
|
|
271
|
+
const getClient = await initializeOpencodeForDirectory(context.dir)
|
|
272
|
+
if (getClient instanceof Error) {
|
|
273
|
+
await interaction.editReply({ content: getClient.message })
|
|
274
|
+
return
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
const agentsResponse = await getClient().app.agents({
|
|
278
|
+
directory: context.dir,
|
|
279
|
+
})
|
|
280
|
+
|
|
281
|
+
if (!agentsResponse.data || agentsResponse.data.length === 0) {
|
|
282
|
+
await interaction.editReply({ content: 'No agents available' })
|
|
283
|
+
return
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
const agents = agentsResponse.data
|
|
287
|
+
.filter((agent) => {
|
|
288
|
+
const hidden = (agent as { hidden?: boolean }).hidden
|
|
289
|
+
return (agent.mode === 'primary' || agent.mode === 'all') && !hidden
|
|
290
|
+
})
|
|
291
|
+
.slice(0, 25)
|
|
292
|
+
|
|
293
|
+
if (agents.length === 0) {
|
|
294
|
+
await interaction.editReply({ content: 'No primary agents available' })
|
|
295
|
+
return
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
const currentAgentInfo = await getCurrentAgentInfo({
|
|
299
|
+
sessionId: context.sessionId,
|
|
300
|
+
channelId: context.channelId,
|
|
301
|
+
})
|
|
302
|
+
|
|
303
|
+
const currentAgentText = (() => {
|
|
304
|
+
switch (currentAgentInfo.type) {
|
|
305
|
+
case 'session':
|
|
306
|
+
return `**Current (session override):** \`${currentAgentInfo.agent}\``
|
|
307
|
+
case 'channel':
|
|
308
|
+
return `**Current (channel override):** \`${currentAgentInfo.agent}\``
|
|
309
|
+
case 'none':
|
|
310
|
+
return '**Current:** none'
|
|
311
|
+
}
|
|
312
|
+
})()
|
|
313
|
+
|
|
314
|
+
const contextHash = crypto.randomBytes(8).toString('hex')
|
|
315
|
+
pendingAgentContexts.set(contextHash, context)
|
|
316
|
+
setTimeout(() => {
|
|
317
|
+
pendingAgentContexts.delete(contextHash)
|
|
318
|
+
}, AGENT_CONTEXT_TTL_MS).unref()
|
|
319
|
+
|
|
320
|
+
const options = agents.map((agent) => ({
|
|
321
|
+
label: agent.name.slice(0, 100),
|
|
322
|
+
value: agent.name,
|
|
323
|
+
description: (agent.description || `${agent.mode} agent`).slice(0, 100),
|
|
324
|
+
}))
|
|
325
|
+
|
|
326
|
+
const selectMenu = new StringSelectMenuBuilder()
|
|
327
|
+
.setCustomId(`agent_select:${contextHash}`)
|
|
328
|
+
.setPlaceholder('Select an agent')
|
|
329
|
+
.addOptions(options)
|
|
330
|
+
|
|
331
|
+
const actionRow =
|
|
332
|
+
new ActionRowBuilder<StringSelectMenuBuilder>().addComponents(selectMenu)
|
|
333
|
+
|
|
334
|
+
await interaction.editReply({
|
|
335
|
+
content: `**Set Agent Preference**\n${currentAgentText}\nSelect an agent:`,
|
|
336
|
+
components: [actionRow],
|
|
337
|
+
})
|
|
338
|
+
} catch (error) {
|
|
339
|
+
agentLogger.error('Error loading agents:', error)
|
|
340
|
+
await interaction.editReply({
|
|
341
|
+
content: `Failed to load agents: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
342
|
+
})
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
export async function handleAgentSelectMenu(
|
|
347
|
+
interaction: StringSelectMenuInteraction,
|
|
348
|
+
): Promise<void> {
|
|
349
|
+
const customId = interaction.customId
|
|
350
|
+
|
|
351
|
+
if (!customId.startsWith('agent_select:')) {
|
|
352
|
+
return
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
await interaction.deferUpdate()
|
|
356
|
+
|
|
357
|
+
const contextHash = customId.replace('agent_select:', '')
|
|
358
|
+
const context = pendingAgentContexts.get(contextHash)
|
|
359
|
+
|
|
360
|
+
if (!context) {
|
|
361
|
+
await interaction.editReply({
|
|
362
|
+
content: 'Selection expired. Please run /agent again.',
|
|
363
|
+
components: [],
|
|
364
|
+
})
|
|
365
|
+
return
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
const selectedAgent = interaction.values[0]
|
|
369
|
+
if (!selectedAgent) {
|
|
370
|
+
await interaction.editReply({
|
|
371
|
+
content: 'No agent selected',
|
|
372
|
+
components: [],
|
|
373
|
+
})
|
|
374
|
+
return
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
try {
|
|
378
|
+
await setAgentForContext({ context, agentName: selectedAgent })
|
|
379
|
+
|
|
380
|
+
if (context.isThread && context.sessionId) {
|
|
381
|
+
await interaction.editReply({
|
|
382
|
+
content: `Agent preference set for this session next messages: **${selectedAgent}**`,
|
|
383
|
+
components: [],
|
|
384
|
+
})
|
|
385
|
+
} else {
|
|
386
|
+
await interaction.editReply({
|
|
387
|
+
content: `Agent preference set for this channel: **${selectedAgent}**\nAll new sessions in this channel will use this agent.`,
|
|
388
|
+
components: [],
|
|
389
|
+
})
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
pendingAgentContexts.delete(contextHash)
|
|
393
|
+
} catch (error) {
|
|
394
|
+
agentLogger.error('Error saving agent preference:', error)
|
|
395
|
+
await interaction.editReply({
|
|
396
|
+
content: `Failed to save agent preference: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
397
|
+
components: [],
|
|
398
|
+
})
|
|
399
|
+
}
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
/**
|
|
403
|
+
* Handle quick agent commands like /plan-agent, /build-agent.
|
|
404
|
+
* These instantly switch to the specified agent without showing a dropdown.
|
|
405
|
+
*
|
|
406
|
+
* The slash command name is sanitized for Discord and can be lossy
|
|
407
|
+
* (for example gpt5.4 -> gpt5-4-agent). To keep the original agent name,
|
|
408
|
+
* registration stores [agent:<name>] metadata in the description and this
|
|
409
|
+
* handler resolves from that metadata first.
|
|
410
|
+
*/
|
|
411
|
+
export async function handleQuickAgentCommand({
|
|
412
|
+
command,
|
|
413
|
+
appId,
|
|
414
|
+
}: {
|
|
415
|
+
command: ChatInputCommandInteraction
|
|
416
|
+
appId: string
|
|
417
|
+
}): Promise<void> {
|
|
418
|
+
const fallbackAgentName = command.commandName.replace(/-agent$/, '')
|
|
419
|
+
|
|
420
|
+
await command.deferReply({ flags: MessageFlags.Ephemeral })
|
|
421
|
+
|
|
422
|
+
const context = await resolveAgentCommandContext({
|
|
423
|
+
interaction: command,
|
|
424
|
+
appId,
|
|
425
|
+
})
|
|
426
|
+
if (!context) {
|
|
427
|
+
return
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
try {
|
|
431
|
+
const resolvedAgentName =
|
|
432
|
+
(await resolveQuickAgentNameFromInteraction({ command })) ||
|
|
433
|
+
fallbackAgentName
|
|
434
|
+
|
|
435
|
+
// Check current agent and set new one.
|
|
436
|
+
// getCurrentAgentInfo is fast (DB only), use it for the "was X" text.
|
|
437
|
+
const previousAgent = await getCurrentAgentInfo({
|
|
438
|
+
sessionId: context.sessionId,
|
|
439
|
+
channelId: context.channelId,
|
|
440
|
+
})
|
|
441
|
+
const previousAgentName =
|
|
442
|
+
previousAgent.type !== 'none' ? previousAgent.agent : undefined
|
|
443
|
+
|
|
444
|
+
if (previousAgentName === resolvedAgentName) {
|
|
445
|
+
await command.editReply({
|
|
446
|
+
content: `Already using **${resolvedAgentName}** agent`,
|
|
447
|
+
})
|
|
448
|
+
return
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
// Set the agent preference in DB for this context.
|
|
452
|
+
await setAgentForContext({ context, agentName: resolvedAgentName })
|
|
453
|
+
|
|
454
|
+
const previousText = previousAgentName
|
|
455
|
+
? ` (was **${previousAgentName}**)`
|
|
456
|
+
: ''
|
|
457
|
+
|
|
458
|
+
if (context.isThread && context.sessionId) {
|
|
459
|
+
await command.editReply({
|
|
460
|
+
content: `Switched to **${resolvedAgentName}** agent for this session next messages${previousText}`,
|
|
461
|
+
})
|
|
462
|
+
} else {
|
|
463
|
+
await command.editReply({
|
|
464
|
+
content: `Switched to **${resolvedAgentName}** agent for this channel${previousText}\nAll new sessions will use this agent.`,
|
|
465
|
+
})
|
|
466
|
+
}
|
|
467
|
+
} catch (error) {
|
|
468
|
+
agentLogger.error('Error in quick agent command:', error)
|
|
469
|
+
await command.editReply({
|
|
470
|
+
content: `Failed to switch agent: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
471
|
+
})
|
|
472
|
+
}
|
|
473
|
+
}
|