coding-tool-x 3.2.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 (185) hide show
  1. package/CHANGELOG.md +599 -0
  2. package/LICENSE +21 -0
  3. package/README.md +439 -0
  4. package/bin/ctx.js +8 -0
  5. package/dist/web/assets/Analytics-DN_YsnkW.js +39 -0
  6. package/dist/web/assets/Analytics-DuYvId7u.css +1 -0
  7. package/dist/web/assets/ConfigTemplates-Bidwfdf2.css +1 -0
  8. package/dist/web/assets/ConfigTemplates-DpXIMy0p.js +1 -0
  9. package/dist/web/assets/Home-38JTUlYt.js +1 -0
  10. package/dist/web/assets/Home-CjupSEWE.css +1 -0
  11. package/dist/web/assets/PluginManager-CX2tgq2H.js +1 -0
  12. package/dist/web/assets/PluginManager-ROyoZ-6m.css +1 -0
  13. package/dist/web/assets/ProjectList-C1lDcsn6.js +1 -0
  14. package/dist/web/assets/ProjectList-oJIyIRkP.css +1 -0
  15. package/dist/web/assets/SessionList-C55tjV7i.css +1 -0
  16. package/dist/web/assets/SessionList-CZ7T6rVx.js +1 -0
  17. package/dist/web/assets/SkillManager-D7pd-d_P.css +1 -0
  18. package/dist/web/assets/SkillManager-DLN9f79y.js +1 -0
  19. package/dist/web/assets/WorkspaceManager-CrwgQgmP.css +1 -0
  20. package/dist/web/assets/WorkspaceManager-DxlHZkpZ.js +1 -0
  21. package/dist/web/assets/icons-DRrXwWZi.js +1 -0
  22. package/dist/web/assets/index-CetESrXw.css +1 -0
  23. package/dist/web/assets/index-Cfvn-2Gb.js +2 -0
  24. package/dist/web/assets/markdown-BfC0goYb.css +10 -0
  25. package/dist/web/assets/markdown-C9MYpaSi.js +1 -0
  26. package/dist/web/assets/naive-ui-DlpKk-8M.js +1 -0
  27. package/dist/web/assets/vendors-DMjSfzlv.js +7 -0
  28. package/dist/web/assets/vue-vendor-DET08QYg.js +45 -0
  29. package/dist/web/favicon.ico +0 -0
  30. package/dist/web/index.html +20 -0
  31. package/dist/web/logo.png +0 -0
  32. package/docs/bannel.png +0 -0
  33. package/docs/home.png +0 -0
  34. package/docs/logo.png +0 -0
  35. package/docs/model-redirection.md +251 -0
  36. package/docs/multi-channel-load-balancing.md +249 -0
  37. package/package.json +80 -0
  38. package/src/commands/channels.js +551 -0
  39. package/src/commands/cli-type.js +101 -0
  40. package/src/commands/daemon.js +365 -0
  41. package/src/commands/doctor.js +333 -0
  42. package/src/commands/export-config.js +205 -0
  43. package/src/commands/list.js +222 -0
  44. package/src/commands/logs.js +261 -0
  45. package/src/commands/plugin.js +585 -0
  46. package/src/commands/port-config.js +135 -0
  47. package/src/commands/proxy-control.js +264 -0
  48. package/src/commands/proxy.js +152 -0
  49. package/src/commands/resume.js +137 -0
  50. package/src/commands/search.js +190 -0
  51. package/src/commands/security.js +37 -0
  52. package/src/commands/stats.js +398 -0
  53. package/src/commands/switch.js +48 -0
  54. package/src/commands/toggle-proxy.js +247 -0
  55. package/src/commands/ui.js +99 -0
  56. package/src/commands/update.js +97 -0
  57. package/src/commands/workspace.js +454 -0
  58. package/src/config/default.js +69 -0
  59. package/src/config/loader.js +149 -0
  60. package/src/config/model-metadata.js +167 -0
  61. package/src/config/model-metadata.json +125 -0
  62. package/src/config/model-pricing.js +35 -0
  63. package/src/config/paths.js +190 -0
  64. package/src/index.js +680 -0
  65. package/src/plugins/constants.js +15 -0
  66. package/src/plugins/event-bus.js +54 -0
  67. package/src/plugins/manifest-validator.js +129 -0
  68. package/src/plugins/plugin-api.js +128 -0
  69. package/src/plugins/plugin-installer.js +601 -0
  70. package/src/plugins/plugin-loader.js +229 -0
  71. package/src/plugins/plugin-manager.js +170 -0
  72. package/src/plugins/registry.js +152 -0
  73. package/src/plugins/schema/plugin-manifest.json +115 -0
  74. package/src/reset-config.js +94 -0
  75. package/src/server/api/agents.js +826 -0
  76. package/src/server/api/aliases.js +36 -0
  77. package/src/server/api/channels.js +368 -0
  78. package/src/server/api/claude-hooks.js +480 -0
  79. package/src/server/api/codex-channels.js +417 -0
  80. package/src/server/api/codex-projects.js +104 -0
  81. package/src/server/api/codex-proxy.js +195 -0
  82. package/src/server/api/codex-sessions.js +483 -0
  83. package/src/server/api/codex-statistics.js +57 -0
  84. package/src/server/api/commands.js +482 -0
  85. package/src/server/api/config-export.js +212 -0
  86. package/src/server/api/config-registry.js +357 -0
  87. package/src/server/api/config-sync.js +155 -0
  88. package/src/server/api/config-templates.js +248 -0
  89. package/src/server/api/config.js +521 -0
  90. package/src/server/api/convert.js +260 -0
  91. package/src/server/api/dashboard.js +142 -0
  92. package/src/server/api/env.js +144 -0
  93. package/src/server/api/favorites.js +77 -0
  94. package/src/server/api/gemini-channels.js +366 -0
  95. package/src/server/api/gemini-projects.js +91 -0
  96. package/src/server/api/gemini-proxy.js +173 -0
  97. package/src/server/api/gemini-sessions.js +376 -0
  98. package/src/server/api/gemini-statistics.js +57 -0
  99. package/src/server/api/health-check.js +31 -0
  100. package/src/server/api/mcp.js +399 -0
  101. package/src/server/api/opencode-channels.js +419 -0
  102. package/src/server/api/opencode-projects.js +99 -0
  103. package/src/server/api/opencode-proxy.js +207 -0
  104. package/src/server/api/opencode-sessions.js +327 -0
  105. package/src/server/api/opencode-statistics.js +57 -0
  106. package/src/server/api/plugins.js +463 -0
  107. package/src/server/api/pm2-autostart.js +269 -0
  108. package/src/server/api/projects.js +124 -0
  109. package/src/server/api/prompts.js +279 -0
  110. package/src/server/api/proxy.js +306 -0
  111. package/src/server/api/security.js +53 -0
  112. package/src/server/api/sessions.js +514 -0
  113. package/src/server/api/settings.js +142 -0
  114. package/src/server/api/skills.js +570 -0
  115. package/src/server/api/statistics.js +238 -0
  116. package/src/server/api/ui-config.js +64 -0
  117. package/src/server/api/workspaces.js +456 -0
  118. package/src/server/codex-proxy-server.js +681 -0
  119. package/src/server/dev-server.js +26 -0
  120. package/src/server/gemini-proxy-server.js +610 -0
  121. package/src/server/index.js +422 -0
  122. package/src/server/opencode-proxy-server.js +4771 -0
  123. package/src/server/proxy-server.js +669 -0
  124. package/src/server/services/agents-service.js +1137 -0
  125. package/src/server/services/alias.js +71 -0
  126. package/src/server/services/channel-health.js +234 -0
  127. package/src/server/services/channel-scheduler.js +240 -0
  128. package/src/server/services/channels.js +447 -0
  129. package/src/server/services/codex-channels.js +705 -0
  130. package/src/server/services/codex-config.js +90 -0
  131. package/src/server/services/codex-parser.js +322 -0
  132. package/src/server/services/codex-sessions.js +936 -0
  133. package/src/server/services/codex-settings-manager.js +619 -0
  134. package/src/server/services/codex-speed-test-template.json +24 -0
  135. package/src/server/services/codex-statistics-service.js +161 -0
  136. package/src/server/services/commands-service.js +574 -0
  137. package/src/server/services/config-export-service.js +1165 -0
  138. package/src/server/services/config-registry-service.js +828 -0
  139. package/src/server/services/config-sync-manager.js +941 -0
  140. package/src/server/services/config-sync-service.js +504 -0
  141. package/src/server/services/config-templates-service.js +913 -0
  142. package/src/server/services/enhanced-cache.js +196 -0
  143. package/src/server/services/env-checker.js +409 -0
  144. package/src/server/services/env-manager.js +436 -0
  145. package/src/server/services/favorites.js +165 -0
  146. package/src/server/services/format-converter.js +620 -0
  147. package/src/server/services/gemini-channels.js +459 -0
  148. package/src/server/services/gemini-config.js +73 -0
  149. package/src/server/services/gemini-sessions.js +689 -0
  150. package/src/server/services/gemini-settings-manager.js +263 -0
  151. package/src/server/services/gemini-statistics-service.js +157 -0
  152. package/src/server/services/health-check.js +85 -0
  153. package/src/server/services/mcp-client.js +790 -0
  154. package/src/server/services/mcp-service.js +1732 -0
  155. package/src/server/services/model-detector.js +1245 -0
  156. package/src/server/services/network-access.js +80 -0
  157. package/src/server/services/opencode-channels.js +366 -0
  158. package/src/server/services/opencode-gateway-adapters.js +1168 -0
  159. package/src/server/services/opencode-gateway-converter.js +639 -0
  160. package/src/server/services/opencode-sessions.js +931 -0
  161. package/src/server/services/opencode-settings-manager.js +478 -0
  162. package/src/server/services/opencode-statistics-service.js +161 -0
  163. package/src/server/services/plugins-service.js +1268 -0
  164. package/src/server/services/prompts-service.js +534 -0
  165. package/src/server/services/proxy-runtime.js +79 -0
  166. package/src/server/services/repo-scanner-base.js +708 -0
  167. package/src/server/services/request-logger.js +130 -0
  168. package/src/server/services/response-decoder.js +21 -0
  169. package/src/server/services/security-config.js +131 -0
  170. package/src/server/services/session-cache.js +127 -0
  171. package/src/server/services/session-converter.js +577 -0
  172. package/src/server/services/sessions.js +900 -0
  173. package/src/server/services/settings-manager.js +163 -0
  174. package/src/server/services/skill-service.js +1482 -0
  175. package/src/server/services/speed-test.js +1146 -0
  176. package/src/server/services/statistics-service.js +1043 -0
  177. package/src/server/services/ui-config.js +132 -0
  178. package/src/server/services/workspace-service.js +830 -0
  179. package/src/server/utils/pricing.js +73 -0
  180. package/src/server/websocket-server.js +513 -0
  181. package/src/ui/menu.js +139 -0
  182. package/src/ui/prompts.js +100 -0
  183. package/src/utils/format.js +43 -0
  184. package/src/utils/port-helper.js +108 -0
  185. package/src/utils/session.js +240 -0
@@ -0,0 +1,376 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const {
4
+ getProjectSessions,
5
+ getSessionById,
6
+ searchSessions,
7
+ forkSession,
8
+ deleteSession,
9
+ getRecentSessions,
10
+ saveSessionOrder,
11
+ getProjectPath,
12
+ getAllSessions
13
+ } = require('../services/gemini-sessions');
14
+ const { isGeminiInstalled } = require('../services/gemini-config');
15
+ const { loadAliases } = require('../services/alias');
16
+
17
+ module.exports = (config) => {
18
+ /**
19
+ * GET /api/gemini/sessions/search/global?keyword=xxx
20
+ * 全局搜索
21
+ */
22
+ router.get('/search/global', (req, res) => {
23
+ try {
24
+ if (!isGeminiInstalled()) {
25
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
26
+ }
27
+
28
+ const { keyword } = req.query;
29
+
30
+ if (!keyword) {
31
+ return res.status(400).json({ error: 'Keyword is required' });
32
+ }
33
+
34
+ const results = searchSessions(keyword);
35
+
36
+ res.json({
37
+ keyword,
38
+ totalMatches: results.reduce((sum, r) => sum + r.matchCount, 0),
39
+ sessions: results,
40
+ source: 'gemini'
41
+ });
42
+ } catch (err) {
43
+ console.error('[Gemini API] Failed to search sessions:', err);
44
+ res.status(500).json({ error: err.message });
45
+ }
46
+ });
47
+
48
+ /**
49
+ * GET /api/gemini/sessions/recent/list?limit=10
50
+ * 获取最近会话
51
+ */
52
+ router.get('/recent/list', (req, res) => {
53
+ try {
54
+ if (!isGeminiInstalled()) {
55
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
56
+ }
57
+
58
+ const limit = parseInt(req.query.limit) || 5;
59
+ const sessions = getRecentSessions(limit);
60
+
61
+ res.json({
62
+ sessions,
63
+ source: 'gemini'
64
+ });
65
+ } catch (err) {
66
+ console.error('[Gemini API] Failed to get recent sessions:', err);
67
+ res.status(500).json({ error: err.message });
68
+ }
69
+ });
70
+
71
+ /**
72
+ * GET /api/gemini/sessions/:projectHash
73
+ * 获取项目的所有会话
74
+ */
75
+ router.get('/:projectHash', (req, res) => {
76
+ try {
77
+ if (!isGeminiInstalled()) {
78
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
79
+ }
80
+
81
+ const { projectHash } = req.params;
82
+ const sessions = getProjectSessions(projectHash);
83
+
84
+ // 计算总大小
85
+ const totalSize = sessions.reduce((sum, session) => {
86
+ return sum + (session.size || 0);
87
+ }, 0);
88
+
89
+ // 获取别名
90
+ const aliases = loadAliases();
91
+
92
+ // 使用彩虹表解析真实路径
93
+ const realPath = getProjectPath(projectHash);
94
+ const path = require('path');
95
+ const displayName = realPath ? path.basename(realPath) : `Project ${projectHash.substring(0, 8)}`;
96
+
97
+ res.json({
98
+ sessions,
99
+ totalSize,
100
+ aliases,
101
+ projectInfo: {
102
+ name: projectHash,
103
+ fullPath: realPath || projectHash,
104
+ path: realPath || projectHash,
105
+ displayName
106
+ }
107
+ });
108
+ } catch (err) {
109
+ console.error('[Gemini API] Failed to get sessions:', err);
110
+ res.status(500).json({ error: err.message });
111
+ }
112
+ });
113
+
114
+ /**
115
+ * GET /api/gemini/sessions/:projectHash/search
116
+ * 搜索项目内会话内容
117
+ */
118
+ router.get('/:projectHash/search', (req, res) => {
119
+ try {
120
+ if (!isGeminiInstalled()) {
121
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
122
+ }
123
+
124
+ const { projectHash } = req.params;
125
+ const { keyword, context } = req.query;
126
+
127
+ if (!keyword) {
128
+ return res.status(400).json({ error: 'Keyword is required' });
129
+ }
130
+
131
+ const contextLength = context ? parseInt(context) : 35;
132
+
133
+ // 搜索所有会话,然后过滤该项目的会话
134
+ const allResults = searchSessions(keyword, contextLength);
135
+ const results = allResults.filter(r => r.projectHash === projectHash);
136
+
137
+ res.json({
138
+ keyword,
139
+ totalMatches: results.reduce((sum, r) => sum + r.matchCount, 0),
140
+ sessions: results
141
+ });
142
+ } catch (err) {
143
+ console.error('[Gemini API] Failed to search sessions:', err);
144
+ res.status(500).json({ error: err.message });
145
+ }
146
+ });
147
+
148
+ /**
149
+ * GET /api/gemini/sessions/:projectHash/:sessionId/messages
150
+ * 获取会话的消息列表
151
+ */
152
+ router.get('/:projectHash/:sessionId/messages', (req, res) => {
153
+ try {
154
+ if (!isGeminiInstalled()) {
155
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
156
+ }
157
+
158
+ const { sessionId } = req.params;
159
+ const { page = 1, limit = 20, order = 'desc' } = req.query;
160
+
161
+ const session = getSessionById(sessionId);
162
+
163
+ if (!session) {
164
+ return res.status(404).json({ error: 'Session not found' });
165
+ }
166
+
167
+ // 转换消息格式为前端期望的格式
168
+ const convertedMessages = [];
169
+
170
+ for (const msg of session.messages || []) {
171
+ // 用户消息
172
+ if (msg.type === 'user') {
173
+ convertedMessages.push({
174
+ type: 'user',
175
+ content: msg.content || '[空消息]',
176
+ timestamp: msg.timestamp,
177
+ model: null
178
+ });
179
+ }
180
+ // Gemini 助手消息(type 是 'gemini' 而不是 'assistant')
181
+ else if (msg.type === 'gemini' || msg.type === 'assistant') {
182
+ let content = msg.content || '[空消息]';
183
+
184
+ // 如果有 thoughts(思考过程),添加到内容前面
185
+ if (msg.thoughts && Array.isArray(msg.thoughts) && msg.thoughts.length > 0) {
186
+ const thoughtsText = msg.thoughts.map(t =>
187
+ `**[思考: ${t.subject}]**\n${t.description}`
188
+ ).join('\n\n');
189
+
190
+ content = `**[思考过程]**\n${thoughtsText}\n\n---\n\n${content}`;
191
+ }
192
+
193
+ convertedMessages.push({
194
+ type: 'assistant',
195
+ content,
196
+ timestamp: msg.timestamp,
197
+ model: msg.model || session.model || 'gemini'
198
+ });
199
+ }
200
+ }
201
+
202
+ // 分页处理
203
+ const pageNum = parseInt(page);
204
+ const limitNum = parseInt(limit);
205
+
206
+ // 排序
207
+ let messages = convertedMessages;
208
+ if (order === 'desc') {
209
+ messages = [...messages].reverse();
210
+ }
211
+
212
+ // 分页
213
+ const totalMessages = messages.length;
214
+ const start = (pageNum - 1) * limitNum;
215
+ const end = start + limitNum;
216
+ const paginatedMessages = messages.slice(start, end);
217
+
218
+ res.json({
219
+ messages: paginatedMessages,
220
+ metadata: {
221
+ gitBranch: null,
222
+ gitRepository: null,
223
+ cwd: null,
224
+ model: session.model || 'gemini'
225
+ },
226
+ pagination: {
227
+ page: pageNum,
228
+ limit: limitNum,
229
+ total: totalMessages,
230
+ hasMore: end < totalMessages
231
+ }
232
+ });
233
+ } catch (err) {
234
+ console.error('[Gemini API] Failed to get session messages:', err);
235
+ res.status(500).json({ error: err.message });
236
+ }
237
+ });
238
+
239
+ /**
240
+ * DELETE /api/gemini/sessions/:projectHash/:sessionId
241
+ * 删除会话
242
+ */
243
+ router.delete('/:projectHash/:sessionId', (req, res) => {
244
+ try {
245
+ if (!isGeminiInstalled()) {
246
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
247
+ }
248
+
249
+ const { sessionId } = req.params;
250
+ const result = deleteSession(sessionId);
251
+
252
+ res.json(result);
253
+ } catch (err) {
254
+ console.error('[Gemini API] Failed to delete session:', err);
255
+ res.status(500).json({ error: err.message });
256
+ }
257
+ });
258
+
259
+ /**
260
+ * POST /api/gemini/sessions/:projectHash/:sessionId/fork
261
+ * Fork 一个会话
262
+ */
263
+ router.post('/:projectHash/:sessionId/fork', (req, res) => {
264
+ try {
265
+ if (!isGeminiInstalled()) {
266
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
267
+ }
268
+
269
+ const { sessionId } = req.params;
270
+ const result = forkSession(sessionId);
271
+
272
+ res.json(result);
273
+ } catch (err) {
274
+ console.error('[Gemini API] Failed to fork session:', err);
275
+ res.status(500).json({ error: err.message });
276
+ }
277
+ });
278
+
279
+ /**
280
+ * POST /api/gemini/sessions/:projectHash/order
281
+ * 保存会话排序
282
+ */
283
+ router.post('/:projectHash/order', (req, res) => {
284
+ try {
285
+ if (!isGeminiInstalled()) {
286
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
287
+ }
288
+
289
+ const { projectHash } = req.params;
290
+ const { order } = req.body;
291
+
292
+ if (!Array.isArray(order)) {
293
+ return res.status(400).json({ error: 'order must be an array' });
294
+ }
295
+
296
+ saveSessionOrder(projectHash, order);
297
+
298
+ res.json({ success: true });
299
+ } catch (err) {
300
+ console.error('[Gemini API] Failed to save session order:', err);
301
+ res.status(500).json({ error: err.message });
302
+ }
303
+ });
304
+
305
+ /**
306
+ * POST /api/gemini/sessions/:projectHash/:sessionId/launch
307
+ * 获取会话启动命令(用于复制)
308
+ */
309
+ router.post('/:projectHash/:sessionId/launch', (req, res) => {
310
+ try {
311
+ if (!isGeminiInstalled()) {
312
+ return res.status(404).json({ error: 'Gemini CLI not installed' });
313
+ }
314
+
315
+ const { projectHash, sessionId } = req.params;
316
+
317
+ // 获取会话详情
318
+ const session = getSessionById(sessionId);
319
+
320
+ if (!session) {
321
+ return res.status(404).json({ error: 'Session not found' });
322
+ }
323
+
324
+ // 使用彩虹表方法获取项目路径
325
+ const projectPath = getProjectPath(projectHash);
326
+
327
+ if (!projectPath) {
328
+ return res.status(400).json({
329
+ error: 'Could not resolve project path. The original directory may have been moved or deleted.'
330
+ });
331
+ }
332
+
333
+ // 获取该项目的所有会话文件,按 startTime 升序排列(与 gemini --list-sessions 一致)
334
+ const allSessions = getAllSessions()
335
+ .filter(s => s.projectHash === projectHash)
336
+ .sort((a, b) => new Date(a.startTime) - new Date(b.startTime));
337
+
338
+ // 找到该 sessionId 对应的最新文件的索引
339
+ // 注意:同一个 sessionId 可能有多个文件(继续对话),我们要找最新的那个
340
+ let sessionIndex = -1;
341
+ for (let i = allSessions.length - 1; i >= 0; i--) {
342
+ if (allSessions[i].sessionId === sessionId) {
343
+ sessionIndex = i;
344
+ break;
345
+ }
346
+ }
347
+
348
+ if (sessionIndex === -1) {
349
+ return res.status(404).json({ error: 'Session not found in project sessions list' });
350
+ }
351
+
352
+ // Gemini 的索引从 1 开始
353
+ const resumeIndex = sessionIndex + 1;
354
+
355
+ // 构建 Gemini CLI 命令(使用 --resume <index> 恢复特定会话)
356
+ const command = `gemini --resume ${resumeIndex}`;
357
+ const quotedCwd = `"${String(projectPath).replace(/"/g, '\\"')}"`;
358
+ const copyCommand = `cd ${quotedCwd} && ${command}`;
359
+
360
+ res.json({
361
+ success: true,
362
+ sessionId,
363
+ projectPath,
364
+ cwd: projectPath,
365
+ tool: 'gemini',
366
+ command,
367
+ copyCommand
368
+ });
369
+ } catch (err) {
370
+ console.error('[Gemini API] Failed to prepare launch command:', err);
371
+ res.status(500).json({ error: err.message });
372
+ }
373
+ });
374
+
375
+ return router;
376
+ };
@@ -0,0 +1,57 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const {
4
+ getStatistics,
5
+ getDailyStatistics,
6
+ getTodayStatistics
7
+ } = require('../services/gemini-statistics-service');
8
+
9
+ /**
10
+ * 获取 Gemini 总体统计数据
11
+ * GET /api/gemini/statistics/summary
12
+ */
13
+ router.get('/summary', (req, res) => {
14
+ try {
15
+ const stats = getStatistics();
16
+ res.json(stats);
17
+ } catch (error) {
18
+ console.error('[Gemini] Failed to get statistics:', error);
19
+ res.status(500).json({ error: 'Failed to get statistics' });
20
+ }
21
+ });
22
+
23
+ /**
24
+ * 获取 Gemini 今日统计数据
25
+ * GET /api/gemini/statistics/today
26
+ */
27
+ router.get('/today', (req, res) => {
28
+ try {
29
+ const stats = getTodayStatistics();
30
+ res.json(stats);
31
+ } catch (error) {
32
+ console.error('[Gemini] Failed to get today statistics:', error);
33
+ res.status(500).json({ error: 'Failed to get today statistics' });
34
+ }
35
+ });
36
+
37
+ /**
38
+ * 获取 Gemini 指定日期的统计数据
39
+ * GET /api/gemini/statistics/daily/:date
40
+ */
41
+ router.get('/daily/:date', (req, res) => {
42
+ try {
43
+ const { date } = req.params;
44
+
45
+ if (!/^\d{4}-\d{2}-\d{2}$/.test(date)) {
46
+ return res.status(400).json({ error: 'Invalid date format. Expected YYYY-MM-DD' });
47
+ }
48
+
49
+ const stats = getDailyStatistics(date);
50
+ res.json(stats);
51
+ } catch (error) {
52
+ console.error('[Gemini] Failed to get daily statistics:', error);
53
+ res.status(500).json({ error: 'Failed to get daily statistics' });
54
+ }
55
+ });
56
+
57
+ module.exports = router;
@@ -0,0 +1,31 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { healthCheckAllProjects } = require('../services/health-check');
4
+ const { getProjects } = require('../services/sessions');
5
+
6
+ module.exports = (config) => {
7
+ /**
8
+ * GET /api/health-check - 健康检查所有项目
9
+ */
10
+ router.get('/', async (req, res) => {
11
+ try {
12
+ const projects = await getProjects(config);
13
+ const projectList = Array.isArray(projects) ? projects : [];
14
+ const result = healthCheckAllProjects(projectList);
15
+
16
+ res.json({
17
+ success: true,
18
+ timestamp: new Date().toISOString(),
19
+ ...result
20
+ });
21
+ } catch (error) {
22
+ console.error('Health check failed:', error);
23
+ res.status(500).json({
24
+ success: false,
25
+ error: error.message
26
+ });
27
+ }
28
+ });
29
+
30
+ return router;
31
+ };