upfynai-code 3.0.3 → 3.1.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 (243) hide show
  1. package/README.md +66 -91
  2. package/bin/cli.js +191 -0
  3. package/{client/dist → dist/client}/api-docs.html +838 -838
  4. package/{client/dist/assets/AppContent-Bvg0CPCO.js → dist/client/assets/AppContent-BofJquUs.js} +43 -43
  5. package/dist/client/assets/BrowserPanel-CSvD4jOX.js +2 -0
  6. package/dist/client/assets/CanvasFullScreen-onRfarpc.js +1 -0
  7. package/dist/client/assets/CanvasWorkspace-DvGKdL-k.js +259 -0
  8. package/dist/client/assets/DashboardPanel-DqAHbXDO.js +1 -0
  9. package/dist/client/assets/FileTree-BE0h-9M9.js +1 -0
  10. package/{client/dist/assets/GitPanel-RtyZUIWS.js → dist/client/assets/GitPanel-DdeJ0bp5.js} +2 -2
  11. package/{client/dist/assets/LoginModal-BWep8a6g.js → dist/client/assets/LoginModal-BP0pCTrH.js} +3 -3
  12. package/{client/dist/assets/MarkdownPreview-DHmk3qzu.js → dist/client/assets/MarkdownPreview-CESjI261.js} +1 -1
  13. package/dist/client/assets/MermaidBlock-D0rfEhrT.js +2 -0
  14. package/dist/client/assets/Onboarding-B2zQy-_6.js +1 -0
  15. package/dist/client/assets/SetupForm-Be7-WBe-.js +1 -0
  16. package/dist/client/assets/WorkflowsPanel-CusLbVJ6.js +1 -0
  17. package/{client/dist/assets/index-C5ptjuTl.js → dist/client/assets/index-BQy15irW.js} +25 -25
  18. package/dist/client/assets/index-CS0fDqEC.js +1 -0
  19. package/dist/client/assets/index-DYLSCCCp.css +1 -0
  20. package/dist/client/assets/vendor-canvas-QWTduIvM.js +23 -0
  21. package/{client/dist/assets/vendor-codemirror-CbtmxxaB.js → dist/client/assets/vendor-codemirror-D2ALgpaX.js} +1 -1
  22. package/{client/dist/assets/vendor-icons-BaD0x9SL.js → dist/client/assets/vendor-icons-kix3Gb31.js} +178 -138
  23. package/{client/dist/assets/vendor-mermaid-CH7SGc99.js → dist/client/assets/vendor-mermaid-CS3J4_Bz.js} +329 -326
  24. package/{client/dist/assets/vendor-syntax-DuHI9Ok6.js → dist/client/assets/vendor-syntax-LS_Nt30I.js} +1 -1
  25. package/{client/dist → dist/client}/clear-cache.html +85 -85
  26. package/dist/client/favicon.png +0 -0
  27. package/dist/client/favicon.svg +15 -0
  28. package/{client/dist → dist/client}/index.html +17 -17
  29. package/{client/dist → dist/client}/manifest.json +15 -15
  30. package/{client/dist → dist/client}/mcp-docs.html +108 -108
  31. package/{client/dist → dist/client}/offline.html +84 -84
  32. package/{client/dist → dist/client}/sw.js +82 -82
  33. package/package.json +55 -104
  34. package/scripts/postinstall.js +9 -0
  35. package/scripts/prepublish.js +77 -0
  36. package/src/animation.js +228 -0
  37. package/src/auth.js +142 -0
  38. package/src/config.js +40 -0
  39. package/src/connect.js +416 -0
  40. package/src/launch.js +81 -0
  41. package/src/mcp.js +57 -0
  42. package/src/permissions.js +140 -0
  43. package/src/persistent-shell.js +261 -0
  44. package/src/server.js +54 -0
  45. package/client/dist/assets/CanvasFullScreen-BdiJ35aq.js +0 -1
  46. package/client/dist/assets/CanvasWorkspace-Bk9R9_e0.js +0 -163
  47. package/client/dist/assets/DashboardPanel-CblJfTGi.js +0 -1
  48. package/client/dist/assets/FileTree-BDUnBheV.js +0 -1
  49. package/client/dist/assets/MermaidBlock-BuBc_G-F.js +0 -2
  50. package/client/dist/assets/Onboarding-Drnlt75a.js +0 -1
  51. package/client/dist/assets/SetupForm-CtCKitZG.js +0 -1
  52. package/client/dist/assets/WorkflowsPanel-B2mIXDvD.js +0 -1
  53. package/client/dist/assets/index-BFuqS0tY.css +0 -1
  54. package/client/dist/assets/vendor-canvas-D39yWul6.js +0 -49
  55. package/client/dist/favicon.png +0 -0
  56. package/client/dist/favicon.svg +0 -5
  57. package/commands/upfynai-connect.md +0 -59
  58. package/commands/upfynai-disconnect.md +0 -31
  59. package/commands/upfynai-doctor.md +0 -99
  60. package/commands/upfynai-export.md +0 -49
  61. package/commands/upfynai-local.md +0 -82
  62. package/commands/upfynai-status.md +0 -75
  63. package/commands/upfynai-stop.md +0 -49
  64. package/commands/upfynai-uninstall.md +0 -58
  65. package/commands/upfynai.md +0 -69
  66. package/scripts/build-client.js +0 -17
  67. package/scripts/fix-node-pty.js +0 -67
  68. package/scripts/install-commands.js +0 -78
  69. package/server/agent-loop.js +0 -242
  70. package/server/auto-compact.js +0 -99
  71. package/server/claude-sdk.js +0 -797
  72. package/server/cli-ui.js +0 -798
  73. package/server/cli.js +0 -751
  74. package/server/constants/config.js +0 -31
  75. package/server/cursor-cli.js +0 -270
  76. package/server/database/auth.db +0 -0
  77. package/server/database/db.js +0 -1451
  78. package/server/database/init.sql +0 -70
  79. package/server/index.js +0 -3814
  80. package/server/load-env.js +0 -26
  81. package/server/mcp-server.js +0 -621
  82. package/server/middleware/auth.js +0 -181
  83. package/server/middleware/relayHelpers.js +0 -44
  84. package/server/middleware/sandboxRouter.js +0 -174
  85. package/server/openai-codex.js +0 -403
  86. package/server/openrouter.js +0 -137
  87. package/server/projects.js +0 -1807
  88. package/server/provider-factory.js +0 -174
  89. package/server/relay-client.js +0 -390
  90. package/server/routes/agent.js +0 -1234
  91. package/server/routes/auth.js +0 -559
  92. package/server/routes/canvas.js +0 -53
  93. package/server/routes/cli-auth.js +0 -263
  94. package/server/routes/codex.js +0 -396
  95. package/server/routes/commands.js +0 -707
  96. package/server/routes/composio.js +0 -176
  97. package/server/routes/cursor.js +0 -770
  98. package/server/routes/dashboard.js +0 -295
  99. package/server/routes/git.js +0 -1208
  100. package/server/routes/keys.js +0 -34
  101. package/server/routes/mcp-utils.js +0 -48
  102. package/server/routes/mcp.js +0 -661
  103. package/server/routes/payments.js +0 -227
  104. package/server/routes/projects.js +0 -655
  105. package/server/routes/sessions.js +0 -146
  106. package/server/routes/settings.js +0 -261
  107. package/server/routes/taskmaster.js +0 -1928
  108. package/server/routes/user.js +0 -106
  109. package/server/routes/vapi-chat.js +0 -624
  110. package/server/routes/voice.js +0 -235
  111. package/server/routes/webhooks.js +0 -166
  112. package/server/routes/workflows.js +0 -312
  113. package/server/sandbox.js +0 -120
  114. package/server/services/composio.js +0 -204
  115. package/server/services/sessionRegistry.js +0 -139
  116. package/server/services/whisperService.js +0 -84
  117. package/server/services/workflowScheduler.js +0 -211
  118. package/server/tests/relay-flow.test.js +0 -570
  119. package/server/tests/sessions.test.js +0 -259
  120. package/server/utils/commandParser.js +0 -303
  121. package/server/utils/email.js +0 -66
  122. package/server/utils/gitConfig.js +0 -24
  123. package/server/utils/mcp-detector.js +0 -198
  124. package/server/utils/taskmaster-websocket.js +0 -129
  125. package/shared/integrationCatalog.d.ts +0 -12
  126. package/shared/integrationCatalog.js +0 -172
  127. package/shared/modelConstants.js +0 -96
  128. /package/{shared → dist}/agents/claude.js +0 -0
  129. /package/{shared → dist}/agents/codex.js +0 -0
  130. /package/{shared → dist}/agents/cursor.js +0 -0
  131. /package/{shared → dist}/agents/detect.js +0 -0
  132. /package/{shared → dist}/agents/exec.js +0 -0
  133. /package/{shared → dist}/agents/files.js +0 -0
  134. /package/{shared → dist}/agents/git.js +0 -0
  135. /package/{shared → dist}/agents/gitagent.js +0 -0
  136. /package/{shared → dist}/agents/index.js +0 -0
  137. /package/{shared → dist}/agents/shell.js +0 -0
  138. /package/{shared → dist}/agents/utils.js +0 -0
  139. /package/{client/dist → dist/client}/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  140. /package/{client/dist → dist/client}/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  141. /package/{client/dist → dist/client}/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  142. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  143. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  144. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  145. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  146. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  147. /package/{client/dist → dist/client}/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  148. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  149. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  150. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  151. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  152. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  153. /package/{client/dist → dist/client}/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  154. /package/{client/dist → dist/client}/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  155. /package/{client/dist → dist/client}/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  156. /package/{client/dist → dist/client}/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  157. /package/{client/dist → dist/client}/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  158. /package/{client/dist → dist/client}/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  159. /package/{client/dist → dist/client}/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  160. /package/{client/dist → dist/client}/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  161. /package/{client/dist → dist/client}/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  162. /package/{client/dist → dist/client}/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  163. /package/{client/dist → dist/client}/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  164. /package/{client/dist → dist/client}/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  165. /package/{client/dist → dist/client}/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  166. /package/{client/dist → dist/client}/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  167. /package/{client/dist → dist/client}/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  168. /package/{client/dist → dist/client}/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  169. /package/{client/dist → dist/client}/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  170. /package/{client/dist → dist/client}/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  171. /package/{client/dist → dist/client}/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  172. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  173. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  174. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  175. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  176. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  177. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  178. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  179. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  180. /package/{client/dist → dist/client}/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  181. /package/{client/dist → dist/client}/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  182. /package/{client/dist → dist/client}/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  183. /package/{client/dist → dist/client}/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  184. /package/{client/dist → dist/client}/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  185. /package/{client/dist → dist/client}/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  186. /package/{client/dist → dist/client}/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  187. /package/{client/dist → dist/client}/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  188. /package/{client/dist → dist/client}/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  189. /package/{client/dist → dist/client}/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  190. /package/{client/dist → dist/client}/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  191. /package/{client/dist → dist/client}/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  192. /package/{client/dist → dist/client}/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  193. /package/{client/dist → dist/client}/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  194. /package/{client/dist → dist/client}/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  195. /package/{client/dist → dist/client}/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  196. /package/{client/dist → dist/client}/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  197. /package/{client/dist → dist/client}/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  198. /package/{client/dist → dist/client}/assets/PreviewPanel-CqCa92Tf.js +0 -0
  199. /package/{client/dist → dist/client}/assets/pdf-CE_K4jFx.js +0 -0
  200. /package/{client/dist → dist/client}/assets/vendor-canvas-BZV40eAE.css +0 -0
  201. /package/{client/dist → dist/client}/assets/vendor-diff-DNQpbhrT.js +0 -0
  202. /package/{client/dist → dist/client}/assets/vendor-i18n-DCFGyhQR.js +0 -0
  203. /package/{client/dist → dist/client}/assets/vendor-markdown-CimbIo6Y.js +0 -0
  204. /package/{client/dist → dist/client}/assets/vendor-react-96lCPsRK.js +0 -0
  205. /package/{client/dist → dist/client}/assets/vendor-xterm-CZq1hqo1.js +0 -0
  206. /package/{client/dist → dist/client}/assets/vendor-xterm-qxJ8_QYu.css +0 -0
  207. /package/{client/dist → dist/client}/convert-icons.md +0 -0
  208. /package/{client/dist → dist/client}/generate-icons.js +0 -0
  209. /package/{client/dist → dist/client}/icons/claude-ai-icon.svg +0 -0
  210. /package/{client/dist → dist/client}/icons/codex-white.svg +0 -0
  211. /package/{client/dist → dist/client}/icons/codex.svg +0 -0
  212. /package/{client/dist → dist/client}/icons/cursor-white.svg +0 -0
  213. /package/{client/dist → dist/client}/icons/cursor.svg +0 -0
  214. /package/{client/dist → dist/client}/icons/icon-128x128.png +0 -0
  215. /package/{client/dist → dist/client}/icons/icon-128x128.svg +0 -0
  216. /package/{client/dist → dist/client}/icons/icon-144x144.png +0 -0
  217. /package/{client/dist → dist/client}/icons/icon-144x144.svg +0 -0
  218. /package/{client/dist → dist/client}/icons/icon-152x152.png +0 -0
  219. /package/{client/dist → dist/client}/icons/icon-152x152.svg +0 -0
  220. /package/{client/dist → dist/client}/icons/icon-192x192.png +0 -0
  221. /package/{client/dist → dist/client}/icons/icon-192x192.svg +0 -0
  222. /package/{client/dist → dist/client}/icons/icon-384x384.png +0 -0
  223. /package/{client/dist → dist/client}/icons/icon-384x384.svg +0 -0
  224. /package/{client/dist → dist/client}/icons/icon-512x512.png +0 -0
  225. /package/{client/dist → dist/client}/icons/icon-512x512.svg +0 -0
  226. /package/{client/dist → dist/client}/icons/icon-72x72.png +0 -0
  227. /package/{client/dist → dist/client}/icons/icon-72x72.svg +0 -0
  228. /package/{client/dist → dist/client}/icons/icon-96x96.png +0 -0
  229. /package/{client/dist → dist/client}/icons/icon-96x96.svg +0 -0
  230. /package/{client/dist → dist/client}/icons/icon-template.svg +0 -0
  231. /package/{client/dist → dist/client}/logo-128.png +0 -0
  232. /package/{client/dist → dist/client}/logo-256.png +0 -0
  233. /package/{client/dist → dist/client}/logo-32.png +0 -0
  234. /package/{client/dist → dist/client}/logo-512.png +0 -0
  235. /package/{client/dist → dist/client}/logo-64.png +0 -0
  236. /package/{client/dist → dist/client}/logo.svg +0 -0
  237. /package/{client/dist → dist/client}/screenshots/cli-selection.png +0 -0
  238. /package/{client/dist → dist/client}/screenshots/desktop-main.png +0 -0
  239. /package/{client/dist → dist/client}/screenshots/mobile-chat.png +0 -0
  240. /package/{client/dist → dist/client}/screenshots/tools-modal.png +0 -0
  241. /package/{shared → dist}/gitagent/index.js +0 -0
  242. /package/{shared → dist}/gitagent/parser.js +0 -0
  243. /package/{shared → dist}/gitagent/prompt-builder.js +0 -0
@@ -1,263 +0,0 @@
1
- import express from 'express';
2
- import { spawn } from 'child_process';
3
- import fs from 'fs/promises';
4
- import path from 'path';
5
- import os from 'os';
6
-
7
- const router = express.Router();
8
-
9
- router.get('/claude/status', async (req, res) => {
10
- try {
11
- const credentialsResult = await checkClaudeCredentials();
12
-
13
- if (credentialsResult.authenticated) {
14
- return res.json({
15
- authenticated: true,
16
- email: credentialsResult.email || 'Authenticated',
17
- method: 'credentials_file'
18
- });
19
- }
20
-
21
- return res.json({
22
- authenticated: false,
23
- email: null,
24
- error: credentialsResult.error || 'Not authenticated'
25
- });
26
-
27
- } catch (error) {
28
- // auth status error
29
- res.status(500).json({
30
- authenticated: false,
31
- email: null,
32
- error: 'An error occurred'
33
- });
34
- }
35
- });
36
-
37
- router.get('/cursor/status', async (req, res) => {
38
- try {
39
- const result = await checkCursorStatus();
40
-
41
- res.json({
42
- authenticated: result.authenticated,
43
- email: result.email,
44
- error: result.error
45
- });
46
-
47
- } catch (error) {
48
- // auth status error
49
- res.status(500).json({
50
- authenticated: false,
51
- email: null,
52
- error: 'An error occurred'
53
- });
54
- }
55
- });
56
-
57
- router.get('/codex/status', async (req, res) => {
58
- try {
59
- const result = await checkCodexCredentials();
60
-
61
- res.json({
62
- authenticated: result.authenticated,
63
- email: result.email,
64
- error: result.error
65
- });
66
-
67
- } catch (error) {
68
- // auth status error
69
- res.status(500).json({
70
- authenticated: false,
71
- email: null,
72
- error: 'An error occurred'
73
- });
74
- }
75
- });
76
-
77
- async function checkClaudeCredentials() {
78
- try {
79
- const credPath = path.join(os.homedir(), '.claude', '.credentials.json');
80
- const content = await fs.readFile(credPath, 'utf8');
81
- const creds = JSON.parse(content);
82
-
83
- const oauth = creds.claudeAiOauth;
84
- if (oauth && oauth.accessToken) {
85
- const isExpired = oauth.expiresAt && Date.now() >= oauth.expiresAt;
86
-
87
- if (!isExpired) {
88
- return {
89
- authenticated: true,
90
- email: creds.email || creds.user || null
91
- };
92
- }
93
- }
94
-
95
- return {
96
- authenticated: false,
97
- email: null
98
- };
99
- } catch (error) {
100
- return {
101
- authenticated: false,
102
- email: null
103
- };
104
- }
105
- }
106
-
107
- function checkCursorStatus() {
108
- return new Promise((resolve) => {
109
- let processCompleted = false;
110
-
111
- const timeout = setTimeout(() => {
112
- if (!processCompleted) {
113
- processCompleted = true;
114
- if (childProcess) {
115
- childProcess.kill();
116
- }
117
- resolve({
118
- authenticated: false,
119
- email: null,
120
- error: 'Command timeout'
121
- });
122
- }
123
- }, 5000);
124
-
125
- let childProcess;
126
- try {
127
- childProcess = spawn('cursor-agent', ['status']);
128
- } catch (err) {
129
- clearTimeout(timeout);
130
- processCompleted = true;
131
- resolve({
132
- authenticated: false,
133
- email: null,
134
- error: 'Cursor CLI not found or not installed'
135
- });
136
- return;
137
- }
138
-
139
- let stdout = '';
140
- let stderr = '';
141
-
142
- childProcess.stdout.on('data', (data) => {
143
- stdout += data.toString();
144
- });
145
-
146
- childProcess.stderr.on('data', (data) => {
147
- stderr += data.toString();
148
- });
149
-
150
- childProcess.on('close', (code) => {
151
- if (processCompleted) return;
152
- processCompleted = true;
153
- clearTimeout(timeout);
154
-
155
- if (code === 0) {
156
- const emailMatch = stdout.match(/Logged in as ([a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,})/i);
157
-
158
- if (emailMatch) {
159
- resolve({
160
- authenticated: true,
161
- email: emailMatch[1],
162
- output: stdout
163
- });
164
- } else if (stdout.includes('Logged in')) {
165
- resolve({
166
- authenticated: true,
167
- email: 'Logged in',
168
- output: stdout
169
- });
170
- } else {
171
- resolve({
172
- authenticated: false,
173
- email: null,
174
- error: 'Not logged in'
175
- });
176
- }
177
- } else {
178
- resolve({
179
- authenticated: false,
180
- email: null,
181
- error: stderr || 'Not logged in'
182
- });
183
- }
184
- });
185
-
186
- childProcess.on('error', (err) => {
187
- if (processCompleted) return;
188
- processCompleted = true;
189
- clearTimeout(timeout);
190
-
191
- resolve({
192
- authenticated: false,
193
- email: null,
194
- error: 'Cursor CLI not found or not installed'
195
- });
196
- });
197
- });
198
- }
199
-
200
- async function checkCodexCredentials() {
201
- try {
202
- const authPath = path.join(os.homedir(), '.codex', 'auth.json');
203
- const content = await fs.readFile(authPath, 'utf8');
204
- const auth = JSON.parse(content);
205
-
206
- // Tokens are nested under 'tokens' key
207
- const tokens = auth.tokens || {};
208
-
209
- // Check for valid tokens (id_token or access_token)
210
- if (tokens.id_token || tokens.access_token) {
211
- // Try to extract email from id_token JWT payload
212
- let email = 'Authenticated';
213
- if (tokens.id_token) {
214
- try {
215
- // JWT is base64url encoded: header.payload.signature
216
- const parts = tokens.id_token.split('.');
217
- if (parts.length >= 2) {
218
- // Decode the payload (second part)
219
- const payload = JSON.parse(Buffer.from(parts[1], 'base64url').toString('utf8'));
220
- email = payload.email || payload.user || 'Authenticated';
221
- }
222
- } catch {
223
- // If JWT decoding fails, use fallback
224
- email = 'Authenticated';
225
- }
226
- }
227
-
228
- return {
229
- authenticated: true,
230
- email
231
- };
232
- }
233
-
234
- // Also check for OPENAI_API_KEY as fallback auth method
235
- if (auth.OPENAI_API_KEY) {
236
- return {
237
- authenticated: true,
238
- email: 'API Key Auth'
239
- };
240
- }
241
-
242
- return {
243
- authenticated: false,
244
- email: null,
245
- error: 'No valid tokens found'
246
- };
247
- } catch (error) {
248
- if (error.code === 'ENOENT') {
249
- return {
250
- authenticated: false,
251
- email: null,
252
- error: 'Codex not configured'
253
- };
254
- }
255
- return {
256
- authenticated: false,
257
- email: null,
258
- error: 'An error occurred'
259
- };
260
- }
261
- }
262
-
263
- export default router;
@@ -1,396 +0,0 @@
1
- import express from 'express';
2
- import { spawn } from 'child_process';
3
- import { promises as fs } from 'fs';
4
- import path from 'path';
5
- import os from 'os';
6
- import TOML from '@iarna/toml';
7
- import { getCodexSessions, getCodexSessionMessages, deleteCodexSession } from '../projects.js';
8
-
9
- const router = express.Router();
10
-
11
- function createCliResponder(res) {
12
- let responded = false;
13
- return (status, payload) => {
14
- if (responded || res.headersSent) {
15
- return;
16
- }
17
- responded = true;
18
- res.status(status).json(payload);
19
- };
20
- }
21
-
22
- router.get('/config', async (req, res) => {
23
- try {
24
- // Cloud mode: read config from user's machine
25
- if (req.isCloud) {
26
- if (!req.requireRelay()) return;
27
- try {
28
- const result = await req.sendRelay('file-read', { filePath: '~/.codex/config.toml' }, 15000);
29
- const config = TOML.parse(result.content);
30
- return res.json({ success: true, config: { model: config.model || null, mcpServers: config.mcp_servers || {}, approvalMode: config.approval_mode || 'suggest' } });
31
- } catch {
32
- return res.json({ success: true, config: { model: null, mcpServers: {}, approvalMode: 'suggest' } });
33
- }
34
- }
35
-
36
- // Local mode
37
- const configPath = path.join(os.homedir(), '.codex', 'config.toml');
38
- const content = await fs.readFile(configPath, 'utf8');
39
- const config = TOML.parse(content);
40
- res.json({ success: true, config: { model: config.model || null, mcpServers: config.mcp_servers || {}, approvalMode: config.approval_mode || 'suggest' } });
41
- } catch (error) {
42
- if (error.code === 'ENOENT') {
43
- res.json({ success: true, config: { model: null, mcpServers: {}, approvalMode: 'suggest' } });
44
- } else {
45
- res.status(500).json({ success: false, error: 'Internal server error' });
46
- }
47
- }
48
- });
49
-
50
- router.get('/sessions', async (req, res) => {
51
- try {
52
- const { projectPath } = req.query;
53
-
54
- if (!projectPath) {
55
- return res.status(400).json({ success: false, error: 'projectPath query parameter required' });
56
- }
57
-
58
- const sessions = await getCodexSessions(projectPath);
59
- res.json({ success: true, sessions });
60
- } catch (error) {
61
- // sessions fetch error
62
- res.status(500).json({ success: false, error: 'Internal server error' });
63
- }
64
- });
65
-
66
- router.get('/sessions/:sessionId/messages', async (req, res) => {
67
- try {
68
- const { sessionId } = req.params;
69
- const { limit, offset } = req.query;
70
-
71
- const result = await getCodexSessionMessages(
72
- sessionId,
73
- limit ? parseInt(limit, 10) : null,
74
- offset ? parseInt(offset, 10) : 0
75
- );
76
-
77
- res.json({ success: true, ...result });
78
- } catch (error) {
79
- // session messages fetch error
80
- res.status(500).json({ success: false, error: 'Internal server error' });
81
- }
82
- });
83
-
84
- router.delete('/sessions/:sessionId', async (req, res) => {
85
- try {
86
- const { sessionId } = req.params;
87
- await deleteCodexSession(sessionId);
88
- res.json({ success: true });
89
- } catch (error) {
90
- // session delete error
91
- res.status(500).json({ success: false, error: 'Internal server error' });
92
- }
93
- });
94
-
95
- // MCP Server Management Routes
96
-
97
- router.get('/mcp/cli/list', async (req, res) => {
98
- try {
99
- // Cloud mode: run codex CLI on user's machine via relay
100
- if (req.isCloud) {
101
- if (!req.requireRelay()) return;
102
- try {
103
- const result = await req.sendRelay('shell-command', { command: 'codex mcp list' }, 30000);
104
- const output = result.stdout || result.output || '';
105
- return res.json({ success: true, output, servers: parseCodexListOutput(output) });
106
- } catch (err) {
107
- return res.status(500).json({ error: 'Codex CLI command failed via relay', details: err.message });
108
- }
109
- }
110
-
111
- // Local mode
112
- const respond = createCliResponder(res);
113
- const proc = spawn('codex', ['mcp', 'list'], { stdio: ['pipe', 'pipe', 'pipe'] });
114
-
115
- let stdout = '';
116
- let stderr = '';
117
-
118
- proc.stdout?.on('data', (data) => { stdout += data.toString(); });
119
- proc.stderr?.on('data', (data) => { stderr += data.toString(); });
120
-
121
- proc.on('close', (code) => {
122
- if (code === 0) {
123
- respond(200, { success: true, output: stdout, servers: parseCodexListOutput(stdout) });
124
- } else {
125
- respond(500, { error: 'Codex CLI command failed', details: stderr || `Exited with code ${code}` });
126
- }
127
- });
128
-
129
- proc.on('error', (error) => {
130
- const isMissing = error?.code === 'ENOENT';
131
- respond(isMissing ? 503 : 500, {
132
- error: isMissing ? 'Codex CLI not installed' : 'Failed to run Codex CLI',
133
- details: 'An error occurred',
134
- code: error.code
135
- });
136
- });
137
- } catch (error) {
138
- res.status(500).json({ error: 'Failed to list MCP servers', details: 'An error occurred' });
139
- }
140
- });
141
-
142
- router.post('/mcp/cli/add', async (req, res) => {
143
- try {
144
- const { name, command, args = [], env = {} } = req.body;
145
-
146
- if (!name || !command) {
147
- return res.status(400).json({ error: 'name and command are required' });
148
- }
149
-
150
- let cliArgs = ['mcp', 'add', name];
151
- Object.entries(env).forEach(([key, value]) => { cliArgs.push('-e', `${key}=${value}`); });
152
- cliArgs.push('--', command);
153
- if (args && args.length > 0) cliArgs.push(...args);
154
-
155
- // Cloud mode
156
- if (req.isCloud) {
157
- if (!req.requireRelay()) return;
158
- try {
159
- const fullCmd = `codex ${cliArgs.map(a => a.includes(' ') ? `"${a}"` : a).join(' ')}`;
160
- const result = await req.sendRelay('shell-command', { command: fullCmd }, 30000);
161
- return res.json({ success: true, output: result.stdout || result.output || '', message: `MCP server "${name}" added successfully` });
162
- } catch (err) {
163
- return res.status(500).json({ error: 'Codex CLI command failed via relay', details: err.message });
164
- }
165
- }
166
-
167
- // Local mode
168
- const respond = createCliResponder(res);
169
- const proc = spawn('codex', cliArgs, { stdio: ['pipe', 'pipe', 'pipe'] });
170
-
171
- let stdout = '';
172
- let stderr = '';
173
-
174
- proc.stdout?.on('data', (data) => { stdout += data.toString(); });
175
- proc.stderr?.on('data', (data) => { stderr += data.toString(); });
176
-
177
- proc.on('close', (code) => {
178
- if (code === 0) {
179
- respond(200, { success: true, output: stdout, message: `MCP server "${name}" added successfully` });
180
- } else {
181
- respond(400, { error: 'Codex CLI command failed', details: stderr || `Exited with code ${code}` });
182
- }
183
- });
184
-
185
- proc.on('error', (error) => {
186
- const isMissing = error?.code === 'ENOENT';
187
- respond(isMissing ? 503 : 500, {
188
- error: isMissing ? 'Codex CLI not installed' : 'Failed to run Codex CLI',
189
- details: 'An error occurred',
190
- code: error.code
191
- });
192
- });
193
- } catch (error) {
194
- res.status(500).json({ error: 'Failed to add MCP server', details: 'An error occurred' });
195
- }
196
- });
197
-
198
- router.delete('/mcp/cli/remove/:name', async (req, res) => {
199
- try {
200
- const { name } = req.params;
201
-
202
- if (req.isCloud) {
203
- if (!req.requireRelay()) return;
204
- try {
205
- const result = await req.sendRelay('shell-command', { command: `codex mcp remove ${name}` }, 30000);
206
- return res.json({ success: true, output: result.stdout || result.output || '', message: `MCP server "${name}" removed successfully` });
207
- } catch (err) {
208
- return res.status(500).json({ error: 'Codex CLI command failed via relay', details: err.message });
209
- }
210
- }
211
-
212
- const respond = createCliResponder(res);
213
- const proc = spawn('codex', ['mcp', 'remove', name], { stdio: ['pipe', 'pipe', 'pipe'] });
214
-
215
- let stdout = '';
216
- let stderr = '';
217
-
218
- proc.stdout?.on('data', (data) => { stdout += data.toString(); });
219
- proc.stderr?.on('data', (data) => { stderr += data.toString(); });
220
-
221
- proc.on('close', (code) => {
222
- if (code === 0) {
223
- respond(200, { success: true, output: stdout, message: `MCP server "${name}" removed successfully` });
224
- } else {
225
- respond(400, { error: 'Codex CLI command failed', details: stderr || `Exited with code ${code}` });
226
- }
227
- });
228
-
229
- proc.on('error', (error) => {
230
- const isMissing = error?.code === 'ENOENT';
231
- respond(isMissing ? 503 : 500, {
232
- error: isMissing ? 'Codex CLI not installed' : 'Failed to run Codex CLI',
233
- details: 'An error occurred',
234
- code: error.code
235
- });
236
- });
237
- } catch (error) {
238
- res.status(500).json({ error: 'Failed to remove MCP server', details: 'An error occurred' });
239
- }
240
- });
241
-
242
- router.get('/mcp/cli/get/:name', async (req, res) => {
243
- try {
244
- const { name } = req.params;
245
-
246
- if (req.isCloud) {
247
- if (!req.requireRelay()) return;
248
- try {
249
- const result = await req.sendRelay('shell-command', { command: `codex mcp get ${name}` }, 30000);
250
- const output = result.stdout || result.output || '';
251
- return res.json({ success: true, output, server: parseCodexGetOutput(output) });
252
- } catch (err) {
253
- return res.status(404).json({ error: 'Codex CLI command failed via relay', details: err.message });
254
- }
255
- }
256
-
257
- const respond = createCliResponder(res);
258
- const proc = spawn('codex', ['mcp', 'get', name], { stdio: ['pipe', 'pipe', 'pipe'] });
259
-
260
- let stdout = '';
261
- let stderr = '';
262
-
263
- proc.stdout?.on('data', (data) => { stdout += data.toString(); });
264
- proc.stderr?.on('data', (data) => { stderr += data.toString(); });
265
-
266
- proc.on('close', (code) => {
267
- if (code === 0) {
268
- respond(200, { success: true, output: stdout, server: parseCodexGetOutput(stdout) });
269
- } else {
270
- respond(404, { error: 'Codex CLI command failed', details: stderr || `Exited with code ${code}` });
271
- }
272
- });
273
-
274
- proc.on('error', (error) => {
275
- const isMissing = error?.code === 'ENOENT';
276
- respond(isMissing ? 503 : 500, {
277
- error: isMissing ? 'Codex CLI not installed' : 'Failed to run Codex CLI',
278
- details: 'An error occurred',
279
- code: error.code
280
- });
281
- });
282
- } catch (error) {
283
- res.status(500).json({ error: 'Failed to get MCP server details', details: 'An error occurred' });
284
- }
285
- });
286
-
287
- router.get('/mcp/config/read', async (req, res) => {
288
- try {
289
- // Cloud mode
290
- if (req.isCloud) {
291
- if (!req.requireRelay()) return;
292
- try {
293
- const result = await req.sendRelay('file-read', { filePath: '~/.codex/config.toml' }, 15000);
294
- const configData = TOML.parse(result.content);
295
- // Parse servers from config (same logic as local below)
296
- const servers = [];
297
- if (configData.mcp_servers && typeof configData.mcp_servers === 'object') {
298
- for (const [name, config] of Object.entries(configData.mcp_servers)) {
299
- servers.push({ name, command: config.command || '', args: config.args || [], env: config.env || {} });
300
- }
301
- }
302
- return res.json({ success: true, configPath: '~/.codex/config.toml', servers });
303
- } catch {
304
- return res.json({ success: true, configPath: '~/.codex/config.toml', servers: [] });
305
- }
306
- }
307
-
308
- // Local mode
309
- const configPath = path.join(os.homedir(), '.codex', 'config.toml');
310
- let configData = null;
311
- try {
312
- const fileContent = await fs.readFile(configPath, 'utf8');
313
- configData = TOML.parse(fileContent);
314
- } catch (error) { /* doesn't exist */ }
315
-
316
- if (!configData) {
317
- return res.json({ success: true, configPath, servers: [] }); }
318
-
319
- const servers = [];
320
-
321
- if (configData.mcp_servers && typeof configData.mcp_servers === 'object') {
322
- for (const [name, config] of Object.entries(configData.mcp_servers)) {
323
- servers.push({
324
- id: name,
325
- name: name,
326
- type: 'stdio',
327
- scope: 'user',
328
- config: {
329
- command: config.command || '',
330
- args: config.args || [],
331
- env: config.env || {}
332
- },
333
- raw: config
334
- });
335
- }
336
- }
337
-
338
- res.json({ success: true, configPath, servers });
339
- } catch (error) {
340
- res.status(500).json({ error: 'Failed to read Codex configuration', details: 'An error occurred' });
341
- }
342
- });
343
-
344
- function parseCodexListOutput(output) {
345
- const servers = [];
346
- const lines = output.split('\n').filter(line => line.trim());
347
-
348
- for (const line of lines) {
349
- if (line.includes(':')) {
350
- const colonIndex = line.indexOf(':');
351
- const name = line.substring(0, colonIndex).trim();
352
-
353
- if (!name) continue;
354
-
355
- const rest = line.substring(colonIndex + 1).trim();
356
- let description = rest;
357
- let status = 'unknown';
358
-
359
- if (rest.includes('✓') || rest.includes('✗')) {
360
- const statusMatch = rest.match(/(.*?)\s*-\s*([✓✗].*)$/);
361
- if (statusMatch) {
362
- description = statusMatch[1].trim();
363
- status = statusMatch[2].includes('✓') ? 'connected' : 'failed';
364
- }
365
- }
366
-
367
- servers.push({ name, type: 'stdio', status, description });
368
- }
369
- }
370
-
371
- return servers;
372
- }
373
-
374
- function parseCodexGetOutput(output) {
375
- try {
376
- const jsonMatch = output.match(/\{[\s\S]*\}/);
377
- if (jsonMatch) {
378
- return JSON.parse(jsonMatch[0]);
379
- }
380
-
381
- const server = { raw_output: output };
382
- const lines = output.split('\n');
383
-
384
- for (const line of lines) {
385
- if (line.includes('Name:')) server.name = line.split(':')[1]?.trim();
386
- else if (line.includes('Type:')) server.type = line.split(':')[1]?.trim();
387
- else if (line.includes('Command:')) server.command = line.split(':')[1]?.trim();
388
- }
389
-
390
- return server;
391
- } catch (error) {
392
- return { raw_output: output, parse_error: error.message };
393
- }
394
- }
395
-
396
- export default router;