upfynai-code 2.5.1 → 2.6.1

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 (293) hide show
  1. package/README.md +123 -88
  2. package/bin/cli.js +63 -0
  3. package/package.json +48 -112
  4. package/src/auth.js +115 -0
  5. package/src/config.js +33 -0
  6. package/src/connect.js +314 -0
  7. package/src/launch.js +54 -0
  8. package/src/mcp.js +57 -0
  9. package/src/server.js +54 -0
  10. package/client/dist/api-docs.html +0 -879
  11. package/client/dist/assets/AppContent-C0CyP3g5.js +0 -513
  12. package/client/dist/assets/CanvasPanel-0u9QR7U-.js +0 -34
  13. package/client/dist/assets/CanvasPanel-WhZulBJw.css +0 -1
  14. package/client/dist/assets/DashboardPanel-Dgqw1yZk.js +0 -1
  15. package/client/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  16. package/client/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  17. package/client/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  18. package/client/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  19. package/client/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  20. package/client/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  21. package/client/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  22. package/client/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  23. package/client/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  24. package/client/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  25. package/client/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  26. package/client/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  27. package/client/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  28. package/client/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  29. package/client/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  30. package/client/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  31. package/client/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  32. package/client/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  33. package/client/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  34. package/client/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  35. package/client/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  36. package/client/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  37. package/client/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  38. package/client/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  39. package/client/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  40. package/client/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  41. package/client/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  42. package/client/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  43. package/client/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  44. package/client/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  45. package/client/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  46. package/client/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  47. package/client/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  48. package/client/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  49. package/client/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  50. package/client/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  51. package/client/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  52. package/client/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  53. package/client/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  54. package/client/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  55. package/client/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  56. package/client/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  57. package/client/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  58. package/client/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  59. package/client/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  60. package/client/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  61. package/client/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  62. package/client/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  63. package/client/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  64. package/client/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  65. package/client/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  66. package/client/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  67. package/client/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  68. package/client/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  69. package/client/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  70. package/client/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  71. package/client/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  72. package/client/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  73. package/client/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  74. package/client/dist/assets/LoginModal-CZDEzqjK.js +0 -19
  75. package/client/dist/assets/MarkdownPreview-CYdvwJaV.js +0 -1
  76. package/client/dist/assets/Onboarding-DR6NZ4Vz.js +0 -1
  77. package/client/dist/assets/SetupForm-D49gtWY4.js +0 -1
  78. package/client/dist/assets/Tableau10-B-NsZVaP.js +0 -1
  79. package/client/dist/assets/WorkflowsPanel-CqlbEJA_.js +0 -1
  80. package/client/dist/assets/_commonjs-dynamic-modules-TDtrdbi3.js +0 -1
  81. package/client/dist/assets/ar-SA-G6X2FPQ2-BWqa1yBH.js +0 -10
  82. package/client/dist/assets/arc-BegSKqEW.js +0 -1
  83. package/client/dist/assets/array-BKyUJesY.js +0 -1
  84. package/client/dist/assets/az-AZ-76LH7QW2-DrVlbZDP.js +0 -1
  85. package/client/dist/assets/bg-BG-XCXSNQG7-DdunjBgT.js +0 -5
  86. package/client/dist/assets/blockDiagram-38ab4fdb-BKMbwGHu.js +0 -118
  87. package/client/dist/assets/bn-BD-2XOGV67Q-_7DtmvwO.js +0 -5
  88. package/client/dist/assets/c4Diagram-3d4e48cf-hJuiHhSn.js +0 -10
  89. package/client/dist/assets/ca-ES-6MX7JW3Y-BFIrmojG.js +0 -8
  90. package/client/dist/assets/channel-Bur-rRTp.js +0 -1
  91. package/client/dist/assets/classDiagram-70f12bd4-BjiAf9cM.js +0 -2
  92. package/client/dist/assets/classDiagram-v2-f2320105-pwBewejc.js +0 -2
  93. package/client/dist/assets/clone-BtqXeoBJ.js +0 -1
  94. package/client/dist/assets/createText-2e5e7dd3-Dq_acOWe.js +0 -5
  95. package/client/dist/assets/cs-CZ-2BRQDIVT-B-x4F6TJ.js +0 -11
  96. package/client/dist/assets/da-DK-5WZEPLOC-Btlc8Dgn.js +0 -5
  97. package/client/dist/assets/de-DE-XR44H4JA-BVu3ZIoD.js +0 -8
  98. package/client/dist/assets/directory-open-01563666-DWU9wJ6I.js +0 -1
  99. package/client/dist/assets/directory-open-4ed118d0-CunoC1EB.js +0 -1
  100. package/client/dist/assets/edges-e0da2a9e-DH0wVTXR.js +0 -4
  101. package/client/dist/assets/el-GR-BZB4AONW-h2ll8_ZC.js +0 -10
  102. package/client/dist/assets/erDiagram-9861fffd-BYezLIR7.js +0 -51
  103. package/client/dist/assets/es-ES-U4NZUMDT-Cveiulwt.js +0 -9
  104. package/client/dist/assets/eu-ES-A7QVB2H4-DQluL2PY.js +0 -11
  105. package/client/dist/assets/fa-IR-HGAKTJCU-BJtcMBSv.js +0 -8
  106. package/client/dist/assets/fi-FI-Z5N7JZ37-D8NfbVXV.js +0 -6
  107. package/client/dist/assets/file-open-002ab408-DIuFHtCF.js +0 -1
  108. package/client/dist/assets/file-open-7c801643-684qeFg4.js +0 -1
  109. package/client/dist/assets/file-save-3189631c-C1wFhQhH.js +0 -1
  110. package/client/dist/assets/file-save-745eba88-Bb9F9Kg7.js +0 -1
  111. package/client/dist/assets/flowDb-956e92f1-scnUykhM.js +0 -10
  112. package/client/dist/assets/flowDiagram-66a62f08-jVyWsfyU.js +0 -4
  113. package/client/dist/assets/flowDiagram-v2-96b9c2cf-N6xgi25h.js +0 -1
  114. package/client/dist/assets/flowchart-elk-definition-4a651766-gKGX3HqR.js +0 -139
  115. package/client/dist/assets/fr-FR-RHASNOE6-vdj42kC6.js +0 -9
  116. package/client/dist/assets/ganttDiagram-c361ad54-C2CiWFUP.js +0 -257
  117. package/client/dist/assets/gitGraphDiagram-72cf32ee-C59Yz2LK.js +0 -70
  118. package/client/dist/assets/gl-ES-HMX3MZ6V-DQo0TzoP.js +0 -10
  119. package/client/dist/assets/graph-Dx_H43Kv.js +0 -1
  120. package/client/dist/assets/he-IL-6SHJWFNN-DKXK5e33.js +0 -10
  121. package/client/dist/assets/hi-IN-IWLTKZ5I-C2Qgqc0R.js +0 -4
  122. package/client/dist/assets/hu-HU-A5ZG7DT2-Ss-6vX0m.js +0 -7
  123. package/client/dist/assets/id-ID-SAP4L64H-D7Wsg1S2.js +0 -10
  124. package/client/dist/assets/image-blob-reduce.esm-D6s-rqMO.js +0 -7
  125. package/client/dist/assets/index-3862675e-u8Nv7hHC.js +0 -1
  126. package/client/dist/assets/index-BVowJdZF.js +0 -97
  127. package/client/dist/assets/index-ce18TYkg.js +0 -27
  128. package/client/dist/assets/index-kQoJx-bc.css +0 -1
  129. package/client/dist/assets/infoDiagram-f8f76790-LmoJYsxo.js +0 -7
  130. package/client/dist/assets/init-Gi6I4Gst.js +0 -1
  131. package/client/dist/assets/it-IT-JPQ66NNP-CAPTVl7M.js +0 -11
  132. package/client/dist/assets/ja-JP-DBVTYXUO-eNVPawR2.js +0 -8
  133. package/client/dist/assets/journeyDiagram-49397b02-BaJqehpR.js +0 -139
  134. package/client/dist/assets/kaa-6HZHGXH3-tpuNkKhS.js +0 -1
  135. package/client/dist/assets/kab-KAB-ZGHBKWFO-Dp83kx4x.js +0 -8
  136. package/client/dist/assets/kk-KZ-P5N5QNE5-B9IlC6YN.js +0 -1
  137. package/client/dist/assets/km-KH-HSX4SM5Z-B_KMYaMj.js +0 -11
  138. package/client/dist/assets/ko-KR-MTYHY66A-yebnUNdb.js +0 -9
  139. package/client/dist/assets/ku-TR-6OUDTVRD-BR6fh6-5.js +0 -9
  140. package/client/dist/assets/layout-DLl5Jwcl.js +0 -1
  141. package/client/dist/assets/line-FpB7omSK.js +0 -1
  142. package/client/dist/assets/linear-CkXqUFJ8.js +0 -1
  143. package/client/dist/assets/lt-LT-XHIRWOB4-SutZSWtR.js +0 -3
  144. package/client/dist/assets/lv-LV-5QDEKY6T-DuAxdcZL.js +0 -7
  145. package/client/dist/assets/mindmap-definition-fc14e90a-DyxXOExh.js +0 -425
  146. package/client/dist/assets/mr-IN-CRQNXWMA-DqDUWM_8.js +0 -13
  147. package/client/dist/assets/my-MM-5M5IBNSE-C40kMFMR.js +0 -1
  148. package/client/dist/assets/nb-NO-T6EIAALU-DVij32Ju.js +0 -10
  149. package/client/dist/assets/nl-NL-IS3SIHDZ-rT84mDYq.js +0 -8
  150. package/client/dist/assets/nn-NO-6E72VCQL-BBZXBW8V.js +0 -8
  151. package/client/dist/assets/oc-FR-POXYY2M6-DzjOugOf.js +0 -8
  152. package/client/dist/assets/ordinal-Cboi1Yqb.js +0 -1
  153. package/client/dist/assets/pa-IN-N4M65BXN-DD1iU8_F.js +0 -4
  154. package/client/dist/assets/path-CbwjOpE9.js +0 -1
  155. package/client/dist/assets/pdf-CE_K4jFx.js +0 -12
  156. package/client/dist/assets/pdf.worker-BA9kU3Pw.mjs +0 -61080
  157. package/client/dist/assets/percentages-BXMCSKIN-WVlHS4wx.js +0 -207
  158. package/client/dist/assets/pica-CQIY57Tf.js +0 -7
  159. package/client/dist/assets/pieDiagram-8a3498a8-Dd_85qBH.js +0 -35
  160. package/client/dist/assets/pl-PL-T2D74RX3-ukVXa48G.js +0 -9
  161. package/client/dist/assets/pt-BR-5N22H2LF-BibawarT.js +0 -9
  162. package/client/dist/assets/pt-PT-UZXXM6DQ-So3i9l9w.js +0 -9
  163. package/client/dist/assets/quadrantDiagram-120e2f19-C4dFVDEx.js +0 -7
  164. package/client/dist/assets/requirementDiagram-deff3bca-DrTO7yFl.js +0 -52
  165. package/client/dist/assets/ro-RO-JPDTUUEW-DY0Xq_Hd.js +0 -11
  166. package/client/dist/assets/roundRect-0PYZxl1G.js +0 -1
  167. package/client/dist/assets/ru-RU-B4JR7IUQ-B7u_Zvkd.js +0 -9
  168. package/client/dist/assets/sankeyDiagram-04a897e0-D24gfzuS.js +0 -8
  169. package/client/dist/assets/sequenceDiagram-704730f1-Dgji2XLQ.js +0 -122
  170. package/client/dist/assets/si-LK-N5RQ5JYF-OejsLzQ_.js +0 -1
  171. package/client/dist/assets/sk-SK-C5VTKIMK-_vy2Bt-M.js +0 -6
  172. package/client/dist/assets/sl-SI-NN7IZMDC-DKOl_u2M.js +0 -6
  173. package/client/dist/assets/stateDiagram-587899a1-CJ8eBaiU.js +0 -1
  174. package/client/dist/assets/stateDiagram-v2-d93cdb3a-C5K3l-Nt.js +0 -1
  175. package/client/dist/assets/styles-6aaf32cf-DAKE0jbx.js +0 -207
  176. package/client/dist/assets/styles-9a916d00-LFAJCgEy.js +0 -160
  177. package/client/dist/assets/styles-c10674c1-CllKO8NG.js +0 -116
  178. package/client/dist/assets/subset-shared.chunk-Uy-J87FQ.js +0 -84
  179. package/client/dist/assets/subset-worker.chunk-dvgDvqt9.js +0 -1
  180. package/client/dist/assets/sv-SE-XGPEYMSR-CDCB2ZV5.js +0 -10
  181. package/client/dist/assets/svgDrawCommon-08f97a94-CObOzbFQ.js +0 -1
  182. package/client/dist/assets/ta-IN-2NMHFXQM-DHUNdO69.js +0 -9
  183. package/client/dist/assets/th-TH-HPSO5L25-zI2hnBq3.js +0 -2
  184. package/client/dist/assets/timeline-definition-85554ec2-C2XHRmxK.js +0 -61
  185. package/client/dist/assets/tr-TR-DEFEU3FU-l-6Hu4-D.js +0 -7
  186. package/client/dist/assets/uk-UA-QMV73CPH-CqSOwrl7.js +0 -6
  187. package/client/dist/assets/vendor-codemirror-D_s0aGBu.js +0 -35
  188. package/client/dist/assets/vendor-i18n-DCFGyhQR.js +0 -1
  189. package/client/dist/assets/vendor-icons-Lb69KSFJ.js +0 -646
  190. package/client/dist/assets/vendor-markdown-BXEi_H3G.js +0 -298
  191. package/client/dist/assets/vendor-react-9mUTKBHH.js +0 -67
  192. package/client/dist/assets/vendor-syntax-DnmwQQJF.js +0 -16
  193. package/client/dist/assets/vendor-xterm-CZq1hqo1.js +0 -66
  194. package/client/dist/assets/vendor-xterm-qxJ8_QYu.css +0 -32
  195. package/client/dist/assets/vi-VN-M7AON7JQ-CUL8-mBZ.js +0 -5
  196. package/client/dist/assets/xychartDiagram-e933f94c-1fmf6slj.js +0 -7
  197. package/client/dist/assets/zh-CN-LNUGB5OW-CB5y5VVU.js +0 -10
  198. package/client/dist/assets/zh-HK-E62DVLB3-BHcrrEeJ.js +0 -1
  199. package/client/dist/assets/zh-TW-RAJ6MFWO-DoDUdkaJ.js +0 -9
  200. package/client/dist/clear-cache.html +0 -85
  201. package/client/dist/convert-icons.md +0 -53
  202. package/client/dist/favicon.png +0 -0
  203. package/client/dist/favicon.svg +0 -9
  204. package/client/dist/generate-icons.js +0 -49
  205. package/client/dist/icons/claude-ai-icon.svg +0 -1
  206. package/client/dist/icons/codex-white.svg +0 -3
  207. package/client/dist/icons/codex.svg +0 -3
  208. package/client/dist/icons/cursor-white.svg +0 -12
  209. package/client/dist/icons/cursor.svg +0 -1
  210. package/client/dist/icons/icon-128x128.png +0 -0
  211. package/client/dist/icons/icon-128x128.svg +0 -12
  212. package/client/dist/icons/icon-144x144.png +0 -0
  213. package/client/dist/icons/icon-144x144.svg +0 -12
  214. package/client/dist/icons/icon-152x152.png +0 -0
  215. package/client/dist/icons/icon-152x152.svg +0 -12
  216. package/client/dist/icons/icon-192x192.png +0 -0
  217. package/client/dist/icons/icon-192x192.svg +0 -12
  218. package/client/dist/icons/icon-384x384.png +0 -0
  219. package/client/dist/icons/icon-384x384.svg +0 -12
  220. package/client/dist/icons/icon-512x512.png +0 -0
  221. package/client/dist/icons/icon-512x512.svg +0 -12
  222. package/client/dist/icons/icon-72x72.png +0 -0
  223. package/client/dist/icons/icon-72x72.svg +0 -12
  224. package/client/dist/icons/icon-96x96.png +0 -0
  225. package/client/dist/icons/icon-96x96.svg +0 -12
  226. package/client/dist/icons/icon-template.svg +0 -12
  227. package/client/dist/index.html +0 -128
  228. package/client/dist/logo-128.png +0 -0
  229. package/client/dist/logo-256.png +0 -0
  230. package/client/dist/logo-32.png +0 -0
  231. package/client/dist/logo-512.png +0 -0
  232. package/client/dist/logo-64.png +0 -0
  233. package/client/dist/logo.svg +0 -17
  234. package/client/dist/manifest.json +0 -61
  235. package/client/dist/mcp-docs.html +0 -119
  236. package/client/dist/screenshots/cli-selection.png +0 -0
  237. package/client/dist/screenshots/desktop-main.png +0 -0
  238. package/client/dist/screenshots/mobile-chat.png +0 -0
  239. package/client/dist/screenshots/tools-modal.png +0 -0
  240. package/client/dist/sw.js +0 -19
  241. package/commands/upfynai-connect.md +0 -59
  242. package/commands/upfynai-disconnect.md +0 -31
  243. package/commands/upfynai-doctor.md +0 -99
  244. package/commands/upfynai-export.md +0 -49
  245. package/commands/upfynai-local.md +0 -82
  246. package/commands/upfynai-status.md +0 -75
  247. package/commands/upfynai-stop.md +0 -49
  248. package/commands/upfynai-uninstall.md +0 -58
  249. package/commands/upfynai.md +0 -69
  250. package/scripts/build-client.js +0 -17
  251. package/scripts/fix-node-pty.js +0 -67
  252. package/scripts/install-commands.js +0 -78
  253. package/server/claude-sdk.js +0 -714
  254. package/server/cli-ui.js +0 -785
  255. package/server/cli.js +0 -596
  256. package/server/constants/config.js +0 -31
  257. package/server/cursor-cli.js +0 -270
  258. package/server/database/auth.db +0 -0
  259. package/server/database/db.js +0 -808
  260. package/server/database/init.sql +0 -70
  261. package/server/index.js +0 -2621
  262. package/server/load-env.js +0 -26
  263. package/server/mcp-server.js +0 -621
  264. package/server/middleware/auth.js +0 -173
  265. package/server/openai-codex.js +0 -403
  266. package/server/openrouter.js +0 -137
  267. package/server/projects.js +0 -1742
  268. package/server/relay-client.js +0 -619
  269. package/server/routes/agent.js +0 -1266
  270. package/server/routes/auth.js +0 -263
  271. package/server/routes/cli-auth.js +0 -263
  272. package/server/routes/codex.js +0 -344
  273. package/server/routes/commands.js +0 -601
  274. package/server/routes/cursor.js +0 -808
  275. package/server/routes/dashboard.js +0 -52
  276. package/server/routes/git.js +0 -1165
  277. package/server/routes/mcp-utils.js +0 -48
  278. package/server/routes/mcp.js +0 -552
  279. package/server/routes/payments.js +0 -172
  280. package/server/routes/projects.js +0 -552
  281. package/server/routes/settings.js +0 -269
  282. package/server/routes/taskmaster.js +0 -1964
  283. package/server/routes/user.js +0 -106
  284. package/server/routes/voice.js +0 -198
  285. package/server/routes/webhooks.js +0 -166
  286. package/server/routes/workflows.js +0 -118
  287. package/server/services/whisperService.js +0 -84
  288. package/server/services/workflowScheduler.js +0 -186
  289. package/server/utils/commandParser.js +0 -303
  290. package/server/utils/gitConfig.js +0 -24
  291. package/server/utils/mcp-detector.js +0 -198
  292. package/server/utils/taskmaster-websocket.js +0 -129
  293. package/shared/modelConstants.js +0 -96
@@ -1,601 +0,0 @@
1
- import express from 'express';
2
- import { promises as fs } from 'fs';
3
- import path from 'path';
4
- import { fileURLToPath } from 'url';
5
- import os from 'os';
6
- import matter from 'gray-matter';
7
- import { CLAUDE_MODELS, CURSOR_MODELS, CODEX_MODELS } from '../../shared/modelConstants.js';
8
-
9
- const __filename = fileURLToPath(import.meta.url);
10
- const __dirname = path.dirname(__filename);
11
-
12
- const router = express.Router();
13
-
14
- /**
15
- * Recursively scan directory for command files (.md)
16
- * @param {string} dir - Directory to scan
17
- * @param {string} baseDir - Base directory for relative paths
18
- * @param {string} namespace - Namespace for commands (e.g., 'project', 'user')
19
- * @returns {Promise<Array>} Array of command objects
20
- */
21
- async function scanCommandsDirectory(dir, baseDir, namespace) {
22
- const commands = [];
23
-
24
- try {
25
- // Check if directory exists
26
- await fs.access(dir);
27
-
28
- const entries = await fs.readdir(dir, { withFileTypes: true });
29
-
30
- for (const entry of entries) {
31
- const fullPath = path.join(dir, entry.name);
32
-
33
- if (entry.isDirectory()) {
34
- // Recursively scan subdirectories
35
- const subCommands = await scanCommandsDirectory(fullPath, baseDir, namespace);
36
- commands.push(...subCommands);
37
- } else if (entry.isFile() && entry.name.endsWith('.md')) {
38
- // Parse markdown file for metadata
39
- try {
40
- const content = await fs.readFile(fullPath, 'utf8');
41
- const { data: frontmatter, content: commandContent } = matter(content);
42
-
43
- // Calculate relative path from baseDir for command name
44
- const relativePath = path.relative(baseDir, fullPath);
45
- // Remove .md extension and convert to command name
46
- const commandName = '/' + relativePath.replace(/\.md$/, '').replace(/\\/g, '/');
47
-
48
- // Extract description from frontmatter or first line of content
49
- let description = frontmatter.description || '';
50
- if (!description) {
51
- const firstLine = commandContent.trim().split('\n')[0];
52
- description = firstLine.replace(/^#+\s*/, '').trim();
53
- }
54
-
55
- commands.push({
56
- name: commandName,
57
- path: fullPath,
58
- relativePath,
59
- description,
60
- namespace,
61
- metadata: frontmatter
62
- });
63
- } catch (err) {
64
- console.error(`Error parsing command file ${fullPath}:`, err.message);
65
- }
66
- }
67
- }
68
- } catch (err) {
69
- // Directory doesn't exist or can't be accessed - this is okay
70
- if (err.code !== 'ENOENT' && err.code !== 'EACCES') {
71
- console.error(`Error scanning directory ${dir}:`, err.message);
72
- }
73
- }
74
-
75
- return commands;
76
- }
77
-
78
- /**
79
- * Built-in commands that are always available
80
- */
81
- const builtInCommands = [
82
- {
83
- name: '/help',
84
- description: 'Show help documentation for Upfyn-Code',
85
- namespace: 'builtin',
86
- metadata: { type: 'builtin' }
87
- },
88
- {
89
- name: '/clear',
90
- description: 'Clear the conversation history',
91
- namespace: 'builtin',
92
- metadata: { type: 'builtin' }
93
- },
94
- {
95
- name: '/model',
96
- description: 'Switch or view the current AI model',
97
- namespace: 'builtin',
98
- metadata: { type: 'builtin' }
99
- },
100
- {
101
- name: '/cost',
102
- description: 'Display token usage and cost information',
103
- namespace: 'builtin',
104
- metadata: { type: 'builtin' }
105
- },
106
- {
107
- name: '/memory',
108
- description: 'Open CLAUDE.md memory file for editing',
109
- namespace: 'builtin',
110
- metadata: { type: 'builtin' }
111
- },
112
- {
113
- name: '/config',
114
- description: 'Open settings and configuration',
115
- namespace: 'builtin',
116
- metadata: { type: 'builtin' }
117
- },
118
- {
119
- name: '/status',
120
- description: 'Show system status and version information',
121
- namespace: 'builtin',
122
- metadata: { type: 'builtin' }
123
- },
124
- {
125
- name: '/rewind',
126
- description: 'Rewind the conversation to a previous state',
127
- namespace: 'builtin',
128
- metadata: { type: 'builtin' }
129
- }
130
- ];
131
-
132
- /**
133
- * Built-in command handlers
134
- * Each handler returns { type: 'builtin', action: string, data: any }
135
- */
136
- const builtInHandlers = {
137
- '/help': async (args, context) => {
138
- const helpText = `# Upfyn-Code Commands
139
-
140
- ## Built-in Commands
141
-
142
- ${builtInCommands.map(cmd => `### ${cmd.name}
143
- ${cmd.description}
144
- `).join('\n')}
145
-
146
- ## Custom Commands
147
-
148
- Custom commands can be created in:
149
- - Project: \`.claude/commands/\` (project-specific)
150
- - User: \`~/.claude/commands/\` (available in all projects)
151
-
152
- ### Command Syntax
153
-
154
- - **Arguments**: Use \`$ARGUMENTS\` for all args or \`$1\`, \`$2\`, etc. for positional
155
- - **File Includes**: Use \`@filename\` to include file contents
156
- - **Bash Commands**: Use \`!command\` to execute bash commands
157
-
158
- ### Examples
159
-
160
- \`\`\`markdown
161
- /mycommand arg1 arg2
162
- \`\`\`
163
- `;
164
-
165
- return {
166
- type: 'builtin',
167
- action: 'help',
168
- data: {
169
- content: helpText,
170
- format: 'markdown'
171
- }
172
- };
173
- },
174
-
175
- '/clear': async (args, context) => {
176
- return {
177
- type: 'builtin',
178
- action: 'clear',
179
- data: {
180
- message: 'Conversation history cleared'
181
- }
182
- };
183
- },
184
-
185
- '/model': async (args, context) => {
186
- // Read available models from centralized constants
187
- const availableModels = {
188
- claude: CLAUDE_MODELS.OPTIONS.map(o => o.value),
189
- cursor: CURSOR_MODELS.OPTIONS.map(o => o.value),
190
- codex: CODEX_MODELS.OPTIONS.map(o => o.value)
191
- };
192
-
193
- const currentProvider = context?.provider || 'claude';
194
- const currentModel = context?.model || CLAUDE_MODELS.DEFAULT;
195
-
196
- return {
197
- type: 'builtin',
198
- action: 'model',
199
- data: {
200
- current: {
201
- provider: currentProvider,
202
- model: currentModel
203
- },
204
- available: availableModels,
205
- message: args.length > 0
206
- ? `Switching to model: ${args[0]}`
207
- : `Current model: ${currentModel}`
208
- }
209
- };
210
- },
211
-
212
- '/cost': async (args, context) => {
213
- const tokenUsage = context?.tokenUsage || {};
214
- const provider = context?.provider || 'claude';
215
- const model =
216
- context?.model ||
217
- (provider === 'cursor'
218
- ? CURSOR_MODELS.DEFAULT
219
- : provider === 'codex'
220
- ? CODEX_MODELS.DEFAULT
221
- : CLAUDE_MODELS.DEFAULT);
222
-
223
- const used = Number(tokenUsage.used ?? tokenUsage.totalUsed ?? tokenUsage.total_tokens ?? 0) || 0;
224
- const total =
225
- Number(
226
- tokenUsage.total ??
227
- tokenUsage.contextWindow ??
228
- parseInt(process.env.CONTEXT_WINDOW || '160000', 10),
229
- ) || 160000;
230
- const percentage = total > 0 ? Number(((used / total) * 100).toFixed(1)) : 0;
231
-
232
- const inputTokensRaw =
233
- Number(
234
- tokenUsage.inputTokens ??
235
- tokenUsage.input ??
236
- tokenUsage.cumulativeInputTokens ??
237
- tokenUsage.promptTokens ??
238
- 0,
239
- ) || 0;
240
- const outputTokens =
241
- Number(
242
- tokenUsage.outputTokens ??
243
- tokenUsage.output ??
244
- tokenUsage.cumulativeOutputTokens ??
245
- tokenUsage.completionTokens ??
246
- 0,
247
- ) || 0;
248
- const cacheTokens =
249
- Number(
250
- tokenUsage.cacheReadTokens ??
251
- tokenUsage.cacheCreationTokens ??
252
- tokenUsage.cacheTokens ??
253
- tokenUsage.cachedTokens ??
254
- 0,
255
- ) || 0;
256
-
257
- // If we only have total used tokens, treat them as input for display/estimation.
258
- const inputTokens =
259
- inputTokensRaw > 0 || outputTokens > 0 || cacheTokens > 0 ? inputTokensRaw + cacheTokens : used;
260
-
261
- // Rough default rates by provider (USD / 1M tokens).
262
- const pricingByProvider = {
263
- claude: { input: 3, output: 15 },
264
- cursor: { input: 3, output: 15 },
265
- codex: { input: 1.5, output: 6 },
266
- };
267
- const rates = pricingByProvider[provider] || pricingByProvider.claude;
268
-
269
- const inputCost = (inputTokens / 1_000_000) * rates.input;
270
- const outputCost = (outputTokens / 1_000_000) * rates.output;
271
- const totalCost = inputCost + outputCost;
272
-
273
- return {
274
- type: 'builtin',
275
- action: 'cost',
276
- data: {
277
- tokenUsage: {
278
- used,
279
- total,
280
- percentage,
281
- },
282
- cost: {
283
- input: inputCost.toFixed(4),
284
- output: outputCost.toFixed(4),
285
- total: totalCost.toFixed(4),
286
- },
287
- model,
288
- },
289
- };
290
- },
291
-
292
- '/status': async (args, context) => {
293
- // Read version from package.json
294
- const packageJsonPath = path.join(path.dirname(__dirname), '..', 'package.json');
295
- let version = 'unknown';
296
- let packageName = 'upfynai-code';
297
-
298
- try {
299
- const packageJson = JSON.parse(await fs.readFile(packageJsonPath, 'utf8'));
300
- version = packageJson.version;
301
- packageName = packageJson.name;
302
- } catch (err) {
303
- console.error('Error reading package.json:', err);
304
- }
305
-
306
- const uptime = process.uptime();
307
- const uptimeMinutes = Math.floor(uptime / 60);
308
- const uptimeHours = Math.floor(uptimeMinutes / 60);
309
- const uptimeFormatted = uptimeHours > 0
310
- ? `${uptimeHours}h ${uptimeMinutes % 60}m`
311
- : `${uptimeMinutes}m`;
312
-
313
- return {
314
- type: 'builtin',
315
- action: 'status',
316
- data: {
317
- version,
318
- packageName,
319
- uptime: uptimeFormatted,
320
- uptimeSeconds: Math.floor(uptime),
321
- model: context?.model || 'claude-sonnet-4.5',
322
- provider: context?.provider || 'claude',
323
- nodeVersion: process.version,
324
- platform: process.platform
325
- }
326
- };
327
- },
328
-
329
- '/memory': async (args, context) => {
330
- const projectPath = context?.projectPath;
331
-
332
- if (!projectPath) {
333
- return {
334
- type: 'builtin',
335
- action: 'memory',
336
- data: {
337
- error: 'No project selected',
338
- message: 'Please select a project to access its CLAUDE.md file'
339
- }
340
- };
341
- }
342
-
343
- const claudeMdPath = path.join(projectPath, 'CLAUDE.md');
344
-
345
- // Check if CLAUDE.md exists
346
- let exists = false;
347
- try {
348
- await fs.access(claudeMdPath);
349
- exists = true;
350
- } catch (err) {
351
- // File doesn't exist
352
- }
353
-
354
- return {
355
- type: 'builtin',
356
- action: 'memory',
357
- data: {
358
- path: claudeMdPath,
359
- exists,
360
- message: exists
361
- ? `Opening CLAUDE.md at ${claudeMdPath}`
362
- : `CLAUDE.md not found at ${claudeMdPath}. Create it to store project-specific instructions.`
363
- }
364
- };
365
- },
366
-
367
- '/config': async (args, context) => {
368
- return {
369
- type: 'builtin',
370
- action: 'config',
371
- data: {
372
- message: 'Opening settings...'
373
- }
374
- };
375
- },
376
-
377
- '/rewind': async (args, context) => {
378
- const steps = args[0] ? parseInt(args[0]) : 1;
379
-
380
- if (isNaN(steps) || steps < 1) {
381
- return {
382
- type: 'builtin',
383
- action: 'rewind',
384
- data: {
385
- error: 'Invalid steps parameter',
386
- message: 'Usage: /rewind [number] - Rewind conversation by N steps (default: 1)'
387
- }
388
- };
389
- }
390
-
391
- return {
392
- type: 'builtin',
393
- action: 'rewind',
394
- data: {
395
- steps,
396
- message: `Rewinding conversation by ${steps} step${steps > 1 ? 's' : ''}...`
397
- }
398
- };
399
- }
400
- };
401
-
402
- /**
403
- * POST /api/commands/list
404
- * List all available commands from project and user directories
405
- */
406
- router.post('/list', async (req, res) => {
407
- try {
408
- const { projectPath } = req.body;
409
- const allCommands = [...builtInCommands];
410
-
411
- // Scan project-level commands (.claude/commands/)
412
- if (projectPath) {
413
- const projectCommandsDir = path.join(projectPath, '.claude', 'commands');
414
- const projectCommands = await scanCommandsDirectory(
415
- projectCommandsDir,
416
- projectCommandsDir,
417
- 'project'
418
- );
419
- allCommands.push(...projectCommands);
420
- }
421
-
422
- // Scan user-level commands (~/.claude/commands/)
423
- const homeDir = os.homedir();
424
- const userCommandsDir = path.join(homeDir, '.claude', 'commands');
425
- const userCommands = await scanCommandsDirectory(
426
- userCommandsDir,
427
- userCommandsDir,
428
- 'user'
429
- );
430
- allCommands.push(...userCommands);
431
-
432
- // Separate built-in and custom commands
433
- const customCommands = allCommands.filter(cmd => cmd.namespace !== 'builtin');
434
-
435
- // Sort commands alphabetically by name
436
- customCommands.sort((a, b) => a.name.localeCompare(b.name));
437
-
438
- res.json({
439
- builtIn: builtInCommands,
440
- custom: customCommands,
441
- count: allCommands.length
442
- });
443
- } catch (error) {
444
- // command list error
445
- res.status(500).json({
446
- error: 'Failed to list commands',
447
- message: 'An error occurred'
448
- });
449
- }
450
- });
451
-
452
- /**
453
- * POST /api/commands/load
454
- * Load a specific command file and return its content and metadata
455
- */
456
- router.post('/load', async (req, res) => {
457
- try {
458
- const { commandPath } = req.body;
459
-
460
- if (!commandPath) {
461
- return res.status(400).json({
462
- error: 'Command path is required'
463
- });
464
- }
465
-
466
- // Security: Prevent path traversal
467
- const resolvedPath = path.resolve(commandPath);
468
- if (!resolvedPath.startsWith(path.resolve(os.homedir())) &&
469
- !resolvedPath.includes('.claude/commands')) {
470
- return res.status(403).json({
471
- error: 'Access denied',
472
- message: 'Command must be in .claude/commands directory'
473
- });
474
- }
475
-
476
- // Read and parse the command file
477
- const content = await fs.readFile(commandPath, 'utf8');
478
- const { data: metadata, content: commandContent } = matter(content);
479
-
480
- res.json({
481
- path: commandPath,
482
- metadata,
483
- content: commandContent
484
- });
485
- } catch (error) {
486
- if (error.code === 'ENOENT') {
487
- return res.status(404).json({
488
- error: 'Command not found',
489
- message: `Command file not found: ${req.body.commandPath}`
490
- });
491
- }
492
-
493
- // command load error
494
- res.status(500).json({
495
- error: 'Failed to load command',
496
- message: 'An error occurred'
497
- });
498
- }
499
- });
500
-
501
- /**
502
- * POST /api/commands/execute
503
- * Execute a command with argument replacement
504
- * This endpoint prepares the command content but doesn't execute bash commands yet
505
- * (that will be handled in the command parser utility)
506
- */
507
- router.post('/execute', async (req, res) => {
508
- try {
509
- const { commandName, commandPath, args = [], context = {} } = req.body;
510
-
511
- if (!commandName) {
512
- return res.status(400).json({
513
- error: 'Command name is required'
514
- });
515
- }
516
-
517
- // Handle built-in commands
518
- const handler = builtInHandlers[commandName];
519
- if (handler) {
520
- try {
521
- const result = await handler(args, context);
522
- return res.json({
523
- ...result,
524
- command: commandName
525
- });
526
- } catch (error) {
527
- // built-in command error
528
- return res.status(500).json({
529
- error: 'Command execution failed',
530
- message: 'An error occurred',
531
- command: commandName
532
- });
533
- }
534
- }
535
-
536
- // Handle custom commands
537
- if (!commandPath) {
538
- return res.status(400).json({
539
- error: 'Command path is required for custom commands'
540
- });
541
- }
542
-
543
- // Load command content
544
- // Security: validate commandPath is within allowed directories
545
- {
546
- const resolvedPath = path.resolve(commandPath);
547
- const userBase = path.resolve(path.join(os.homedir(), '.claude', 'commands'));
548
- const projectBase = context?.projectPath
549
- ? path.resolve(path.join(context.projectPath, '.claude', 'commands'))
550
- : null;
551
- const isUnder = (base) => {
552
- const rel = path.relative(base, resolvedPath);
553
- return rel !== '' && !rel.startsWith('..') && !path.isAbsolute(rel);
554
- };
555
- if (!(isUnder(userBase) || (projectBase && isUnder(projectBase)))) {
556
- return res.status(403).json({
557
- error: 'Access denied',
558
- message: 'Command must be in .claude/commands directory'
559
- });
560
- }
561
- }
562
- const content = await fs.readFile(commandPath, 'utf8');
563
- const { data: metadata, content: commandContent } = matter(content);
564
- // Basic argument replacement (will be enhanced in command parser utility)
565
- let processedContent = commandContent;
566
-
567
- // Replace $ARGUMENTS with all arguments joined
568
- const argsString = args.join(' ');
569
- processedContent = processedContent.replace(/\$ARGUMENTS/g, argsString);
570
-
571
- // Replace $1, $2, etc. with positional arguments
572
- args.forEach((arg, index) => {
573
- const placeholder = `$${index + 1}`;
574
- processedContent = processedContent.replace(new RegExp(`\\${placeholder}\\b`, 'g'), arg);
575
- });
576
-
577
- res.json({
578
- type: 'custom',
579
- command: commandName,
580
- content: processedContent,
581
- metadata,
582
- hasFileIncludes: processedContent.includes('@'),
583
- hasBashCommands: processedContent.includes('!')
584
- });
585
- } catch (error) {
586
- if (error.code === 'ENOENT') {
587
- return res.status(404).json({
588
- error: 'Command not found',
589
- message: `Command file not found: ${req.body.commandPath}`
590
- });
591
- }
592
-
593
- // command execution error
594
- res.status(500).json({
595
- error: 'Failed to execute command',
596
- message: 'An error occurred'
597
- });
598
- }
599
- });
600
-
601
- export default router;