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,483 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const {
4
+ getSessionsByProject,
5
+ getSessionById,
6
+ searchSessions,
7
+ forkSession,
8
+ deleteSession,
9
+ getRecentSessions,
10
+ saveSessionOrder
11
+ } = require('../services/codex-sessions');
12
+ const { isCodexInstalled } = require('../services/codex-config');
13
+ const { loadAliases } = require('../services/alias');
14
+
15
+ const DEBUG_CODEX_PERF = process.env.DEBUG_CODEX_PERF === '1';
16
+
17
+ function logPerf(route, startMs, detail = '') {
18
+ if (!DEBUG_CODEX_PERF) return;
19
+ const duration = Date.now() - startMs;
20
+ const suffix = detail ? ` | ${detail}` : '';
21
+ console.log(`[Codex Perf] ${route}: ${duration}ms${suffix}`);
22
+ }
23
+
24
+ module.exports = (config) => {
25
+ // ============================================
26
+ // 静态路由必须放在参数路由之前
27
+ // ============================================
28
+
29
+ /**
30
+ * GET /api/codex/sessions/search/global?keyword=xxx
31
+ * 全局搜索
32
+ */
33
+ router.get('/search/global', (req, res) => {
34
+ try {
35
+ if (!isCodexInstalled()) {
36
+ return res.status(404).json({ error: 'Codex CLI not installed' });
37
+ }
38
+
39
+ const { keyword } = req.query;
40
+
41
+ if (!keyword) {
42
+ return res.status(400).json({ error: 'Keyword is required' });
43
+ }
44
+
45
+ const results = searchSessions(keyword);
46
+
47
+ // 按会话分组,统计每个会话的匹配数
48
+ const sessionMap = new Map();
49
+ results.forEach(match => {
50
+ if (!sessionMap.has(match.sessionId)) {
51
+ sessionMap.set(match.sessionId, {
52
+ sessionId: match.sessionId,
53
+ projectName: match.projectName,
54
+ matchCount: 0,
55
+ matches: []
56
+ });
57
+ }
58
+ const session = sessionMap.get(match.sessionId);
59
+ session.matchCount++;
60
+ session.matches.push({
61
+ messageIndex: match.messageIndex,
62
+ role: match.role,
63
+ context: match.context,
64
+ timestamp: match.timestamp
65
+ });
66
+ });
67
+
68
+ const sessions = Array.from(sessionMap.values());
69
+
70
+ res.json({
71
+ keyword,
72
+ totalMatches: results.length,
73
+ sessions,
74
+ source: 'codex'
75
+ });
76
+ } catch (err) {
77
+ console.error('[Codex API] Failed to search sessions:', err);
78
+ res.status(500).json({ error: err.message });
79
+ }
80
+ });
81
+
82
+ /**
83
+ * GET /api/codex/sessions/recent/list?limit=10
84
+ * 获取最近会话
85
+ */
86
+ router.get('/recent/list', (req, res) => {
87
+ try {
88
+ if (!isCodexInstalled()) {
89
+ return res.status(404).json({ error: 'Codex CLI not installed' });
90
+ }
91
+
92
+ const limit = parseInt(req.query.limit) || 5;
93
+ const sessions = getRecentSessions(limit);
94
+
95
+ res.json({
96
+ sessions,
97
+ source: 'codex'
98
+ });
99
+ } catch (err) {
100
+ console.error('[Codex API] Failed to get recent sessions:', err);
101
+ res.status(500).json({ error: err.message });
102
+ }
103
+ });
104
+
105
+ // ============================================
106
+ // 参数路由
107
+ // ============================================
108
+
109
+ /**
110
+ * GET /api/codex/sessions/:projectName
111
+ * 获取项目的所有会话
112
+ */
113
+ router.get('/:projectName', (req, res) => {
114
+ const startMs = Date.now();
115
+ try {
116
+ if (!isCodexInstalled()) {
117
+ logPerf('GET /api/codex/sessions/:projectName', startMs, 'codex not installed');
118
+ return res.status(404).json({ error: 'Codex CLI not installed' });
119
+ }
120
+
121
+ const { projectName } = req.params;
122
+ const sessions = getSessionsByProject(projectName);
123
+
124
+ // 计算总大小
125
+ const totalSize = sessions.reduce((sum, session) => {
126
+ return sum + (session.size || 0);
127
+ }, 0);
128
+
129
+ // 获取别名
130
+ const aliases = loadAliases();
131
+ logPerf('GET /api/codex/sessions/:projectName', startMs, `project=${projectName}, sessions=${sessions.length}`);
132
+
133
+ res.json({
134
+ sessions,
135
+ totalSize,
136
+ aliases, // 返回所有别名
137
+ projectInfo: {
138
+ name: projectName,
139
+ fullPath: projectName,
140
+ displayName: projectName
141
+ }
142
+ });
143
+ } catch (err) {
144
+ console.error('[Codex API] Failed to get sessions:', err);
145
+ logPerf('GET /api/codex/sessions/:projectName', startMs, `error=${err.message}`);
146
+ res.status(500).json({ error: err.message });
147
+ }
148
+ });
149
+
150
+ /**
151
+ * GET /api/codex/sessions/:projectName/search
152
+ * 项目级搜索
153
+ */
154
+ router.get('/:projectName/search', (req, res) => {
155
+ try {
156
+ if (!isCodexInstalled()) {
157
+ return res.status(404).json({ error: 'Codex CLI not installed' });
158
+ }
159
+
160
+ const { projectName } = req.params;
161
+ const { keyword, context } = req.query;
162
+
163
+ if (!keyword) {
164
+ return res.status(400).json({ error: 'Keyword is required' });
165
+ }
166
+
167
+ // 使用全局搜索,然后过滤项目
168
+ const allResults = searchSessions(keyword);
169
+ const filteredResults = allResults.filter(r => r.projectName === projectName);
170
+
171
+ // 按会话分组
172
+ const sessionMap = new Map();
173
+ filteredResults.forEach(match => {
174
+ if (!sessionMap.has(match.sessionId)) {
175
+ sessionMap.set(match.sessionId, {
176
+ sessionId: match.sessionId,
177
+ projectName: match.projectName,
178
+ matchCount: 0,
179
+ matches: []
180
+ });
181
+ }
182
+ const session = sessionMap.get(match.sessionId);
183
+ session.matchCount++;
184
+ session.matches.push({
185
+ messageIndex: match.messageIndex,
186
+ role: match.role,
187
+ context: match.context,
188
+ timestamp: match.timestamp
189
+ });
190
+ });
191
+
192
+ const sessions = Array.from(sessionMap.values());
193
+
194
+ res.json({
195
+ keyword,
196
+ totalMatches: filteredResults.length,
197
+ sessions
198
+ });
199
+ } catch (err) {
200
+ console.error('[Codex API] Failed to search project sessions:', err);
201
+ res.status(500).json({ error: err.message });
202
+ }
203
+ });
204
+
205
+ /**
206
+ * GET /api/codex/sessions/:projectName/:sessionId/messages
207
+ * 获取会话的消息列表
208
+ */
209
+ router.get('/:projectName/:sessionId/messages', (req, res) => {
210
+ const startMs = Date.now();
211
+ try {
212
+ if (!isCodexInstalled()) {
213
+ logPerf('GET /api/codex/sessions/:projectName/:sessionId/messages', startMs, 'codex not installed');
214
+ return res.status(404).json({ error: 'Codex CLI not installed' });
215
+ }
216
+
217
+ const { sessionId } = req.params;
218
+ const { page = 1, limit = 20, order = 'desc' } = req.query;
219
+
220
+ const session = getSessionById(sessionId);
221
+
222
+ if (!session) {
223
+ logPerf('GET /api/codex/sessions/:projectName/:sessionId/messages', startMs, `session=${sessionId}, not found`);
224
+ return res.status(404).json({ error: 'Session not found' });
225
+ }
226
+
227
+ // 转换消息格式为前端期望的格式
228
+ const convertedMessages = [];
229
+
230
+ for (const msg of session.messages) {
231
+ // 用户消息
232
+ if (msg.role === 'user') {
233
+ convertedMessages.push({
234
+ type: 'user',
235
+ content: msg.content || '[空消息]',
236
+ timestamp: msg.timestamp,
237
+ model: null
238
+ });
239
+ }
240
+ // 助手消息(普通回复)
241
+ else if (msg.role === 'assistant') {
242
+ convertedMessages.push({
243
+ type: 'assistant',
244
+ content: msg.content || '[空消息]',
245
+ timestamp: msg.timestamp,
246
+ model: session.provider || 'codex'
247
+ });
248
+ }
249
+ // 推理内容
250
+ else if (msg.role === 'reasoning') {
251
+ convertedMessages.push({
252
+ type: 'assistant',
253
+ content: `**[推理]**\n${msg.content || '[空推理]'}`,
254
+ timestamp: msg.timestamp,
255
+ model: session.provider || 'codex'
256
+ });
257
+ }
258
+ // 工具调用
259
+ else if (msg.role === 'tool_call') {
260
+ const argsStr = typeof msg.arguments === 'object'
261
+ ? JSON.stringify(msg.arguments, null, 2)
262
+ : msg.arguments;
263
+ convertedMessages.push({
264
+ type: 'assistant',
265
+ content: `**[调用工具: ${msg.name}]**\n\`\`\`json\n${argsStr}\n\`\`\``,
266
+ timestamp: msg.timestamp,
267
+ model: session.provider || 'codex'
268
+ });
269
+ }
270
+ // 工具输出
271
+ else if (msg.role === 'tool_output') {
272
+ let outputStr = '';
273
+ if (typeof msg.output === 'object' && msg.output.output) {
274
+ // 标准格式:{ output: '...', metadata: {...} }
275
+ outputStr = msg.output.output;
276
+ if (msg.output.metadata) {
277
+ const meta = msg.output.metadata;
278
+ outputStr += `\n\n**[元数据]**\n- 退出码: ${meta.exit_code || 0}\n- 耗时: ${meta.duration_seconds || 0}s`;
279
+ }
280
+ } else if (typeof msg.output === 'string') {
281
+ outputStr = msg.output;
282
+ } else {
283
+ outputStr = JSON.stringify(msg.output, null, 2);
284
+ }
285
+
286
+ convertedMessages.push({
287
+ type: 'user',
288
+ content: `**[工具结果]**\n\`\`\`\n${outputStr}\n\`\`\``,
289
+ timestamp: msg.timestamp,
290
+ model: null
291
+ });
292
+ }
293
+ }
294
+
295
+ // 分页处理
296
+ const pageNum = parseInt(page);
297
+ const limitNum = parseInt(limit);
298
+
299
+ // 排序
300
+ let messages = convertedMessages;
301
+ if (order === 'desc') {
302
+ messages = [...messages].reverse();
303
+ }
304
+
305
+ // 分页
306
+ const totalMessages = messages.length;
307
+ const start = (pageNum - 1) * limitNum;
308
+ const end = start + limitNum;
309
+ const paginatedMessages = messages.slice(start, end);
310
+
311
+ res.json({
312
+ messages: paginatedMessages,
313
+ metadata: {
314
+ gitBranch: session.gitBranch,
315
+ gitRepository: session.gitRepository,
316
+ cwd: session.cwd,
317
+ provider: session.provider
318
+ },
319
+ pagination: {
320
+ page: pageNum,
321
+ limit: limitNum,
322
+ total: totalMessages,
323
+ hasMore: end < totalMessages
324
+ }
325
+ });
326
+ logPerf(
327
+ 'GET /api/codex/sessions/:projectName/:sessionId/messages',
328
+ startMs,
329
+ `session=${sessionId}, total=${totalMessages}, page=${pageNum}, returned=${paginatedMessages.length}`
330
+ );
331
+ } catch (err) {
332
+ console.error('[Codex API] Failed to get session messages:', err);
333
+ logPerf('GET /api/codex/sessions/:projectName/:sessionId/messages', startMs, `error=${err.message}`);
334
+ res.status(500).json({ error: err.message });
335
+ }
336
+ });
337
+
338
+ /**
339
+ * DELETE /api/codex/sessions/:projectName/:sessionId
340
+ * 删除会话
341
+ */
342
+ router.delete('/:projectName/:sessionId', (req, res) => {
343
+ try {
344
+ if (!isCodexInstalled()) {
345
+ return res.status(404).json({ error: 'Codex CLI not installed' });
346
+ }
347
+
348
+ const { sessionId } = req.params;
349
+ const result = deleteSession(sessionId);
350
+
351
+ res.json(result);
352
+ } catch (err) {
353
+ console.error('[Codex API] Failed to delete session:', err);
354
+ res.status(500).json({ error: err.message });
355
+ }
356
+ });
357
+
358
+ /**
359
+ * POST /api/codex/sessions/:projectName/:sessionId/fork
360
+ * Fork 一个会话
361
+ */
362
+ router.post('/:projectName/:sessionId/fork', (req, res) => {
363
+ try {
364
+ if (!isCodexInstalled()) {
365
+ return res.status(404).json({ error: 'Codex CLI not installed' });
366
+ }
367
+
368
+ const { sessionId } = req.params;
369
+ const result = forkSession(sessionId);
370
+
371
+ res.json(result);
372
+ } catch (err) {
373
+ console.error('[Codex API] Failed to fork session:', err);
374
+ res.status(500).json({ error: err.message });
375
+ }
376
+ });
377
+
378
+ /**
379
+ * POST /api/codex/sessions/:projectName/order
380
+ * 保存会话排序
381
+ */
382
+ router.post('/:projectName/order', (req, res) => {
383
+ try {
384
+ if (!isCodexInstalled()) {
385
+ return res.status(404).json({ error: 'Codex CLI not installed' });
386
+ }
387
+
388
+ const { projectName } = req.params;
389
+ const { order } = req.body;
390
+
391
+ if (!Array.isArray(order)) {
392
+ return res.status(400).json({ error: 'order must be an array' });
393
+ }
394
+
395
+ saveSessionOrder(projectName, order);
396
+
397
+ res.json({ success: true });
398
+ } catch (err) {
399
+ console.error('[Codex API] Failed to save session order:', err);
400
+ res.status(500).json({ error: err.message });
401
+ }
402
+ });
403
+
404
+ /**
405
+ * POST /api/codex/sessions/:projectName/:sessionId/launch
406
+ * 获取会话启动命令(用于复制)
407
+ */
408
+ router.post('/:projectName/:sessionId/launch', (req, res) => {
409
+ try {
410
+ if (!isCodexInstalled()) {
411
+ return res.status(404).json({ error: 'Codex CLI not installed' });
412
+ }
413
+
414
+ const { sessionId } = req.params;
415
+ const fs = require('fs');
416
+
417
+ // 获取会话详情
418
+ const session = getSessionById(sessionId);
419
+
420
+ if (!session) {
421
+ return res.status(404).json({ error: 'Session not found' });
422
+ }
423
+
424
+ // 从会话文件提取 cwd
425
+ let cwd = null;
426
+ try {
427
+ if (session.filePath && fs.existsSync(session.filePath)) {
428
+ const content = fs.readFileSync(session.filePath, 'utf8');
429
+ const firstLine = content.split('\n')[0];
430
+ if (firstLine) {
431
+ const json = JSON.parse(firstLine);
432
+ if (json.type === 'session_meta' && json.payload?.cwd) {
433
+ cwd = json.payload.cwd;
434
+ }
435
+ }
436
+ }
437
+ } catch (e) {
438
+ console.warn('Unable to extract cwd from Codex session:', e.message);
439
+ }
440
+
441
+ if (!cwd) {
442
+ return res.status(400).json({
443
+ error: 'Unable to determine working directory from session'
444
+ });
445
+ }
446
+
447
+ // 获取别名
448
+ const { loadAliases } = require('../services/alias');
449
+ const aliases = loadAliases();
450
+ const alias = aliases[sessionId];
451
+
452
+ // 广播行为日志
453
+ const { broadcastLog } = require('../websocket-server');
454
+ broadcastLog({
455
+ type: 'action',
456
+ action: 'launch_codex_session',
457
+ message: `复制 Codex 会话启动命令 ${alias || sessionId.substring(0, 8)}`,
458
+ sessionId: sessionId,
459
+ alias: alias || null,
460
+ timestamp: Date.now()
461
+ });
462
+
463
+ const command = `codex resume ${sessionId}`;
464
+ const quotedCwd = `"${String(cwd).replace(/"/g, '\\"')}"`;
465
+ const copyCommand = `cd ${quotedCwd} && ${command}`;
466
+
467
+ res.json({
468
+ success: true,
469
+ cwd,
470
+ sessionFile: session.filePath,
471
+ sessionId,
472
+ tool: 'codex',
473
+ command,
474
+ copyCommand
475
+ });
476
+ } catch (err) {
477
+ console.error('[Codex API] Failed to prepare launch command:', err);
478
+ res.status(500).json({ error: err.message });
479
+ }
480
+ });
481
+
482
+ return router;
483
+ };
@@ -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/codex-statistics-service');
8
+
9
+ /**
10
+ * 获取 Codex 总体统计数据
11
+ * GET /api/codex/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('[Codex] Failed to get statistics:', error);
19
+ res.status(500).json({ error: 'Failed to get statistics' });
20
+ }
21
+ });
22
+
23
+ /**
24
+ * 获取 Codex 今日统计数据
25
+ * GET /api/codex/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('[Codex] Failed to get today statistics:', error);
33
+ res.status(500).json({ error: 'Failed to get today statistics' });
34
+ }
35
+ });
36
+
37
+ /**
38
+ * 获取 Codex 指定日期的统计数据
39
+ * GET /api/codex/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('[Codex] Failed to get daily statistics:', error);
53
+ res.status(500).json({ error: 'Failed to get daily statistics' });
54
+ }
55
+ });
56
+
57
+ module.exports = router;