@glwhappen/web-code 1.32.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 (479) hide show
  1. package/LICENSE +718 -0
  2. package/README.de.md +250 -0
  3. package/README.ja.md +242 -0
  4. package/README.ko.md +242 -0
  5. package/README.md +252 -0
  6. package/README.ru.md +250 -0
  7. package/README.tr.md +252 -0
  8. package/README.zh-CN.md +242 -0
  9. package/dist/api-docs.html +879 -0
  10. package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  11. package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  12. package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  13. package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  14. package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  15. package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  16. package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  17. package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  18. package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  19. package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  20. package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  21. package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  22. package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  23. package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  24. package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  25. package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  26. package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  27. package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  28. package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  29. package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  30. package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  31. package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  32. package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  33. package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  34. package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  35. package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  36. package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  37. package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  38. package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  39. package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  40. package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  41. package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  42. package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  43. package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  44. package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  45. package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  46. package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  47. package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  48. package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  49. package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  50. package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  51. package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  52. package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  53. package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  54. package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  55. package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  56. package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  57. package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  58. package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  59. package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  60. package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  61. package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  62. package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  63. package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  64. package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  65. package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  66. package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  67. package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  68. package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  69. package/dist/assets/index-Ct6oPUQk.css +32 -0
  70. package/dist/assets/index-u6XmIqLb.js +1346 -0
  71. package/dist/assets/vendor-codemirror-OwyKSvPE.js +41 -0
  72. package/dist/assets/vendor-react-BGZc9oRE.js +59 -0
  73. package/dist/assets/vendor-xterm-CJZjLICi.js +66 -0
  74. package/dist/clear-cache.html +85 -0
  75. package/dist/convert-icons.md +53 -0
  76. package/dist/favicon.png +0 -0
  77. package/dist/favicon.svg +9 -0
  78. package/dist/generate-icons.js +49 -0
  79. package/dist/icons/claude-ai-icon.svg +1 -0
  80. package/dist/icons/codex-white.svg +3 -0
  81. package/dist/icons/codex.svg +3 -0
  82. package/dist/icons/cursor-white.svg +12 -0
  83. package/dist/icons/cursor.svg +1 -0
  84. package/dist/icons/gemini-ai-icon.svg +1 -0
  85. package/dist/icons/icon-128x128.png +0 -0
  86. package/dist/icons/icon-128x128.svg +12 -0
  87. package/dist/icons/icon-144x144.png +0 -0
  88. package/dist/icons/icon-144x144.svg +12 -0
  89. package/dist/icons/icon-152x152.png +0 -0
  90. package/dist/icons/icon-152x152.svg +12 -0
  91. package/dist/icons/icon-192x192.png +0 -0
  92. package/dist/icons/icon-192x192.svg +12 -0
  93. package/dist/icons/icon-384x384.png +0 -0
  94. package/dist/icons/icon-384x384.svg +12 -0
  95. package/dist/icons/icon-512x512.png +0 -0
  96. package/dist/icons/icon-512x512.svg +12 -0
  97. package/dist/icons/icon-72x72.png +0 -0
  98. package/dist/icons/icon-72x72.svg +12 -0
  99. package/dist/icons/icon-96x96.png +0 -0
  100. package/dist/icons/icon-96x96.svg +12 -0
  101. package/dist/icons/icon-template.svg +12 -0
  102. package/dist/index.html +52 -0
  103. package/dist/logo-128.png +0 -0
  104. package/dist/logo-256.png +0 -0
  105. package/dist/logo-32.png +0 -0
  106. package/dist/logo-512.png +0 -0
  107. package/dist/logo-64.png +0 -0
  108. package/dist/logo.svg +17 -0
  109. package/dist/manifest.json +61 -0
  110. package/dist/screenshots/cli-selection.png +0 -0
  111. package/dist/screenshots/desktop-main.png +0 -0
  112. package/dist/screenshots/mobile-chat.png +0 -0
  113. package/dist/screenshots/tools-modal.png +0 -0
  114. package/dist/sw.js +124 -0
  115. package/dist-server/server/claude-sdk.js +738 -0
  116. package/dist-server/server/claude-sdk.js.map +1 -0
  117. package/dist-server/server/cli.js +641 -0
  118. package/dist-server/server/cli.js.map +1 -0
  119. package/dist-server/server/constants/config.js +6 -0
  120. package/dist-server/server/constants/config.js.map +1 -0
  121. package/dist-server/server/cursor-cli.js +271 -0
  122. package/dist-server/server/cursor-cli.js.map +1 -0
  123. package/dist-server/server/gemini-cli.js +539 -0
  124. package/dist-server/server/gemini-cli.js.map +1 -0
  125. package/dist-server/server/gemini-response-handler.js +72 -0
  126. package/dist-server/server/gemini-response-handler.js.map +1 -0
  127. package/dist-server/server/index.js +1340 -0
  128. package/dist-server/server/index.js.map +1 -0
  129. package/dist-server/server/load-env.js +32 -0
  130. package/dist-server/server/load-env.js.map +1 -0
  131. package/dist-server/server/middleware/auth.js +117 -0
  132. package/dist-server/server/middleware/auth.js.map +1 -0
  133. package/dist-server/server/modules/database/connection.js +125 -0
  134. package/dist-server/server/modules/database/connection.js.map +1 -0
  135. package/dist-server/server/modules/database/index.js +13 -0
  136. package/dist-server/server/modules/database/index.js.map +1 -0
  137. package/dist-server/server/modules/database/init-db.js +18 -0
  138. package/dist-server/server/modules/database/init-db.js.map +1 -0
  139. package/dist-server/server/modules/database/migrations.js +419 -0
  140. package/dist-server/server/modules/database/migrations.js.map +1 -0
  141. package/dist-server/server/modules/database/repositories/api-keys.js +72 -0
  142. package/dist-server/server/modules/database/repositories/api-keys.js.map +1 -0
  143. package/dist-server/server/modules/database/repositories/app-config.js +47 -0
  144. package/dist-server/server/modules/database/repositories/app-config.js.map +1 -0
  145. package/dist-server/server/modules/database/repositories/credentials.js +68 -0
  146. package/dist-server/server/modules/database/repositories/credentials.js.map +1 -0
  147. package/dist-server/server/modules/database/repositories/github-tokens.js +54 -0
  148. package/dist-server/server/modules/database/repositories/github-tokens.js.map +1 -0
  149. package/dist-server/server/modules/database/repositories/notification-preferences.js +72 -0
  150. package/dist-server/server/modules/database/repositories/notification-preferences.js.map +1 -0
  151. package/dist-server/server/modules/database/repositories/projects.db.integration.test.js +67 -0
  152. package/dist-server/server/modules/database/repositories/projects.db.integration.test.js.map +1 -0
  153. package/dist-server/server/modules/database/repositories/projects.db.js +185 -0
  154. package/dist-server/server/modules/database/repositories/projects.db.js.map +1 -0
  155. package/dist-server/server/modules/database/repositories/push-subscriptions.js +49 -0
  156. package/dist-server/server/modules/database/repositories/push-subscriptions.js.map +1 -0
  157. package/dist-server/server/modules/database/repositories/scan-state.db.js +31 -0
  158. package/dist-server/server/modules/database/repositories/scan-state.db.js.map +1 -0
  159. package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js +64 -0
  160. package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js.map +1 -0
  161. package/dist-server/server/modules/database/repositories/sessions.db.js +150 -0
  162. package/dist-server/server/modules/database/repositories/sessions.db.js.map +1 -0
  163. package/dist-server/server/modules/database/repositories/users.js +116 -0
  164. package/dist-server/server/modules/database/repositories/users.js.map +1 -0
  165. package/dist-server/server/modules/database/repositories/vapid-keys.js +38 -0
  166. package/dist-server/server/modules/database/repositories/vapid-keys.js.map +1 -0
  167. package/dist-server/server/modules/database/schema.js +150 -0
  168. package/dist-server/server/modules/database/schema.js.map +1 -0
  169. package/dist-server/server/modules/projects/index.js +4 -0
  170. package/dist-server/server/modules/projects/index.js.map +1 -0
  171. package/dist-server/server/modules/projects/projects.routes.js +225 -0
  172. package/dist-server/server/modules/projects/projects.routes.js.map +1 -0
  173. package/dist-server/server/modules/projects/services/project-clone.service.js +220 -0
  174. package/dist-server/server/modules/projects/services/project-clone.service.js.map +1 -0
  175. package/dist-server/server/modules/projects/services/project-delete.service.js +83 -0
  176. package/dist-server/server/modules/projects/services/project-delete.service.js.map +1 -0
  177. package/dist-server/server/modules/projects/services/project-management.service.js +99 -0
  178. package/dist-server/server/modules/projects/services/project-management.service.js.map +1 -0
  179. package/dist-server/server/modules/projects/services/project-star.service.js +60 -0
  180. package/dist-server/server/modules/projects/services/project-star.service.js.map +1 -0
  181. package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js +171 -0
  182. package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js.map +1 -0
  183. package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js +213 -0
  184. package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js.map +1 -0
  185. package/dist-server/server/modules/projects/tests/project-clone.service.test.js +129 -0
  186. package/dist-server/server/modules/projects/tests/project-clone.service.test.js.map +1 -0
  187. package/dist-server/server/modules/projects/tests/project-management.service.test.js +89 -0
  188. package/dist-server/server/modules/projects/tests/project-management.service.test.js.map +1 -0
  189. package/dist-server/server/modules/projects/tests/project-star.service.test.js +99 -0
  190. package/dist-server/server/modules/projects/tests/project-star.service.test.js.map +1 -0
  191. package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js +88 -0
  192. package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js.map +1 -0
  193. package/dist-server/server/modules/providers/index.js +5 -0
  194. package/dist-server/server/modules/providers/index.js.map +1 -0
  195. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +104 -0
  196. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -0
  197. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js +103 -0
  198. package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js.map +1 -0
  199. package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js +116 -0
  200. package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js.map +1 -0
  201. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js +546 -0
  202. package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js.map +1 -0
  203. package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js +198 -0
  204. package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js.map +1 -0
  205. package/dist-server/server/modules/providers/list/claude/claude.provider.js +17 -0
  206. package/dist-server/server/modules/providers/list/claude/claude.provider.js.map +1 -0
  207. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +84 -0
  208. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -0
  209. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js +107 -0
  210. package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js.map +1 -0
  211. package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js +123 -0
  212. package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js.map +1 -0
  213. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js +513 -0
  214. package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js.map +1 -0
  215. package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js +82 -0
  216. package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js.map +1 -0
  217. package/dist-server/server/modules/providers/list/codex/codex.provider.js +17 -0
  218. package/dist-server/server/modules/providers/list/codex/codex.provider.js.map +1 -0
  219. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +118 -0
  220. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -0
  221. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js +80 -0
  222. package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js.map +1 -0
  223. package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js +105 -0
  224. package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js.map +1 -0
  225. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js +545 -0
  226. package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js.map +1 -0
  227. package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js +28 -0
  228. package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js.map +1 -0
  229. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js +17 -0
  230. package/dist-server/server/modules/providers/list/cursor/cursor.provider.js.map +1 -0
  231. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +254 -0
  232. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -0
  233. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js +82 -0
  234. package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js.map +1 -0
  235. package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js +312 -0
  236. package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js.map +1 -0
  237. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js +484 -0
  238. package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js.map +1 -0
  239. package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js +33 -0
  240. package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js.map +1 -0
  241. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js +17 -0
  242. package/dist-server/server/modules/providers/list/gemini/gemini.provider.js.map +1 -0
  243. package/dist-server/server/modules/providers/provider.registry.js +31 -0
  244. package/dist-server/server/modules/providers/provider.registry.js.map +1 -0
  245. package/dist-server/server/modules/providers/provider.routes.js +377 -0
  246. package/dist-server/server/modules/providers/provider.routes.js.map +1 -0
  247. package/dist-server/server/modules/providers/services/mcp.service.js +69 -0
  248. package/dist-server/server/modules/providers/services/mcp.service.js.map +1 -0
  249. package/dist-server/server/modules/providers/services/provider-auth.service.js +25 -0
  250. package/dist-server/server/modules/providers/services/provider-auth.service.js.map +1 -0
  251. package/dist-server/server/modules/providers/services/session-conversations-search.service.js +984 -0
  252. package/dist-server/server/modules/providers/services/session-conversations-search.service.js.map +1 -0
  253. package/dist-server/server/modules/providers/services/session-synchronizer.service.js +56 -0
  254. package/dist-server/server/modules/providers/services/session-synchronizer.service.js.map +1 -0
  255. package/dist-server/server/modules/providers/services/sessions-watcher.service.js +269 -0
  256. package/dist-server/server/modules/providers/services/sessions-watcher.service.js.map +1 -0
  257. package/dist-server/server/modules/providers/services/sessions.service.js +179 -0
  258. package/dist-server/server/modules/providers/services/sessions.service.js.map +1 -0
  259. package/dist-server/server/modules/providers/services/skills.service.js +11 -0
  260. package/dist-server/server/modules/providers/services/skills.service.js.map +1 -0
  261. package/dist-server/server/modules/providers/shared/base/abstract.provider.js +14 -0
  262. package/dist-server/server/modules/providers/shared/base/abstract.provider.js.map +1 -0
  263. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js +102 -0
  264. package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js.map +1 -0
  265. package/dist-server/server/modules/providers/shared/skills/skills.provider.js +45 -0
  266. package/dist-server/server/modules/providers/shared/skills/skills.provider.js.map +1 -0
  267. package/dist-server/server/modules/providers/tests/mcp.test.js +250 -0
  268. package/dist-server/server/modules/providers/tests/mcp.test.js.map +1 -0
  269. package/dist-server/server/modules/providers/tests/skills.test.js +226 -0
  270. package/dist-server/server/modules/providers/tests/skills.test.js.map +1 -0
  271. package/dist-server/server/modules/websocket/index.js +3 -0
  272. package/dist-server/server/modules/websocket/index.js.map +1 -0
  273. package/dist-server/server/modules/websocket/services/chat-websocket.service.js +192 -0
  274. package/dist-server/server/modules/websocket/services/chat-websocket.service.js.map +1 -0
  275. package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js +52 -0
  276. package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js.map +1 -0
  277. package/dist-server/server/modules/websocket/services/shell-websocket.service.js +360 -0
  278. package/dist-server/server/modules/websocket/services/shell-websocket.service.js.map +1 -0
  279. package/dist-server/server/modules/websocket/services/websocket-auth.service.js +32 -0
  280. package/dist-server/server/modules/websocket/services/websocket-auth.service.js.map +1 -0
  281. package/dist-server/server/modules/websocket/services/websocket-server.service.js +36 -0
  282. package/dist-server/server/modules/websocket/services/websocket-server.service.js.map +1 -0
  283. package/dist-server/server/modules/websocket/services/websocket-state.service.js +14 -0
  284. package/dist-server/server/modules/websocket/services/websocket-state.service.js.map +1 -0
  285. package/dist-server/server/modules/websocket/services/websocket-writer.service.js +32 -0
  286. package/dist-server/server/modules/websocket/services/websocket-writer.service.js.map +1 -0
  287. package/dist-server/server/openai-codex.js +418 -0
  288. package/dist-server/server/openai-codex.js.map +1 -0
  289. package/dist-server/server/routes/admin.js +109 -0
  290. package/dist-server/server/routes/admin.js.map +1 -0
  291. package/dist-server/server/routes/agent.js +1145 -0
  292. package/dist-server/server/routes/agent.js.map +1 -0
  293. package/dist-server/server/routes/auth.js +123 -0
  294. package/dist-server/server/routes/auth.js.map +1 -0
  295. package/dist-server/server/routes/commands.js +487 -0
  296. package/dist-server/server/routes/commands.js.map +1 -0
  297. package/dist-server/server/routes/cursor.js +49 -0
  298. package/dist-server/server/routes/cursor.js.map +1 -0
  299. package/dist-server/server/routes/gemini.js +25 -0
  300. package/dist-server/server/routes/gemini.js.map +1 -0
  301. package/dist-server/server/routes/git.js +1263 -0
  302. package/dist-server/server/routes/git.js.map +1 -0
  303. package/dist-server/server/routes/mcp-utils.js +29 -0
  304. package/dist-server/server/routes/mcp-utils.js.map +1 -0
  305. package/dist-server/server/routes/plugins.js +266 -0
  306. package/dist-server/server/routes/plugins.js.map +1 -0
  307. package/dist-server/server/routes/settings.js +259 -0
  308. package/dist-server/server/routes/settings.js.map +1 -0
  309. package/dist-server/server/routes/taskmaster.js +1360 -0
  310. package/dist-server/server/routes/taskmaster.js.map +1 -0
  311. package/dist-server/server/routes/user.js +115 -0
  312. package/dist-server/server/routes/user.js.map +1 -0
  313. package/dist-server/server/services/notification-orchestrator.js +177 -0
  314. package/dist-server/server/services/notification-orchestrator.js.map +1 -0
  315. package/dist-server/server/services/vapid-keys.js +27 -0
  316. package/dist-server/server/services/vapid-keys.js.map +1 -0
  317. package/dist-server/server/sessionManager.js +215 -0
  318. package/dist-server/server/sessionManager.js.map +1 -0
  319. package/dist-server/server/shared/claude-cli-path.js +103 -0
  320. package/dist-server/server/shared/claude-cli-path.js.map +1 -0
  321. package/dist-server/server/shared/claude-cli-path.test.js +45 -0
  322. package/dist-server/server/shared/claude-cli-path.test.js.map +1 -0
  323. package/dist-server/server/shared/default-user.js +29 -0
  324. package/dist-server/server/shared/default-user.js.map +1 -0
  325. package/dist-server/server/shared/frontmatter.js +16 -0
  326. package/dist-server/server/shared/frontmatter.js.map +1 -0
  327. package/dist-server/server/shared/interfaces.js +2 -0
  328. package/dist-server/server/shared/interfaces.js.map +1 -0
  329. package/dist-server/server/shared/types.js +2 -0
  330. package/dist-server/server/shared/types.js.map +1 -0
  331. package/dist-server/server/shared/utils.js +633 -0
  332. package/dist-server/server/shared/utils.js.map +1 -0
  333. package/dist-server/server/utils/colors.js +20 -0
  334. package/dist-server/server/utils/colors.js.map +1 -0
  335. package/dist-server/server/utils/commandParser.js +255 -0
  336. package/dist-server/server/utils/commandParser.js.map +1 -0
  337. package/dist-server/server/utils/gitConfig.js +36 -0
  338. package/dist-server/server/utils/gitConfig.js.map +1 -0
  339. package/dist-server/server/utils/mcp-detector.js +134 -0
  340. package/dist-server/server/utils/mcp-detector.js.map +1 -0
  341. package/dist-server/server/utils/plugin-loader.js +413 -0
  342. package/dist-server/server/utils/plugin-loader.js.map +1 -0
  343. package/dist-server/server/utils/plugin-process-manager.js +163 -0
  344. package/dist-server/server/utils/plugin-process-manager.js.map +1 -0
  345. package/dist-server/server/utils/runtime-paths.js +30 -0
  346. package/dist-server/server/utils/runtime-paths.js.map +1 -0
  347. package/dist-server/server/utils/taskmaster-websocket.js +124 -0
  348. package/dist-server/server/utils/taskmaster-websocket.js.map +1 -0
  349. package/dist-server/server/utils/url-detection.js +58 -0
  350. package/dist-server/server/utils/url-detection.js.map +1 -0
  351. package/dist-server/shared/modelConstants.js +99 -0
  352. package/dist-server/shared/modelConstants.js.map +1 -0
  353. package/dist-server/shared/networkHosts.js +20 -0
  354. package/dist-server/shared/networkHosts.js.map +1 -0
  355. package/package.json +169 -0
  356. package/scripts/fix-node-pty.js +67 -0
  357. package/server/claude-sdk.js +864 -0
  358. package/server/cli.js +688 -0
  359. package/server/constants/config.js +5 -0
  360. package/server/cursor-cli.js +334 -0
  361. package/server/gemini-cli.js +622 -0
  362. package/server/gemini-response-handler.js +79 -0
  363. package/server/index.js +1505 -0
  364. package/server/load-env.js +34 -0
  365. package/server/middleware/auth.js +142 -0
  366. package/server/modules/database/connection.ts +143 -0
  367. package/server/modules/database/index.ts +12 -0
  368. package/server/modules/database/init-db.ts +17 -0
  369. package/server/modules/database/migrations.ts +496 -0
  370. package/server/modules/database/repositories/api-keys.ts +119 -0
  371. package/server/modules/database/repositories/app-config.ts +53 -0
  372. package/server/modules/database/repositories/credentials.ts +106 -0
  373. package/server/modules/database/repositories/github-tokens.ts +100 -0
  374. package/server/modules/database/repositories/notification-preferences.ts +103 -0
  375. package/server/modules/database/repositories/projects.db.integration.test.ts +78 -0
  376. package/server/modules/database/repositories/projects.db.ts +210 -0
  377. package/server/modules/database/repositories/push-subscriptions.ts +80 -0
  378. package/server/modules/database/repositories/scan-state.db.ts +42 -0
  379. package/server/modules/database/repositories/sessions.db.integration.test.ts +78 -0
  380. package/server/modules/database/repositories/sessions.db.ts +230 -0
  381. package/server/modules/database/repositories/users.ts +186 -0
  382. package/server/modules/database/repositories/vapid-keys.ts +57 -0
  383. package/server/modules/database/schema.ts +159 -0
  384. package/server/modules/projects/index.ts +6 -0
  385. package/server/modules/projects/projects.routes.ts +292 -0
  386. package/server/modules/projects/services/project-clone.service.ts +327 -0
  387. package/server/modules/projects/services/project-delete.service.ts +95 -0
  388. package/server/modules/projects/services/project-management.service.ts +158 -0
  389. package/server/modules/projects/services/project-star.service.ts +78 -0
  390. package/server/modules/projects/services/projects-has-taskmaster.service.ts +257 -0
  391. package/server/modules/projects/services/projects-with-sessions-fetch.service.ts +355 -0
  392. package/server/modules/projects/tests/project-clone.service.test.ts +186 -0
  393. package/server/modules/projects/tests/project-management.service.test.ts +122 -0
  394. package/server/modules/projects/tests/project-star.service.test.ts +128 -0
  395. package/server/modules/projects/tests/projects-has-taskmaster.service.test.ts +107 -0
  396. package/server/modules/providers/README.md +346 -0
  397. package/server/modules/providers/index.ts +5 -0
  398. package/server/modules/providers/list/claude/claude-auth.provider.ts +124 -0
  399. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -0
  400. package/server/modules/providers/list/claude/claude-session-synchronizer.provider.ts +179 -0
  401. package/server/modules/providers/list/claude/claude-sessions.provider.ts +642 -0
  402. package/server/modules/providers/list/claude/claude-skills.provider.ts +257 -0
  403. package/server/modules/providers/list/claude/claude.provider.ts +24 -0
  404. package/server/modules/providers/list/codex/codex-auth.provider.ts +100 -0
  405. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -0
  406. package/server/modules/providers/list/codex/codex-session-synchronizer.provider.ts +182 -0
  407. package/server/modules/providers/list/codex/codex-sessions.provider.ts +589 -0
  408. package/server/modules/providers/list/codex/codex-skills.provider.ts +100 -0
  409. package/server/modules/providers/list/codex/codex.provider.ts +24 -0
  410. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -0
  411. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -0
  412. package/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.ts +155 -0
  413. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +624 -0
  414. package/server/modules/providers/list/cursor/cursor-skills.provider.ts +31 -0
  415. package/server/modules/providers/list/cursor/cursor.provider.ts +24 -0
  416. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +307 -0
  417. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -0
  418. package/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.ts +407 -0
  419. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +552 -0
  420. package/server/modules/providers/list/gemini/gemini-skills.provider.ts +36 -0
  421. package/server/modules/providers/list/gemini/gemini.provider.ts +24 -0
  422. package/server/modules/providers/provider.registry.ts +36 -0
  423. package/server/modules/providers/provider.routes.ts +488 -0
  424. package/server/modules/providers/services/mcp.service.ts +94 -0
  425. package/server/modules/providers/services/provider-auth.service.ts +26 -0
  426. package/server/modules/providers/services/session-conversations-search.service.ts +1319 -0
  427. package/server/modules/providers/services/session-synchronizer.service.ts +75 -0
  428. package/server/modules/providers/services/sessions-watcher.service.ts +318 -0
  429. package/server/modules/providers/services/sessions.service.ts +240 -0
  430. package/server/modules/providers/services/skills.service.ts +15 -0
  431. package/server/modules/providers/shared/base/abstract.provider.ts +29 -0
  432. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -0
  433. package/server/modules/providers/shared/skills/skills.provider.ts +64 -0
  434. package/server/modules/providers/tests/mcp.test.ts +293 -0
  435. package/server/modules/providers/tests/skills.test.ts +446 -0
  436. package/server/modules/websocket/README.md +267 -0
  437. package/server/modules/websocket/index.ts +2 -0
  438. package/server/modules/websocket/services/chat-websocket.service.ts +275 -0
  439. package/server/modules/websocket/services/plugin-websocket-proxy.service.ts +65 -0
  440. package/server/modules/websocket/services/shell-websocket.service.ts +489 -0
  441. package/server/modules/websocket/services/websocket-auth.service.ts +54 -0
  442. package/server/modules/websocket/services/websocket-server.service.ts +58 -0
  443. package/server/modules/websocket/services/websocket-state.service.ts +16 -0
  444. package/server/modules/websocket/services/websocket-writer.service.ts +38 -0
  445. package/server/openai-codex.js +474 -0
  446. package/server/routes/admin.js +128 -0
  447. package/server/routes/agent.js +1246 -0
  448. package/server/routes/auth.js +144 -0
  449. package/server/routes/commands.js +556 -0
  450. package/server/routes/cursor.js +52 -0
  451. package/server/routes/gemini.js +30 -0
  452. package/server/routes/git.js +1493 -0
  453. package/server/routes/mcp-utils.js +31 -0
  454. package/server/routes/plugins.js +307 -0
  455. package/server/routes/settings.js +286 -0
  456. package/server/routes/taskmaster.js +1468 -0
  457. package/server/routes/user.js +123 -0
  458. package/server/services/notification-orchestrator.js +228 -0
  459. package/server/services/vapid-keys.js +36 -0
  460. package/server/sessionManager.js +248 -0
  461. package/server/shared/claude-cli-path.test.ts +61 -0
  462. package/server/shared/claude-cli-path.ts +139 -0
  463. package/server/shared/default-user.ts +30 -0
  464. package/server/shared/frontmatter.ts +18 -0
  465. package/server/shared/interfaces.ts +111 -0
  466. package/server/shared/types.ts +406 -0
  467. package/server/shared/utils.ts +763 -0
  468. package/server/tsconfig.json +36 -0
  469. package/server/utils/colors.js +21 -0
  470. package/server/utils/commandParser.js +305 -0
  471. package/server/utils/gitConfig.js +34 -0
  472. package/server/utils/mcp-detector.js +147 -0
  473. package/server/utils/plugin-loader.js +457 -0
  474. package/server/utils/plugin-process-manager.js +184 -0
  475. package/server/utils/runtime-paths.js +37 -0
  476. package/server/utils/taskmaster-websocket.js +135 -0
  477. package/server/utils/url-detection.js +71 -0
  478. package/shared/modelConstants.js +107 -0
  479. package/shared/networkHosts.js +22 -0
package/server/cli.js ADDED
@@ -0,0 +1,688 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * CloudCLI CLI
4
+ *
5
+ * Provides command-line utilities for managing CloudCLI
6
+ *
7
+ * Commands:
8
+ * (no args) - Start the server (default)
9
+ * start - Start the server
10
+ * sandbox - Manage Docker sandbox environments
11
+ * status - Show configuration and data locations
12
+ * help - Show help information
13
+ * version - Show version information
14
+ */
15
+
16
+ import fs from 'fs';
17
+ import path from 'path';
18
+ import os from 'os';
19
+ import { findAppRoot, getModuleDir } from './utils/runtime-paths.js';
20
+
21
+ const __dirname = getModuleDir(import.meta.url);
22
+ // The CLI is compiled into dist-server/server, but it still needs to read the top-level
23
+ // package.json and .env file. Resolving the app root once keeps those lookups stable.
24
+ const APP_ROOT = findAppRoot(__dirname);
25
+
26
+ // ANSI color codes for terminal output
27
+ const colors = {
28
+ reset: '\x1b[0m',
29
+ bright: '\x1b[1m',
30
+ dim: '\x1b[2m',
31
+
32
+ // Foreground colors
33
+ cyan: '\x1b[36m',
34
+ green: '\x1b[32m',
35
+ yellow: '\x1b[33m',
36
+ blue: '\x1b[34m',
37
+ magenta: '\x1b[35m',
38
+ white: '\x1b[37m',
39
+ gray: '\x1b[90m',
40
+ };
41
+
42
+ // Helper to colorize text
43
+ const c = {
44
+ info: (text) => `${colors.cyan}${text}${colors.reset}`,
45
+ ok: (text) => `${colors.green}${text}${colors.reset}`,
46
+ warn: (text) => `${colors.yellow}${text}${colors.reset}`,
47
+ error: (text) => `${colors.yellow}${text}${colors.reset}`,
48
+ tip: (text) => `${colors.blue}${text}${colors.reset}`,
49
+ bright: (text) => `${colors.bright}${text}${colors.reset}`,
50
+ dim: (text) => `${colors.dim}${text}${colors.reset}`,
51
+ };
52
+
53
+ // Load package.json for version info
54
+ const packageJsonPath = path.join(APP_ROOT, 'package.json');
55
+ const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
56
+ // Match the runtime fallback in load-env.js so "cloudcli status" reports the same default
57
+ // database location that the backend will actually use when no DATABASE_PATH is configured.
58
+ const DEFAULT_DATABASE_PATH = path.join(os.homedir(), '.cloudcli', 'auth.db');
59
+
60
+ // Load environment variables from .env file if it exists
61
+ function loadEnvFile() {
62
+ try {
63
+ const envPath = path.join(APP_ROOT, '.env');
64
+ const envFile = fs.readFileSync(envPath, 'utf8');
65
+ envFile.split('\n').forEach(line => {
66
+ const trimmedLine = line.trim();
67
+ if (trimmedLine && !trimmedLine.startsWith('#')) {
68
+ const [key, ...valueParts] = trimmedLine.split('=');
69
+ if (key && valueParts.length > 0 && !process.env[key]) {
70
+ process.env[key] = valueParts.join('=').trim();
71
+ }
72
+ }
73
+ });
74
+ } catch (e) {
75
+ // .env file is optional
76
+ }
77
+ }
78
+
79
+ // Get the database path (same logic as db.js)
80
+ function getDatabasePath() {
81
+ loadEnvFile();
82
+ return process.env.DATABASE_PATH || DEFAULT_DATABASE_PATH;
83
+ }
84
+
85
+ // Get the installation directory
86
+ function getInstallDir() {
87
+ return APP_ROOT;
88
+ }
89
+
90
+ // Show status command
91
+ function showStatus() {
92
+ console.log(`\n${c.bright('CloudCLI UI - Status')}\n`);
93
+ console.log(c.dim('═'.repeat(60)));
94
+
95
+ // Version info
96
+ console.log(`\n${c.info('[INFO]')} Version: ${c.bright(packageJson.version)}`);
97
+
98
+ // Installation location
99
+ const installDir = getInstallDir();
100
+ console.log(`\n${c.info('[INFO]')} Installation Directory:`);
101
+ console.log(` ${c.dim(installDir)}`);
102
+
103
+ // Database location
104
+ const dbPath = getDatabasePath();
105
+ const dbExists = fs.existsSync(dbPath);
106
+ console.log(`\n${c.info('[INFO]')} Database Location:`);
107
+ console.log(` ${c.dim(dbPath)}`);
108
+ console.log(` Status: ${dbExists ? c.ok('[OK] Exists') : c.warn('[WARN] Not created yet (will be created on first run)')}`);
109
+
110
+ if (dbExists) {
111
+ const stats = fs.statSync(dbPath);
112
+ console.log(` Size: ${c.dim((stats.size / 1024).toFixed(2) + ' KB')}`);
113
+ console.log(` Modified: ${c.dim(stats.mtime.toLocaleString())}`);
114
+ }
115
+
116
+ // Environment variables
117
+ console.log(`\n${c.info('[INFO]')} Configuration:`);
118
+ console.log(` SERVER_PORT: ${c.bright(process.env.SERVER_PORT || process.env.PORT || '3001')} ${c.dim(process.env.SERVER_PORT || process.env.PORT ? '' : '(default)')}`);
119
+ console.log(` DATABASE_PATH: ${c.dim(process.env.DATABASE_PATH || '(using default location)')}`);
120
+ console.log(` CLAUDE_CLI_PATH: ${c.dim(process.env.CLAUDE_CLI_PATH || 'claude (default)')}`);
121
+ console.log(` CONTEXT_WINDOW: ${c.dim(process.env.CONTEXT_WINDOW || '160000 (default)')}`);
122
+
123
+ // Claude projects folder
124
+ const claudeProjectsPath = path.join(os.homedir(), '.claude', 'projects');
125
+ const projectsExists = fs.existsSync(claudeProjectsPath);
126
+ console.log(`\n${c.info('[INFO]')} Claude Projects Folder:`);
127
+ console.log(` ${c.dim(claudeProjectsPath)}`);
128
+ console.log(` Status: ${projectsExists ? c.ok('[OK] Exists') : c.warn('[WARN] Not found')}`);
129
+
130
+ // Config file location
131
+ const envFilePath = path.join(APP_ROOT, '.env');
132
+ const envExists = fs.existsSync(envFilePath);
133
+ console.log(`\n${c.info('[INFO]')} Configuration File:`);
134
+ console.log(` ${c.dim(envFilePath)}`);
135
+ console.log(` Status: ${envExists ? c.ok('[OK] Exists') : c.warn('[WARN] Not found (using defaults)')}`);
136
+
137
+ console.log('\n' + c.dim('═'.repeat(60)));
138
+ console.log(`\n${c.tip('[TIP]')} Hints:`);
139
+ console.log(` ${c.dim('>')} Use ${c.bright('web-code --port 8080')} to run on a custom port`);
140
+ console.log(` ${c.dim('>')} Use ${c.bright('web-code --database-path /path/to/db')} for custom database`);
141
+ console.log(` ${c.dim('>')} Run ${c.bright('web-code help')} for all options`);
142
+ console.log(` ${c.dim('>')} Access the UI at http://localhost:${process.env.SERVER_PORT || process.env.PORT || '3001'}\n`);
143
+ }
144
+
145
+ // Show help
146
+ function showHelp() {
147
+ console.log(`
148
+ ╔═══════════════════════════════════════════════════════════════╗
149
+ ║ CloudCLI - Command Line Tool ║
150
+ ╚═══════════════════════════════════════════════════════════════╝
151
+
152
+ Usage:
153
+ web-code [command] [options]
154
+
155
+ Commands:
156
+ start Start the CloudCLI server (default)
157
+ sandbox Manage Docker sandbox environments
158
+ status Show configuration and data locations
159
+ update Update to the latest version
160
+ help Show this help information
161
+ version Show version information
162
+
163
+ Options:
164
+ -p, --port <port> Set server port (default: 3001)
165
+ --database-path <path> Set custom database location
166
+ -h, --help Show this help information
167
+ -v, --version Show version information
168
+
169
+ Examples:
170
+ $ web-code # Start with defaults
171
+ $ web-code --port 8080 # Start on port 8080
172
+ $ web-code sandbox ~/my-project # Run in a Docker sandbox
173
+ $ web-code status # Show configuration
174
+
175
+ Environment Variables:
176
+ SERVER_PORT Set server port (default: 3001)
177
+ PORT Set server port (default: 3001) (LEGACY)
178
+ DATABASE_PATH Set custom database location
179
+ CLAUDE_CLI_PATH Set custom Claude CLI path
180
+ CONTEXT_WINDOW Set context window size (default: 160000)
181
+
182
+ Documentation:
183
+ ${packageJson.homepage || 'https://github.com/siteboon/claudecodeui'}
184
+
185
+ Report Issues:
186
+ ${packageJson.bugs?.url || 'https://github.com/siteboon/claudecodeui/issues'}
187
+ `);
188
+ }
189
+
190
+ // Show version
191
+ function showVersion() {
192
+ console.log(`${packageJson.version}`);
193
+ }
194
+
195
+ // Compare semver versions, returns true if v1 > v2
196
+ function isNewerVersion(v1, v2) {
197
+ const parts1 = v1.split('.').map(Number);
198
+ const parts2 = v2.split('.').map(Number);
199
+ for (let i = 0; i < 3; i++) {
200
+ if (parts1[i] > parts2[i]) return true;
201
+ if (parts1[i] < parts2[i]) return false;
202
+ }
203
+ return false;
204
+ }
205
+
206
+ // Check for updates
207
+ async function checkForUpdates(silent = false) {
208
+ try {
209
+ const { execSync } = await import('child_process');
210
+ const latestVersion = execSync('npm show @glwhappen/web-code version', { encoding: 'utf8' }).trim();
211
+ const currentVersion = packageJson.version;
212
+
213
+ if (isNewerVersion(latestVersion, currentVersion)) {
214
+ console.log(`\n${c.warn('[UPDATE]')} New version available: ${c.bright(latestVersion)} (current: ${currentVersion})`);
215
+ console.log(` Run ${c.bright('web-code update')} to update\n`);
216
+ return { hasUpdate: true, latestVersion, currentVersion };
217
+ } else if (!silent) {
218
+ console.log(`${c.ok('[OK]')} You are on the latest version (${currentVersion})`);
219
+ }
220
+ return { hasUpdate: false, latestVersion, currentVersion };
221
+ } catch (e) {
222
+ if (!silent) {
223
+ console.log(`${c.warn('[WARN]')} Could not check for updates`);
224
+ }
225
+ return { hasUpdate: false, error: e.message };
226
+ }
227
+ }
228
+
229
+ // Update the package
230
+ async function updatePackage() {
231
+ try {
232
+ const { execSync } = await import('child_process');
233
+ console.log(`${c.info('[INFO]')} Checking for updates...`);
234
+
235
+ const { hasUpdate, latestVersion, currentVersion } = await checkForUpdates(true);
236
+
237
+ if (!hasUpdate) {
238
+ console.log(`${c.ok('[OK]')} Already on the latest version (${currentVersion})`);
239
+ return;
240
+ }
241
+
242
+ console.log(`${c.info('[INFO]')} Updating from ${currentVersion} to ${latestVersion}...`);
243
+ execSync('npm update -g @glwhappen/web-code', { stdio: 'inherit' });
244
+ console.log(`${c.ok('[OK]')} Update complete! Restart web-code to use the new version.`);
245
+ } catch (e) {
246
+ console.error(`${c.error('[ERROR]')} Update failed: ${e.message}`);
247
+ console.log(`${c.tip('[TIP]')} Try running manually: npm update -g @glwhappen/web-code`);
248
+ }
249
+ }
250
+
251
+ // ── Sandbox command ─────────────────────────────────────────
252
+
253
+ const SANDBOX_TEMPLATES = {
254
+ claude: 'docker.io/cloudcliai/sandbox:claude-code',
255
+ codex: 'docker.io/cloudcliai/sandbox:codex',
256
+ gemini: 'docker.io/cloudcliai/sandbox:gemini',
257
+ };
258
+
259
+ const SANDBOX_SECRETS = {
260
+ claude: 'anthropic',
261
+ codex: 'openai',
262
+ gemini: 'google',
263
+ };
264
+
265
+ function parseSandboxArgs(args) {
266
+ const result = {
267
+ subcommand: null,
268
+ workspace: null,
269
+ agent: 'claude',
270
+ name: null,
271
+ port: 3001,
272
+ template: null,
273
+ env: [],
274
+ };
275
+
276
+ const subcommands = ['ls', 'stop', 'start', 'rm', 'logs', 'help'];
277
+
278
+ for (let i = 0; i < args.length; i++) {
279
+ const arg = args[i];
280
+
281
+ if (i === 0 && subcommands.includes(arg)) {
282
+ result.subcommand = arg;
283
+ } else if (arg === '--agent' || arg === '-a') {
284
+ result.agent = args[++i];
285
+ } else if (arg === '--name' || arg === '-n') {
286
+ result.name = args[++i];
287
+ } else if (arg === '--port') {
288
+ result.port = parseInt(args[++i], 10);
289
+ } else if (arg === '--template' || arg === '-t') {
290
+ result.template = args[++i];
291
+ } else if (arg === '--env' || arg === '-e') {
292
+ result.env.push(args[++i]);
293
+ } else if (!arg.startsWith('-')) {
294
+ if (!result.subcommand) {
295
+ result.workspace = arg;
296
+ } else {
297
+ result.name = arg; // for stop/start/rm/logs <name>
298
+ }
299
+ }
300
+ }
301
+
302
+ // Default subcommand based on what we got
303
+ if (!result.subcommand) {
304
+ result.subcommand = 'create';
305
+ }
306
+
307
+ // Derive name from workspace path if not set
308
+ if (!result.name && result.workspace) {
309
+ result.name = path.basename(path.resolve(result.workspace.replace(/^~/, os.homedir())));
310
+ }
311
+
312
+ // Default template from agent
313
+ if (!result.template) {
314
+ result.template = SANDBOX_TEMPLATES[result.agent] || SANDBOX_TEMPLATES.claude;
315
+ }
316
+
317
+ return result;
318
+ }
319
+
320
+ function showSandboxHelp() {
321
+ console.log(`
322
+ ${c.bright('CloudCLI Sandbox')} — Run CloudCLI inside Docker Sandboxes
323
+
324
+ Usage:
325
+ web-code sandbox <workspace> Create and start a sandbox
326
+ web-code sandbox <subcommand> [name] Manage sandboxes
327
+
328
+ Subcommands:
329
+ ${c.bright('(default)')} Create a sandbox and start the web UI
330
+ ${c.bright('ls')} List all sandboxes
331
+ ${c.bright('start')} Restart a stopped sandbox and re-launch the web UI
332
+ ${c.bright('stop')} Stop a sandbox (preserves state)
333
+ ${c.bright('rm')} Remove a sandbox
334
+ ${c.bright('logs')} Show CloudCLI server logs
335
+ ${c.bright('help')} Show this help
336
+
337
+ Options:
338
+ -a, --agent <agent> Agent to use: claude, codex, gemini (default: claude)
339
+ -n, --name <name> Sandbox name (default: derived from workspace folder)
340
+ -t, --template <image> Custom template image
341
+ -e, --env <KEY=VALUE> Set environment variable (repeatable)
342
+ --port <port> Host port for the web UI (default: 3001)
343
+
344
+ Examples:
345
+ $ web-code sandbox ~/my-project
346
+ $ web-code sandbox ~/my-project --agent codex --port 8080
347
+ $ web-code sandbox ~/my-project --env SERVER_PORT=8080 --env HOST=0.0.0.0
348
+ $ web-code sandbox ls
349
+ $ web-code sandbox stop my-project
350
+ $ web-code sandbox start my-project
351
+ $ web-code sandbox rm my-project
352
+
353
+ Prerequisites:
354
+ 1. Install sbx CLI: https://docs.docker.com/ai/sandboxes/get-started/
355
+ 2. Authenticate and store your API key:
356
+ sbx login
357
+ sbx secret set -g anthropic # for Claude
358
+ sbx secret set -g openai # for Codex
359
+ sbx secret set -g google # for Gemini
360
+
361
+ Advanced usage:
362
+ For branch mode, multiple workspaces, memory limits, network policies,
363
+ or passing prompts to the agent, use sbx directly with the template:
364
+
365
+ sbx run --template docker.io/cloudcliai/sandbox:claude-code claude ~/my-project --branch my-feature
366
+ sbx run --template docker.io/cloudcliai/sandbox:claude-code claude ~/project ~/libs:ro --memory 8g
367
+
368
+ Full Docker Sandboxes docs: https://docs.docker.com/ai/sandboxes/usage/
369
+ `);
370
+ }
371
+
372
+ async function sandboxCommand(args) {
373
+ const { execFileSync, spawn: spawnProcess } = await import('child_process');
374
+
375
+ // Safe execution — uses execFileSync (no shell) to prevent injection
376
+ const sbx = (subcmd, opts = {}) => {
377
+ const result = execFileSync('sbx', subcmd, {
378
+ encoding: 'utf8',
379
+ stdio: opts.inherit ? 'inherit' : 'pipe',
380
+ });
381
+ return result || '';
382
+ };
383
+
384
+ const opts = parseSandboxArgs(args);
385
+
386
+ if (opts.subcommand === 'help') {
387
+ showSandboxHelp();
388
+ return;
389
+ }
390
+
391
+ // Validate name (alphanumeric, hyphens, underscores only)
392
+ if (opts.name && !/^[\w-]+$/.test(opts.name)) {
393
+ console.error(`\n${c.error('❌')} Invalid sandbox name: ${opts.name}`);
394
+ console.log(` Names may only contain letters, numbers, hyphens, and underscores.\n`);
395
+ process.exit(1);
396
+ }
397
+
398
+ // Check sbx is installed
399
+ try {
400
+ sbx(['version']);
401
+ } catch {
402
+ console.error(`\n${c.error('❌')} ${c.bright('sbx')} CLI not found.\n`);
403
+ console.log(` Install it from: ${c.info('https://docs.docker.com/ai/sandboxes/get-started/')}`);
404
+ console.log(` Then run: ${c.bright('sbx login')}`);
405
+ console.log(` And store your API key: ${c.bright('sbx secret set -g anthropic')}\n`);
406
+ process.exit(1);
407
+ }
408
+
409
+ switch (opts.subcommand) {
410
+
411
+ case 'ls':
412
+ sbx(['ls'], { inherit: true });
413
+ break;
414
+
415
+ case 'stop':
416
+ if (!opts.name) {
417
+ console.error(`\n${c.error('❌')} Sandbox name required: web-code sandbox stop <name>\n`);
418
+ process.exit(1);
419
+ }
420
+ sbx(['stop', opts.name], { inherit: true });
421
+ break;
422
+
423
+ case 'rm':
424
+ if (!opts.name) {
425
+ console.error(`\n${c.error('❌')} Sandbox name required: web-code sandbox rm <name>\n`);
426
+ process.exit(1);
427
+ }
428
+ sbx(['rm', opts.name], { inherit: true });
429
+ break;
430
+
431
+ case 'logs':
432
+ if (!opts.name) {
433
+ console.error(`\n${c.error('❌')} Sandbox name required: web-code sandbox logs <name>\n`);
434
+ process.exit(1);
435
+ }
436
+ try {
437
+ sbx(['exec', opts.name, 'bash', '-c', 'cat /tmp/cloudcli-ui.log'], { inherit: true });
438
+ } catch (e) {
439
+ console.error(`\n${c.error('❌')} Could not read logs: ${e.message || 'Is the sandbox running?'}\n`);
440
+ }
441
+ break;
442
+
443
+ case 'start': {
444
+ if (!opts.name) {
445
+ console.error(`\n${c.error('❌')} Sandbox name required: web-code sandbox start <name>\n`);
446
+ process.exit(1);
447
+ }
448
+ console.log(`\n${c.info('▶')} Starting sandbox ${c.bright(opts.name)}...`);
449
+ const restartRun = spawnProcess('sbx', ['run', opts.name], {
450
+ detached: true,
451
+ stdio: ['ignore', 'ignore', 'ignore'],
452
+ });
453
+ restartRun.unref();
454
+ await new Promise(resolve => setTimeout(resolve, 5000));
455
+
456
+ console.log(`${c.info('▶')} Launching CloudCLI web server...`);
457
+ sbx(['exec', opts.name, 'bash', '-c', 'web-code start --port 3001 &']);
458
+
459
+ console.log(`${c.info('▶')} Forwarding port ${opts.port} → 3001...`);
460
+ try {
461
+ sbx(['ports', opts.name, '--publish', `${opts.port}:3001`]);
462
+ } catch (e) {
463
+ const msg = e.stdout || e.stderr || e.message || '';
464
+ if (msg.includes('address already in use')) {
465
+ const altPort = opts.port + 1;
466
+ console.log(`${c.warn('⚠')} Port ${opts.port} in use, trying ${altPort}...`);
467
+ try {
468
+ sbx(['ports', opts.name, '--publish', `${altPort}:3001`]);
469
+ opts.port = altPort;
470
+ } catch {
471
+ console.error(`${c.error('❌')} Ports ${opts.port} and ${altPort} both in use. Use --port to specify a free port.`);
472
+ process.exit(1);
473
+ }
474
+ } else {
475
+ throw e;
476
+ }
477
+ }
478
+
479
+ console.log(`\n${c.ok('✔')} ${c.bright('CloudCLI is ready!')}`);
480
+ console.log(` ${c.info('→')} ${c.bright(`http://localhost:${opts.port}`)}\n`);
481
+ break;
482
+ }
483
+
484
+ case 'create': {
485
+ if (!opts.workspace) {
486
+ console.error(`\n${c.error('❌')} Workspace path required: web-code sandbox <path>\n`);
487
+ console.log(` Example: ${c.bright('web-code sandbox ~/my-project')}\n`);
488
+ process.exit(1);
489
+ }
490
+
491
+ const workspace = opts.workspace.startsWith('~')
492
+ ? opts.workspace.replace(/^~/, os.homedir())
493
+ : path.resolve(opts.workspace);
494
+
495
+ if (!fs.existsSync(workspace)) {
496
+ console.error(`\n${c.error('❌')} Workspace path not found: ${c.dim(workspace)}\n`);
497
+ process.exit(1);
498
+ }
499
+
500
+ const secret = SANDBOX_SECRETS[opts.agent] || 'anthropic';
501
+
502
+ // Check if the required secret is stored
503
+ try {
504
+ const secretList = sbx(['secret', 'ls']);
505
+ if (!secretList.includes(secret)) {
506
+ console.error(`\n${c.error('❌')} No ${c.bright(secret)} API key found.\n`);
507
+ console.log(` Run: ${c.bright(`sbx secret set -g ${secret}`)}\n`);
508
+ process.exit(1);
509
+ }
510
+ } catch { /* sbx secret ls not available, skip check */ }
511
+
512
+ console.log(`\n${c.bright('CloudCLI Sandbox')}`);
513
+ console.log(c.dim('─'.repeat(50)));
514
+ console.log(` Agent: ${c.info(opts.agent)} ${c.dim(`(${secret} credentials)`)}`);
515
+ console.log(` Workspace: ${c.dim(workspace)}`);
516
+ console.log(` Name: ${c.dim(opts.name)}`);
517
+ console.log(` Template: ${c.dim(opts.template)}`);
518
+ console.log(` Port: ${c.dim(String(opts.port))}`);
519
+ if (opts.env.length > 0) {
520
+ console.log(` Env: ${c.dim(opts.env.join(', '))}`);
521
+ }
522
+ console.log(c.dim('─'.repeat(50)));
523
+
524
+ // Step 1: Launch sandbox with sbx run in background.
525
+ // sbx run creates the sandbox (or reconnects) AND holds an active session,
526
+ // which prevents the sandbox from auto-stopping.
527
+ console.log(`\n${c.info('▶')} Creating sandbox ${c.bright(opts.name)}...`);
528
+ const bgRun = spawnProcess('sbx', [
529
+ 'run', '--template', opts.template, '--name', opts.name, opts.agent, workspace,
530
+ ], {
531
+ detached: true,
532
+ stdio: ['ignore', 'ignore', 'ignore'],
533
+ });
534
+ bgRun.unref();
535
+ // Wait for sandbox to be ready
536
+ await new Promise(resolve => setTimeout(resolve, 5000));
537
+
538
+ // Step 2: Inject environment variables
539
+ if (opts.env.length > 0) {
540
+ console.log(`${c.info('▶')} Setting environment variables...`);
541
+ const exports = opts.env
542
+ .filter(e => /^\w+=.+$/.test(e))
543
+ .map(e => `export ${e}`)
544
+ .join('\n');
545
+ if (exports) {
546
+ sbx(['exec', opts.name, 'bash', '-c', `echo '${exports}' >> /etc/sandbox-persistent.sh`]);
547
+ }
548
+ const invalid = opts.env.filter(e => !/^\w+=.+$/.test(e));
549
+ if (invalid.length > 0) {
550
+ console.log(`${c.warn('⚠')} Skipped invalid env vars: ${invalid.join(', ')} (expected KEY=VALUE)`);
551
+ }
552
+ }
553
+
554
+ // Step 3: Start CloudCLI inside the sandbox
555
+ console.log(`${c.info('▶')} Launching CloudCLI web server...`);
556
+ sbx(['exec', opts.name, 'bash', '-c', 'cloudcli start --port 3001 &']);
557
+
558
+ // Step 4: Forward port
559
+ console.log(`${c.info('▶')} Forwarding port ${opts.port} → 3001...`);
560
+ try {
561
+ sbx(['ports', opts.name, '--publish', `${opts.port}:3001`]);
562
+ } catch (e) {
563
+ const msg = e.stdout || e.stderr || e.message || '';
564
+ if (msg.includes('address already in use')) {
565
+ const altPort = opts.port + 1;
566
+ console.log(`${c.warn('⚠')} Port ${opts.port} in use, trying ${altPort}...`);
567
+ try {
568
+ sbx(['ports', opts.name, '--publish', `${altPort}:3001`]);
569
+ opts.port = altPort;
570
+ } catch {
571
+ console.error(`${c.error('❌')} Ports ${opts.port} and ${altPort} both in use. Use --port to specify a free port.`);
572
+ process.exit(1);
573
+ }
574
+ } else {
575
+ throw e;
576
+ }
577
+ }
578
+
579
+ // Done
580
+ console.log(`\n${c.ok('✔')} ${c.bright('CloudCLI is ready!')}`);
581
+ console.log(` ${c.info('→')} Open ${c.bright(`http://localhost:${opts.port}`)}`);
582
+ console.log(`\n${c.dim(' Manage with:')}`);
583
+ console.log(` ${c.dim('$')} sbx ls`);
584
+ console.log(` ${c.dim('$')} sbx stop ${opts.name}`);
585
+ console.log(` ${c.dim('$')} sbx start ${opts.name}`);
586
+ console.log(` ${c.dim('$')} sbx rm ${opts.name}`);
587
+ console.log(`\n${c.dim(' Or install globally:')} npm install -g @glwhappen/web-code\n`);
588
+ break;
589
+ }
590
+
591
+ default:
592
+ showSandboxHelp();
593
+ }
594
+ }
595
+
596
+ // ── Server ──────────────────────────────────────────────────
597
+
598
+ // Start the server
599
+ async function startServer() {
600
+ // Check for updates silently on startup
601
+ checkForUpdates(true);
602
+
603
+ // Import and run the server
604
+ await import('./index.js');
605
+ }
606
+
607
+ // Parse CLI arguments
608
+ function parseArgs(args) {
609
+ const parsed = { command: 'start', options: {} };
610
+
611
+ for (let i = 0; i < args.length; i++) {
612
+ const arg = args[i];
613
+
614
+ if (arg === '--port' || arg === '-p') {
615
+ parsed.options.serverPort = args[++i];
616
+ } else if (arg.startsWith('--port=')) {
617
+ parsed.options.serverPort = arg.split('=')[1];
618
+ } else if (arg === '--database-path') {
619
+ parsed.options.databasePath = args[++i];
620
+ } else if (arg.startsWith('--database-path=')) {
621
+ parsed.options.databasePath = arg.split('=')[1];
622
+ } else if (arg === '--help' || arg === '-h') {
623
+ parsed.command = 'help';
624
+ } else if (arg === '--version' || arg === '-v') {
625
+ parsed.command = 'version';
626
+ } else if (!arg.startsWith('-')) {
627
+ parsed.command = arg;
628
+ if (arg === 'sandbox') {
629
+ parsed.remainingArgs = args.slice(i + 1);
630
+ break;
631
+ }
632
+ }
633
+ }
634
+
635
+ return parsed;
636
+ }
637
+
638
+ // Main CLI handler
639
+ async function main() {
640
+ const args = process.argv.slice(2);
641
+ const { command, options, remainingArgs } = parseArgs(args);
642
+
643
+ // Apply CLI options to environment variables
644
+ if (options.serverPort) {
645
+ process.env.SERVER_PORT = options.serverPort;
646
+ } else if (!process.env.SERVER_PORT && process.env.PORT) {
647
+ process.env.SERVER_PORT = process.env.PORT;
648
+ }
649
+ if (options.databasePath) {
650
+ process.env.DATABASE_PATH = options.databasePath;
651
+ }
652
+
653
+ switch (command) {
654
+ case 'start':
655
+ await startServer();
656
+ break;
657
+ case 'sandbox':
658
+ await sandboxCommand(remainingArgs || []);
659
+ break;
660
+ case 'status':
661
+ case 'info':
662
+ showStatus();
663
+ break;
664
+ case 'help':
665
+ case '-h':
666
+ case '--help':
667
+ showHelp();
668
+ break;
669
+ case 'version':
670
+ case '-v':
671
+ case '--version':
672
+ showVersion();
673
+ break;
674
+ case 'update':
675
+ await updatePackage();
676
+ break;
677
+ default:
678
+ console.error(`\n❌ Unknown command: ${command}`);
679
+ console.log(' Run "cloudcli help" for usage information.\n');
680
+ process.exit(1);
681
+ }
682
+ }
683
+
684
+ // Run the CLI
685
+ main().catch(error => {
686
+ console.error('\n❌ Error:', error.message);
687
+ process.exit(1);
688
+ });
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Environment Flag: Is Platform
3
+ * Indicates if the app is running in Platform mode (hosted) or OSS mode (self-hosted)
4
+ */
5
+ export const IS_PLATFORM = process.env.VITE_IS_PLATFORM === 'true';