zardbot-telegram 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. package/.env.example +116 -0
  2. package/LICENSE +21 -0
  3. package/README.md +250 -0
  4. package/dist/agent/manager.js +88 -0
  5. package/dist/agent/types.js +26 -0
  6. package/dist/app/start-bot-app.js +49 -0
  7. package/dist/bot/commands/abort.js +121 -0
  8. package/dist/bot/commands/commands.js +480 -0
  9. package/dist/bot/commands/definitions.js +27 -0
  10. package/dist/bot/commands/help.js +10 -0
  11. package/dist/bot/commands/models.js +38 -0
  12. package/dist/bot/commands/new.js +70 -0
  13. package/dist/bot/commands/opencode-start.js +101 -0
  14. package/dist/bot/commands/opencode-stop.js +44 -0
  15. package/dist/bot/commands/projects.js +223 -0
  16. package/dist/bot/commands/rename.js +139 -0
  17. package/dist/bot/commands/sessions.js +351 -0
  18. package/dist/bot/commands/start.js +43 -0
  19. package/dist/bot/commands/status.js +95 -0
  20. package/dist/bot/commands/task.js +399 -0
  21. package/dist/bot/commands/tasklist.js +220 -0
  22. package/dist/bot/commands/voice.js +145 -0
  23. package/dist/bot/handlers/agent.js +118 -0
  24. package/dist/bot/handlers/context.js +100 -0
  25. package/dist/bot/handlers/document.js +65 -0
  26. package/dist/bot/handlers/inline-menu.js +119 -0
  27. package/dist/bot/handlers/model.js +143 -0
  28. package/dist/bot/handlers/permission.js +235 -0
  29. package/dist/bot/handlers/prompt.js +240 -0
  30. package/dist/bot/handlers/question.js +390 -0
  31. package/dist/bot/handlers/tts.js +89 -0
  32. package/dist/bot/handlers/variant.js +138 -0
  33. package/dist/bot/handlers/voice.js +173 -0
  34. package/dist/bot/index.js +977 -0
  35. package/dist/bot/message-patterns.js +4 -0
  36. package/dist/bot/middleware/auth.js +30 -0
  37. package/dist/bot/middleware/interaction-guard.js +95 -0
  38. package/dist/bot/middleware/unknown-command.js +22 -0
  39. package/dist/bot/streaming/response-streamer.js +286 -0
  40. package/dist/bot/streaming/tool-call-streamer.js +285 -0
  41. package/dist/bot/utils/busy-guard.js +15 -0
  42. package/dist/bot/utils/commands.js +21 -0
  43. package/dist/bot/utils/file-download.js +91 -0
  44. package/dist/bot/utils/finalize-assistant-response.js +52 -0
  45. package/dist/bot/utils/keyboard.js +69 -0
  46. package/dist/bot/utils/send-with-markdown-fallback.js +165 -0
  47. package/dist/bot/utils/telegram-text.js +28 -0
  48. package/dist/bot/utils/thinking-message.js +8 -0
  49. package/dist/cli/args.js +98 -0
  50. package/dist/cli.js +80 -0
  51. package/dist/config.js +97 -0
  52. package/dist/i18n/de.js +357 -0
  53. package/dist/i18n/en.js +357 -0
  54. package/dist/i18n/es.js +357 -0
  55. package/dist/i18n/fr.js +357 -0
  56. package/dist/i18n/index.js +109 -0
  57. package/dist/i18n/ru.js +357 -0
  58. package/dist/i18n/zh.js +357 -0
  59. package/dist/index.js +26 -0
  60. package/dist/interaction/busy.js +8 -0
  61. package/dist/interaction/cleanup.js +32 -0
  62. package/dist/interaction/guard.js +140 -0
  63. package/dist/interaction/manager.js +106 -0
  64. package/dist/interaction/types.js +1 -0
  65. package/dist/keyboard/manager.js +172 -0
  66. package/dist/keyboard/types.js +1 -0
  67. package/dist/model/capabilities.js +62 -0
  68. package/dist/model/context-limit.js +57 -0
  69. package/dist/model/manager.js +259 -0
  70. package/dist/model/types.js +24 -0
  71. package/dist/opencode/client.js +13 -0
  72. package/dist/opencode/events.js +140 -0
  73. package/dist/permission/manager.js +100 -0
  74. package/dist/permission/types.js +1 -0
  75. package/dist/pinned/format.js +29 -0
  76. package/dist/pinned/manager.js +682 -0
  77. package/dist/pinned/types.js +1 -0
  78. package/dist/process/manager.js +273 -0
  79. package/dist/process/types.js +1 -0
  80. package/dist/project/manager.js +88 -0
  81. package/dist/question/manager.js +176 -0
  82. package/dist/question/types.js +1 -0
  83. package/dist/rename/manager.js +53 -0
  84. package/dist/runtime/bootstrap.js +350 -0
  85. package/dist/runtime/mode.js +74 -0
  86. package/dist/runtime/paths.js +37 -0
  87. package/dist/scheduled-task/creation-manager.js +113 -0
  88. package/dist/scheduled-task/display.js +239 -0
  89. package/dist/scheduled-task/executor.js +87 -0
  90. package/dist/scheduled-task/foreground-state.js +32 -0
  91. package/dist/scheduled-task/next-run.js +207 -0
  92. package/dist/scheduled-task/runtime.js +368 -0
  93. package/dist/scheduled-task/schedule-parser.js +169 -0
  94. package/dist/scheduled-task/store.js +65 -0
  95. package/dist/scheduled-task/types.js +19 -0
  96. package/dist/session/cache-manager.js +455 -0
  97. package/dist/session/manager.js +10 -0
  98. package/dist/settings/manager.js +158 -0
  99. package/dist/stt/client.js +97 -0
  100. package/dist/summary/aggregator.js +1136 -0
  101. package/dist/summary/formatter.js +491 -0
  102. package/dist/summary/subagent-formatter.js +63 -0
  103. package/dist/summary/tool-message-batcher.js +90 -0
  104. package/dist/tts/client.js +130 -0
  105. package/dist/utils/error-format.js +29 -0
  106. package/dist/utils/logger.js +127 -0
  107. package/dist/utils/safe-background-task.js +33 -0
  108. package/dist/utils/telegram-rate-limit-retry.js +93 -0
  109. package/dist/variant/manager.js +103 -0
  110. package/dist/variant/types.js +1 -0
  111. package/package.json +79 -0
@@ -0,0 +1,357 @@
1
+ export const zh = {
2
+ "cmd.description.status": "服务器和会话状态",
3
+ "cmd.description.new": "创建新会话",
4
+ "cmd.description.stop": "停止当前操作",
5
+ "cmd.description.sessions": "列出会话",
6
+ "cmd.description.projects": "列出项目",
7
+ "cmd.description.task": "创建定时任务",
8
+ "cmd.description.tasklist": "查看定时任务",
9
+ "cmd.description.commands": "自定义命令",
10
+ "cmd.description.voice": "选择 TTS 语音",
11
+ "cmd.description.opencode_start": "启动 OpenCode 服务器",
12
+ "cmd.description.opencode_stop": "停止 OpenCode 服务器",
13
+ "cmd.description.help": "帮助",
14
+ "callback.unknown_command": "未知命令",
15
+ "callback.processing_error": "处理错误",
16
+ "error.load_agents": "❌ 加载代理列表失败",
17
+ "error.load_models": "❌ 加载模型列表失败",
18
+ "error.load_variants": "❌ 加载变体列表失败",
19
+ "error.context_button": "❌ 处理上下文按钮失败",
20
+ "error.generic": "🔴 出现了一些问题。",
21
+ "interaction.blocked.expired": "⚠️ 此交互已过期。请重新开始。",
22
+ "interaction.blocked.expected_callback": "⚠️ 此步骤请使用内联按钮,或点击取消。",
23
+ "interaction.blocked.expected_text": "⚠️ 此步骤请发送一条文本消息。",
24
+ "interaction.blocked.expected_command": "⚠️ 此步骤请发送一条命令。",
25
+ "interaction.blocked.command_not_allowed": "⚠️ 当前步骤不可用此命令。",
26
+ "interaction.blocked.finish_current": "⚠️ 请先完成当前交互(回答或取消),然后再打开其他菜单。",
27
+ "inline.blocked.expected_choice": "⚠️ 请使用内联按钮选择一个选项,或点击取消。",
28
+ "inline.blocked.command_not_allowed": "⚠️ 内联菜单激活期间不可用此命令。",
29
+ "question.blocked.expected_answer": "⚠️ 请使用按钮、自定义回答或取消来回答当前问题。",
30
+ "question.blocked.command_not_allowed": "⚠️ 在当前问答流程完成之前不可用此命令。",
31
+ "inline.button.cancel": "❌ 取消",
32
+ "inline.inactive_callback": "此菜单已失效",
33
+ "inline.cancelled_callback": "已取消",
34
+ "common.unknown": "未知",
35
+ "common.unknown_error": "未知错误",
36
+ "start.welcome": "👋 欢迎使用 OpenCode Telegram Bot!\n\n可用命令:\n/projects — 选择项目\n/sessions — 会话列表\n/new — 新建会话\n/task — 定时任务\n/tasklist — 定时任务列表\n/status — 状态\n/help — 帮助\n\n请使用底部按钮选择模式、模型和变体。",
37
+ "help.keyboard_hint": "💡 代理模式、模型、变体和上下文操作请使用底部键盘按钮。",
38
+ "help.text": "📖 **帮助**\n\n/status - 查看服务器状态\n/sessions - 会话列表\n/new - 创建新会话\n/help - 帮助",
39
+ "bot.thinking": "💭 思考中...",
40
+ "bot.project_not_selected": "🏗 未选择项目。\n\n请先使用 /projects 选择一个项目。",
41
+ "bot.creating_session": "🔄 正在创建新会话...",
42
+ "bot.create_session_error": "🔴 创建会话失败。请重试 /new,或使用 /status 检查服务器状态。",
43
+ "bot.session_created": "✅ 会话已创建:{title}",
44
+ "bot.session_busy": "⏳ 代理正在执行任务。请等待完成,或使用 /abort 中断当前运行。",
45
+ "bot.session_reset_project_mismatch": "⚠️ 活动会话与所选项目不匹配,因此已重置。使用 /sessions 选择一个会话,或 /new 创建新会话。",
46
+ "bot.prompt_send_error": "向 OpenCode 发送请求失败。",
47
+ "bot.session_error": "🔴 OpenCode 返回错误:{message}",
48
+ "bot.session_retry": "🔁 {message}\n\n提供方在重复重试时持续返回同一错误。使用 /abort 可停止。",
49
+ "bot.unknown_command": "⚠️ 未知命令:{command}。使用 /help 查看可用命令。",
50
+ "bot.photo_downloading": "⏳ 正在下载照片...",
51
+ "bot.photo_too_large": "⚠️ 照片过大(最大 {maxSizeMb}MB)",
52
+ "bot.photo_model_no_image": "⚠️ 当前模型不支持图像输入。将仅发送文本。",
53
+ "bot.photo_download_error": "🔴 下载照片失败",
54
+ "bot.photo_no_caption": "💡 提示:添加说明文字以描述你希望对这张照片做什么。",
55
+ "bot.file_downloading": "⏳ 正在下载文件...",
56
+ "bot.file_too_large": "⚠️ 文件过大(最大 {maxSizeMb}MB)",
57
+ "bot.file_download_error": "🔴 下载文件失败",
58
+ "bot.model_no_pdf": "⚠️ 当前模型不支持PDF输入。将仅发送文本。",
59
+ "bot.text_file_too_large": "⚠️ 文本文件过大(最大 {maxSizeKb}KB)",
60
+ "status.header_running": "🟢 OpenCode 服务器正在运行",
61
+ "status.health.healthy": "健康",
62
+ "status.health.unhealthy": "不健康",
63
+ "status.line.health": "状态:{health}",
64
+ "status.line.version": "版本:{version}",
65
+ "status.line.managed_yes": "由机器人启动:是",
66
+ "status.line.managed_no": "由机器人启动:否",
67
+ "status.line.pid": "PID:{pid}",
68
+ "status.line.uptime_sec": "运行时间:{seconds} 秒",
69
+ "status.line.mode": "模式:{mode}",
70
+ "status.line.model": "模型:{model}",
71
+ "status.agent_not_set": "未设置",
72
+ "status.project_selected": "项目:{project}",
73
+ "status.project_not_selected": "项目:未选择",
74
+ "status.project_hint": "使用 /projects 选择项目",
75
+ "status.session_selected": "当前会话:{title}",
76
+ "status.session_not_selected": "当前会话:未选择",
77
+ "status.session_hint": "使用 /sessions 选择一个会话,或 /new 创建",
78
+ "status.server_unavailable": "🔴 OpenCode 服务器不可用\n\n使用 /opencode_start 启动服务器。",
79
+ "projects.empty": "📭 未找到项目。\n\n在 OpenCode 中打开一个目录并至少创建一个会话,然后它会出现在这里。",
80
+ "projects.select": "请选择一个项目:",
81
+ "projects.select_with_current": "请选择一个项目:\n\n当前:🏗 {project}",
82
+ "projects.page_indicator": "第 {current}/{total} 页",
83
+ "projects.prev_page": "⬅️ 上一页",
84
+ "projects.next_page": "下一页 ➡️",
85
+ "projects.fetch_error": "🔴 OpenCode 服务器不可用,或加载项目时发生错误。",
86
+ "projects.page_load_error": "无法加载此页面。请重试。",
87
+ "projects.selected": "✅ 已选择项目:{project}\n\n📋 会话已重置。请在此项目中使用 /sessions 或 /new。",
88
+ "projects.select_error": "🔴 选择项目失败。",
89
+ "sessions.project_not_selected": "🏗 未选择项目。\n\n请先使用 /projects 选择一个项目。",
90
+ "sessions.empty": "📭 未找到会话。\n\n使用 /new 创建新会话。",
91
+ "sessions.select": "请选择一个会话:",
92
+ "sessions.select_page": "请选择一个会话(第 {page} 页):",
93
+ "sessions.fetch_error": "🔴 OpenCode 服务器不可用,或加载会话时发生错误。",
94
+ "sessions.select_project_first": "🔴 未选择项目。使用 /projects。",
95
+ "sessions.page_empty_callback": "这一页没有会话",
96
+ "sessions.page_load_error_callback": "无法加载此页面。请重试。",
97
+ "sessions.button.prev_page": "⬅️ 上一页",
98
+ "sessions.button.next_page": "下一页 ➡️",
99
+ "sessions.loading_context": "⏳ 正在加载上下文和最新消息...",
100
+ "sessions.selected": "✅ 已选择会话:{title}",
101
+ "sessions.select_error": "🔴 选择会话失败。",
102
+ "sessions.preview.empty": "没有最近消息。",
103
+ "sessions.preview.title": "最近消息:",
104
+ "sessions.preview.you": "你:",
105
+ "sessions.preview.agent": "代理:",
106
+ "new.project_not_selected": "🏗 未选择项目。\n\n请先使用 /projects 选择一个项目。",
107
+ "new.created": "✅ 新会话已创建:{title}",
108
+ "new.create_error": "🔴 OpenCode 服务器不可用,或创建会话时发生错误。",
109
+ "stop.no_active_session": "🛑 代理尚未启动\n\n使用 /new 创建会话,或通过 /sessions 选择一个。",
110
+ "stop.in_progress": "🛑 已停止事件流,正在发送中止信号...\n\n等待代理停止。",
111
+ "stop.warn_unconfirmed": "⚠️ 事件流已停止,但服务器未确认中止。\n\n检查 /status,并在几秒后重试 /abort。",
112
+ "stop.warn_maybe_finished": "⚠️ 事件流已停止,但代理可能已完成。",
113
+ "stop.success": "✅ 代理操作已中断。本次运行的后续消息将不再发送。",
114
+ "stop.warn_still_busy": "⚠️ 信号已发送,但代理仍在忙。\n\n事件流已禁用,因此不会发送中间消息。",
115
+ "stop.warn_timeout": "⚠️ 中止请求超时。\n\n事件流已禁用,请在几秒后重试 /abort。",
116
+ "stop.warn_local_only": "⚠️ 已在本地停止事件流,但服务器端中止失败。",
117
+ "stop.error": "🔴 停止操作失败。\n\n事件流已停止,请再次尝试 /abort。",
118
+ "opencode_start.already_running_managed": "⚠️ OpenCode 服务器已在运行\n\nPID:{pid}\n运行时间:{seconds} 秒",
119
+ "opencode_start.already_running_external": "✅ OpenCode 服务器正作为外部进程运行\n\n版本:{version}\n\n该服务器不是由机器人启动,因此 /opencode-stop 无法停止它。",
120
+ "opencode_start.starting": "🔄 正在启动 OpenCode 服务器...",
121
+ "opencode_start.start_error": "🔴 启动 OpenCode 服务器失败\n\n错误:{error}\n\n请检查 OpenCode CLI 已安装且在 PATH 中可用:\nopencode --version\nnpm install -g @opencode-ai/cli",
122
+ "opencode_start.started_not_ready": "⚠️ OpenCode 服务器已启动,但未响应\n\nPID:{pid}\n\n服务器可能仍在启动中。几秒后试试 /status。",
123
+ "opencode_start.success": "✅ OpenCode 服务器启动成功\n\nPID:{pid}\n版本:{version}",
124
+ "opencode_start.error": "🔴 启动服务器时发生错误。\n\n请查看应用日志了解详情。",
125
+ "opencode_stop.external_running": "⚠️ OpenCode 服务器正作为外部进程运行\n\n该服务器不是通过 /opencode-start 启动的。\n请手动停止它,或使用 /status 检查状态。",
126
+ "opencode_stop.not_running": "⚠️ OpenCode 服务器未运行",
127
+ "opencode_stop.stopping": "🛑 正在停止 OpenCode 服务器...\n\nPID:{pid}",
128
+ "opencode_stop.stop_error": "🔴 停止 OpenCode 服务器失败\n\n错误:{error}",
129
+ "opencode_stop.success": "✅ OpenCode 服务器已成功停止",
130
+ "opencode_stop.error": "🔴 停止服务器时发生错误。\n\n请查看应用日志了解详情。",
131
+ "agent.changed_callback": "模式已更改:{name}",
132
+ "agent.changed_message": "✅ 模式已切换为:{name}",
133
+ "agent.change_error_callback": "切换模式失败",
134
+ "agent.menu.current": "当前模式:{name}\n\n请选择模式:",
135
+ "agent.menu.select": "请选择工作模式:",
136
+ "agent.menu.empty": "⚠️ 没有可用的代理",
137
+ "agent.menu.error": "🔴 获取代理列表失败",
138
+ "model.changed_callback": "模型已更改:{name}",
139
+ "model.changed_message": "✅ 模型已切换为:{name}",
140
+ "model.change_error_callback": "切换模型失败",
141
+ "model.menu.empty": "⚠️ 没有可用模型",
142
+ "model.menu.select": "请选择模型:",
143
+ "model.menu.current": "当前模型:{name}\n\n请选择模型:",
144
+ "model.menu.favorites_title": "⭐ 收藏(可在 OpenCode CLI 中将模型加入收藏)",
145
+ "model.menu.favorites_empty": "— 列表为空。",
146
+ "model.menu.recent_title": "🕘 最近使用",
147
+ "model.menu.recent_empty": "— 列表为空。",
148
+ "model.menu.favorites_hint": "ℹ️ 可在 OpenCode CLI 中将模型加入收藏,使其显示在列表顶部。",
149
+ "model.menu.error": "🔴 获取模型列表失败",
150
+ "variant.model_not_selected_callback": "错误:未选择模型",
151
+ "variant.changed_callback": "变体已更改:{name}",
152
+ "variant.changed_message": "✅ 变体已切换为:{name}",
153
+ "variant.change_error_callback": "切换变体失败",
154
+ "variant.select_model_first": "⚠️ 请先选择一个模型",
155
+ "variant.menu.empty": "⚠️ 没有可用变体",
156
+ "variant.menu.current": "当前变体:{name}\n\n请选择变体:",
157
+ "variant.menu.error": "🔴 获取变体列表失败",
158
+ "context.button.confirm": "✅ 是的,压缩上下文",
159
+ "context.no_active_session": "⚠️ 没有活动会话。使用 /new 创建会话",
160
+ "context.confirm_text": '📊 会话 "{title}" 的上下文压缩\n\n这会通过移除历史中的旧消息来减少上下文占用。当前任务不会被中断。\n\n继续?',
161
+ "context.callback_session_not_found": "未找到会话",
162
+ "context.callback_compacting": "正在压缩上下文...",
163
+ "context.progress": "⏳ 正在压缩上下文...",
164
+ "context.error": "❌ 上下文压缩失败",
165
+ "context.success": "✅ 上下文压缩成功",
166
+ "permission.inactive_callback": "权限请求已失效",
167
+ "permission.processing_error_callback": "处理错误",
168
+ "permission.no_active_request_callback": "错误:没有活动请求",
169
+ "permission.reply.once": "仅允许一次",
170
+ "permission.reply.always": "始终允许",
171
+ "permission.reply.reject": "已拒绝",
172
+ "permission.send_reply_error": "❌ 发送权限回复失败",
173
+ "permission.blocked.expected_reply": "⚠️ 请先使用上方按钮回答权限请求。",
174
+ "permission.blocked.command_not_allowed": "⚠️ 在你回答权限请求之前不可用此命令。",
175
+ "permission.header": "{emoji} 权限请求:{name}\n\n",
176
+ "permission.button.allow": "✅ 允许一次",
177
+ "permission.button.always": "🔓 始终允许",
178
+ "permission.button.reject": "❌ 拒绝",
179
+ "permission.name.bash": "Bash",
180
+ "permission.name.edit": "编辑",
181
+ "permission.name.write": "写入",
182
+ "permission.name.read": "读取",
183
+ "permission.name.webfetch": "Web 获取",
184
+ "permission.name.websearch": "Web 搜索",
185
+ "permission.name.glob": "文件搜索",
186
+ "permission.name.grep": "内容搜索",
187
+ "permission.name.list": "列出目录",
188
+ "permission.name.task": "任务",
189
+ "permission.name.lsp": "LSP",
190
+ "permission.name.external_directory": "外部目录",
191
+ "question.inactive_callback": "投票已失效",
192
+ "question.processing_error_callback": "处理错误",
193
+ "question.select_one_required_callback": "请至少选择一个选项",
194
+ "question.enter_custom_callback": "请发送你的自定义回答",
195
+ "question.cancelled": "❌ 投票已取消",
196
+ "question.answer_already_received": "已收到答案,请稍候...",
197
+ "question.completed_no_answers": "✅ 投票完成(无答案)",
198
+ "question.no_active_project": "❌ 没有活动项目",
199
+ "question.no_active_request": "❌ 没有活动请求",
200
+ "question.send_answers_error": "❌ 向代理发送答案失败",
201
+ "question.multi_hint": "\n(你可以选择多个选项)",
202
+ "question.button.submit": "✅ 完成",
203
+ "question.button.custom": "🔤 自定义回答",
204
+ "question.button.cancel": "❌ 取消",
205
+ "question.use_custom_button_first": '⚠️ 要发送文本,请先点击当前问题的 "自定义回答" 按钮。',
206
+ "question.summary.title": "✅ 投票已完成!\n\n",
207
+ "question.summary.question": "问题 {index}:\n{question}\n\n",
208
+ "question.summary.answer": "回答:\n{answer}\n\n",
209
+ "keyboard.agent_mode": "{emoji} {name} 模式",
210
+ "keyboard.context": "📊 {used} / {limit} ({percent}%)",
211
+ "keyboard.context_empty": "📊 0",
212
+ "keyboard.variant": "💭 {name}",
213
+ "keyboard.variant_default": "💡 默认",
214
+ "keyboard.updated": "⌨️ 键盘已更新",
215
+ "pinned.default_session_title": "新会话",
216
+ "pinned.unknown": "未知",
217
+ "pinned.line.project": "项目: {project}",
218
+ "pinned.line.model": "模型: {model}",
219
+ "pinned.line.context": "上下文: {used} / {limit} ({percent}%)",
220
+ "pinned.line.cost": "费用: {cost}",
221
+ "subagent.header": "子代理 {agent}: {description}",
222
+ "subagent.line.status": "状态: {status}",
223
+ "subagent.line.task": "任务: {task}",
224
+ "subagent.line.agent": "代理: {agent}",
225
+ "subagent.working": "执行中...",
226
+ "subagent.working_with_details": "执行中: {details}",
227
+ "subagent.completed": "已完成",
228
+ "subagent.failed": "任务失败",
229
+ "subagent.status.pending": "等待中",
230
+ "subagent.status.running": "运行中",
231
+ "subagent.status.completed": "已完成",
232
+ "subagent.status.error": "错误",
233
+ "pinned.files.title": "文件({count}):",
234
+ "pinned.files.item": " {path}{diff}",
235
+ "pinned.files.more": " ... 还有 {count} 个",
236
+ "tool.todo.overflow": "*(还有 {count} 个任务)*",
237
+ "tool.file_header.write": "写入文件/路径: {path}\n============================================================\n\n",
238
+ "tool.file_header.edit": "编辑文件/路径: {path}\n============================================================\n\n",
239
+ "runtime.wizard.ask_token": "请输入 Telegram 机器人 token(从 @BotFather 获取)。\n> ",
240
+ "runtime.wizard.ask_language": "请选择界面语言。\n输入列表中的语言编号或 locale code。\n按 Enter 保持默认语言:{defaultLocale}\n{options}\n> ",
241
+ "runtime.wizard.language_invalid": "请输入列表中的语言编号或受支持的 locale code。\n",
242
+ "runtime.wizard.language_selected": "已选择语言:{language}\n",
243
+ "runtime.wizard.token_required": "必须提供 token。请重试。\n",
244
+ "runtime.wizard.token_invalid": "token 看起来无效(期望格式 <id>:<secret>)。请重试。\n",
245
+ "runtime.wizard.ask_user_id": "请输入你的 Telegram User ID(可从 @userinfobot 获取)。\n> ",
246
+ "runtime.wizard.user_id_invalid": "请输入一个正整数(> 0)。\n",
247
+ "runtime.wizard.ask_api_url": "请输入 OpenCode API URL(可选)。\n按 Enter 使用默认值:{defaultUrl}\n> ",
248
+ "runtime.wizard.ask_server_username": "请输入 OpenCode 服务器用户名(可选)。\n按 Enter 使用默认值:{defaultUsername}\n> ",
249
+ "runtime.wizard.ask_server_password": "请输入 OpenCode 服务器密码(可选)。\n按 Enter 保持为空。\n> ",
250
+ "runtime.wizard.api_url_invalid": "请输入有效 URL(http/https),或按 Enter 使用默认值。\n",
251
+ "runtime.wizard.start": "OpenCode Telegram Bot 设置。\n",
252
+ "runtime.wizard.saved": "配置已保存:\n- {envPath}\n- {settingsPath}\n",
253
+ "runtime.wizard.not_configured_starting": "应用尚未配置。正在启动向导...\n",
254
+ "runtime.wizard.tty_required": "交互式向导需要 TTY 终端。请在交互式 shell 中运行 `opencode-telegram config`。",
255
+ "rename.no_session": "⚠️ 没有活动会话。请先创建或选择一个会话。",
256
+ "rename.prompt": "📝 请输入会话的新标题:\n\n当前:{title}",
257
+ "rename.empty_title": "⚠️ 标题不能为空。",
258
+ "rename.success": "✅ 会话已重命名为:{title}",
259
+ "rename.error": "🔴 重命名会话失败。",
260
+ "rename.cancelled": "❌ 重命名已取消。",
261
+ "rename.inactive_callback": "重命名请求已失效",
262
+ "rename.inactive": "⚠️ 重命名请求未激活。请再次运行 /rename。",
263
+ "rename.blocked.expected_name": "⚠️ 请以文本输入新会话名称,或在重命名消息中点击取消。",
264
+ "rename.blocked.command_not_allowed": "⚠️ 重命名等待新名称期间不可用此命令。",
265
+ "rename.button.cancel": "❌ 取消",
266
+ "task.prompt.schedule": "⏰ 请用自然语言发送任务的时间安排。\n\n示例:\n- 每 5 分钟\n- 每天 17:00\n- 明天 12:00",
267
+ "task.schedule_empty": "⚠️ 时间安排不能为空。",
268
+ "task.parse.in_progress": "⏳ 正在解析时间安排...",
269
+ "task.parse_error": "🔴 无法解析时间安排。\n\n{message}\n\n请用更清晰的方式重新发送。",
270
+ "task.schedule_preview": "✅ 时间安排已解析\n\n理解为:{summary}\n{cronLine}时区:{timezone}\n类型:{kind}\n下次运行:{nextRunAt}",
271
+ "task.schedule_preview.cron": "Cron: {cron}",
272
+ "task.prompt.body": "📝 现在发送机器人按此时间安排需要执行的内容。",
273
+ "task.prompt_empty": "⚠️ 任务文本不能为空。",
274
+ "task.created": "✅ 定时任务已创建\n\n任务:{description}\n项目:{project}\n模型:{model}\n时间安排:{schedule}\n{cronLine}下次运行:{nextRunAt}",
275
+ "task.created.cron": "Cron: {cron}",
276
+ "task.button.retry_schedule": "🔁 重新输入时间安排",
277
+ "task.button.cancel": "❌ 取消",
278
+ "task.retry_schedule_callback": "正在重新输入时间安排...",
279
+ "task.cancel_callback": "正在取消...",
280
+ "task.cancelled": "❌ 定时任务创建已取消。",
281
+ "task.inactive_callback": "这个定时任务流程已失效",
282
+ "task.inactive": "⚠️ 定时任务创建流程未激活。请重新运行 /task。",
283
+ "task.blocked.expected_input": "⚠️ 请先完成当前定时任务设置:发送文本,或使用时间安排消息中的按钮。",
284
+ "task.blocked.command_not_allowed": "⚠️ 定时任务创建进行中时,此命令不可用。",
285
+ "task.limit_reached": "⚠️ 已达到任务数量上限({limit})。请先删除一个现有定时任务。",
286
+ "task.schedule_too_frequent": "重复任务过于频繁。最小允许间隔为每 5 分钟一次。",
287
+ "task.kind.cron": "重复",
288
+ "task.kind.once": "一次性",
289
+ "task.run.success": "⏰ 定时任务已完成: {description}",
290
+ "task.run.error": "🔴 定时任务执行失败: {description}\n\n错误: {error}",
291
+ "tasklist.empty": "📭 还没有定时任务。",
292
+ "tasklist.select": "请选择一个定时任务:",
293
+ "tasklist.details": "⏰ 定时任务\n\n任务:{prompt}\n项目:{project}\n计划:{schedule}\n{cronLine}时区:{timezone}\n下次运行:{nextRunAt}\n上次运行:{lastRunAt}\n运行次数:{runCount}",
294
+ "tasklist.details.cron": "Cron: {cron}",
295
+ "tasklist.button.delete": "🗑 删除",
296
+ "tasklist.button.cancel": "❌ 取消",
297
+ "tasklist.deleted_callback": "已删除",
298
+ "tasklist.cancelled_callback": "已取消",
299
+ "tasklist.inactive_callback": "此定时任务菜单已失效",
300
+ "tasklist.load_error": "🔴 无法加载定时任务。",
301
+ "commands.select": "请选择一个 OpenCode 命令:",
302
+ "commands.empty": "📭 当前项目没有可用的 OpenCode 命令。",
303
+ "commands.fetch_error": "🔴 加载 OpenCode 命令失败。",
304
+ "commands.no_description": "无描述",
305
+ "commands.button.execute": "✅ 执行",
306
+ "commands.button.cancel": "❌ 取消",
307
+ "commands.confirm": "请确认执行命令 {command}。若需带参数执行,请发送一条包含参数的消息。",
308
+ "commands.inactive_callback": "该命令菜单已失效",
309
+ "commands.cancelled_callback": "已取消",
310
+ "commands.execute_callback": "正在执行命令...",
311
+ "commands.executing_prefix": "⚡ 执行命令:",
312
+ "commands.arguments_empty": "⚠️ 参数不能为空。请发送文本或点击执行。",
313
+ "commands.execute_error": "🔴 执行 OpenCode 命令失败。",
314
+ "commands.select_page": "请选择一个 OpenCode 命令(第 {page} 页):",
315
+ "commands.button.prev_page": "⬅️ 上一页",
316
+ "commands.button.next_page": "下一页 ➡️",
317
+ "commands.page_empty_callback": "这一页没有命令",
318
+ "commands.page_load_error_callback": "无法加载此页面。请重试。",
319
+ "cmd.description.rename": "重命名当前会话",
320
+ "cli.usage": "用法:\n opencode-telegram [start] [--mode sources|installed]\n opencode-telegram status\n opencode-telegram stop\n opencode-telegram config\n\n注意:\n - 无命令时默认为 `start`\n - `--mode` 当前仅支持 `start`",
321
+ "cli.placeholder.status": "`status` 命令当前为占位符。实际状态检查将在服务层中添加(第5阶段)。",
322
+ "cli.placeholder.stop": "`stop` 命令当前为占位符。实际后台进程停止功能将在服务层中添加(第5阶段)。",
323
+ "cli.placeholder.unavailable": "命令不可用。",
324
+ "cli.error.prefix": "CLI 错误:{message}",
325
+ "cli.args.unknown_command": "未知命令:{value}",
326
+ "cli.args.mode_requires_value": "选项 --mode 需要一个值:sources|installed",
327
+ "cli.args.invalid_mode": "无效的 --mode 值:{value}。期望 sources|installed",
328
+ "cli.args.unknown_option": "未知选项:{value}",
329
+ "cli.args.mode_only_start": "选项 --mode 仅支持 start 命令",
330
+ "legacy.models.fetch_error": "🔴 获取模型列表失败。请使用 /status 检查服务器状态。",
331
+ "legacy.models.empty": "📋 没有可用模型。请在 OpenCode 中配置 providers。",
332
+ "legacy.models.header": "📋 可用模型:\n\n",
333
+ "legacy.models.no_provider_models": " ⚠️ 没有可用模型\n",
334
+ "legacy.models.env_hint": "💡 在 .env 中使用该模型:\n",
335
+ "legacy.models.error": "🔴 加载模型列表时发生错误。",
336
+ "stt.recognizing": "🎤 正在识别音频...",
337
+ "stt.recognized": "🎤 识别结果:\n{text}",
338
+ "stt.not_configured": "🎤 语音识别尚未配置。\n\n在 .env 中设置 STT_API_URL 和 STT_API_KEY 以启用。",
339
+ "stt.error": "🔴 识别音频失败:{error}",
340
+ "stt.empty_result": "🎤 音频消息中未检测到语音。",
341
+ "tts.not_configured": "🔊 语音合成尚未配置。\n\n在 .env 中设置 TTS_API_URL 以启用。\n\n推荐:使用 pocket-tts-server 进行本地 TTS。\nhttps://github.com/ai-joe-git/pocket-tts-server",
342
+ "tts.voice_changed": "🔊 语音已更改:{voice}",
343
+ "tts.voice_error": "🔴 无法更改语音:{error}",
344
+ "tts.fetch_voices_error": "🔴 无法从 TTS 服务器获取语音。",
345
+ "tts.synthesis_error": "🔴 语音合成失败:{error}",
346
+ "tts.speaking": "🔊 正在播放...",
347
+ "tts.menu_title": "选择语音:",
348
+ "tts.menu_current": "当前语音:{voice}\n\n选择语音:",
349
+ "tts.menu_empty": "⚠️ 没有可用的语音。请检查 TTS 服务器配置。",
350
+ "tts.menu_loading": "⏳ 加载语音中...",
351
+ "tts.menu_error": "🔴 无法加载语音。",
352
+ "tts.button.off": "🔇 关闭 TTS",
353
+ "tts.button.off_description": "禁用语音回复",
354
+ "tts.button.prev_page": "⬅️ 上一页",
355
+ "tts.button.next_page": "下一页 ➡️",
356
+ "tts.off_success": "🔇 语音回复已禁用。",
357
+ };
package/dist/index.js ADDED
@@ -0,0 +1,26 @@
1
+ import { resolveRuntimeMode, setRuntimeMode } from "./runtime/mode.js";
2
+ const EXIT_RUNTIME_ERROR = 1;
3
+ const EXIT_INVALID_ARGS = 2;
4
+ async function main() {
5
+ const modeResult = resolveRuntimeMode({
6
+ defaultMode: "sources",
7
+ argv: process.argv.slice(2),
8
+ });
9
+ if (modeResult.error) {
10
+ process.stderr.write(`${modeResult.error}\n`);
11
+ process.exit(EXIT_INVALID_ARGS);
12
+ return;
13
+ }
14
+ setRuntimeMode(modeResult.mode);
15
+ const { startBotApp } = await import("./app/start-bot-app.js");
16
+ await startBotApp();
17
+ }
18
+ void main().catch((error) => {
19
+ if (error instanceof Error) {
20
+ process.stderr.write(`Failed to start bot: ${error.message}\n`);
21
+ }
22
+ else {
23
+ process.stderr.write(`Failed to start bot: ${String(error)}\n`);
24
+ }
25
+ process.exit(EXIT_RUNTIME_ERROR);
26
+ });
@@ -0,0 +1,8 @@
1
+ export const BUSY_ALLOWED_COMMANDS = ["/abort", "/status", "/help"];
2
+ const BUSY_ALLOWED_COMMAND_SET = new Set(BUSY_ALLOWED_COMMANDS);
3
+ export function isBusyAllowedCommand(command) {
4
+ return Boolean(command && BUSY_ALLOWED_COMMAND_SET.has(command));
5
+ }
6
+ export function allowsBusyInteraction(kind) {
7
+ return kind === "question" || kind === "permission";
8
+ }
@@ -0,0 +1,32 @@
1
+ import { permissionManager } from "../permission/manager.js";
2
+ import { questionManager } from "../question/manager.js";
3
+ import { renameManager } from "../rename/manager.js";
4
+ import { taskCreationManager } from "../scheduled-task/creation-manager.js";
5
+ import { interactionManager } from "./manager.js";
6
+ import { logger } from "../utils/logger.js";
7
+ export function clearAllInteractionState(reason) {
8
+ const questionActive = questionManager.isActive();
9
+ const permissionActive = permissionManager.isActive();
10
+ const renameActive = renameManager.isWaitingForName();
11
+ const taskCreationActive = taskCreationManager.isActive();
12
+ const interactionSnapshot = interactionManager.getSnapshot();
13
+ questionManager.clear();
14
+ permissionManager.clear();
15
+ renameManager.clear();
16
+ taskCreationManager.clear();
17
+ interactionManager.clear(reason);
18
+ const hasAnyActiveState = questionActive ||
19
+ permissionActive ||
20
+ renameActive ||
21
+ taskCreationActive ||
22
+ interactionSnapshot !== null;
23
+ const message = `[InteractionCleanup] Cleared state: reason=${reason}, ` +
24
+ `questionActive=${questionActive}, permissionActive=${permissionActive}, ` +
25
+ `renameActive=${renameActive}, taskCreationActive=${taskCreationActive}, ` +
26
+ `interactionKind=${interactionSnapshot?.kind || "none"}`;
27
+ if (hasAnyActiveState) {
28
+ logger.info(message);
29
+ return;
30
+ }
31
+ logger.debug(message);
32
+ }
@@ -0,0 +1,140 @@
1
+ import { interactionManager } from "./manager.js";
2
+ import { allowsBusyInteraction, isBusyAllowedCommand } from "./busy.js";
3
+ import { foregroundSessionState } from "../scheduled-task/foreground-state.js";
4
+ function normalizeIncomingCommand(text) {
5
+ const trimmed = text.trim();
6
+ if (!trimmed.startsWith("/")) {
7
+ return null;
8
+ }
9
+ const token = trimmed.split(/\s+/)[0];
10
+ const withoutMention = token.split("@")[0].toLowerCase();
11
+ if (withoutMention.length <= 1) {
12
+ return null;
13
+ }
14
+ return withoutMention;
15
+ }
16
+ function classifyIncomingInput(ctx) {
17
+ if (ctx.callbackQuery?.data) {
18
+ return { inputType: "callback" };
19
+ }
20
+ const text = ctx.message?.text;
21
+ if (typeof text === "string") {
22
+ const command = normalizeIncomingCommand(text);
23
+ if (command) {
24
+ return { inputType: "command", command };
25
+ }
26
+ return { inputType: "text" };
27
+ }
28
+ // Photo, voice, audio, and other non-text messages are classified as "other"
29
+ if (ctx.message?.photo) {
30
+ return { inputType: "other" };
31
+ }
32
+ return { inputType: "other" };
33
+ }
34
+ function getExpectedInputBlockReason(expectedInput) {
35
+ switch (expectedInput) {
36
+ case "callback":
37
+ return "expected_callback";
38
+ case "command":
39
+ return "expected_command";
40
+ case "text":
41
+ case "mixed":
42
+ return "expected_text";
43
+ }
44
+ }
45
+ function createAllowDecision(inputType, state, command, busy) {
46
+ return {
47
+ allow: true,
48
+ inputType,
49
+ state,
50
+ command,
51
+ busy,
52
+ };
53
+ }
54
+ function createBlockDecision(inputType, state, reason, command, busy) {
55
+ return {
56
+ allow: false,
57
+ inputType,
58
+ state,
59
+ reason,
60
+ command,
61
+ busy,
62
+ };
63
+ }
64
+ function createBusyBlockDecision(inputType, state, reason, command) {
65
+ return {
66
+ allow: false,
67
+ inputType,
68
+ state,
69
+ reason,
70
+ command,
71
+ busy: true,
72
+ };
73
+ }
74
+ function isAllowedRenameCancelCallback(ctx, state) {
75
+ return (state.kind === "rename" &&
76
+ state.expectedInput === "text" &&
77
+ ctx.callbackQuery?.data === "rename:cancel");
78
+ }
79
+ function isAllowedTaskCallback(ctx, state) {
80
+ return (state.kind === "task" &&
81
+ (ctx.callbackQuery?.data === "task:cancel" || ctx.callbackQuery?.data === "task:retry-schedule"));
82
+ }
83
+ export function resolveInteractionGuardDecision(ctx) {
84
+ const state = interactionManager.getSnapshot();
85
+ const { inputType, command } = classifyIncomingInput(ctx);
86
+ const isBusy = foregroundSessionState.isBusy();
87
+ if (state && interactionManager.isExpired()) {
88
+ interactionManager.clear("expired");
89
+ return createBlockDecision(inputType, state, "expired", command, isBusy);
90
+ }
91
+ if (isBusy) {
92
+ if (inputType === "command") {
93
+ if (isBusyAllowedCommand(command)) {
94
+ return createAllowDecision(inputType, state, command, true);
95
+ }
96
+ return createBusyBlockDecision(inputType, state, "command_not_allowed", command);
97
+ }
98
+ if (state && allowsBusyInteraction(state.kind)) {
99
+ if (state.expectedInput === "mixed") {
100
+ if (inputType === "callback" || inputType === "text") {
101
+ return createAllowDecision(inputType, state, command, true);
102
+ }
103
+ return createBusyBlockDecision(inputType, state, "expected_text", command);
104
+ }
105
+ if (state.expectedInput === inputType) {
106
+ return createAllowDecision(inputType, state, command, true);
107
+ }
108
+ return createBusyBlockDecision(inputType, state, getExpectedInputBlockReason(state.expectedInput), command);
109
+ }
110
+ return createBusyBlockDecision(inputType, state, "expected_text", command);
111
+ }
112
+ if (!state) {
113
+ return createAllowDecision(inputType, null, command);
114
+ }
115
+ if (inputType === "command") {
116
+ if (command === "/start") {
117
+ return createAllowDecision(inputType, state, command);
118
+ }
119
+ if (command && state.allowedCommands.includes(command)) {
120
+ return createAllowDecision(inputType, state, command);
121
+ }
122
+ return createBlockDecision(inputType, state, "command_not_allowed", command);
123
+ }
124
+ if (state.expectedInput === "mixed") {
125
+ if (inputType === "callback" || inputType === "text") {
126
+ return createAllowDecision(inputType, state, command);
127
+ }
128
+ return createBlockDecision(inputType, state, "expected_text", command);
129
+ }
130
+ if (inputType === "callback" && isAllowedRenameCancelCallback(ctx, state)) {
131
+ return createAllowDecision(inputType, state, command);
132
+ }
133
+ if (inputType === "callback" && isAllowedTaskCallback(ctx, state)) {
134
+ return createAllowDecision(inputType, state, command);
135
+ }
136
+ if (state.expectedInput === inputType) {
137
+ return createAllowDecision(inputType, state, command);
138
+ }
139
+ return createBlockDecision(inputType, state, getExpectedInputBlockReason(state.expectedInput), command);
140
+ }