@thevinci/web 1.0.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 (337) hide show
  1. package/README.md +197 -0
  2. package/bin/cli-entry.js +55 -0
  3. package/bin/cli-output.js +145 -0
  4. package/bin/cli.js +4887 -0
  5. package/bin/cli.test.js +64 -0
  6. package/dist/apple-touch-icon-120x120.png +0 -0
  7. package/dist/apple-touch-icon-152x152.png +0 -0
  8. package/dist/apple-touch-icon-167x167.png +0 -0
  9. package/dist/apple-touch-icon-180x180.png +0 -0
  10. package/dist/apple-touch-icon.png +0 -0
  11. package/dist/apple-touch-icon.svg +528 -0
  12. package/dist/assets/JsonTreeView-CSm9OzXG.js +1 -0
  13. package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  14. package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  15. package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  16. package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  17. package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  18. package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  19. package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  20. package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  21. package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  22. package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  23. package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  24. package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  25. package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  26. package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  27. package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  28. package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  29. package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  30. package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  31. package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  32. package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  33. package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  34. package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  35. package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  36. package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  37. package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  38. package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  39. package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  40. package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  41. package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  42. package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  43. package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  44. package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  45. package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  46. package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  47. package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  48. package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  49. package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  50. package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  51. package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  52. package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  53. package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  54. package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  55. package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  56. package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  57. package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  58. package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  59. package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  60. package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  61. package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  62. package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  63. package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  64. package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  65. package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  66. package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  67. package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  68. package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  69. package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  70. package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  71. package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  72. package/dist/assets/MarkdownRendererImpl-DensKOLc.js +6 -0
  73. package/dist/assets/MultiRunWindow-Bo7THayo.js +1 -0
  74. package/dist/assets/OnboardingScreen-BDqmzTVR.js +2 -0
  75. package/dist/assets/SettingsWindow-coz__Ykw.js +1 -0
  76. package/dist/assets/TerminalView-DrZ-i3Dr.js +1 -0
  77. package/dist/assets/ToolOutputDialog-Eglzslt3.js +16 -0
  78. package/dist/assets/es-4o9ciP61.js +15 -0
  79. package/dist/assets/ibm-plex-mono-latin-400-normal-CvHOgSBP.woff +0 -0
  80. package/dist/assets/ibm-plex-mono-latin-400-normal-DMJ8VG8y.woff2 +0 -0
  81. package/dist/assets/ibm-plex-mono-latin-500-normal-CB9ihrfo.woff +0 -0
  82. package/dist/assets/ibm-plex-mono-latin-500-normal-DSY6xOcd.woff2 +0 -0
  83. package/dist/assets/ibm-plex-mono-latin-600-normal-BgSNZQsw.woff2 +0 -0
  84. package/dist/assets/ibm-plex-mono-latin-600-normal-DWFSQ4vo.woff +0 -0
  85. package/dist/assets/ibm-plex-sans-latin-400-normal-CDDApCn2.woff2 +0 -0
  86. package/dist/assets/ibm-plex-sans-latin-400-normal-CYLoc0-x.woff +0 -0
  87. package/dist/assets/ibm-plex-sans-latin-500-normal-6ng42L7E.woff2 +0 -0
  88. package/dist/assets/ibm-plex-sans-latin-500-normal-BgVn5rGT.woff +0 -0
  89. package/dist/assets/ibm-plex-sans-latin-600-normal-Cu4Hd6ag.woff +0 -0
  90. package/dist/assets/ibm-plex-sans-latin-600-normal-CuJfVYMP.woff2 +0 -0
  91. package/dist/assets/index-DLTDToSP.css +1 -0
  92. package/dist/assets/index-DgiFEKGN.js +1 -0
  93. package/dist/assets/ko-B20imCHE.js +15 -0
  94. package/dist/assets/main-BV3KOtdA.css +1 -0
  95. package/dist/assets/main-CDKJj0sH.js +226 -0
  96. package/dist/assets/main-LC-PSNVM.js +2 -0
  97. package/dist/assets/miniChat-CQUiG_cr.js +2 -0
  98. package/dist/assets/modelPrefsAutoSave-Dm799vzR.js +6986 -0
  99. package/dist/assets/pl-DQJ7LSzj.js +15 -0
  100. package/dist/assets/pt-BR-OmjHUz9y.js +15 -0
  101. package/dist/assets/renderElectronMiniChatApp-CARbeW0G.js +2 -0
  102. package/dist/assets/uk-BNFxOlO4.js +15 -0
  103. package/dist/assets/vendor--DBfsbEis.css +1 -0
  104. package/dist/assets/vendor-.bun-B9l0ZNi2.js +4094 -0
  105. package/dist/assets/wasm-CG6Dc4jp.js +1 -0
  106. package/dist/assets/wasmSttWorker-Dtlxac_K.js +1 -0
  107. package/dist/assets/wasmSttWorker-oo7Dm_jy.js +1806 -0
  108. package/dist/assets/worker-CbT6TVo7.js +155 -0
  109. package/dist/assets/zh-CN-C6T-Ac7F.js +15 -0
  110. package/dist/favicon-16.png +0 -0
  111. package/dist/favicon-32.png +0 -0
  112. package/dist/favicon.png +0 -0
  113. package/dist/favicon.svg +528 -0
  114. package/dist/index.html +607 -0
  115. package/dist/logo-dark-192x192.png +0 -0
  116. package/dist/logo-dark-512x512.png +0 -0
  117. package/dist/logo-dark-512x512.svg +528 -0
  118. package/dist/logo-light-192x192.png +0 -0
  119. package/dist/logo-light-512x512.png +0 -0
  120. package/dist/logo-light-512x512.svg +528 -0
  121. package/dist/mini-chat.html +16 -0
  122. package/dist/pwa-192.png +0 -0
  123. package/dist/pwa-512.png +0 -0
  124. package/dist/pwa-maskable-192.png +0 -0
  125. package/dist/pwa-maskable-512.png +0 -0
  126. package/dist/site.webmanifest +21 -0
  127. package/dist/sw.js +1 -0
  128. package/package.json +118 -0
  129. package/public/apple-touch-icon-120x120.png +0 -0
  130. package/public/apple-touch-icon-152x152.png +0 -0
  131. package/public/apple-touch-icon-167x167.png +0 -0
  132. package/public/apple-touch-icon-180x180.png +0 -0
  133. package/public/apple-touch-icon.png +0 -0
  134. package/public/apple-touch-icon.svg +528 -0
  135. package/public/favicon-16.png +0 -0
  136. package/public/favicon-32.png +0 -0
  137. package/public/favicon.png +0 -0
  138. package/public/favicon.svg +528 -0
  139. package/public/logo-dark-192x192.png +0 -0
  140. package/public/logo-dark-512x512.png +0 -0
  141. package/public/logo-dark-512x512.svg +528 -0
  142. package/public/logo-light-192x192.png +0 -0
  143. package/public/logo-light-512x512.png +0 -0
  144. package/public/logo-light-512x512.svg +528 -0
  145. package/public/pwa-192.png +0 -0
  146. package/public/pwa-512.png +0 -0
  147. package/public/pwa-maskable-192.png +0 -0
  148. package/public/pwa-maskable-512.png +0 -0
  149. package/public/site.webmanifest +21 -0
  150. package/server/TERMINAL_WS_PROTOCOL.md +48 -0
  151. package/server/index.d.ts +39 -0
  152. package/server/index.js +1311 -0
  153. package/server/lib/cloudflare-tunnel.js +650 -0
  154. package/server/lib/event-stream/DOCUMENTATION.md +61 -0
  155. package/server/lib/event-stream/directory-ws-bridge.js +185 -0
  156. package/server/lib/event-stream/global-hub.js +158 -0
  157. package/server/lib/event-stream/global-hub.test.js +140 -0
  158. package/server/lib/event-stream/global-ws-bridge.js +206 -0
  159. package/server/lib/event-stream/index.js +25 -0
  160. package/server/lib/event-stream/protocol.js +131 -0
  161. package/server/lib/event-stream/protocol.test.js +182 -0
  162. package/server/lib/event-stream/runtime.js +180 -0
  163. package/server/lib/event-stream/runtime.test.js +512 -0
  164. package/server/lib/event-stream/upstream-reader.js +226 -0
  165. package/server/lib/event-stream/upstream-reader.test.js +276 -0
  166. package/server/lib/fs/DOCUMENTATION.md +36 -0
  167. package/server/lib/fs/routes.js +1040 -0
  168. package/server/lib/fs/search.js +238 -0
  169. package/server/lib/git/DOCUMENTATION.md +152 -0
  170. package/server/lib/git/credentials.js +74 -0
  171. package/server/lib/git/identity-storage.js +112 -0
  172. package/server/lib/git/index.js +6 -0
  173. package/server/lib/git/routes.js +972 -0
  174. package/server/lib/git/service.js +3432 -0
  175. package/server/lib/git/service.test.js +39 -0
  176. package/server/lib/github/DOCUMENTATION.md +171 -0
  177. package/server/lib/github/auth.js +307 -0
  178. package/server/lib/github/device-flow.js +50 -0
  179. package/server/lib/github/index.js +24 -0
  180. package/server/lib/github/octokit.js +10 -0
  181. package/server/lib/github/pr-status.js +519 -0
  182. package/server/lib/github/repo/fork-detection.js +102 -0
  183. package/server/lib/github/repo/index.js +55 -0
  184. package/server/lib/github/routes.js +1560 -0
  185. package/server/lib/magic-prompts/routes.js +63 -0
  186. package/server/lib/magic-prompts/runtime.js +119 -0
  187. package/server/lib/notifications/DOCUMENTATION.md +122 -0
  188. package/server/lib/notifications/emitter-runtime.js +102 -0
  189. package/server/lib/notifications/index.js +4 -0
  190. package/server/lib/notifications/message.js +52 -0
  191. package/server/lib/notifications/message.test.js +34 -0
  192. package/server/lib/notifications/push-runtime.js +304 -0
  193. package/server/lib/notifications/routes.js +315 -0
  194. package/server/lib/notifications/runtime.js +566 -0
  195. package/server/lib/notifications/template-runtime.js +349 -0
  196. package/server/lib/notifications/template-runtime.test.js +26 -0
  197. package/server/lib/opencode/DOCUMENTATION.md +362 -0
  198. package/server/lib/opencode/agents.js +634 -0
  199. package/server/lib/opencode/auth-state-runtime.js +88 -0
  200. package/server/lib/opencode/auth.js +83 -0
  201. package/server/lib/opencode/bootstrap-runtime.js +131 -0
  202. package/server/lib/opencode/cli-entry-runtime.js +43 -0
  203. package/server/lib/opencode/cli-options.js +128 -0
  204. package/server/lib/opencode/commands.js +339 -0
  205. package/server/lib/opencode/config-entity-routes.js +370 -0
  206. package/server/lib/opencode/core-routes.js +500 -0
  207. package/server/lib/opencode/core-routes.test.js +26 -0
  208. package/server/lib/opencode/env-config.js +74 -0
  209. package/server/lib/opencode/env-keys.js +68 -0
  210. package/server/lib/opencode/env-runtime.js +1162 -0
  211. package/server/lib/opencode/env-runtime.test.js +116 -0
  212. package/server/lib/opencode/feature-routes-runtime.js +244 -0
  213. package/server/lib/opencode/hmr-state-runtime.js +85 -0
  214. package/server/lib/opencode/index.js +66 -0
  215. package/server/lib/opencode/lifecycle.js +1019 -0
  216. package/server/lib/opencode/lifecycle.test.js +240 -0
  217. package/server/lib/opencode/mcp.js +278 -0
  218. package/server/lib/opencode/network-runtime.js +104 -0
  219. package/server/lib/opencode/network-runtime.test.js +37 -0
  220. package/server/lib/opencode/opencode-resolution-runtime.js +71 -0
  221. package/server/lib/opencode/path-utils.js +100 -0
  222. package/server/lib/opencode/path-utils.test.js +71 -0
  223. package/server/lib/opencode/project-directory-runtime.js +124 -0
  224. package/server/lib/opencode/project-icon-routes.js +399 -0
  225. package/server/lib/opencode/project-icon-routes.test.js +107 -0
  226. package/server/lib/opencode/providers.js +96 -0
  227. package/server/lib/opencode/proxy.js +445 -0
  228. package/server/lib/opencode/pwa-manifest-routes.js +257 -0
  229. package/server/lib/opencode/pwa-manifest-routes.test.js +133 -0
  230. package/server/lib/opencode/routes.js +541 -0
  231. package/server/lib/opencode/server-startup-runtime.js +156 -0
  232. package/server/lib/opencode/server-utils-runtime.js +168 -0
  233. package/server/lib/opencode/server-utils-runtime.test.js +135 -0
  234. package/server/lib/opencode/session-runtime.js +356 -0
  235. package/server/lib/opencode/session-runtime.test.js +151 -0
  236. package/server/lib/opencode/settings-helpers.js +770 -0
  237. package/server/lib/opencode/settings-helpers.test.js +109 -0
  238. package/server/lib/opencode/settings-normalization-runtime.js +428 -0
  239. package/server/lib/opencode/settings-runtime.js +826 -0
  240. package/server/lib/opencode/settings-runtime.test.js +85 -0
  241. package/server/lib/opencode/shared.js +615 -0
  242. package/server/lib/opencode/shutdown-runtime.js +139 -0
  243. package/server/lib/opencode/shutdown-runtime.test.js +58 -0
  244. package/server/lib/opencode/skill-routes.js +701 -0
  245. package/server/lib/opencode/skills.js +548 -0
  246. package/server/lib/opencode/startup-pipeline-runtime.js +130 -0
  247. package/server/lib/opencode/static-routes-runtime.js +65 -0
  248. package/server/lib/opencode/theme-runtime.js +167 -0
  249. package/server/lib/opencode/tunnel-auth.js +591 -0
  250. package/server/lib/opencode/tunnel-wiring-runtime.js +94 -0
  251. package/server/lib/opencode/vinci-routes.js +76 -0
  252. package/server/lib/opencode/watcher.js +115 -0
  253. package/server/lib/opencode/watcher.test.js +239 -0
  254. package/server/lib/preview/proxy-runtime.js +1333 -0
  255. package/server/lib/preview/proxy-runtime.test.js +144 -0
  256. package/server/lib/projects/project-config.js +567 -0
  257. package/server/lib/projects/project-config.test.js +175 -0
  258. package/server/lib/projects/project-id.js +13 -0
  259. package/server/lib/quota/DOCUMENTATION.md +58 -0
  260. package/server/lib/quota/index.js +25 -0
  261. package/server/lib/quota/providers/claude.js +107 -0
  262. package/server/lib/quota/providers/codex.js +113 -0
  263. package/server/lib/quota/providers/copilot.js +165 -0
  264. package/server/lib/quota/providers/google/api.js +92 -0
  265. package/server/lib/quota/providers/google/auth.js +108 -0
  266. package/server/lib/quota/providers/google/index.js +124 -0
  267. package/server/lib/quota/providers/google/transforms.js +109 -0
  268. package/server/lib/quota/providers/index.js +168 -0
  269. package/server/lib/quota/providers/interface.js +55 -0
  270. package/server/lib/quota/providers/kimi.js +108 -0
  271. package/server/lib/quota/providers/minimax-cn-coding-plan.js +140 -0
  272. package/server/lib/quota/providers/minimax-coding-plan.js +139 -0
  273. package/server/lib/quota/providers/nanogpt.js +124 -0
  274. package/server/lib/quota/providers/ollama-cloud.js +112 -0
  275. package/server/lib/quota/providers/openai.js +91 -0
  276. package/server/lib/quota/providers/openrouter.js +92 -0
  277. package/server/lib/quota/providers/zai.js +91 -0
  278. package/server/lib/quota/providers/zhipuai-coding-plan.js +133 -0
  279. package/server/lib/quota/providers/zhipuai.js +114 -0
  280. package/server/lib/quota/routes.js +27 -0
  281. package/server/lib/quota/utils/auth.js +50 -0
  282. package/server/lib/quota/utils/formatters.js +85 -0
  283. package/server/lib/quota/utils/formatters.test.js +54 -0
  284. package/server/lib/quota/utils/index.js +10 -0
  285. package/server/lib/quota/utils/transformers.js +55 -0
  286. package/server/lib/scheduled-tasks/DOCUMENTATION.md +44 -0
  287. package/server/lib/scheduled-tasks/routes.js +235 -0
  288. package/server/lib/scheduled-tasks/runtime.js +773 -0
  289. package/server/lib/scheduled-tasks/runtime.test.js +100 -0
  290. package/server/lib/security/request-security.js +115 -0
  291. package/server/lib/session-folders/routes.js +63 -0
  292. package/server/lib/session-folders/routes.test.js +102 -0
  293. package/server/lib/skills-catalog/DOCUMENTATION.md +178 -0
  294. package/server/lib/skills-catalog/cache.js +29 -0
  295. package/server/lib/skills-catalog/clawdhub/api.js +158 -0
  296. package/server/lib/skills-catalog/clawdhub/index.js +30 -0
  297. package/server/lib/skills-catalog/clawdhub/install.js +238 -0
  298. package/server/lib/skills-catalog/clawdhub/scan.js +113 -0
  299. package/server/lib/skills-catalog/curated-sources.js +21 -0
  300. package/server/lib/skills-catalog/git.js +77 -0
  301. package/server/lib/skills-catalog/index.js +42 -0
  302. package/server/lib/skills-catalog/install.js +294 -0
  303. package/server/lib/skills-catalog/scan.js +221 -0
  304. package/server/lib/skills-catalog/source.js +87 -0
  305. package/server/lib/terminal/DOCUMENTATION.md +76 -0
  306. package/server/lib/terminal/index.js +31 -0
  307. package/server/lib/terminal/output-replay-buffer.js +78 -0
  308. package/server/lib/terminal/output-replay-buffer.test.js +75 -0
  309. package/server/lib/terminal/runtime.js +850 -0
  310. package/server/lib/terminal/runtime.test.js +96 -0
  311. package/server/lib/terminal/terminal-ws-protocol.js +68 -0
  312. package/server/lib/terminal/terminal-ws-protocol.test.js +145 -0
  313. package/server/lib/text/DOCUMENTATION.md +35 -0
  314. package/server/lib/text/summarization.js +138 -0
  315. package/server/lib/text/summarization.test.js +34 -0
  316. package/server/lib/tts/DOCUMENTATION.md +146 -0
  317. package/server/lib/tts/base-url.js +62 -0
  318. package/server/lib/tts/capability-runtime.js +31 -0
  319. package/server/lib/tts/index.js +19 -0
  320. package/server/lib/tts/routes.js +261 -0
  321. package/server/lib/tts/routes.test.js +53 -0
  322. package/server/lib/tts/service.js +178 -0
  323. package/server/lib/tts/stt.js +75 -0
  324. package/server/lib/tunnels/DOCUMENTATION.md +18 -0
  325. package/server/lib/tunnels/index.js +166 -0
  326. package/server/lib/tunnels/managed-config.js +201 -0
  327. package/server/lib/tunnels/providers/cloudflare.js +260 -0
  328. package/server/lib/tunnels/registry.js +51 -0
  329. package/server/lib/tunnels/routes.js +605 -0
  330. package/server/lib/tunnels/types.js +219 -0
  331. package/server/lib/ui-auth/DOCUMENTATION.md +38 -0
  332. package/server/lib/ui-auth/ui-auth.js +673 -0
  333. package/server/lib/ui-auth/ui-passkeys.js +545 -0
  334. package/server/opencode-proxy.test.js +151 -0
  335. package/server/proxy-headers.js +61 -0
  336. package/server/proxy-headers.test.js +58 -0
  337. package/server/sse-routes.test.js +152 -0
@@ -0,0 +1,133 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { registerPwaManifestRoute } from './pwa-manifest-routes.js';
3
+
4
+ const createResponse = () => ({
5
+ headers: new Map(),
6
+ contentType: '',
7
+ body: '',
8
+ setHeader(name, value) {
9
+ this.headers.set(name, value);
10
+ return this;
11
+ },
12
+ type(value) {
13
+ this.contentType = value;
14
+ return this;
15
+ },
16
+ send(value) {
17
+ this.body = value;
18
+ return this;
19
+ },
20
+ });
21
+
22
+ describe('PWA manifest route', () => {
23
+ it('does not fall back to unrelated global session shortcuts for scoped manifests', async () => {
24
+ const routes = new Map();
25
+ const app = {
26
+ get(route, handler) {
27
+ routes.set(route, handler);
28
+ },
29
+ };
30
+ const originalFetch = globalThis.fetch;
31
+ const fetchCalls = [];
32
+ globalThis.fetch = async (url) => {
33
+ fetchCalls.push(String(url));
34
+ const sessions = String(url).includes('?directory=')
35
+ ? []
36
+ : [
37
+ {
38
+ id: 'other-session',
39
+ title: 'Other project',
40
+ directory: '/workspace/other',
41
+ time: { updated: 2 },
42
+ },
43
+ ];
44
+ return {
45
+ ok: true,
46
+ json: async () => sessions,
47
+ };
48
+ };
49
+
50
+ try {
51
+ registerPwaManifestRoute(app, {
52
+ process: { platform: 'darwin' },
53
+ resolveProjectDirectory: async () => ({ directory: '/workspace/app' }),
54
+ buildOpenCodeUrl: (route) => route,
55
+ getOpenCodeAuthHeaders: () => ({}),
56
+ readSettingsFromDiskMigrated: async () => ({}),
57
+ normalizePwaAppName: (value, fallback) => typeof value === 'string' && value.trim() ? value.trim() : fallback,
58
+ normalizePwaOrientation: (value, fallback) => typeof value === 'string' && value.trim() ? value.trim() : fallback,
59
+ });
60
+
61
+ const handler = routes.get('/manifest.webmanifest');
62
+ const res = createResponse();
63
+ await handler({ query: {} }, res);
64
+
65
+ const manifest = JSON.parse(res.body);
66
+ expect(fetchCalls).toHaveLength(2);
67
+ expect(manifest.shortcuts).toEqual([
68
+ {
69
+ name: 'Appearance Settings',
70
+ short_name: 'Settings',
71
+ description: 'Open appearance settings',
72
+ url: '/?settings=appearance',
73
+ icons: [{ src: '/pwa-192.png', sizes: '192x192', type: 'image/png' }],
74
+ },
75
+ ]);
76
+ } finally {
77
+ globalThis.fetch = originalFetch;
78
+ }
79
+ });
80
+
81
+ it('includes child session shortcuts for root-scoped manifests', async () => {
82
+ const routes = new Map();
83
+ const app = {
84
+ get(route, handler) {
85
+ routes.set(route, handler);
86
+ },
87
+ };
88
+ const originalFetch = globalThis.fetch;
89
+ const fetchCalls = [];
90
+ globalThis.fetch = async (url) => {
91
+ fetchCalls.push(String(url));
92
+ return {
93
+ ok: true,
94
+ json: async () => [
95
+ {
96
+ id: 'root-child',
97
+ title: 'Root child',
98
+ directory: '/workspace/app',
99
+ time: { updated: 2 },
100
+ },
101
+ ],
102
+ };
103
+ };
104
+
105
+ try {
106
+ registerPwaManifestRoute(app, {
107
+ process: { platform: 'darwin' },
108
+ resolveProjectDirectory: async () => ({ directory: '/' }),
109
+ buildOpenCodeUrl: (route) => route,
110
+ getOpenCodeAuthHeaders: () => ({}),
111
+ readSettingsFromDiskMigrated: async () => ({}),
112
+ normalizePwaAppName: (value, fallback) => typeof value === 'string' && value.trim() ? value.trim() : fallback,
113
+ normalizePwaOrientation: (value, fallback) => typeof value === 'string' && value.trim() ? value.trim() : fallback,
114
+ });
115
+
116
+ const handler = routes.get('/manifest.webmanifest');
117
+ const res = createResponse();
118
+ await handler({ query: {} }, res);
119
+
120
+ const manifest = JSON.parse(res.body);
121
+ expect(fetchCalls).toEqual(['/session?directory=%2F']);
122
+ expect(manifest.shortcuts).toContainEqual({
123
+ name: 'Root child',
124
+ short_name: 'Root child',
125
+ description: 'Open recent session',
126
+ url: '/?session=root-child',
127
+ icons: [{ src: '/pwa-192.png', sizes: '192x192', type: 'image/png' }],
128
+ });
129
+ } finally {
130
+ globalThis.fetch = originalFetch;
131
+ }
132
+ });
133
+ });
@@ -0,0 +1,541 @@
1
+ import { createProjectIdFromPath } from '../projects/project-id.js';
2
+ import fs from 'fs';
3
+ import os from 'os';
4
+ import path from 'path';
5
+ import { writeEnvKeys } from './env-keys.js';
6
+
7
+ export const registerOpenCodeRoutes = (app, dependencies) => {
8
+ const {
9
+ crypto,
10
+ clientReloadDelayMs,
11
+ getOpenCodeResolutionSnapshot,
12
+ formatSettingsResponse,
13
+ readSettingsFromDisk,
14
+ readSettingsFromDiskMigrated,
15
+ persistSettings,
16
+ sanitizeProjects,
17
+ validateDirectoryPath,
18
+ resolveProjectDirectory,
19
+ getProviderSources,
20
+ removeProviderConfig,
21
+ refreshOpenCodeAfterConfigChange,
22
+ buildOpenCodeUrl,
23
+ getOpenCodeAuthHeaders,
24
+ } = dependencies;
25
+
26
+ let authLibrary = null;
27
+ const pendingMcpAuthContextByState = new Map();
28
+ const PENDING_MCP_AUTH_TTL_MS = 30 * 60 * 1000;
29
+ const getAuthLibrary = async () => {
30
+ if (!authLibrary) {
31
+ authLibrary = await import('./auth.js');
32
+ }
33
+ return authLibrary;
34
+ };
35
+
36
+ const normalizePendingString = (value) => {
37
+ if (typeof value !== 'string') {
38
+ return null;
39
+ }
40
+
41
+ const trimmed = value.trim();
42
+ return trimmed || null;
43
+ };
44
+
45
+ const parseVersionForComparison = (value) => {
46
+ const normalized = String(value || '').replace(/^v/, '').split('+')[0];
47
+ const prereleaseIndex = normalized.indexOf('-');
48
+ const core = prereleaseIndex >= 0 ? normalized.slice(0, prereleaseIndex) : normalized;
49
+ const parts = core.split('.').map((part) => {
50
+ const parsed = Number.parseInt(part || '0', 10);
51
+ return Number.isFinite(parsed) ? parsed : 0;
52
+ });
53
+ return { parts, prerelease: prereleaseIndex >= 0 };
54
+ };
55
+
56
+ const compareVersions = (left, right) => {
57
+ const a = parseVersionForComparison(left);
58
+ const b = parseVersionForComparison(right);
59
+ const length = Math.max(a.parts.length, b.parts.length);
60
+ for (let index = 0; index < length; index += 1) {
61
+ const diff = (a.parts[index] || 0) - (b.parts[index] || 0);
62
+ if (diff !== 0) return diff;
63
+ }
64
+ if (a.prerelease !== b.prerelease) return a.prerelease ? -1 : 1;
65
+ return 0;
66
+ };
67
+
68
+ const fetchLatestOpenCodeVersionFromGithub = async () => {
69
+ const response = await fetch('https://api.github.com/repos/anomalyco/opencode/releases/latest', {
70
+ headers: { Accept: 'application/json' },
71
+ signal: AbortSignal.timeout(10_000),
72
+ });
73
+ if (!response.ok) {
74
+ throw new Error(`OpenCode releases responded with ${response.status}`);
75
+ }
76
+ const payload = await response.json();
77
+ const tag = typeof payload?.tag_name === 'string' ? payload.tag_name.trim() : '';
78
+ return tag.replace(/^v/, '');
79
+ };
80
+
81
+ const fetchLatestOpenCodeVersionFromNpm = async () => {
82
+ const response = await fetch('https://registry.npmjs.org/opencode-ai/latest', {
83
+ headers: { Accept: 'application/json' },
84
+ signal: AbortSignal.timeout(10_000),
85
+ });
86
+ if (!response.ok) {
87
+ throw new Error(`OpenCode npm registry responded with ${response.status}`);
88
+ }
89
+ const payload = await response.json();
90
+ return typeof payload?.version === 'string' ? payload.version.trim().replace(/^v/, '') : '';
91
+ };
92
+
93
+ const fetchLatestOpenCodeVersion = async () => {
94
+ const results = await Promise.allSettled([
95
+ fetchLatestOpenCodeVersionFromNpm(),
96
+ fetchLatestOpenCodeVersionFromGithub(),
97
+ ]);
98
+ const versions = results
99
+ .filter((result) => result.status === 'fulfilled' && result.value)
100
+ .map((result) => result.value);
101
+ if (versions.length === 0) {
102
+ const failure = results.find((result) => result.status === 'rejected');
103
+ throw failure?.reason instanceof Error ? failure.reason : new Error('Failed to resolve latest OpenCode version');
104
+ }
105
+ return versions.sort((left, right) => compareVersions(right, left))[0];
106
+ };
107
+
108
+ const pruneExpiredPendingMcpAuthContexts = () => {
109
+ const now = Date.now();
110
+ for (const [state, entry] of pendingMcpAuthContextByState.entries()) {
111
+ if (!entry || typeof entry.expiresAt !== 'number' || entry.expiresAt <= now) {
112
+ pendingMcpAuthContextByState.delete(state);
113
+ }
114
+ }
115
+ };
116
+
117
+ app.get('/api/config/settings', async (_req, res) => {
118
+ try {
119
+ const settings = await readSettingsFromDiskMigrated();
120
+ res.json(formatSettingsResponse(settings));
121
+ } catch (error) {
122
+ console.error('Failed to read settings:', error);
123
+ res.status(500).json({ error: 'Failed to read settings' });
124
+ }
125
+ });
126
+
127
+ app.get('/api/config/opencode-resolution', async (_req, res) => {
128
+ try {
129
+ const settings = await readSettingsFromDiskMigrated();
130
+ const resolution = await getOpenCodeResolutionSnapshot(settings);
131
+ res.json(resolution);
132
+ } catch (error) {
133
+ console.error('Failed to resolve OpenCode binary:', error);
134
+ res.status(500).json({ error: 'Failed to resolve OpenCode binary' });
135
+ }
136
+ });
137
+
138
+ app.post('/api/opencode/upgrade', async (req, res) => {
139
+ try {
140
+ const target = typeof req.body?.target === 'string' && req.body.target.trim().length > 0
141
+ ? req.body.target.trim()
142
+ : undefined;
143
+ const response = await fetch(buildOpenCodeUrl('/global/upgrade', ''), {
144
+ method: 'POST',
145
+ headers: {
146
+ 'Content-Type': 'application/json',
147
+ Accept: 'application/json',
148
+ ...getOpenCodeAuthHeaders(),
149
+ },
150
+ body: JSON.stringify(target ? { target } : {}),
151
+ });
152
+ const payload = await response.json().catch(() => null);
153
+ if (!response.ok) {
154
+ return res.status(response.status).json({
155
+ success: false,
156
+ error: payload?.error || response.statusText || 'Failed to upgrade OpenCode',
157
+ });
158
+ }
159
+ return res.json(payload ?? { success: true });
160
+ } catch (error) {
161
+ console.error('Failed to upgrade OpenCode:', error);
162
+ return res.status(500).json({
163
+ success: false,
164
+ error: error instanceof Error ? error.message : 'Failed to upgrade OpenCode',
165
+ });
166
+ }
167
+ });
168
+
169
+ app.get('/api/opencode/upgrade-status', async (_req, res) => {
170
+ try {
171
+ const [healthResponse, latestVersion] = await Promise.all([
172
+ fetch(buildOpenCodeUrl('/global/health', ''), {
173
+ method: 'GET',
174
+ headers: { Accept: 'application/json', ...getOpenCodeAuthHeaders() },
175
+ }),
176
+ fetchLatestOpenCodeVersion(),
177
+ ]);
178
+ const health = await healthResponse.json().catch(() => null);
179
+ if (!healthResponse.ok) {
180
+ return res.status(healthResponse.status).json({
181
+ available: null,
182
+ error: health?.error || healthResponse.statusText || 'Failed to read OpenCode version',
183
+ });
184
+ }
185
+ const currentVersion = typeof health?.version === 'string' ? health.version.replace(/^v/, '') : null;
186
+ if (!currentVersion || !latestVersion) {
187
+ return res.json({ available: null, currentVersion, latestVersion: latestVersion || null });
188
+ }
189
+ const available = compareVersions(latestVersion, currentVersion) > 0;
190
+ return res.json({
191
+ available,
192
+ currentVersion,
193
+ latestVersion,
194
+ });
195
+ } catch (error) {
196
+ return res.status(500).json({
197
+ available: null,
198
+ error: error instanceof Error ? error.message : 'Failed to check OpenCode upgrade status',
199
+ });
200
+ }
201
+ });
202
+
203
+ app.put('/api/config/settings', async (req, res) => {
204
+ console.log('[API:PUT /api/config/settings] Received request');
205
+ try {
206
+ const oldSettings = await readSettingsFromDiskMigrated();
207
+ const updated = await persistSettings(req.body ?? {});
208
+ console.log(`[API:PUT /api/config/settings] Success, returning ${updated.projects?.length || 0} projects`);
209
+
210
+ // Sync settings-level API keys to env-keys.json and process.env
211
+ // so they're available to custom opencode tools regardless of settings.json state.
212
+ const envKeyMappings = {
213
+ klipyAppKey: 'KLIPY_APP_KEY',
214
+ imageGenApiKey: 'GOOGLE_GENERATIVE_AI_API_KEY',
215
+ elevenlabsApiKey: 'ELEVENLABS_API_KEY',
216
+ };
217
+ const body = req.body ?? {};
218
+ const envBatch = {};
219
+ for (const [settingsKey, envVar] of Object.entries(envKeyMappings)) {
220
+ const val = body[settingsKey];
221
+ if (typeof val === 'string' && val.trim().length > 0) {
222
+ const trimmed = val.trim();
223
+ process.env[envVar] = trimmed;
224
+ envBatch[envVar] = trimmed;
225
+ } else if (val === '' || val === null) {
226
+ delete process.env[envVar];
227
+ envBatch[envVar] = '';
228
+ }
229
+ }
230
+ if (Object.keys(envBatch).length > 0) {
231
+ writeEnvKeys(envBatch);
232
+ }
233
+
234
+ // Sync generic extra env vars from settings to env-keys.json and process.env
235
+ if (body.extraEnvVars && typeof body.extraEnvVars === 'object' && !Array.isArray(body.extraEnvVars)) {
236
+ const extraBatch = {};
237
+ for (const [key, value] of Object.entries(body.extraEnvVars)) {
238
+ if (typeof key === 'string' && typeof value === 'string' && key.trim()) {
239
+ const trimmed = value.trim();
240
+ process.env[key] = trimmed;
241
+ extraBatch[key] = trimmed;
242
+ }
243
+ }
244
+ if (Object.keys(extraBatch).length > 0) {
245
+ writeEnvKeys(extraBatch);
246
+ }
247
+ }
248
+
249
+ const keysChanged =
250
+ oldSettings.klipyAppKey !== updated.klipyAppKey ||
251
+ oldSettings.imageGenApiKey !== updated.imageGenApiKey ||
252
+ oldSettings.elevenlabsApiKey !== updated.elevenlabsApiKey ||
253
+ oldSettings.opencodeBinary !== updated.opencodeBinary ||
254
+ JSON.stringify(oldSettings.extraEnvVars) !== JSON.stringify(updated.extraEnvVars);
255
+
256
+ if (keysChanged) {
257
+ console.log('[API:PUT /api/config/settings] API keys or OpenCode binary changed, restarting OpenCode...');
258
+ await refreshOpenCodeAfterConfigChange('API keys or OpenCode binary changed').catch((err) => {
259
+ console.error('[API:PUT /api/config/settings] Failed to refresh OpenCode after settings change:', err);
260
+ });
261
+ }
262
+
263
+ res.json(updated);
264
+ } catch (error) {
265
+ console.error('[API:PUT /api/config/settings] Failed to save settings:', error);
266
+ console.error('[API:PUT /api/config/settings] Error stack:', error.stack);
267
+ res.status(500).json({ error: 'Failed to save settings' });
268
+ }
269
+ });
270
+
271
+ app.post('/api/mcp/auth/pending', async (req, res) => {
272
+ try {
273
+ pruneExpiredPendingMcpAuthContexts();
274
+
275
+ const state = normalizePendingString(req.body?.state);
276
+ if (!state) {
277
+ return res.json({ success: true, context: null });
278
+ }
279
+
280
+ const name = normalizePendingString(req.body?.name);
281
+ if (!name) {
282
+ return res.status(400).json({ error: 'MCP server name is required' });
283
+ }
284
+
285
+ const entry = {
286
+ name,
287
+ directory: normalizePendingString(req.body?.directory),
288
+ expiresAt: Date.now() + PENDING_MCP_AUTH_TTL_MS,
289
+ };
290
+ pendingMcpAuthContextByState.set(state, entry);
291
+
292
+ return res.json({
293
+ success: true,
294
+ context: {
295
+ name: entry.name,
296
+ directory: entry.directory,
297
+ },
298
+ });
299
+ } catch (error) {
300
+ console.error('Failed to store pending MCP auth context:', error);
301
+ return res.status(500).json({ error: error.message || 'Failed to store pending MCP auth context' });
302
+ }
303
+ });
304
+
305
+ app.get('/api/mcp/auth/pending', async (req, res) => {
306
+ try {
307
+ pruneExpiredPendingMcpAuthContexts();
308
+
309
+ const state = normalizePendingString(Array.isArray(req.query?.state) ? req.query.state[0] : req.query?.state);
310
+ if (!state) {
311
+ return res.json(null);
312
+ }
313
+
314
+ const pendingMcpAuthContext = pendingMcpAuthContextByState.get(state) ?? null;
315
+ if (!pendingMcpAuthContext) {
316
+ return res.status(404).json({ error: 'No pending MCP auth context' });
317
+ }
318
+
319
+ return res.json(pendingMcpAuthContext);
320
+ } catch (error) {
321
+ console.error('Failed to read pending MCP auth context:', error);
322
+ return res.status(500).json({ error: error.message || 'Failed to read pending MCP auth context' });
323
+ }
324
+ });
325
+
326
+ app.delete('/api/mcp/auth/pending', async (req, res) => {
327
+ try {
328
+ const state = normalizePendingString(Array.isArray(req.query?.state) ? req.query.state[0] : req.query?.state);
329
+ if (!state) {
330
+ return res.json({ success: true });
331
+ }
332
+
333
+ pendingMcpAuthContextByState.delete(state);
334
+ return res.json({ success: true });
335
+ } catch (error) {
336
+ console.error('Failed to clear pending MCP auth context:', error);
337
+ return res.status(500).json({ error: error.message || 'Failed to clear pending MCP auth context' });
338
+ }
339
+ });
340
+
341
+ app.get('/api/provider/:providerId/source', async (req, res) => {
342
+ try {
343
+ const { providerId } = req.params;
344
+ if (!providerId) {
345
+ return res.status(400).json({ error: 'Provider ID is required' });
346
+ }
347
+
348
+ const headerDirectory = typeof req.get === 'function' ? req.get('x-opencode-directory') : null;
349
+ const queryDirectory = Array.isArray(req.query?.directory)
350
+ ? req.query.directory[0]
351
+ : req.query?.directory;
352
+ const requestedDirectory = headerDirectory || queryDirectory || null;
353
+
354
+ let directory = null;
355
+ const resolved = await resolveProjectDirectory(req);
356
+ if (resolved.directory) {
357
+ directory = resolved.directory;
358
+ } else if (requestedDirectory) {
359
+ return res.status(400).json({ error: resolved.error });
360
+ }
361
+
362
+ const sources = getProviderSources(providerId, directory);
363
+ const { getProviderAuth } = await getAuthLibrary();
364
+ const auth = getProviderAuth(providerId);
365
+ sources.sources.auth.exists = Boolean(auth);
366
+
367
+ return res.json({
368
+ providerId,
369
+ sources: sources.sources,
370
+ });
371
+ } catch (error) {
372
+ console.error('Failed to get provider sources:', error);
373
+ return res.status(500).json({ error: error.message || 'Failed to get provider sources' });
374
+ }
375
+ });
376
+
377
+ app.delete('/api/provider/:providerId/auth', async (req, res) => {
378
+ try {
379
+ const { providerId } = req.params;
380
+ if (!providerId) {
381
+ return res.status(400).json({ error: 'Provider ID is required' });
382
+ }
383
+
384
+ const scope = typeof req.query?.scope === 'string' ? req.query.scope : 'auth';
385
+ const headerDirectory = typeof req.get === 'function' ? req.get('x-opencode-directory') : null;
386
+ const queryDirectory = Array.isArray(req.query?.directory)
387
+ ? req.query.directory[0]
388
+ : req.query?.directory;
389
+ const requestedDirectory = headerDirectory || queryDirectory || null;
390
+ let directory = null;
391
+
392
+ if (scope === 'project' || requestedDirectory) {
393
+ const resolved = await resolveProjectDirectory(req);
394
+ if (!resolved.directory) {
395
+ return res.status(400).json({ error: resolved.error });
396
+ }
397
+ directory = resolved.directory;
398
+ } else {
399
+ const resolved = await resolveProjectDirectory(req);
400
+ if (resolved.directory) {
401
+ directory = resolved.directory;
402
+ }
403
+ }
404
+
405
+ let removed = false;
406
+ if (scope === 'auth') {
407
+ const { removeProviderAuth } = await getAuthLibrary();
408
+ removed = removeProviderAuth(providerId);
409
+ } else if (scope === 'user' || scope === 'project' || scope === 'custom') {
410
+ removed = removeProviderConfig(providerId, directory, scope);
411
+ } else if (scope === 'all') {
412
+ const { removeProviderAuth } = await getAuthLibrary();
413
+ const authRemoved = removeProviderAuth(providerId);
414
+ const userRemoved = removeProviderConfig(providerId, directory, 'user');
415
+ const projectRemoved = directory ? removeProviderConfig(providerId, directory, 'project') : false;
416
+ const customRemoved = removeProviderConfig(providerId, directory, 'custom');
417
+ removed = authRemoved || userRemoved || projectRemoved || customRemoved;
418
+ } else {
419
+ return res.status(400).json({ error: 'Invalid scope' });
420
+ }
421
+
422
+ if (removed) {
423
+ await refreshOpenCodeAfterConfigChange(`provider ${providerId} disconnected (${scope})`);
424
+ }
425
+
426
+ return res.json({
427
+ success: true,
428
+ removed,
429
+ requiresReload: removed,
430
+ message: removed ? 'Provider disconnected successfully' : 'Provider was not connected',
431
+ reloadDelayMs: removed ? clientReloadDelayMs : undefined,
432
+ });
433
+ } catch (error) {
434
+ console.error('Failed to disconnect provider:', error);
435
+ return res.status(500).json({ error: error.message || 'Failed to disconnect provider' });
436
+ }
437
+ });
438
+
439
+ app.post('/api/opencode/directory', async (req, res) => {
440
+ try {
441
+ const requestedPath = typeof req.body?.path === 'string' ? req.body.path.trim() : '';
442
+ if (!requestedPath) {
443
+ return res.status(400).json({ error: 'Path is required' });
444
+ }
445
+
446
+ const validated = await validateDirectoryPath(requestedPath);
447
+ if (!validated.ok) {
448
+ return res.status(400).json({ error: validated.error });
449
+ }
450
+
451
+ const resolvedPath = validated.directory;
452
+ const currentSettings = await readSettingsFromDisk();
453
+ const existingProjects = sanitizeProjects(currentSettings.projects) || [];
454
+ const existing = existingProjects.find((project) => project.path === resolvedPath) || null;
455
+
456
+ const nextProjects = existing
457
+ ? existingProjects
458
+ : [
459
+ ...existingProjects,
460
+ {
461
+ id: createProjectIdFromPath(resolvedPath),
462
+ path: resolvedPath,
463
+ addedAt: Date.now(),
464
+ lastOpenedAt: Date.now(),
465
+ },
466
+ ];
467
+
468
+ const activeProjectId = existing ? existing.id : nextProjects[nextProjects.length - 1].id;
469
+
470
+ const updated = await persistSettings({
471
+ projects: nextProjects,
472
+ activeProjectId,
473
+ lastDirectory: resolvedPath,
474
+ });
475
+
476
+ return res.json({
477
+ success: true,
478
+ restarted: false,
479
+ path: resolvedPath,
480
+ settings: updated,
481
+ });
482
+ } catch (error) {
483
+ console.error('Failed to update OpenCode working directory:', error);
484
+ return res.status(500).json({ error: error.message || 'Failed to update working directory' });
485
+ }
486
+ });
487
+
488
+ // Behavior / Global AGENTS.md endpoints
489
+ const opencodeConfigDir = process.env.OPENCODE_CONFIG_DIR
490
+ ? path.resolve(process.env.OPENCODE_CONFIG_DIR)
491
+ : path.join(os.homedir(), '.vinci', '.opencode');
492
+ const AGENTS_MD_PATH = path.join(opencodeConfigDir, 'AGENTS.md');
493
+ const MAX_BEHAVIOR_PROMPT_SIZE = 1024 * 1024; // 1 MB
494
+
495
+ app.get('/api/behavior/agents-md', async (_req, res) => {
496
+ try {
497
+ try {
498
+ await fs.promises.access(AGENTS_MD_PATH);
499
+ } catch {
500
+ return res.json({ content: '', exists: false });
501
+ }
502
+ const content = await fs.promises.readFile(AGENTS_MD_PATH, 'utf8');
503
+ return res.json({ content, exists: true });
504
+ } catch (error) {
505
+ console.error('Failed to read AGENTS.md:', error);
506
+ return res.status(500).json({ error: 'Failed to read AGENTS.md' });
507
+ }
508
+ });
509
+
510
+ app.put('/api/behavior/agents-md', async (req, res) => {
511
+ try {
512
+ const content = typeof req.body?.content === 'string' ? req.body.content : '';
513
+
514
+ if (content.length > MAX_BEHAVIOR_PROMPT_SIZE) {
515
+ return res.status(413).json({ error: `Content exceeds maximum size of ${MAX_BEHAVIOR_PROMPT_SIZE} bytes` });
516
+ }
517
+
518
+ // Ensure parent directory exists
519
+ const parentDir = path.dirname(AGENTS_MD_PATH);
520
+ try {
521
+ await fs.promises.access(parentDir);
522
+ } catch {
523
+ await fs.promises.mkdir(parentDir, { recursive: true });
524
+ }
525
+
526
+ await fs.promises.writeFile(AGENTS_MD_PATH, content, 'utf8');
527
+
528
+ // Refresh OpenCode so it picks up the new AGENTS.md without a full restart
529
+ try {
530
+ await refreshOpenCodeAfterConfigChange('global behavior (AGENTS.md) updated');
531
+ } catch {
532
+ // Non-fatal: file was written successfully
533
+ }
534
+
535
+ return res.json({ success: true });
536
+ } catch (error) {
537
+ console.error('Failed to write AGENTS.md:', error);
538
+ return res.status(500).json({ error: error.message || 'Failed to write AGENTS.md' });
539
+ }
540
+ });
541
+ };