@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,18 @@
1
+ # Tunnels Module Documentation
2
+
3
+ ## Purpose
4
+ This module contains tunnel provider orchestration for Vinci, including provider registry/service wiring, managed remote token config lifecycle, and tunnel HTTP route registration.
5
+
6
+ ## Entrypoints and structure
7
+ - `packages/web/server/lib/tunnels/index.js`: tunnel service orchestration.
8
+ - `packages/web/server/lib/tunnels/registry.js`: provider registry.
9
+ - `packages/web/server/lib/tunnels/managed-config.js`: managed remote tunnel token/preset persistence runtime.
10
+ - `packages/web/server/lib/tunnels/routes.js`: tunnel API route registration and request orchestration runtime.
11
+ - `packages/web/server/lib/tunnels/types.js`: tunnel constants, normalization, and shared type helpers.
12
+ - `packages/web/server/lib/tunnels/providers/cloudflare.js`: Cloudflare tunnel provider implementation.
13
+
14
+ ## Public exports (routes.js)
15
+ - `createTunnelRoutesRuntime(dependencies)`: creates tunnel routes runtime and helpers.
16
+ - Returned API:
17
+ - `registerRoutes(app)`
18
+ - `startTunnelWithNormalizedRequest(request)`
@@ -0,0 +1,166 @@
1
+ import {
2
+ TUNNEL_MODE_QUICK,
3
+ TUNNEL_PROVIDER_CLOUDFLARE,
4
+ TunnelServiceError,
5
+ normalizeTunnelStartRequest,
6
+ validateTunnelStartRequest,
7
+ } from './types.js';
8
+
9
+ export function createTunnelService({
10
+ registry,
11
+ getController,
12
+ setController,
13
+ getActivePort,
14
+ onQuickTunnelWarning,
15
+ }) {
16
+ if (!registry) {
17
+ throw new Error('Tunnel service requires a provider registry');
18
+ }
19
+
20
+ const resolveActiveMode = () => {
21
+ const controller = getController();
22
+ if (!controller || typeof controller.mode !== 'string') {
23
+ return null;
24
+ }
25
+ return controller.mode;
26
+ };
27
+
28
+ const resolveActiveProvider = () => {
29
+ const controller = getController();
30
+ if (!controller || typeof controller.provider !== 'string') {
31
+ return null;
32
+ }
33
+ return controller.provider;
34
+ };
35
+
36
+ const stop = () => {
37
+ const controller = getController();
38
+ if (!controller) {
39
+ return false;
40
+ }
41
+
42
+ const providerId = typeof controller.provider === 'string' ? controller.provider : '';
43
+ const provider = providerId ? registry.get(providerId) : null;
44
+ if (provider?.stop) {
45
+ provider.stop(controller);
46
+ } else {
47
+ controller.stop?.();
48
+ }
49
+ setController(null);
50
+ return true;
51
+ };
52
+
53
+ const checkAvailability = async (providerId) => {
54
+ const provider = registry.get(providerId);
55
+ if (!provider) {
56
+ throw new TunnelServiceError('provider_unsupported', `Unsupported tunnel provider: ${providerId}`);
57
+ }
58
+ const result = await provider.checkAvailability();
59
+ return result;
60
+ };
61
+
62
+ // Mutex to prevent concurrent tunnel starts from orphaning child processes.
63
+ let startLock = Promise.resolve();
64
+
65
+ const start = async (rawRequest, options = {}) => {
66
+ let releaseLock;
67
+ const lockPromise = new Promise((resolve) => { releaseLock = resolve; });
68
+ const previousLock = startLock;
69
+ startLock = lockPromise;
70
+
71
+ await previousLock;
72
+
73
+ try {
74
+ const request = normalizeTunnelStartRequest(rawRequest);
75
+ const provider = registry.get(request.provider);
76
+
77
+ if (!provider) {
78
+ throw new TunnelServiceError('provider_unsupported', `Unsupported tunnel provider: ${request.provider}`);
79
+ }
80
+
81
+ validateTunnelStartRequest(request, provider.capabilities);
82
+
83
+ let publicUrl = provider.resolvePublicUrl(getController());
84
+ const activeMode = resolveActiveMode();
85
+
86
+ if (publicUrl && activeMode !== request.mode) {
87
+ stop();
88
+ publicUrl = null;
89
+ }
90
+
91
+ if (!publicUrl) {
92
+ const availability = await provider.checkAvailability();
93
+ if (!availability?.available) {
94
+ const missingDependencyMessage = typeof availability?.message === 'string' && availability.message.trim().length > 0
95
+ ? availability.message
96
+ : (request.provider === TUNNEL_PROVIDER_CLOUDFLARE
97
+ ? 'cloudflared is not installed. Install it with: brew install cloudflared'
98
+ : `Required dependency for provider '${request.provider}' is missing`);
99
+ throw new TunnelServiceError('missing_dependency', missingDependencyMessage);
100
+ }
101
+
102
+ const activePort = Number.isFinite(getActivePort?.()) ? getActivePort() : null;
103
+ const originUrl = activePort !== null ? `http://127.0.0.1:${activePort}` : undefined;
104
+
105
+ const controller = await provider.start(request, {
106
+ activePort,
107
+ originUrl,
108
+ ...options,
109
+ });
110
+ controller.provider = request.provider;
111
+ setController(controller);
112
+
113
+ publicUrl = provider.resolvePublicUrl(controller);
114
+ if (!publicUrl) {
115
+ stop();
116
+ throw new TunnelServiceError('startup_failed', 'Tunnel started but no public URL was assigned');
117
+ }
118
+
119
+ if (request.mode === TUNNEL_MODE_QUICK) {
120
+ onQuickTunnelWarning?.();
121
+ }
122
+ }
123
+
124
+ return {
125
+ publicUrl,
126
+ request,
127
+ activeMode: request.mode,
128
+ provider: request.provider,
129
+ providerMetadata: provider.getMetadata?.(getController()) ?? null,
130
+ };
131
+ } finally {
132
+ releaseLock();
133
+ }
134
+ };
135
+
136
+ const getPublicUrl = () => {
137
+ const controller = getController();
138
+ if (!controller) {
139
+ return null;
140
+ }
141
+ const provider = registry.get(controller.provider);
142
+ if (!provider) {
143
+ return controller.getPublicUrl?.() ?? null;
144
+ }
145
+ return provider.resolvePublicUrl(controller);
146
+ };
147
+
148
+ const getProviderMetadata = () => {
149
+ const controller = getController();
150
+ if (!controller) {
151
+ return null;
152
+ }
153
+ const provider = registry.get(controller.provider);
154
+ return provider?.getMetadata?.(controller) ?? null;
155
+ };
156
+
157
+ return {
158
+ start,
159
+ stop,
160
+ checkAvailability,
161
+ getPublicUrl,
162
+ getProviderMetadata,
163
+ resolveActiveMode,
164
+ resolveActiveProvider,
165
+ };
166
+ }
@@ -0,0 +1,201 @@
1
+ export const createManagedTunnelConfigRuntime = (deps) => {
2
+ const {
3
+ fsPromises,
4
+ path,
5
+ normalizeManagedRemoteTunnelHostname,
6
+ normalizeManagedRemoteTunnelPresets,
7
+ constants,
8
+ } = deps;
9
+
10
+ const {
11
+ CLOUDFLARE_MANAGED_REMOTE_TUNNELS_FILE_PATH,
12
+ CLOUDFLARE_LEGACY_NAMED_TUNNELS_FILE_PATH,
13
+ CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
14
+ } = constants;
15
+
16
+ let persistManagedRemoteTunnelConfigLock = Promise.resolve();
17
+
18
+ const sanitizeManagedRemoteTunnelConfigEntries = (value) => {
19
+ if (!Array.isArray(value)) {
20
+ return [];
21
+ }
22
+
23
+ const result = [];
24
+ const seenIds = new Set();
25
+ const seenHostnames = new Set();
26
+ for (const entry of value) {
27
+ if (!entry || typeof entry !== 'object') {
28
+ continue;
29
+ }
30
+
31
+ const id = typeof entry.id === 'string' ? entry.id.trim() : '';
32
+ const name = typeof entry.name === 'string' ? entry.name.trim() : '';
33
+ const hostname = normalizeManagedRemoteTunnelHostname(entry.hostname);
34
+ const token = typeof entry.token === 'string' ? entry.token.trim() : '';
35
+ const updatedAt = Number.isFinite(entry.updatedAt) ? entry.updatedAt : Date.now();
36
+
37
+ if (!id || !name || !hostname || !token) {
38
+ continue;
39
+ }
40
+ if (seenIds.has(id) || seenHostnames.has(hostname)) {
41
+ continue;
42
+ }
43
+
44
+ seenIds.add(id);
45
+ seenHostnames.add(hostname);
46
+ result.push({ id, name, hostname, token, updatedAt });
47
+ }
48
+
49
+ return result;
50
+ };
51
+
52
+ const writeManagedRemoteTunnelConfigToDisk = async (data) => {
53
+ await fsPromises.mkdir(path.dirname(CLOUDFLARE_MANAGED_REMOTE_TUNNELS_FILE_PATH), { recursive: true });
54
+ await fsPromises.writeFile(CLOUDFLARE_MANAGED_REMOTE_TUNNELS_FILE_PATH, JSON.stringify(data, null, 2), { encoding: 'utf8', mode: 0o600 });
55
+ };
56
+
57
+ const migrateManagedRemoteTunnelConfigFromLegacyFile = async () => {
58
+ try {
59
+ const legacyRaw = await fsPromises.readFile(CLOUDFLARE_LEGACY_NAMED_TUNNELS_FILE_PATH, 'utf8');
60
+ const parsed = JSON.parse(legacyRaw);
61
+ const tunnels = sanitizeManagedRemoteTunnelConfigEntries(parsed?.tunnels);
62
+ const migrated = {
63
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
64
+ tunnels,
65
+ };
66
+ await writeManagedRemoteTunnelConfigToDisk(migrated);
67
+ return migrated;
68
+ } catch (error) {
69
+ if (error && typeof error === 'object' && error.code === 'ENOENT') {
70
+ return { version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION, tunnels: [] };
71
+ }
72
+ console.warn('Failed to migrate legacy named tunnel config file:', error);
73
+ return { version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION, tunnels: [] };
74
+ }
75
+ };
76
+
77
+ const readManagedRemoteTunnelConfigFromDisk = async () => {
78
+ try {
79
+ const raw = await fsPromises.readFile(CLOUDFLARE_MANAGED_REMOTE_TUNNELS_FILE_PATH, 'utf8');
80
+ const parsed = JSON.parse(raw);
81
+ if (!parsed || typeof parsed !== 'object') {
82
+ return { version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION, tunnels: [] };
83
+ }
84
+
85
+ return {
86
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
87
+ tunnels: sanitizeManagedRemoteTunnelConfigEntries(parsed.tunnels),
88
+ };
89
+ } catch (error) {
90
+ if (error && typeof error === 'object' && error.code === 'ENOENT') {
91
+ return migrateManagedRemoteTunnelConfigFromLegacyFile();
92
+ }
93
+ console.warn('Failed to read managed remote tunnel config file:', error);
94
+ return { version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION, tunnels: [] };
95
+ }
96
+ };
97
+
98
+ const updateManagedRemoteTunnelConfig = async (mutate) => {
99
+ persistManagedRemoteTunnelConfigLock = persistManagedRemoteTunnelConfigLock.then(async () => {
100
+ const current = await readManagedRemoteTunnelConfigFromDisk();
101
+ const next = mutate({
102
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
103
+ tunnels: sanitizeManagedRemoteTunnelConfigEntries(current.tunnels),
104
+ });
105
+
106
+ await writeManagedRemoteTunnelConfigToDisk({
107
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
108
+ tunnels: sanitizeManagedRemoteTunnelConfigEntries(next?.tunnels),
109
+ });
110
+ });
111
+
112
+ return persistManagedRemoteTunnelConfigLock;
113
+ };
114
+
115
+ const syncManagedRemoteTunnelConfigWithPresets = async (presets) => {
116
+ const sanitizedPresets = normalizeManagedRemoteTunnelPresets(presets) || [];
117
+
118
+ await updateManagedRemoteTunnelConfig((current) => {
119
+ const byId = new Map(current.tunnels.map((entry) => [entry.id, entry]));
120
+ const byHostname = new Map(current.tunnels.map((entry) => [entry.hostname, entry]));
121
+
122
+ const nextTunnels = [];
123
+ for (const preset of sanitizedPresets) {
124
+ const existing = byId.get(preset.id) || byHostname.get(preset.hostname) || null;
125
+ if (!existing) {
126
+ continue;
127
+ }
128
+
129
+ nextTunnels.push({
130
+ ...existing,
131
+ id: preset.id,
132
+ name: preset.name,
133
+ hostname: preset.hostname,
134
+ });
135
+ }
136
+
137
+ return {
138
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
139
+ tunnels: nextTunnels,
140
+ };
141
+ });
142
+ };
143
+
144
+ const upsertManagedRemoteTunnelToken = async ({ id, name, hostname, token }) => {
145
+ if (typeof id !== 'string' || typeof name !== 'string' || typeof hostname !== 'string' || typeof token !== 'string') {
146
+ return;
147
+ }
148
+ const normalizedId = id.trim();
149
+ const normalizedName = name.trim();
150
+ const normalizedHostname = normalizeManagedRemoteTunnelHostname(hostname);
151
+ const normalizedToken = token.trim();
152
+ if (!normalizedId || !normalizedName || !normalizedHostname || !normalizedToken) {
153
+ return;
154
+ }
155
+
156
+ await updateManagedRemoteTunnelConfig((current) => {
157
+ const withoutConflicts = current.tunnels.filter((entry) => entry.id !== normalizedId && entry.hostname !== normalizedHostname);
158
+ withoutConflicts.push({
159
+ id: normalizedId,
160
+ name: normalizedName,
161
+ hostname: normalizedHostname,
162
+ token: normalizedToken,
163
+ updatedAt: Date.now(),
164
+ });
165
+
166
+ return {
167
+ version: CLOUDFLARE_MANAGED_REMOTE_TUNNELS_VERSION,
168
+ tunnels: withoutConflicts,
169
+ };
170
+ });
171
+ };
172
+
173
+ const resolveManagedRemoteTunnelToken = async ({ presetId, hostname }) => {
174
+ const normalizedPresetId = typeof presetId === 'string' ? presetId.trim() : '';
175
+ const normalizedHostname = normalizeManagedRemoteTunnelHostname(hostname);
176
+ const config = await readManagedRemoteTunnelConfigFromDisk();
177
+
178
+ if (normalizedPresetId) {
179
+ const byId = config.tunnels.find((entry) => entry.id === normalizedPresetId);
180
+ if (byId?.token) {
181
+ return byId.token;
182
+ }
183
+ }
184
+
185
+ if (normalizedHostname) {
186
+ const byHostname = config.tunnels.find((entry) => entry.hostname === normalizedHostname);
187
+ if (byHostname?.token) {
188
+ return byHostname.token;
189
+ }
190
+ }
191
+
192
+ return '';
193
+ };
194
+
195
+ return {
196
+ readManagedRemoteTunnelConfigFromDisk,
197
+ syncManagedRemoteTunnelConfigWithPresets,
198
+ upsertManagedRemoteTunnelToken,
199
+ resolveManagedRemoteTunnelToken,
200
+ };
201
+ };
@@ -0,0 +1,260 @@
1
+ import {
2
+ checkCloudflareApiReachability,
3
+ checkCloudflaredAvailable,
4
+ inspectManagedLocalCloudflareConfig,
5
+ normalizeCloudflareTunnelHostname,
6
+ startCloudflareManagedLocalTunnel,
7
+ startCloudflareManagedRemoteTunnel,
8
+ startCloudflareQuickTunnel,
9
+ } from '../../cloudflare-tunnel.js';
10
+
11
+ import {
12
+ TUNNEL_INTENT_EPHEMERAL_PUBLIC,
13
+ TUNNEL_INTENT_PERSISTENT_PUBLIC,
14
+ TUNNEL_MODE_MANAGED_LOCAL,
15
+ TUNNEL_MODE_MANAGED_REMOTE,
16
+ TUNNEL_MODE_QUICK,
17
+ TUNNEL_PROVIDER_CLOUDFLARE,
18
+ TunnelServiceError,
19
+ } from '../types.js';
20
+
21
+ export const cloudflareTunnelProviderCapabilities = {
22
+ provider: TUNNEL_PROVIDER_CLOUDFLARE,
23
+ defaults: {
24
+ mode: TUNNEL_MODE_QUICK,
25
+ optionDefaults: {},
26
+ },
27
+ modes: [
28
+ {
29
+ key: TUNNEL_MODE_QUICK,
30
+ label: 'Quick Tunnel',
31
+ intent: TUNNEL_INTENT_EPHEMERAL_PUBLIC,
32
+ requires: [],
33
+ supports: ['sessionTTL'],
34
+ stability: 'ga',
35
+ },
36
+ {
37
+ key: TUNNEL_MODE_MANAGED_REMOTE,
38
+ label: 'Managed Remote Tunnel',
39
+ intent: TUNNEL_INTENT_PERSISTENT_PUBLIC,
40
+ requires: ['token', 'hostname'],
41
+ supports: ['customDomain', 'sessionTTL'],
42
+ stability: 'ga',
43
+ },
44
+ {
45
+ key: TUNNEL_MODE_MANAGED_LOCAL,
46
+ label: 'Managed Local Tunnel',
47
+ intent: TUNNEL_INTENT_PERSISTENT_PUBLIC,
48
+ requires: [],
49
+ supports: ['configFile', 'customDomain', 'sessionTTL'],
50
+ stability: 'ga',
51
+ },
52
+ ],
53
+ };
54
+
55
+ export function createCloudflareTunnelProvider() {
56
+ const validateTokenShape = (value) => {
57
+ if (typeof value !== 'string') {
58
+ return { ok: false, detail: 'Managed remote token is missing.' };
59
+ }
60
+ const trimmed = value.trim();
61
+ if (!trimmed) {
62
+ return { ok: false, detail: 'Managed remote token is missing.' };
63
+ }
64
+ if (/\s/.test(trimmed)) {
65
+ return { ok: false, detail: 'Managed remote token has whitespace; provide the raw token value.' };
66
+ }
67
+ return { ok: true, detail: 'Managed remote token looks valid.' };
68
+ };
69
+
70
+ const createModeSummary = (checks) => {
71
+ const failures = checks.filter((entry) => entry.status === 'fail').length;
72
+ const warnings = checks.filter((entry) => entry.status === 'warn').length;
73
+ return {
74
+ ready: failures === 0,
75
+ failures,
76
+ warnings,
77
+ };
78
+ };
79
+
80
+ const describeMode = ({ mode, checks }) => {
81
+ const summary = createModeSummary(checks);
82
+ const blockers = checks
83
+ .filter((entry) => entry.status === 'fail' && entry.id !== 'startup_readiness')
84
+ .map((entry) => entry.detail || entry.label || entry.id);
85
+ return {
86
+ mode,
87
+ checks,
88
+ summary,
89
+ ready: summary.ready,
90
+ blockers,
91
+ };
92
+ };
93
+
94
+ return {
95
+ id: TUNNEL_PROVIDER_CLOUDFLARE,
96
+ capabilities: cloudflareTunnelProviderCapabilities,
97
+ checkAvailability: async () => {
98
+ const result = await checkCloudflaredAvailable();
99
+ if (result.available) {
100
+ return result;
101
+ }
102
+ return {
103
+ ...result,
104
+ message: 'cloudflared is not installed. Install it with: brew install cloudflared',
105
+ };
106
+ },
107
+ diagnose: async (request = {}) => {
108
+ const dependency = await checkCloudflaredAvailable();
109
+ const network = await checkCloudflareApiReachability();
110
+
111
+ const providerChecks = [
112
+ {
113
+ id: 'dependency',
114
+ label: 'cloudflared installed',
115
+ status: dependency.available ? 'pass' : 'fail',
116
+ detail: dependency.available
117
+ ? (dependency.version || dependency.path || 'cloudflared available')
118
+ : 'cloudflared is not installed. Install it with: brew install cloudflared',
119
+ },
120
+ {
121
+ id: 'network',
122
+ label: 'Cloudflare API reachable',
123
+ status: network.reachable ? 'pass' : 'fail',
124
+ detail: network.reachable
125
+ ? (network.status ? `HTTP ${network.status}` : 'Reachable')
126
+ : (network.error || 'Could not reach api.trycloudflare.com'),
127
+ },
128
+ ];
129
+
130
+ const startupReady = dependency.available && network.reachable;
131
+ const startupDetail = startupReady
132
+ ? 'Provider dependency and network checks passed.'
133
+ : 'Resolve provider checks before starting tunnels.';
134
+
135
+ const quickChecks = [
136
+ {
137
+ id: 'startup_readiness',
138
+ label: 'Provider startup readiness',
139
+ status: startupReady ? 'pass' : 'fail',
140
+ detail: startupDetail,
141
+ },
142
+ {
143
+ id: 'quick_mode_prerequisites',
144
+ label: 'Quick tunnel prerequisites',
145
+ status: network.reachable ? 'pass' : 'fail',
146
+ detail: network.reachable
147
+ ? 'Cloudflare edge is reachable for quick tunnels.'
148
+ : 'Cloudflare edge is not reachable for quick tunnels.',
149
+ },
150
+ ];
151
+
152
+ const managedLocalInspection = inspectManagedLocalCloudflareConfig({
153
+ configPath: request.configPath,
154
+ hostname: request.hostname,
155
+ });
156
+ const managedLocalChecks = [
157
+ {
158
+ id: 'startup_readiness',
159
+ label: 'Provider startup readiness',
160
+ status: startupReady ? 'pass' : 'fail',
161
+ detail: startupDetail,
162
+ },
163
+ {
164
+ id: 'managed_local_config',
165
+ label: 'Managed local config',
166
+ status: managedLocalInspection.ok ? 'pass' : 'fail',
167
+ detail: managedLocalInspection.ok
168
+ ? `${managedLocalInspection.effectiveConfigPath}${managedLocalInspection.resolvedHostname ? ` (${managedLocalInspection.resolvedHostname})` : ''}`
169
+ : managedLocalInspection.error,
170
+ },
171
+ ];
172
+
173
+ const normalizedHost = normalizeCloudflareTunnelHostname(request.hostname);
174
+ const hostnameMissing = !normalizedHost;
175
+ const remoteTokenValidation = validateTokenShape(request.token);
176
+ const tokenMissing = typeof request.token !== 'string' || request.token.trim().length === 0;
177
+ const hasSavedManagedRemoteProfile = request.hasSavedManagedRemoteProfile === true;
178
+ const tokenProvided = request.tokenProvided === true;
179
+ const hostnameProvided = request.hostnameProvided === true;
180
+ const hasExplicitManagedRemoteInput = tokenProvided || hostnameProvided;
181
+ const canUseSavedProfileForHostname = !hasExplicitManagedRemoteInput && hostnameMissing && hasSavedManagedRemoteProfile;
182
+ const canUseSavedProfileForToken = !hasExplicitManagedRemoteInput && tokenMissing && hasSavedManagedRemoteProfile;
183
+ const savedProfileReadyDetail = 'at least one saved profile present';
184
+ const managedRemoteChecks = [
185
+ {
186
+ id: 'startup_readiness',
187
+ label: 'Provider startup readiness',
188
+ status: startupReady ? 'pass' : 'fail',
189
+ detail: startupDetail,
190
+ },
191
+ {
192
+ id: 'managed_remote_hostname',
193
+ label: 'Managed remote hostname',
194
+ status: normalizedHost || canUseSavedProfileForHostname ? 'pass' : 'fail',
195
+ detail: normalizedHost
196
+ ? normalizedHost
197
+ : canUseSavedProfileForHostname
198
+ ? savedProfileReadyDetail
199
+ : 'Managed remote hostname is required (use --hostname).',
200
+ },
201
+ {
202
+ id: 'managed_remote_token',
203
+ label: 'Managed remote token',
204
+ status: remoteTokenValidation.ok || canUseSavedProfileForToken ? 'pass' : 'fail',
205
+ detail: canUseSavedProfileForToken
206
+ ? savedProfileReadyDetail
207
+ : remoteTokenValidation.detail,
208
+ },
209
+ ];
210
+
211
+ const allModes = [
212
+ describeMode({ mode: TUNNEL_MODE_QUICK, checks: quickChecks }),
213
+ describeMode({ mode: TUNNEL_MODE_MANAGED_REMOTE, checks: managedRemoteChecks }),
214
+ describeMode({ mode: TUNNEL_MODE_MANAGED_LOCAL, checks: managedLocalChecks }),
215
+ ];
216
+
217
+ const modeFilter = typeof request.mode === 'string' && request.mode.trim().length > 0
218
+ ? request.mode.trim().toLowerCase()
219
+ : null;
220
+ const modes = modeFilter ? allModes.filter((entry) => entry.mode === modeFilter) : allModes;
221
+
222
+ return {
223
+ providerChecks,
224
+ modes,
225
+ };
226
+ },
227
+ start: async (request, context = {}) => {
228
+ if (request.mode === TUNNEL_MODE_MANAGED_REMOTE) {
229
+ return startCloudflareManagedRemoteTunnel({
230
+ token: request.token,
231
+ hostname: request.hostname,
232
+ });
233
+ }
234
+
235
+ if (request.mode === TUNNEL_MODE_MANAGED_LOCAL) {
236
+ return startCloudflareManagedLocalTunnel({
237
+ configPath: request.configPath,
238
+ hostname: request.hostname,
239
+ });
240
+ }
241
+
242
+ if (!context.originUrl) {
243
+ throw new TunnelServiceError('validation_error', 'originUrl is required for quick tunnel mode');
244
+ }
245
+
246
+ return startCloudflareQuickTunnel({
247
+ originUrl: context.originUrl,
248
+ port: context.activePort,
249
+ });
250
+ },
251
+ stop: (controller) => {
252
+ controller?.stop?.();
253
+ },
254
+ resolvePublicUrl: (controller) => controller?.getPublicUrl?.() ?? null,
255
+ getMetadata: (controller) => ({
256
+ configPath: controller?.getEffectiveConfigPath?.() ?? null,
257
+ resolvedHostname: controller?.getResolvedHostname?.() ?? null,
258
+ }),
259
+ };
260
+ }
@@ -0,0 +1,51 @@
1
+ const REQUIRED_PROVIDER_METHODS = ['start', 'stop', 'checkAvailability', 'resolvePublicUrl'];
2
+
3
+ export function createTunnelProviderRegistry(initialProviders = []) {
4
+ const providers = new Map();
5
+ let sealed = false;
6
+
7
+ const register = (provider) => {
8
+ if (sealed) {
9
+ throw new Error('Tunnel provider registry is sealed; no further registrations allowed');
10
+ }
11
+ if (!provider || typeof provider.id !== 'string' || provider.id.trim().length === 0) {
12
+ throw new Error('Tunnel provider must define a non-empty id');
13
+ }
14
+ for (const method of REQUIRED_PROVIDER_METHODS) {
15
+ if (typeof provider[method] !== 'function') {
16
+ throw new Error(`Tunnel provider '${provider.id}' must implement ${method}()`);
17
+ }
18
+ }
19
+ const key = provider.id.trim().toLowerCase();
20
+ if (providers.has(key)) {
21
+ throw new Error(`Tunnel provider '${key}' is already registered`);
22
+ }
23
+ providers.set(key, provider);
24
+ return provider;
25
+ };
26
+
27
+ const get = (providerId) => {
28
+ if (typeof providerId !== 'string' || providerId.trim().length === 0) {
29
+ return null;
30
+ }
31
+ return providers.get(providerId.trim().toLowerCase()) ?? null;
32
+ };
33
+
34
+ const list = () => Array.from(providers.values());
35
+
36
+ const listCapabilities = () => list().map((provider) => ({ ...provider.capabilities }));
37
+
38
+ for (const provider of initialProviders) {
39
+ register(provider);
40
+ }
41
+
42
+ const seal = () => { sealed = true; };
43
+
44
+ return {
45
+ register,
46
+ get,
47
+ list,
48
+ listCapabilities,
49
+ seal,
50
+ };
51
+ }