@pixelbyte-software/pixcode 1.30.2 → 1.31.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 (202) hide show
  1. package/LICENSE +718 -718
  2. package/README.de.md +248 -248
  3. package/README.ja.md +240 -240
  4. package/README.ko.md +240 -240
  5. package/README.md +295 -285
  6. package/README.ru.md +248 -248
  7. package/README.tr.md +250 -250
  8. package/README.zh-CN.md +240 -240
  9. package/dist/api-docs.html +879 -879
  10. package/dist/assets/index-BRRJ47XQ.css +32 -0
  11. package/dist/assets/index-EQohwyiC.js +837 -0
  12. package/dist/clear-cache.html +85 -85
  13. package/dist/convert-icons.md +52 -52
  14. package/dist/favicon.png +0 -0
  15. package/dist/favicon.svg +7 -8
  16. package/dist/generate-icons.js +48 -48
  17. package/dist/icons/codex-white.svg +3 -3
  18. package/dist/icons/codex.svg +3 -3
  19. package/dist/icons/cursor-white.svg +11 -11
  20. package/dist/icons/icon-128x128.png +0 -0
  21. package/dist/icons/icon-128x128.svg +9 -12
  22. package/dist/icons/icon-144x144.png +0 -0
  23. package/dist/icons/icon-144x144.svg +9 -12
  24. package/dist/icons/icon-152x152.png +0 -0
  25. package/dist/icons/icon-152x152.svg +9 -12
  26. package/dist/icons/icon-192x192.png +0 -0
  27. package/dist/icons/icon-192x192.svg +9 -12
  28. package/dist/icons/icon-384x384.png +0 -0
  29. package/dist/icons/icon-384x384.svg +9 -12
  30. package/dist/icons/icon-512x512.png +0 -0
  31. package/dist/icons/icon-512x512.svg +9 -12
  32. package/dist/icons/icon-72x72.png +0 -0
  33. package/dist/icons/icon-72x72.svg +9 -12
  34. package/dist/icons/icon-96x96.png +0 -0
  35. package/dist/icons/icon-96x96.svg +9 -12
  36. package/dist/icons/icon-template.svg +9 -12
  37. package/dist/icons/qwen-ai-icon.png +0 -0
  38. package/dist/index.html +59 -49
  39. package/dist/logo.png +0 -0
  40. package/dist/logo.svg +11 -16
  41. package/dist/manifest.json +60 -60
  42. package/dist/sw.js +124 -124
  43. package/dist-server/server/cli.js +100 -97
  44. package/dist-server/server/cli.js.map +1 -1
  45. package/dist-server/server/daemon/manager.js +33 -33
  46. package/dist-server/server/daemon-manager.js +62 -62
  47. package/dist-server/server/database/db.js +114 -22
  48. package/dist-server/server/database/db.js.map +1 -1
  49. package/dist-server/server/database/schema.js +122 -89
  50. package/dist-server/server/database/schema.js.map +1 -1
  51. package/dist-server/server/gemini-cli.js +6 -1
  52. package/dist-server/server/gemini-cli.js.map +1 -1
  53. package/dist-server/server/index.js +234 -64
  54. package/dist-server/server/index.js.map +1 -1
  55. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +29 -2
  56. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -1
  57. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +22 -2
  58. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -1
  59. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +2 -2
  60. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -1
  61. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +14 -2
  62. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -1
  63. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js +132 -0
  64. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js.map +1 -0
  65. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js +87 -0
  66. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js.map +1 -0
  67. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js +201 -0
  68. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js.map +1 -0
  69. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js +19 -0
  70. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js.map +1 -0
  71. package/dist-server/server/modules/providers/provider.registry.js +2 -0
  72. package/dist-server/server/modules/providers/provider.registry.js.map +1 -1
  73. package/dist-server/server/modules/providers/provider.routes.js +310 -1
  74. package/dist-server/server/modules/providers/provider.routes.js.map +1 -1
  75. package/dist-server/server/projects.js +197 -6
  76. package/dist-server/server/projects.js.map +1 -1
  77. package/dist-server/server/qwen-code-cli.js +350 -0
  78. package/dist-server/server/qwen-code-cli.js.map +1 -0
  79. package/dist-server/server/qwen-response-handler.js +70 -0
  80. package/dist-server/server/qwen-response-handler.js.map +1 -0
  81. package/dist-server/server/routes/commands.js +25 -25
  82. package/dist-server/server/routes/git.js +17 -17
  83. package/dist-server/server/routes/network.js +116 -0
  84. package/dist-server/server/routes/network.js.map +1 -0
  85. package/dist-server/server/routes/projects.js +43 -0
  86. package/dist-server/server/routes/projects.js.map +1 -1
  87. package/dist-server/server/routes/qwen.js +23 -0
  88. package/dist-server/server/routes/qwen.js.map +1 -0
  89. package/dist-server/server/routes/taskmaster.js +419 -419
  90. package/dist-server/server/routes/telegram.js +119 -0
  91. package/dist-server/server/routes/telegram.js.map +1 -0
  92. package/dist-server/server/services/external-access.js +228 -0
  93. package/dist-server/server/services/external-access.js.map +1 -0
  94. package/dist-server/server/services/install-jobs.js +394 -0
  95. package/dist-server/server/services/install-jobs.js.map +1 -0
  96. package/dist-server/server/services/notification-orchestrator.js +19 -5
  97. package/dist-server/server/services/notification-orchestrator.js.map +1 -1
  98. package/dist-server/server/services/provider-credentials.js +154 -0
  99. package/dist-server/server/services/provider-credentials.js.map +1 -0
  100. package/dist-server/server/services/provider-models.js +218 -0
  101. package/dist-server/server/services/provider-models.js.map +1 -0
  102. package/dist-server/server/services/telegram/bot.js +259 -0
  103. package/dist-server/server/services/telegram/bot.js.map +1 -0
  104. package/dist-server/server/services/telegram/translations.js +160 -0
  105. package/dist-server/server/services/telegram/translations.js.map +1 -0
  106. package/dist-server/server/utils/port-access.js +196 -0
  107. package/dist-server/server/utils/port-access.js.map +1 -0
  108. package/dist-server/shared/modelConstants.js +18 -0
  109. package/dist-server/shared/modelConstants.js.map +1 -1
  110. package/package.json +177 -168
  111. package/scripts/fix-node-pty.js +67 -67
  112. package/server/claude-sdk.js +834 -834
  113. package/server/cli.js +940 -937
  114. package/server/constants/config.js +4 -4
  115. package/server/cursor-cli.js +342 -342
  116. package/server/daemon/manager.js +564 -564
  117. package/server/daemon-manager.js +920 -920
  118. package/server/database/db.js +696 -593
  119. package/server/database/schema.js +138 -102
  120. package/server/gemini-cli.js +475 -469
  121. package/server/gemini-response-handler.js +79 -79
  122. package/server/index.js +2730 -2556
  123. package/server/load-env.js +34 -34
  124. package/server/middleware/auth.js +132 -132
  125. package/server/modules/providers/list/claude/claude-auth.provider.ts +145 -123
  126. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  127. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  128. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  129. package/server/modules/providers/list/codex/codex-auth.provider.ts +115 -100
  130. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  131. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  132. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  133. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -143
  134. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  135. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  136. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  137. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +163 -151
  138. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  139. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  140. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  141. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +145 -0
  142. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -0
  143. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +218 -0
  144. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -0
  145. package/server/modules/providers/provider.registry.ts +38 -36
  146. package/server/modules/providers/provider.routes.ts +583 -217
  147. package/server/modules/providers/services/mcp.service.ts +94 -94
  148. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  149. package/server/modules/providers/services/sessions.service.ts +45 -45
  150. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  151. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  152. package/server/modules/providers/tests/mcp.test.ts +293 -293
  153. package/server/openai-codex.js +426 -426
  154. package/server/projects.js +2993 -2792
  155. package/server/qwen-code-cli.js +392 -0
  156. package/server/qwen-response-handler.js +73 -0
  157. package/server/routes/agent.js +1245 -1245
  158. package/server/routes/auth.js +134 -134
  159. package/server/routes/codex.js +19 -19
  160. package/server/routes/commands.js +554 -554
  161. package/server/routes/cursor.js +52 -52
  162. package/server/routes/gemini.js +24 -24
  163. package/server/routes/git.js +1488 -1488
  164. package/server/routes/mcp-utils.js +31 -31
  165. package/server/routes/messages.js +61 -61
  166. package/server/routes/network.js +128 -0
  167. package/server/routes/plugins.js +307 -307
  168. package/server/routes/projects.js +675 -627
  169. package/server/routes/qwen.js +27 -0
  170. package/server/routes/settings.js +286 -286
  171. package/server/routes/taskmaster.js +1471 -1471
  172. package/server/routes/telegram.js +125 -0
  173. package/server/routes/user.js +123 -123
  174. package/server/services/external-access.js +240 -0
  175. package/server/services/install-jobs.js +410 -0
  176. package/server/services/notification-orchestrator.js +242 -227
  177. package/server/services/provider-credentials.js +151 -0
  178. package/server/services/provider-models.js +225 -0
  179. package/server/services/telegram/bot.js +280 -0
  180. package/server/services/telegram/translations.js +170 -0
  181. package/server/services/vapid-keys.js +35 -35
  182. package/server/sessionManager.js +225 -225
  183. package/server/shared/interfaces.ts +54 -54
  184. package/server/shared/types.ts +172 -172
  185. package/server/shared/utils.ts +193 -193
  186. package/server/tsconfig.json +36 -36
  187. package/server/utils/colors.js +21 -21
  188. package/server/utils/commandParser.js +303 -303
  189. package/server/utils/frontmatter.js +18 -18
  190. package/server/utils/gitConfig.js +34 -34
  191. package/server/utils/mcp-detector.js +147 -147
  192. package/server/utils/plugin-loader.js +457 -457
  193. package/server/utils/plugin-process-manager.js +184 -184
  194. package/server/utils/port-access.js +209 -0
  195. package/server/utils/runtime-paths.js +37 -37
  196. package/server/utils/taskmaster-websocket.js +128 -128
  197. package/server/utils/url-detection.js +71 -71
  198. package/server/vite-daemon.js +78 -78
  199. package/shared/modelConstants.js +117 -97
  200. package/shared/networkHosts.js +22 -22
  201. package/dist/assets/index-C2c9QNwK.css +0 -32
  202. package/dist/assets/index-DyXDZED-.js +0 -1277
@@ -1,319 +1,319 @@
1
- import { getCodexSessionMessages } from '@/projects.js';
2
- import type { IProviderSessions } from '@/shared/interfaces.js';
3
- import type { AnyRecord, FetchHistoryOptions, FetchHistoryResult, NormalizedMessage } from '@/shared/types.js';
4
- import { createNormalizedMessage, generateMessageId, readObjectRecord } from '@/shared/utils.js';
5
-
6
- const PROVIDER = 'codex';
7
-
8
- type CodexHistoryResult =
9
- | AnyRecord[]
10
- | {
11
- messages?: AnyRecord[];
12
- total?: number;
13
- hasMore?: boolean;
14
- tokenUsage?: unknown;
15
- };
16
-
17
- const loadCodexSessionMessages = getCodexSessionMessages as unknown as (
18
- sessionId: string,
19
- limit: number | null,
20
- offset: number,
21
- ) => Promise<CodexHistoryResult>;
22
-
23
- export class CodexSessionsProvider implements IProviderSessions {
24
- /**
25
- * Normalizes a persisted Codex JSONL entry.
26
- *
27
- * Live Codex SDK events are transformed before they reach normalizeMessage(),
28
- * while history entries already use a compact message/tool shape from projects.js.
29
- */
30
- private normalizeHistoryEntry(raw: AnyRecord, sessionId: string | null): NormalizedMessage[] {
31
- const ts = raw.timestamp || new Date().toISOString();
32
- const baseId = raw.uuid || generateMessageId('codex');
33
-
34
- if (raw.message?.role === 'user') {
35
- const content = typeof raw.message.content === 'string'
36
- ? raw.message.content
37
- : Array.isArray(raw.message.content)
38
- ? raw.message.content
39
- .map((part: string | AnyRecord) => typeof part === 'string' ? part : part?.text || '')
40
- .filter(Boolean)
41
- .join('\n')
42
- : String(raw.message.content || '');
43
- if (!content.trim()) {
44
- return [];
45
- }
46
- return [createNormalizedMessage({
47
- id: baseId,
48
- sessionId,
49
- timestamp: ts,
50
- provider: PROVIDER,
51
- kind: 'text',
52
- role: 'user',
53
- content,
54
- })];
55
- }
56
-
57
- if (raw.message?.role === 'assistant') {
58
- const content = typeof raw.message.content === 'string'
59
- ? raw.message.content
60
- : Array.isArray(raw.message.content)
61
- ? raw.message.content
62
- .map((part: string | AnyRecord) => typeof part === 'string' ? part : part?.text || '')
63
- .filter(Boolean)
64
- .join('\n')
65
- : '';
66
- if (!content.trim()) {
67
- return [];
68
- }
69
- return [createNormalizedMessage({
70
- id: baseId,
71
- sessionId,
72
- timestamp: ts,
73
- provider: PROVIDER,
74
- kind: 'text',
75
- role: 'assistant',
76
- content,
77
- })];
78
- }
79
-
80
- if (raw.type === 'thinking' || raw.isReasoning) {
81
- return [createNormalizedMessage({
82
- id: baseId,
83
- sessionId,
84
- timestamp: ts,
85
- provider: PROVIDER,
86
- kind: 'thinking',
87
- content: raw.message?.content || '',
88
- })];
89
- }
90
-
91
- if (raw.type === 'tool_use' || raw.toolName) {
92
- return [createNormalizedMessage({
93
- id: baseId,
94
- sessionId,
95
- timestamp: ts,
96
- provider: PROVIDER,
97
- kind: 'tool_use',
98
- toolName: raw.toolName || 'Unknown',
99
- toolInput: raw.toolInput,
100
- toolId: raw.toolCallId || baseId,
101
- })];
102
- }
103
-
104
- if (raw.type === 'tool_result') {
105
- return [createNormalizedMessage({
106
- id: baseId,
107
- sessionId,
108
- timestamp: ts,
109
- provider: PROVIDER,
110
- kind: 'tool_result',
111
- toolId: raw.toolCallId || '',
112
- content: raw.output || '',
113
- isError: Boolean(raw.isError),
114
- })];
115
- }
116
-
117
- return [];
118
- }
119
-
120
- /**
121
- * Normalizes either a Codex history entry or a transformed live SDK event.
122
- */
123
- normalizeMessage(rawMessage: unknown, sessionId: string | null): NormalizedMessage[] {
124
- const raw = readObjectRecord(rawMessage);
125
- if (!raw) {
126
- return [];
127
- }
128
-
129
- if (raw.message?.role) {
130
- return this.normalizeHistoryEntry(raw, sessionId);
131
- }
132
-
133
- const ts = raw.timestamp || new Date().toISOString();
134
- const baseId = raw.uuid || generateMessageId('codex');
135
-
136
- if (raw.type === 'item') {
137
- switch (raw.itemType) {
138
- case 'agent_message':
139
- return [createNormalizedMessage({
140
- id: baseId,
141
- sessionId,
142
- timestamp: ts,
143
- provider: PROVIDER,
144
- kind: 'text',
145
- role: 'assistant',
146
- content: raw.message?.content || '',
147
- })];
148
- case 'reasoning':
149
- return [createNormalizedMessage({
150
- id: baseId,
151
- sessionId,
152
- timestamp: ts,
153
- provider: PROVIDER,
154
- kind: 'thinking',
155
- content: raw.message?.content || '',
156
- })];
157
- case 'command_execution':
158
- return [createNormalizedMessage({
159
- id: baseId,
160
- sessionId,
161
- timestamp: ts,
162
- provider: PROVIDER,
163
- kind: 'tool_use',
164
- toolName: 'Bash',
165
- toolInput: { command: raw.command },
166
- toolId: baseId,
167
- output: raw.output,
168
- exitCode: raw.exitCode,
169
- status: raw.status,
170
- })];
171
- case 'file_change':
172
- return [createNormalizedMessage({
173
- id: baseId,
174
- sessionId,
175
- timestamp: ts,
176
- provider: PROVIDER,
177
- kind: 'tool_use',
178
- toolName: 'FileChanges',
179
- toolInput: raw.changes,
180
- toolId: baseId,
181
- status: raw.status,
182
- })];
183
- case 'mcp_tool_call':
184
- return [createNormalizedMessage({
185
- id: baseId,
186
- sessionId,
187
- timestamp: ts,
188
- provider: PROVIDER,
189
- kind: 'tool_use',
190
- toolName: raw.tool || 'MCP',
191
- toolInput: raw.arguments,
192
- toolId: baseId,
193
- server: raw.server,
194
- result: raw.result,
195
- error: raw.error,
196
- status: raw.status,
197
- })];
198
- case 'web_search':
199
- return [createNormalizedMessage({
200
- id: baseId,
201
- sessionId,
202
- timestamp: ts,
203
- provider: PROVIDER,
204
- kind: 'tool_use',
205
- toolName: 'WebSearch',
206
- toolInput: { query: raw.query },
207
- toolId: baseId,
208
- })];
209
- case 'todo_list':
210
- return [createNormalizedMessage({
211
- id: baseId,
212
- sessionId,
213
- timestamp: ts,
214
- provider: PROVIDER,
215
- kind: 'tool_use',
216
- toolName: 'TodoList',
217
- toolInput: { items: raw.items },
218
- toolId: baseId,
219
- })];
220
- case 'error':
221
- return [createNormalizedMessage({
222
- id: baseId,
223
- sessionId,
224
- timestamp: ts,
225
- provider: PROVIDER,
226
- kind: 'error',
227
- content: raw.message?.content || 'Unknown error',
228
- })];
229
- default:
230
- return [createNormalizedMessage({
231
- id: baseId,
232
- sessionId,
233
- timestamp: ts,
234
- provider: PROVIDER,
235
- kind: 'tool_use',
236
- toolName: raw.itemType || 'Unknown',
237
- toolInput: raw.item || raw,
238
- toolId: baseId,
239
- })];
240
- }
241
- }
242
-
243
- if (raw.type === 'turn_complete') {
244
- return [createNormalizedMessage({
245
- id: baseId,
246
- sessionId,
247
- timestamp: ts,
248
- provider: PROVIDER,
249
- kind: 'complete',
250
- })];
251
- }
252
- if (raw.type === 'turn_failed') {
253
- return [createNormalizedMessage({
254
- id: baseId,
255
- sessionId,
256
- timestamp: ts,
257
- provider: PROVIDER,
258
- kind: 'error',
259
- content: raw.error?.message || 'Turn failed',
260
- })];
261
- }
262
-
263
- return [];
264
- }
265
-
266
- /**
267
- * Loads Codex JSONL history and keeps token usage metadata when projects.js
268
- * provides it.
269
- */
270
- async fetchHistory(
271
- sessionId: string,
272
- options: FetchHistoryOptions = {},
273
- ): Promise<FetchHistoryResult> {
274
- const { limit = null, offset = 0 } = options;
275
-
276
- let result: CodexHistoryResult;
277
- try {
278
- result = await loadCodexSessionMessages(sessionId, limit, offset);
279
- } catch (error) {
280
- const message = error instanceof Error ? error.message : String(error);
281
- console.warn(`[CodexProvider] Failed to load session ${sessionId}:`, message);
282
- return { messages: [], total: 0, hasMore: false, offset: 0, limit: null };
283
- }
284
-
285
- const rawMessages = Array.isArray(result) ? result : (result.messages || []);
286
- const total = Array.isArray(result) ? rawMessages.length : (result.total || 0);
287
- const hasMore = Array.isArray(result) ? false : Boolean(result.hasMore);
288
- const tokenUsage = Array.isArray(result) ? undefined : result.tokenUsage;
289
-
290
- const normalized: NormalizedMessage[] = [];
291
- for (const raw of rawMessages) {
292
- normalized.push(...this.normalizeHistoryEntry(raw, sessionId));
293
- }
294
-
295
- const toolResultMap = new Map<string, NormalizedMessage>();
296
- for (const msg of normalized) {
297
- if (msg.kind === 'tool_result' && msg.toolId) {
298
- toolResultMap.set(msg.toolId, msg);
299
- }
300
- }
301
- for (const msg of normalized) {
302
- if (msg.kind === 'tool_use' && msg.toolId && toolResultMap.has(msg.toolId)) {
303
- const toolResult = toolResultMap.get(msg.toolId);
304
- if (toolResult) {
305
- msg.toolResult = { content: toolResult.content, isError: toolResult.isError };
306
- }
307
- }
308
- }
309
-
310
- return {
311
- messages: normalized,
312
- total,
313
- hasMore,
314
- offset,
315
- limit,
316
- tokenUsage,
317
- };
318
- }
319
- }
1
+ import { getCodexSessionMessages } from '@/projects.js';
2
+ import type { IProviderSessions } from '@/shared/interfaces.js';
3
+ import type { AnyRecord, FetchHistoryOptions, FetchHistoryResult, NormalizedMessage } from '@/shared/types.js';
4
+ import { createNormalizedMessage, generateMessageId, readObjectRecord } from '@/shared/utils.js';
5
+
6
+ const PROVIDER = 'codex';
7
+
8
+ type CodexHistoryResult =
9
+ | AnyRecord[]
10
+ | {
11
+ messages?: AnyRecord[];
12
+ total?: number;
13
+ hasMore?: boolean;
14
+ tokenUsage?: unknown;
15
+ };
16
+
17
+ const loadCodexSessionMessages = getCodexSessionMessages as unknown as (
18
+ sessionId: string,
19
+ limit: number | null,
20
+ offset: number,
21
+ ) => Promise<CodexHistoryResult>;
22
+
23
+ export class CodexSessionsProvider implements IProviderSessions {
24
+ /**
25
+ * Normalizes a persisted Codex JSONL entry.
26
+ *
27
+ * Live Codex SDK events are transformed before they reach normalizeMessage(),
28
+ * while history entries already use a compact message/tool shape from projects.js.
29
+ */
30
+ private normalizeHistoryEntry(raw: AnyRecord, sessionId: string | null): NormalizedMessage[] {
31
+ const ts = raw.timestamp || new Date().toISOString();
32
+ const baseId = raw.uuid || generateMessageId('codex');
33
+
34
+ if (raw.message?.role === 'user') {
35
+ const content = typeof raw.message.content === 'string'
36
+ ? raw.message.content
37
+ : Array.isArray(raw.message.content)
38
+ ? raw.message.content
39
+ .map((part: string | AnyRecord) => typeof part === 'string' ? part : part?.text || '')
40
+ .filter(Boolean)
41
+ .join('\n')
42
+ : String(raw.message.content || '');
43
+ if (!content.trim()) {
44
+ return [];
45
+ }
46
+ return [createNormalizedMessage({
47
+ id: baseId,
48
+ sessionId,
49
+ timestamp: ts,
50
+ provider: PROVIDER,
51
+ kind: 'text',
52
+ role: 'user',
53
+ content,
54
+ })];
55
+ }
56
+
57
+ if (raw.message?.role === 'assistant') {
58
+ const content = typeof raw.message.content === 'string'
59
+ ? raw.message.content
60
+ : Array.isArray(raw.message.content)
61
+ ? raw.message.content
62
+ .map((part: string | AnyRecord) => typeof part === 'string' ? part : part?.text || '')
63
+ .filter(Boolean)
64
+ .join('\n')
65
+ : '';
66
+ if (!content.trim()) {
67
+ return [];
68
+ }
69
+ return [createNormalizedMessage({
70
+ id: baseId,
71
+ sessionId,
72
+ timestamp: ts,
73
+ provider: PROVIDER,
74
+ kind: 'text',
75
+ role: 'assistant',
76
+ content,
77
+ })];
78
+ }
79
+
80
+ if (raw.type === 'thinking' || raw.isReasoning) {
81
+ return [createNormalizedMessage({
82
+ id: baseId,
83
+ sessionId,
84
+ timestamp: ts,
85
+ provider: PROVIDER,
86
+ kind: 'thinking',
87
+ content: raw.message?.content || '',
88
+ })];
89
+ }
90
+
91
+ if (raw.type === 'tool_use' || raw.toolName) {
92
+ return [createNormalizedMessage({
93
+ id: baseId,
94
+ sessionId,
95
+ timestamp: ts,
96
+ provider: PROVIDER,
97
+ kind: 'tool_use',
98
+ toolName: raw.toolName || 'Unknown',
99
+ toolInput: raw.toolInput,
100
+ toolId: raw.toolCallId || baseId,
101
+ })];
102
+ }
103
+
104
+ if (raw.type === 'tool_result') {
105
+ return [createNormalizedMessage({
106
+ id: baseId,
107
+ sessionId,
108
+ timestamp: ts,
109
+ provider: PROVIDER,
110
+ kind: 'tool_result',
111
+ toolId: raw.toolCallId || '',
112
+ content: raw.output || '',
113
+ isError: Boolean(raw.isError),
114
+ })];
115
+ }
116
+
117
+ return [];
118
+ }
119
+
120
+ /**
121
+ * Normalizes either a Codex history entry or a transformed live SDK event.
122
+ */
123
+ normalizeMessage(rawMessage: unknown, sessionId: string | null): NormalizedMessage[] {
124
+ const raw = readObjectRecord(rawMessage);
125
+ if (!raw) {
126
+ return [];
127
+ }
128
+
129
+ if (raw.message?.role) {
130
+ return this.normalizeHistoryEntry(raw, sessionId);
131
+ }
132
+
133
+ const ts = raw.timestamp || new Date().toISOString();
134
+ const baseId = raw.uuid || generateMessageId('codex');
135
+
136
+ if (raw.type === 'item') {
137
+ switch (raw.itemType) {
138
+ case 'agent_message':
139
+ return [createNormalizedMessage({
140
+ id: baseId,
141
+ sessionId,
142
+ timestamp: ts,
143
+ provider: PROVIDER,
144
+ kind: 'text',
145
+ role: 'assistant',
146
+ content: raw.message?.content || '',
147
+ })];
148
+ case 'reasoning':
149
+ return [createNormalizedMessage({
150
+ id: baseId,
151
+ sessionId,
152
+ timestamp: ts,
153
+ provider: PROVIDER,
154
+ kind: 'thinking',
155
+ content: raw.message?.content || '',
156
+ })];
157
+ case 'command_execution':
158
+ return [createNormalizedMessage({
159
+ id: baseId,
160
+ sessionId,
161
+ timestamp: ts,
162
+ provider: PROVIDER,
163
+ kind: 'tool_use',
164
+ toolName: 'Bash',
165
+ toolInput: { command: raw.command },
166
+ toolId: baseId,
167
+ output: raw.output,
168
+ exitCode: raw.exitCode,
169
+ status: raw.status,
170
+ })];
171
+ case 'file_change':
172
+ return [createNormalizedMessage({
173
+ id: baseId,
174
+ sessionId,
175
+ timestamp: ts,
176
+ provider: PROVIDER,
177
+ kind: 'tool_use',
178
+ toolName: 'FileChanges',
179
+ toolInput: raw.changes,
180
+ toolId: baseId,
181
+ status: raw.status,
182
+ })];
183
+ case 'mcp_tool_call':
184
+ return [createNormalizedMessage({
185
+ id: baseId,
186
+ sessionId,
187
+ timestamp: ts,
188
+ provider: PROVIDER,
189
+ kind: 'tool_use',
190
+ toolName: raw.tool || 'MCP',
191
+ toolInput: raw.arguments,
192
+ toolId: baseId,
193
+ server: raw.server,
194
+ result: raw.result,
195
+ error: raw.error,
196
+ status: raw.status,
197
+ })];
198
+ case 'web_search':
199
+ return [createNormalizedMessage({
200
+ id: baseId,
201
+ sessionId,
202
+ timestamp: ts,
203
+ provider: PROVIDER,
204
+ kind: 'tool_use',
205
+ toolName: 'WebSearch',
206
+ toolInput: { query: raw.query },
207
+ toolId: baseId,
208
+ })];
209
+ case 'todo_list':
210
+ return [createNormalizedMessage({
211
+ id: baseId,
212
+ sessionId,
213
+ timestamp: ts,
214
+ provider: PROVIDER,
215
+ kind: 'tool_use',
216
+ toolName: 'TodoList',
217
+ toolInput: { items: raw.items },
218
+ toolId: baseId,
219
+ })];
220
+ case 'error':
221
+ return [createNormalizedMessage({
222
+ id: baseId,
223
+ sessionId,
224
+ timestamp: ts,
225
+ provider: PROVIDER,
226
+ kind: 'error',
227
+ content: raw.message?.content || 'Unknown error',
228
+ })];
229
+ default:
230
+ return [createNormalizedMessage({
231
+ id: baseId,
232
+ sessionId,
233
+ timestamp: ts,
234
+ provider: PROVIDER,
235
+ kind: 'tool_use',
236
+ toolName: raw.itemType || 'Unknown',
237
+ toolInput: raw.item || raw,
238
+ toolId: baseId,
239
+ })];
240
+ }
241
+ }
242
+
243
+ if (raw.type === 'turn_complete') {
244
+ return [createNormalizedMessage({
245
+ id: baseId,
246
+ sessionId,
247
+ timestamp: ts,
248
+ provider: PROVIDER,
249
+ kind: 'complete',
250
+ })];
251
+ }
252
+ if (raw.type === 'turn_failed') {
253
+ return [createNormalizedMessage({
254
+ id: baseId,
255
+ sessionId,
256
+ timestamp: ts,
257
+ provider: PROVIDER,
258
+ kind: 'error',
259
+ content: raw.error?.message || 'Turn failed',
260
+ })];
261
+ }
262
+
263
+ return [];
264
+ }
265
+
266
+ /**
267
+ * Loads Codex JSONL history and keeps token usage metadata when projects.js
268
+ * provides it.
269
+ */
270
+ async fetchHistory(
271
+ sessionId: string,
272
+ options: FetchHistoryOptions = {},
273
+ ): Promise<FetchHistoryResult> {
274
+ const { limit = null, offset = 0 } = options;
275
+
276
+ let result: CodexHistoryResult;
277
+ try {
278
+ result = await loadCodexSessionMessages(sessionId, limit, offset);
279
+ } catch (error) {
280
+ const message = error instanceof Error ? error.message : String(error);
281
+ console.warn(`[CodexProvider] Failed to load session ${sessionId}:`, message);
282
+ return { messages: [], total: 0, hasMore: false, offset: 0, limit: null };
283
+ }
284
+
285
+ const rawMessages = Array.isArray(result) ? result : (result.messages || []);
286
+ const total = Array.isArray(result) ? rawMessages.length : (result.total || 0);
287
+ const hasMore = Array.isArray(result) ? false : Boolean(result.hasMore);
288
+ const tokenUsage = Array.isArray(result) ? undefined : result.tokenUsage;
289
+
290
+ const normalized: NormalizedMessage[] = [];
291
+ for (const raw of rawMessages) {
292
+ normalized.push(...this.normalizeHistoryEntry(raw, sessionId));
293
+ }
294
+
295
+ const toolResultMap = new Map<string, NormalizedMessage>();
296
+ for (const msg of normalized) {
297
+ if (msg.kind === 'tool_result' && msg.toolId) {
298
+ toolResultMap.set(msg.toolId, msg);
299
+ }
300
+ }
301
+ for (const msg of normalized) {
302
+ if (msg.kind === 'tool_use' && msg.toolId && toolResultMap.has(msg.toolId)) {
303
+ const toolResult = toolResultMap.get(msg.toolId);
304
+ if (toolResult) {
305
+ msg.toolResult = { content: toolResult.content, isError: toolResult.isError };
306
+ }
307
+ }
308
+ }
309
+
310
+ return {
311
+ messages: normalized,
312
+ total,
313
+ hasMore,
314
+ offset,
315
+ limit,
316
+ tokenUsage,
317
+ };
318
+ }
319
+ }
@@ -1,15 +1,15 @@
1
- import { AbstractProvider } from '@/modules/providers/shared/base/abstract.provider.js';
2
- import { CodexProviderAuth } from '@/modules/providers/list/codex/codex-auth.provider.js';
3
- import { CodexMcpProvider } from '@/modules/providers/list/codex/codex-mcp.provider.js';
4
- import { CodexSessionsProvider } from '@/modules/providers/list/codex/codex-sessions.provider.js';
5
- import type { IProviderAuth, IProviderSessions } from '@/shared/interfaces.js';
6
-
7
- export class CodexProvider extends AbstractProvider {
8
- readonly mcp = new CodexMcpProvider();
9
- readonly auth: IProviderAuth = new CodexProviderAuth();
10
- readonly sessions: IProviderSessions = new CodexSessionsProvider();
11
-
12
- constructor() {
13
- super('codex');
14
- }
15
- }
1
+ import { AbstractProvider } from '@/modules/providers/shared/base/abstract.provider.js';
2
+ import { CodexProviderAuth } from '@/modules/providers/list/codex/codex-auth.provider.js';
3
+ import { CodexMcpProvider } from '@/modules/providers/list/codex/codex-mcp.provider.js';
4
+ import { CodexSessionsProvider } from '@/modules/providers/list/codex/codex-sessions.provider.js';
5
+ import type { IProviderAuth, IProviderSessions } from '@/shared/interfaces.js';
6
+
7
+ export class CodexProvider extends AbstractProvider {
8
+ readonly mcp = new CodexMcpProvider();
9
+ readonly auth: IProviderAuth = new CodexProviderAuth();
10
+ readonly sessions: IProviderSessions = new CodexSessionsProvider();
11
+
12
+ constructor() {
13
+ super('codex');
14
+ }
15
+ }