@zuzuucodes/cli 1.1.0 → 1.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (462) hide show
  1. package/README.md +7 -6
  2. package/bin/zuzuu-web.mjs +15 -0
  3. package/package.json +19 -8
  4. package/web-app/dist/browse.js +48 -0
  5. package/web-app/dist/config.js +52 -0
  6. package/web-app/dist/file-list.js +87 -0
  7. package/web-app/dist/frames.js +16 -0
  8. package/web-app/dist/fs-api.js +226 -0
  9. package/web-app/dist/git.js +73 -0
  10. package/web-app/dist/history.js +45 -0
  11. package/web-app/dist/index.js +160 -0
  12. package/web-app/dist/protocol/index.js +53 -0
  13. package/web-app/dist/protocol/zuzuu.js +2 -0
  14. package/web-app/dist/safe-path.js +60 -0
  15. package/web-app/dist/search.js +145 -0
  16. package/web-app/dist/server.js +407 -0
  17. package/web-app/dist/sessions.js +334 -0
  18. package/web-app/dist/shell-integration/inject.js +70 -0
  19. package/web-app/dist/shell-integration/webcode.bash +41 -0
  20. package/web-app/dist/shell-integration/webcode.fish +28 -0
  21. package/web-app/dist/shell-integration/webcode.zsh +35 -0
  22. package/web-app/dist/workflows.js +48 -0
  23. package/web-app/dist/ws-fs.js +69 -0
  24. package/web-app/dist/ws-term.js +42 -0
  25. package/web-app/dist/zuzuu-api.js +348 -0
  26. package/web-app/package.json +6 -0
  27. package/web-app/web-dist/assets/DiffTab-NYcg0Jn6.js +1 -0
  28. package/web-app/web-dist/assets/MonacoFile-C_nBCUL-.js +1 -0
  29. package/web-app/web-dist/assets/abap-08VXUWAP.js +1 -0
  30. package/web-app/web-dist/assets/abap-CLvhMVsD.js +1 -0
  31. package/web-app/web-dist/assets/actionscript-3--17pq3dv.js +1 -0
  32. package/web-app/web-dist/assets/ada-C5qYipkI.js +1 -0
  33. package/web-app/web-dist/assets/andromeeda-vGVdxbeo.js +1 -0
  34. package/web-app/web-dist/assets/angular-html-CmT26mqM.js +1 -0
  35. package/web-app/web-dist/assets/angular-ts-CD_OonCa.js +1 -0
  36. package/web-app/web-dist/assets/apache-U0d_L8uA.js +1 -0
  37. package/web-app/web-dist/assets/apex-BWPQTe0t.js +1 -0
  38. package/web-app/web-dist/assets/apex-VAyPSnFM.js +1 -0
  39. package/web-app/web-dist/assets/apl-uOGC3x4e.js +1 -0
  40. package/web-app/web-dist/assets/applescript-CCn79oCD.js +1 -0
  41. package/web-app/web-dist/assets/ara-4CJ0cIlV.js +1 -0
  42. package/web-app/web-dist/assets/asciidoc-DE70LPWp.js +1 -0
  43. package/web-app/web-dist/assets/asciinema-player-CZa1Uf8p.css +1 -0
  44. package/web-app/web-dist/assets/asm-Cmm7eHzH.js +1 -0
  45. package/web-app/web-dist/assets/astro-B6ybQmWG.js +1 -0
  46. package/web-app/web-dist/assets/aurora-x-CDeNXAV0.js +1 -0
  47. package/web-app/web-dist/assets/awk-BWXHIvNe.js +1 -0
  48. package/web-app/web-dist/assets/ayu-dark-DluEY0Gj.js +1 -0
  49. package/web-app/web-dist/assets/ayu-light-C3h-C4tm.js +1 -0
  50. package/web-app/web-dist/assets/ayu-mirage-Bqwy1Gya.js +1 -0
  51. package/web-app/web-dist/assets/azcli-Bc_sGQ0U.js +1 -0
  52. package/web-app/web-dist/assets/ballerina-B7ZEbQpA.js +1 -0
  53. package/web-app/web-dist/assets/bat-Bo4NYOV-.js +1 -0
  54. package/web-app/web-dist/assets/bat-i0X4ZdIN.js +1 -0
  55. package/web-app/web-dist/assets/beancount-D-usSTwE.js +1 -0
  56. package/web-app/web-dist/assets/berry-DKpUyyne.js +1 -0
  57. package/web-app/web-dist/assets/bibtex-Ci_nEsc7.js +1 -0
  58. package/web-app/web-dist/assets/bicep-B5-_aFwp.js +2 -0
  59. package/web-app/web-dist/assets/bicep-CUHmPFLl.js +1 -0
  60. package/web-app/web-dist/assets/bird2-C6vDhewU.js +1 -0
  61. package/web-app/web-dist/assets/blade-B1QGRlVx.js +1 -0
  62. package/web-app/web-dist/assets/bsl-BkkzgIyY.js +1 -0
  63. package/web-app/web-dist/assets/c-BvoqrSVH.js +1 -0
  64. package/web-app/web-dist/assets/c3-BFHwR3_K.js +1 -0
  65. package/web-app/web-dist/assets/cadence-CQ2zXKGN.js +1 -0
  66. package/web-app/web-dist/assets/cairo-DLTphjLi.js +1 -0
  67. package/web-app/web-dist/assets/cameligo-DMUM7wLl.js +1 -0
  68. package/web-app/web-dist/assets/catppuccin-frappe-3VR1Za6u.js +1 -0
  69. package/web-app/web-dist/assets/catppuccin-latte-DwIHMF0Q.js +1 -0
  70. package/web-app/web-dist/assets/catppuccin-macchiato-DYnBP6_5.js +1 -0
  71. package/web-app/web-dist/assets/catppuccin-mocha-DYhrFGRu.js +1 -0
  72. package/web-app/web-dist/assets/chunk-QTnfLwEv.js +1 -0
  73. package/web-app/web-dist/assets/clarity-SemFz856.js +1 -0
  74. package/web-app/web-dist/assets/clojure-Cm7r79vr.js +1 -0
  75. package/web-app/web-dist/assets/clojure-DqKBuwfJ.js +1 -0
  76. package/web-app/web-dist/assets/cmake-Bj61d0ZC.js +1 -0
  77. package/web-app/web-dist/assets/cobol-BgqgtYWn.js +1 -0
  78. package/web-app/web-dist/assets/codeowners-C8r90Shi.js +1 -0
  79. package/web-app/web-dist/assets/codeql-oeQT6MSM.js +1 -0
  80. package/web-app/web-dist/assets/codicon-ngg6Pgfi.ttf +0 -0
  81. package/web-app/web-dist/assets/coffee-0wIRKYlr.js +1 -0
  82. package/web-app/web-dist/assets/coffee-Ba7i2nA0.js +1 -0
  83. package/web-app/web-dist/assets/common-lisp-Cv5bFMCO.js +1 -0
  84. package/web-app/web-dist/assets/coq-BrsZFFmf.js +1 -0
  85. package/web-app/web-dist/assets/cpp-BXsk94m0.js +1 -0
  86. package/web-app/web-dist/assets/cpp-C7h46wYY.js +1 -0
  87. package/web-app/web-dist/assets/crystal-CyTK3qFN.js +1 -0
  88. package/web-app/web-dist/assets/csharp-BKxtCVv1.js +1 -0
  89. package/web-app/web-dist/assets/csharp-oqKa8noW.js +1 -0
  90. package/web-app/web-dist/assets/csp-bTuwJoIa.js +1 -0
  91. package/web-app/web-dist/assets/css-DIMkf-bt.js +3 -0
  92. package/web-app/web-dist/assets/css-Z8oOGxII.js +1 -0
  93. package/web-app/web-dist/assets/css.worker-CvXBzhp8.js +89 -0
  94. package/web-app/web-dist/assets/cssMode-BXZOx0fG.js +1 -0
  95. package/web-app/web-dist/assets/csv-Dx-8-gkx.js +1 -0
  96. package/web-app/web-dist/assets/cue-CE9AQfxI.js +1 -0
  97. package/web-app/web-dist/assets/cypher-CVaqCwHa.js +1 -0
  98. package/web-app/web-dist/assets/cypher-ClKdZ_lG.js +1 -0
  99. package/web-app/web-dist/assets/d-qD-0Kul2.js +1 -0
  100. package/web-app/web-dist/assets/dark-plus-Cs2F2srj.js +1 -0
  101. package/web-app/web-dist/assets/dart-CnvKMtbv.js +1 -0
  102. package/web-app/web-dist/assets/dart-onAF5SnQ.js +1 -0
  103. package/web-app/web-dist/assets/dax-BkyTk9wS.js +1 -0
  104. package/web-app/web-dist/assets/desktop-Dlh5hvp9.js +1 -0
  105. package/web-app/web-dist/assets/diff-woXpYk--.js +1 -0
  106. package/web-app/web-dist/assets/dist-Cj6v7Kj8.js +153 -0
  107. package/web-app/web-dist/assets/dist-DNelLewU.js +4 -0
  108. package/web-app/web-dist/assets/docker-IyjqRm3v.js +1 -0
  109. package/web-app/web-dist/assets/dockerfile-DZFCIeNp.js +1 -0
  110. package/web-app/web-dist/assets/dotenv-_5a1GRtc.js +1 -0
  111. package/web-app/web-dist/assets/dracula-BHWKrbxM.js +1 -0
  112. package/web-app/web-dist/assets/dracula-soft-5eyTD99u.js +1 -0
  113. package/web-app/web-dist/assets/dream-maker-DW3nJb8Q.js +1 -0
  114. package/web-app/web-dist/assets/ecl-D05T4iGw.js +1 -0
  115. package/web-app/web-dist/assets/edge-CvML9pwC.js +1 -0
  116. package/web-app/web-dist/assets/editor-Br_kD0ds.css +1 -0
  117. package/web-app/web-dist/assets/editor.api2-BmGoRSl4.js +872 -0
  118. package/web-app/web-dist/assets/editor.worker-Cn2oRESe.js +26 -0
  119. package/web-app/web-dist/assets/elixir-6RTg0lbw.js +1 -0
  120. package/web-app/web-dist/assets/elixir-CrjqTiSc.js +1 -0
  121. package/web-app/web-dist/assets/elm-C4JtJ0Au.js +1 -0
  122. package/web-app/web-dist/assets/emacs-lisp-B4R74twV.js +1 -0
  123. package/web-app/web-dist/assets/erb-Cmeb-29V.js +1 -0
  124. package/web-app/web-dist/assets/erlang-Cphh6RMH.js +1 -0
  125. package/web-app/web-dist/assets/everforest-dark-sB-x3p7T.js +1 -0
  126. package/web-app/web-dist/assets/everforest-light-Df2xbC6M.js +1 -0
  127. package/web-app/web-dist/assets/fennel-DQxkIbk2.js +1 -0
  128. package/web-app/web-dist/assets/fish-BJitypiv.js +1 -0
  129. package/web-app/web-dist/assets/flow9-C5_-GSwl.js +1 -0
  130. package/web-app/web-dist/assets/fluent-C03EYrpw.js +1 -0
  131. package/web-app/web-dist/assets/fortran-fixed-form-DEKoE2YW.js +1 -0
  132. package/web-app/web-dist/assets/fortran-free-form-CYNrtFtB.js +1 -0
  133. package/web-app/web-dist/assets/freemarker2-B5LAi19B.js +3 -0
  134. package/web-app/web-dist/assets/fsharp-C8Ef5oNN.js +1 -0
  135. package/web-app/web-dist/assets/fsharp-D13ZGOAj.js +1 -0
  136. package/web-app/web-dist/assets/gdresource-C0sCabJj.js +1 -0
  137. package/web-app/web-dist/assets/gdscript-Cp2uCuqX.js +1 -0
  138. package/web-app/web-dist/assets/gdshader-CBce3t8t.js +1 -0
  139. package/web-app/web-dist/assets/genie-CV2tkWYe.js +1 -0
  140. package/web-app/web-dist/assets/gherkin-DExj1W_8.js +1 -0
  141. package/web-app/web-dist/assets/git-commit-BSykSTBG.js +1 -0
  142. package/web-app/web-dist/assets/git-rebase-CXqdToiP.js +1 -0
  143. package/web-app/web-dist/assets/github-dark-C-LZuMrd.js +1 -0
  144. package/web-app/web-dist/assets/github-dark-default-DXG-b-1a.js +1 -0
  145. package/web-app/web-dist/assets/github-dark-dimmed-Bx1FflLF.js +1 -0
  146. package/web-app/web-dist/assets/github-dark-high-contrast-B_tTalzw.js +1 -0
  147. package/web-app/web-dist/assets/github-light-EUqPIrTm.js +1 -0
  148. package/web-app/web-dist/assets/github-light-default-BXViO-2h.js +1 -0
  149. package/web-app/web-dist/assets/github-light-high-contrast-B68TUdTA.js +1 -0
  150. package/web-app/web-dist/assets/gleam-CSRkHgEL.js +1 -0
  151. package/web-app/web-dist/assets/glimmer-js-Kq-kdTyV.js +1 -0
  152. package/web-app/web-dist/assets/glimmer-ts-D0RKLJNf.js +1 -0
  153. package/web-app/web-dist/assets/glsl-KwyfU2aa.js +1 -0
  154. package/web-app/web-dist/assets/gn-ilITqXS6.js +1 -0
  155. package/web-app/web-dist/assets/gnuplot-7GGW24-e.js +1 -0
  156. package/web-app/web-dist/assets/go-C-y9NEjX.js +1 -0
  157. package/web-app/web-dist/assets/go-rLFTqkRN.js +1 -0
  158. package/web-app/web-dist/assets/graphql-DSeOUAa2.js +1 -0
  159. package/web-app/web-dist/assets/graphql-fmXr3nnJ.js +1 -0
  160. package/web-app/web-dist/assets/groovy-CacY0gHj.js +1 -0
  161. package/web-app/web-dist/assets/gruvbox-dark-hard-C820rvS2.js +1 -0
  162. package/web-app/web-dist/assets/gruvbox-dark-medium-BPjhmG05.js +1 -0
  163. package/web-app/web-dist/assets/gruvbox-dark-soft-MrdJrrXF.js +1 -0
  164. package/web-app/web-dist/assets/gruvbox-light-hard-BC_s9l72.js +1 -0
  165. package/web-app/web-dist/assets/gruvbox-light-medium-BAWPOn9u.js +1 -0
  166. package/web-app/web-dist/assets/gruvbox-light-soft-BSMLrYjP.js +1 -0
  167. package/web-app/web-dist/assets/hack-trjVF3Po.js +1 -0
  168. package/web-app/web-dist/assets/haml-azVoxQRV.js +1 -0
  169. package/web-app/web-dist/assets/handlebars-B8_x7Zx7.js +1 -0
  170. package/web-app/web-dist/assets/handlebars-g7ZhGhI_.js +1 -0
  171. package/web-app/web-dist/assets/haskell-D8IpX4py.js +1 -0
  172. package/web-app/web-dist/assets/haxe-OTjmBuCE.js +1 -0
  173. package/web-app/web-dist/assets/hcl-CpzslTdj.js +1 -0
  174. package/web-app/web-dist/assets/hcl-Dh228itO.js +1 -0
  175. package/web-app/web-dist/assets/hjson-CxZEssPk.js +1 -0
  176. package/web-app/web-dist/assets/hlsl-Cvrh5tZx.js +1 -0
  177. package/web-app/web-dist/assets/horizon-CE9ld1lL.js +1 -0
  178. package/web-app/web-dist/assets/horizon-bright-DSNQnXHK.js +1 -0
  179. package/web-app/web-dist/assets/houston-CsvMBhTu.js +1 -0
  180. package/web-app/web-dist/assets/html-CfvRMgoC.js +1 -0
  181. package/web-app/web-dist/assets/html-D_7P5S4m.js +1 -0
  182. package/web-app/web-dist/assets/html-derivative-BYX_F_XH.js +1 -0
  183. package/web-app/web-dist/assets/html.worker-BO6WuOEO.js +502 -0
  184. package/web-app/web-dist/assets/htmlMode-CScDVtoR.js +1 -0
  185. package/web-app/web-dist/assets/http-BIVDpHT-.js +1 -0
  186. package/web-app/web-dist/assets/hurl-CFsshMju.js +1 -0
  187. package/web-app/web-dist/assets/hxml-B0Qn7Nwc.js +1 -0
  188. package/web-app/web-dist/assets/hy-CZbG8q4J.js +1 -0
  189. package/web-app/web-dist/assets/imba-DsUTQ-LC.js +1 -0
  190. package/web-app/web-dist/assets/index--lqPz7z2.css +2 -0
  191. package/web-app/web-dist/assets/index-BTchnWPZ.js +270 -0
  192. package/web-app/web-dist/assets/ini-B5eOa1yu.js +1 -0
  193. package/web-app/web-dist/assets/ini-sBoK_t0W.js +1 -0
  194. package/web-app/web-dist/assets/java-BEtHBSE6.js +1 -0
  195. package/web-app/web-dist/assets/java-D4RbCvBe.js +1 -0
  196. package/web-app/web-dist/assets/javascript-Bxx2wV4w.js +1 -0
  197. package/web-app/web-dist/assets/javascript-Cb010CKM.js +1 -0
  198. package/web-app/web-dist/assets/jetbrains-mono-cyrillic-wght-normal-D73BlboJ.woff2 +0 -0
  199. package/web-app/web-dist/assets/jetbrains-mono-greek-wght-normal-Bw9x6K1M.woff2 +0 -0
  200. package/web-app/web-dist/assets/jetbrains-mono-latin-ext-wght-normal-DBQx-q_a.woff2 +0 -0
  201. package/web-app/web-dist/assets/jetbrains-mono-latin-wght-normal-B9CIFXIH.woff2 +0 -0
  202. package/web-app/web-dist/assets/jetbrains-mono-vietnamese-wght-normal-Bt-aOZkq.woff2 +0 -0
  203. package/web-app/web-dist/assets/jinja-_ZS5zWwe.js +1 -0
  204. package/web-app/web-dist/assets/jison-D8mMEpcs.js +1 -0
  205. package/web-app/web-dist/assets/json-DWgqV4D1.js +1 -0
  206. package/web-app/web-dist/assets/json.worker-BkJRGcCJ.js +58 -0
  207. package/web-app/web-dist/assets/json5-BR5RXkoi.js +1 -0
  208. package/web-app/web-dist/assets/jsonMode-BMZXTtiq.js +7 -0
  209. package/web-app/web-dist/assets/jsonc-CYpm1nAK.js +1 -0
  210. package/web-app/web-dist/assets/jsonl-CmCQp5Yx.js +1 -0
  211. package/web-app/web-dist/assets/jsonnet-CJTPZ8u_.js +1 -0
  212. package/web-app/web-dist/assets/jssm-DXw9l8Rf.js +1 -0
  213. package/web-app/web-dist/assets/jsx-CZjSJa1f.js +1 -0
  214. package/web-app/web-dist/assets/julia-Bri6UV-V.js +1 -0
  215. package/web-app/web-dist/assets/julia-D4h2DZrs.js +1 -0
  216. package/web-app/web-dist/assets/just-bMqQi3xg.js +1 -0
  217. package/web-app/web-dist/assets/kanagawa-dragon-CXtmUGW6.js +1 -0
  218. package/web-app/web-dist/assets/kanagawa-lotus-BN08jTvb.js +1 -0
  219. package/web-app/web-dist/assets/kanagawa-wave-CTweb8Dz.js +1 -0
  220. package/web-app/web-dist/assets/kdl-CsD5j6eV.js +1 -0
  221. package/web-app/web-dist/assets/kotlin-BOotOW0E.js +1 -0
  222. package/web-app/web-dist/assets/kotlin-DhhofPvG.js +1 -0
  223. package/web-app/web-dist/assets/kusto-C7mF5XQf.js +1 -0
  224. package/web-app/web-dist/assets/laserwave-C_8bwKvT.js +1 -0
  225. package/web-app/web-dist/assets/latex-DThYi3CX.js +1 -0
  226. package/web-app/web-dist/assets/lean-CewbzKMR.js +1 -0
  227. package/web-app/web-dist/assets/less-B9JPFI3C.js +2 -0
  228. package/web-app/web-dist/assets/less-DVTAwKKz.js +1 -0
  229. package/web-app/web-dist/assets/lexon-CfSJPG6W.js +1 -0
  230. package/web-app/web-dist/assets/light-plus-DVQuIRkW.js +1 -0
  231. package/web-app/web-dist/assets/liquid-CUjzzP4r.js +1 -0
  232. package/web-app/web-dist/assets/liquid-CesB-zzl.js +1 -0
  233. package/web-app/web-dist/assets/llvm-Cm23YOpf.js +1 -0
  234. package/web-app/web-dist/assets/log-BNLmms1o.js +1 -0
  235. package/web-app/web-dist/assets/logo-Cluzi2Zq.js +1 -0
  236. package/web-app/web-dist/assets/lspLanguageFeatures-gTnJsses.js +4 -0
  237. package/web-app/web-dist/assets/lua-CsQS60Ue.js +1 -0
  238. package/web-app/web-dist/assets/lua-TGj_6NzO.js +1 -0
  239. package/web-app/web-dist/assets/luau-FMPmPwt6.js +1 -0
  240. package/web-app/web-dist/assets/m3-D-oSqn_W.js +1 -0
  241. package/web-app/web-dist/assets/make-Dixweg8N.js +1 -0
  242. package/web-app/web-dist/assets/markdown-BYOwaDjH.js +1 -0
  243. package/web-app/web-dist/assets/markdown-Cimd5fb3.js +1 -0
  244. package/web-app/web-dist/assets/marko-yoGoLK2m.js +1 -0
  245. package/web-app/web-dist/assets/material-theme-Bm3Qr25_.js +1 -0
  246. package/web-app/web-dist/assets/material-theme-darker-2IIEA8gg.js +1 -0
  247. package/web-app/web-dist/assets/material-theme-lighter-uhdI0v04.js +1 -0
  248. package/web-app/web-dist/assets/material-theme-ocean-CHQ94UKr.js +1 -0
  249. package/web-app/web-dist/assets/material-theme-palenight-B5W6OYN7.js +1 -0
  250. package/web-app/web-dist/assets/matlab-D7qyCx1q.js +1 -0
  251. package/web-app/web-dist/assets/mdc-BvtXU6eH.js +1 -0
  252. package/web-app/web-dist/assets/mdx-DQZ5AkYe.js +1 -0
  253. package/web-app/web-dist/assets/mdx-DrXGQbNB.js +1 -0
  254. package/web-app/web-dist/assets/mermaid-Bk4SNUv9.js +1 -0
  255. package/web-app/web-dist/assets/min-dark-BSWPekZh.js +1 -0
  256. package/web-app/web-dist/assets/min-light-DDpmG2fV.js +1 -0
  257. package/web-app/web-dist/assets/mips-CIPQ_RoX.js +1 -0
  258. package/web-app/web-dist/assets/mipsasm-BMqwQI7S.js +1 -0
  259. package/web-app/web-dist/assets/mojo-BgCJLMeH.js +1 -0
  260. package/web-app/web-dist/assets/monaco-setup-DAY1k6DB.js +16 -0
  261. package/web-app/web-dist/assets/monokai-CdkpiU2Y.js +1 -0
  262. package/web-app/web-dist/assets/moonbit-CaWjb8XO.js +1 -0
  263. package/web-app/web-dist/assets/move-B1IS1UjX.js +1 -0
  264. package/web-app/web-dist/assets/msdax-DauUninz.js +1 -0
  265. package/web-app/web-dist/assets/mysql-SOo6toE5.js +1 -0
  266. package/web-app/web-dist/assets/narrat-_X_XdTYD.js +1 -0
  267. package/web-app/web-dist/assets/nextflow-BJtWHP5T.js +1 -0
  268. package/web-app/web-dist/assets/nextflow-groovy-DJMQeKeT.js +1 -0
  269. package/web-app/web-dist/assets/nginx-DoUz032F.js +1 -0
  270. package/web-app/web-dist/assets/night-owl-DhmEMT88.js +1 -0
  271. package/web-app/web-dist/assets/night-owl-light-eJ-hLW7d.js +1 -0
  272. package/web-app/web-dist/assets/nim-B0Pl8B4R.js +1 -0
  273. package/web-app/web-dist/assets/nix-IvuFDN5E.js +1 -0
  274. package/web-app/web-dist/assets/nord-Cb4Vim4T.js +1 -0
  275. package/web-app/web-dist/assets/nushell-DcLAeLz5.js +1 -0
  276. package/web-app/web-dist/assets/objective-c-D1A_Heim.js +1 -0
  277. package/web-app/web-dist/assets/objective-c-FvmIjYaQ.js +1 -0
  278. package/web-app/web-dist/assets/objective-cpp-BsSzOQcm.js +1 -0
  279. package/web-app/web-dist/assets/ocaml-O90oeIOV.js +1 -0
  280. package/web-app/web-dist/assets/odin-B1RWQWA5.js +1 -0
  281. package/web-app/web-dist/assets/one-dark-pro-CLwyXe_n.js +1 -0
  282. package/web-app/web-dist/assets/one-light-D7Lr4KcI.js +1 -0
  283. package/web-app/web-dist/assets/openscad-BUDT5pXO.js +1 -0
  284. package/web-app/web-dist/assets/pascal-4ZHwLPI5.js +1 -0
  285. package/web-app/web-dist/assets/pascal-DrH0SRf2.js +1 -0
  286. package/web-app/web-dist/assets/pascaligo-D-ptJ9y-.js +1 -0
  287. package/web-app/web-dist/assets/perl-D2tfAALb.js +1 -0
  288. package/web-app/web-dist/assets/perl-oz_6vUea.js +1 -0
  289. package/web-app/web-dist/assets/pgsql-DTj74zXo.js +1 -0
  290. package/web-app/web-dist/assets/php-BImCcX5X.js +1 -0
  291. package/web-app/web-dist/assets/php-nr791fC2.js +1 -0
  292. package/web-app/web-dist/assets/pkl-ot-7Btpt.js +1 -0
  293. package/web-app/web-dist/assets/pla-CopQ2nXW.js +1 -0
  294. package/web-app/web-dist/assets/plastic-DQwYfKfQ.js +1 -0
  295. package/web-app/web-dist/assets/plsql-DGHpHOYJ.js +1 -0
  296. package/web-app/web-dist/assets/po-BiJDBrnU.js +1 -0
  297. package/web-app/web-dist/assets/poimandres-DRFjx7u4.js +1 -0
  298. package/web-app/web-dist/assets/polar-C7UOKdEL.js +1 -0
  299. package/web-app/web-dist/assets/postcss-BXeXVLqQ.js +1 -0
  300. package/web-app/web-dist/assets/postiats-43DmfD33.js +1 -0
  301. package/web-app/web-dist/assets/powerquery-D3hlyOfw.js +1 -0
  302. package/web-app/web-dist/assets/powerquery-DNMTfnFr.js +1 -0
  303. package/web-app/web-dist/assets/powershell-DmHpPYUd.js +1 -0
  304. package/web-app/web-dist/assets/powershell-DshXNtvi.js +1 -0
  305. package/web-app/web-dist/assets/prisma-BsRQq5mF.js +1 -0
  306. package/web-app/web-dist/assets/prolog-iXnhIJG7.js +1 -0
  307. package/web-app/web-dist/assets/proto-DB4EqR-F.js +1 -0
  308. package/web-app/web-dist/assets/protobuf-C531GsRP.js +2 -0
  309. package/web-app/web-dist/assets/pug-BcnpC8P_.js +1 -0
  310. package/web-app/web-dist/assets/pug-Z5eAx3Zn.js +1 -0
  311. package/web-app/web-dist/assets/puppet-CDv2pdJW.js +1 -0
  312. package/web-app/web-dist/assets/purescript-9MfHhQsQ.js +1 -0
  313. package/web-app/web-dist/assets/python-gzcpVVnB.js +1 -0
  314. package/web-app/web-dist/assets/python-ypRCBnvu.js +1 -0
  315. package/web-app/web-dist/assets/qml-DFDAunHY.js +1 -0
  316. package/web-app/web-dist/assets/qmldir-DCQb3MpD.js +1 -0
  317. package/web-app/web-dist/assets/qsharp-DkqhCAOL.js +1 -0
  318. package/web-app/web-dist/assets/qss-Fe1Jh2GI.js +1 -0
  319. package/web-app/web-dist/assets/r-BwWrilGY.js +1 -0
  320. package/web-app/web-dist/assets/r-fCpuAR7u.js +1 -0
  321. package/web-app/web-dist/assets/racket-DcIDlBhZ.js +1 -0
  322. package/web-app/web-dist/assets/raku-B3gFvitq.js +1 -0
  323. package/web-app/web-dist/assets/razor-1_376SZM.js +1 -0
  324. package/web-app/web-dist/assets/razor-aqrhpwqZ.js +1 -0
  325. package/web-app/web-dist/assets/red-CJ3rzSJv.js +1 -0
  326. package/web-app/web-dist/assets/redis-ClamHrr6.js +1 -0
  327. package/web-app/web-dist/assets/redshift-DT7zqm-g.js +1 -0
  328. package/web-app/web-dist/assets/reg-CRGYupPL.js +1 -0
  329. package/web-app/web-dist/assets/regexp-B4yxx-Ty.js +1 -0
  330. package/web-app/web-dist/assets/rel-BtDbiS_P.js +1 -0
  331. package/web-app/web-dist/assets/restructuredtext-BYgofb2h.js +1 -0
  332. package/web-app/web-dist/assets/riscv-Ckw8ddFX.js +1 -0
  333. package/web-app/web-dist/assets/ron-VUp2lXgN.js +1 -0
  334. package/web-app/web-dist/assets/rose-pine-BthvhNj6.js +1 -0
  335. package/web-app/web-dist/assets/rose-pine-dawn-Dg85fqjY.js +1 -0
  336. package/web-app/web-dist/assets/rose-pine-moon-hon4tzzS.js +1 -0
  337. package/web-app/web-dist/assets/rosmsg-CAekHB0j.js +1 -0
  338. package/web-app/web-dist/assets/rst-2vG6f11Y.js +1 -0
  339. package/web-app/web-dist/assets/ruby-DezsRK8O.js +1 -0
  340. package/web-app/web-dist/assets/ruby-Dj6bCFXR.js +1 -0
  341. package/web-app/web-dist/assets/rust-Cfkwpbl8.js +1 -0
  342. package/web-app/web-dist/assets/rust-DdL9SqIa.js +1 -0
  343. package/web-app/web-dist/assets/sas-BhVZ4qL2.js +1 -0
  344. package/web-app/web-dist/assets/sass-DXrisJhu.js +1 -0
  345. package/web-app/web-dist/assets/sb-CcwsVR0C.js +1 -0
  346. package/web-app/web-dist/assets/scala-DHpiXF5c.js +1 -0
  347. package/web-app/web-dist/assets/scala-DKOlJaKm.js +1 -0
  348. package/web-app/web-dist/assets/scheme-BeGwcela.js +1 -0
  349. package/web-app/web-dist/assets/scheme-DQCgrYNe.js +1 -0
  350. package/web-app/web-dist/assets/scss-QdjMO_xV.js +1 -0
  351. package/web-app/web-dist/assets/scss-gp-XZpBa.js +3 -0
  352. package/web-app/web-dist/assets/sdbl-bTVj8UrX.js +1 -0
  353. package/web-app/web-dist/assets/shaderlab-TOUzSsQk.js +1 -0
  354. package/web-app/web-dist/assets/shell-CC2rA5mh.js +1 -0
  355. package/web-app/web-dist/assets/shellscript-BnlgeVVx.js +1 -0
  356. package/web-app/web-dist/assets/shellsession-CyO2fnhB.js +1 -0
  357. package/web-app/web-dist/assets/slack-dark-DnToyrRv.js +1 -0
  358. package/web-app/web-dist/assets/slack-ochin-B2OO5cIa.js +1 -0
  359. package/web-app/web-dist/assets/smalltalk-B16xEiuN.js +1 -0
  360. package/web-app/web-dist/assets/snazzy-light-4G7pJPwS.js +1 -0
  361. package/web-app/web-dist/assets/solarized-dark-DV17i1UV.js +1 -0
  362. package/web-app/web-dist/assets/solarized-light-DSh2HLQt.js +1 -0
  363. package/web-app/web-dist/assets/solidity-BEEn4gHE.js +1 -0
  364. package/web-app/web-dist/assets/solidity-CKzVLygQ.js +1 -0
  365. package/web-app/web-dist/assets/sophia-CRfGWb83.js +1 -0
  366. package/web-app/web-dist/assets/soy-DIkw6E88.js +1 -0
  367. package/web-app/web-dist/assets/sparql-D_Lu-MrJ.js +1 -0
  368. package/web-app/web-dist/assets/sparql-D_iOobhT.js +1 -0
  369. package/web-app/web-dist/assets/splunk-BC2Px7Mm.js +1 -0
  370. package/web-app/web-dist/assets/sql-DGnQv6iD.js +1 -0
  371. package/web-app/web-dist/assets/sql-NEE52Syq.js +1 -0
  372. package/web-app/web-dist/assets/ssh-config-BgfXC-Er.js +1 -0
  373. package/web-app/web-dist/assets/st-DbInun42.js +1 -0
  374. package/web-app/web-dist/assets/stata-DvkM932O.js +1 -0
  375. package/web-app/web-dist/assets/stylus-B6D30XZt.js +1 -0
  376. package/web-app/web-dist/assets/surrealql-B4-Q8tqV.js +1 -0
  377. package/web-app/web-dist/assets/svelte-p6yBy-Ki.js +1 -0
  378. package/web-app/web-dist/assets/swift-Bxkupp3x.js +1 -0
  379. package/web-app/web-dist/assets/swift-DonLKvLd.js +1 -0
  380. package/web-app/web-dist/assets/synthwave-84-nFMaYfgc.js +1 -0
  381. package/web-app/web-dist/assets/system-verilog-DJ5XKQeo.js +1 -0
  382. package/web-app/web-dist/assets/systemd-BxMlprV5.js +1 -0
  383. package/web-app/web-dist/assets/systemverilog-Bz4Y3fRF.js +1 -0
  384. package/web-app/web-dist/assets/talonscript-CohzipZa.js +1 -0
  385. package/web-app/web-dist/assets/tasl-DMoTqEGO.js +1 -0
  386. package/web-app/web-dist/assets/tcl-CZd0xW_V.js +1 -0
  387. package/web-app/web-dist/assets/tcl-DISqw1ZD.js +1 -0
  388. package/web-app/web-dist/assets/templ-C7EkuiZr.js +1 -0
  389. package/web-app/web-dist/assets/terraform-DswuEJGm.js +1 -0
  390. package/web-app/web-dist/assets/tex-DkmD8uFC.js +1 -0
  391. package/web-app/web-dist/assets/tokyo-night-oM2G3aXe.js +1 -0
  392. package/web-app/web-dist/assets/toml-CcmNWLt0.js +1 -0
  393. package/web-app/web-dist/assets/ts-tags-U-hncHg4.js +1 -0
  394. package/web-app/web-dist/assets/ts.worker-B0J26iPs.js +67734 -0
  395. package/web-app/web-dist/assets/tsMode-D--bqr5b.js +11 -0
  396. package/web-app/web-dist/assets/tsv-sltzmVWM.js +1 -0
  397. package/web-app/web-dist/assets/tsx-MJ0-9sYG.js +1 -0
  398. package/web-app/web-dist/assets/turtle-ByJddavk.js +1 -0
  399. package/web-app/web-dist/assets/twig-CU0OP-IA.js +1 -0
  400. package/web-app/web-dist/assets/twig-De2hgUGE.js +1 -0
  401. package/web-app/web-dist/assets/typescript-C17ZkDe8.js +1 -0
  402. package/web-app/web-dist/assets/typescript-DnLjiKtn.js +1 -0
  403. package/web-app/web-dist/assets/typespec-B8J7ngcE.js +1 -0
  404. package/web-app/web-dist/assets/typespec-BRdr0IET.js +1 -0
  405. package/web-app/web-dist/assets/typst-DI99ib-x.js +1 -0
  406. package/web-app/web-dist/assets/v-DETTlOr0.js +1 -0
  407. package/web-app/web-dist/assets/vala-zf12oZj6.js +1 -0
  408. package/web-app/web-dist/assets/vb-DV3o63ZY.js +1 -0
  409. package/web-app/web-dist/assets/vb-Djn5o6TS.js +1 -0
  410. package/web-app/web-dist/assets/verilog-CiiDBU1e.js +1 -0
  411. package/web-app/web-dist/assets/vesper-DdrHHSXu.js +1 -0
  412. package/web-app/web-dist/assets/vhdl-BroJfC0k.js +1 -0
  413. package/web-app/web-dist/assets/viml-DvXPmvsu.js +1 -0
  414. package/web-app/web-dist/assets/vitesse-black-fwtXNY1n.js +1 -0
  415. package/web-app/web-dist/assets/vitesse-dark-BZCL-v6S.js +1 -0
  416. package/web-app/web-dist/assets/vitesse-light-VbXTXTou.js +1 -0
  417. package/web-app/web-dist/assets/vue-Db7nY3ba.js +1 -0
  418. package/web-app/web-dist/assets/vue-html-BvAbiAw1.js +1 -0
  419. package/web-app/web-dist/assets/vue-vine-BEaIQIlA.js +1 -0
  420. package/web-app/web-dist/assets/vyper-CgoNMtux.js +1 -0
  421. package/web-app/web-dist/assets/wasm-BnjxR4X6.js +1 -0
  422. package/web-app/web-dist/assets/wasm-ByWQv1Qj.js +1 -0
  423. package/web-app/web-dist/assets/wenyan-C8pVoKbM.js +1 -0
  424. package/web-app/web-dist/assets/wgsl-BsKzXJz4.js +1 -0
  425. package/web-app/web-dist/assets/wgsl-DpFanUEy.js +298 -0
  426. package/web-app/web-dist/assets/wikitext-ClFFjSW2.js +1 -0
  427. package/web-app/web-dist/assets/wit-DdvCle-K.js +1 -0
  428. package/web-app/web-dist/assets/wolfram-DLL8P-h_.js +1 -0
  429. package/web-app/web-dist/assets/xml-CA9lHFQV.js +1 -0
  430. package/web-app/web-dist/assets/xml-an4Nuuqq.js +1 -0
  431. package/web-app/web-dist/assets/xsl-D3NQgH22.js +1 -0
  432. package/web-app/web-dist/assets/yaml-CwRYMJka.js +1 -0
  433. package/web-app/web-dist/assets/yaml-Diiu6O9P.js +1 -0
  434. package/web-app/web-dist/assets/zenscript-BnlCZFoB.js +1 -0
  435. package/web-app/web-dist/assets/zig-CMLA9XwU.js +1 -0
  436. package/web-app/web-dist/index.html +16 -0
  437. package/zuzuu/capture/bin/capture.mjs +99 -0
  438. package/zuzuu/capture/bin/inspect-trace.mjs +13 -0
  439. package/zuzuu/capture-core.mjs +3 -3
  440. package/zuzuu/commands/capture.mjs +1 -1
  441. package/zuzuu/commands/doctor.mjs +1 -1
  442. package/zuzuu/commands/hook.mjs +1 -1
  443. package/zuzuu/commands/init.mjs +1 -1
  444. package/zuzuu/commands/status.mjs +1 -1
  445. package/zuzuu/commands/trace.mjs +1 -1
  446. package/zuzuu/commands/web.mjs +49 -45
  447. package/zuzuu/knowledge/distill.mjs +1 -1
  448. package/zuzuu/live/reconcile.mjs +1 -1
  449. package/zuzuu/store.mjs +1 -1
  450. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/claude-code.mjs +0 -0
  451. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/codex.mjs +0 -0
  452. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/gemini-cli.mjs +0 -0
  453. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/host-adapter.mjs +0 -0
  454. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/opencode.mjs +0 -0
  455. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/pi.mjs +0 -0
  456. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/registry.mjs +0 -0
  457. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/adapters/signals.mjs +0 -0
  458. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/core/event.mjs +0 -0
  459. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/core/ids.mjs +0 -0
  460. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/core/otlp.mjs +0 -0
  461. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/core/render.mjs +0 -0
  462. /package/{experiments/experiment-1-trace-capture → zuzuu/capture}/core/spans.mjs +0 -0
package/README.md CHANGED
@@ -18,6 +18,8 @@ npm install -g @zuzuucodes/cli # zero dependencies — installs the `zuzuu` co
18
18
  # no coding agent yet? one command gives you a fully faculty-equipped one:
19
19
  zuzuu code # scaffold the faculty home, install + wire OpenCode, launch it (capture + gate + grounding)
20
20
 
21
+ zuzuu web # prefer a visual app? the workbench: Home → review ceremony → embedded-terminal agent sessions
22
+
21
23
  # already run Claude Code / Gemini / Codex / OpenCode / pi? wrap the one you have:
22
24
  zuzuu init # scaffold your project's agent home (.zuzuu/) — git-style, hidden like .git
23
25
  zuzuu explain # the 5 faculties + how graduation works (you're always in the loop)
@@ -38,7 +40,7 @@ zuzuu doctor # health + lost-session reconciliation
38
40
 
39
41
  ¹ **Codex is interactive-only** — `codex exec` (headless) fires no hooks (verified, v0.138.0), so live capture + gate work when you run Codex interactively; headless Codex still gets post-hoc `zuzuu capture`.
40
42
 
41
- All five verified against **real sessions** — never fixtures; every host's live capture + gate was wired from **real captured hook payloads** and dogfooded end-to-end ([`experiments/LOG.md`](experiments/LOG.md) exp-11 Gemini/Codex, exp-12 OpenCode/pi). Gate semantics are host-honest: deny hard-blocks everywhere; `ask` maps to a native prompt on Claude, defers to the host elsewhere.
43
+ All five verified against **real sessions** — never fixtures; every host's live capture + gate was wired from **real captured hook payloads** and dogfooded end-to-end ([`docs/LOG.md`](docs/LOG.md) exp-11 Gemini/Codex, exp-12 OpenCode/pi). Gate semantics are host-honest: deny hard-blocks everywhere; `ask` maps to a native prompt on Claude, defers to the host elsewhere.
42
44
 
43
45
  **Prerequisites:** Node ≥ 22 — that's it. You need at least one supported agent you've already used, so a session exists to capture. (Hacking on zuzuu itself? `git clone https://github.com/h1902y/zuzuu && cd zuzuu && npm link`.)
44
46
 
@@ -75,15 +77,14 @@ All five verified against **real sessions** — never fixtures; every host's liv
75
77
 
76
78
  | Path | What |
77
79
  |---|---|
78
- | [`zuzuu/`](zuzuu/) + `bin/zuzuu.mjs` | the CLI — capture, live lifecycle, faculty home (product surface) |
79
- | [`experiments/`](experiments/) | spike code + [`LOG.md`](experiments/LOG.md) the build journal (hypothesis real-data proof conclusions per experiment) |
80
- | [`app/`](app/) | the durable application skeleton (be / run / evolve) — proven code harvests here |
80
+ | [`zuzuu/`](zuzuu/) + `bin/zuzuu.mjs` | the CLI — capture pipeline (`zuzuu/capture/`), live lifecycle, faculty home (product surface) |
81
+ | [`web/`](web/) | the visual workbenchnested project (daemon + React SPA), ships bundled inside the npm package |
81
82
  | [`tests/`](tests/) | hermetic unit + regression (`npm test`) + real-data smoke playgrounds (`npm run playground`) |
82
- | [`docs/`](docs/) | [`DESIGN.md`](docs/DESIGN.md) (the canon) + [`inspiration/`](docs/inspiration/) (the research shelf: 100-project survey + 5 audits) |
83
+ | [`docs/`](docs/) | [`DESIGN.md`](docs/DESIGN.md) (the canon) + [`LOG.md`](docs/LOG.md) (the build journal) + [`inspiration/`](docs/inspiration/) (the research shelf) |
83
84
 
84
85
  ## How this is built (the method)
85
86
 
86
- **Experiment → prove on real data conclude harvest.** Every capability starts as a numbered experiment with a hypothesis; it must be verified against *real* sessions/wire data (never invented fixtures) before it counts; lessons land in the experiment’s Conclusions section; proven parts graduate into `app/`. Built in public — day-by-day on X ([@h1902y](https://x.com/h1902y)).
87
+ **Prove on real data, record in the journal.** Every capability is verified against *real* sessions/wire data (never invented fixtures) before it counts; the record lives in [`docs/LOG.md`](docs/LOG.md) (append-only). Live experimentation happens in disposable project directories outside the repo, keeping the codebase clean. Built in public — day-by-day on X ([@h1902y](https://x.com/h1902y)).
87
88
 
88
89
  ## License & status
89
90
 
@@ -0,0 +1,15 @@
1
+ #!/usr/bin/env node
2
+ // Direct entry to the bundled workbench daemon (the `zuzuu web` command is the
3
+ // porcelain; this bin exists for people who want the daemon itself).
4
+ import { existsSync } from 'node:fs';
5
+ import { join, dirname } from 'node:path';
6
+ import { fileURLToPath } from 'node:url';
7
+
8
+ const entry = join(dirname(fileURLToPath(import.meta.url)), '..', 'web-app', 'dist', 'index.js');
9
+ if (!existsSync(entry)) {
10
+ console.error('zuzuu-web: the bundled workbench is not staged.');
11
+ console.error(' installed package → reinstall: npm i -g @zuzuucodes/cli (without --omit=optional)');
12
+ console.error(' repo checkout → build it: npm run build:web');
13
+ process.exit(1);
14
+ }
15
+ await import(entry);
package/package.json CHANGED
@@ -1,12 +1,12 @@
1
1
  {
2
2
  "name": "@zuzuucodes/cli",
3
- "version": "1.1.0",
3
+ "version": "1.2.1",
4
4
  "private": false,
5
5
  "publishConfig": {
6
6
  "access": "public"
7
7
  },
8
8
  "type": "module",
9
- "description": "Give the coding agent you already run (Claude Code, Codex, Gemini CLI, OpenCode) evolving faculties knowledge, memory, actions, instructions, guardrails with host-agnostic OpenTelemetry session tracing. Zero dependencies.",
9
+ "description": "Give the coding agent you already run (Claude Code, Codex, Gemini CLI, OpenCode) evolving faculties \u2014 knowledge, memory, actions, instructions, guardrails \u2014 with host-agnostic OpenTelemetry session tracing. Zero dependencies.",
10
10
  "keywords": [
11
11
  "ai-agents",
12
12
  "claude-code",
@@ -35,22 +35,33 @@
35
35
  },
36
36
  "bin": {
37
37
  "zz": "bin/zuzuu.mjs",
38
- "zuzuu": "bin/zuzuu.mjs"
38
+ "zuzuu": "bin/zuzuu.mjs",
39
+ "zuzuu-web": "bin/zuzuu-web.mjs"
39
40
  },
40
41
  "files": [
41
42
  "bin/",
42
43
  "zuzuu/",
43
- "experiments/experiment-1-trace-capture/core/",
44
- "experiments/experiment-1-trace-capture/adapters/",
44
+ "web-app/",
45
45
  "LICENSE",
46
46
  "README.md"
47
47
  ],
48
48
  "scripts": {
49
49
  "zuzuu": "node bin/zuzuu.mjs",
50
- "capture": "node experiments/experiment-1-trace-capture/bin/capture.mjs",
51
- "inspect": "node experiments/experiment-1-trace-capture/bin/inspect-trace.mjs",
50
+ "capture": "node zuzuu/capture/bin/capture.mjs",
51
+ "inspect": "node zuzuu/capture/bin/inspect-trace.mjs",
52
52
  "test": "node --test 'tests/**/*.test.mjs'",
53
53
  "playground": "node tests/playground/run.mjs",
54
- "prepublishOnly": "npm test"
54
+ "prepublishOnly": "npm test",
55
+ "build:web": "node scripts/build-web.mjs"
56
+ },
57
+ "optionalDependencies": {
58
+ "@hono/node-server": "^2.0.4",
59
+ "@lydell/node-pty": "^1.2.0-beta.12",
60
+ "@xterm/addon-serialize": "^0.14.0",
61
+ "@xterm/headless": "^6.0.0",
62
+ "archiver": "^8.0.0",
63
+ "chokidar": "^5.0.0",
64
+ "hono": "^4.12.25",
65
+ "ws": "^8.21.0"
55
66
  }
56
67
  }
@@ -0,0 +1,48 @@
1
+ import fsp from "node:fs/promises";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ const MAX_ENTRIES = 1000;
5
+ /**
6
+ * List the subdirectories of an absolute path, for the vault folder picker.
7
+ *
8
+ * SECURITY: this deliberately reads OUTSIDE the workspace sandbox — choosing
9
+ * a new vault means browsing the user's own machine. It is intentionally not
10
+ * routed through `safePath`. Mitigations: auth-gated + loopback-only at the
11
+ * server layer, and it returns DIRECTORY NAMES ONLY (never file contents).
12
+ */
13
+ export async function listDirs(input) {
14
+ const abs = input && path.isAbsolute(input) ? path.resolve(input) : os.homedir();
15
+ const parent = path.dirname(abs);
16
+ const dirs = [];
17
+ const entries = await fsp.readdir(abs, { withFileTypes: true });
18
+ for (const e of entries) {
19
+ if (dirs.length >= MAX_ENTRIES)
20
+ break;
21
+ if (e.name.startsWith("."))
22
+ continue;
23
+ let isDir = e.isDirectory();
24
+ if (e.isSymbolicLink()) {
25
+ try {
26
+ isDir = (await fsp.stat(path.join(abs, e.name))).isDirectory();
27
+ }
28
+ catch {
29
+ isDir = false;
30
+ }
31
+ }
32
+ if (isDir)
33
+ dirs.push({ name: e.name, path: path.join(abs, e.name) });
34
+ }
35
+ dirs.sort((a, b) => a.name.localeCompare(b.name));
36
+ return { path: abs, parent: parent === abs ? null : parent, dirs };
37
+ }
38
+ /** Create a directory under an absolute parent, returning its absolute path. */
39
+ export async function mkdirIn(parent, name) {
40
+ if (!path.isAbsolute(parent))
41
+ throw new Error("parent must be absolute");
42
+ if (!name || name.includes("/") || name.includes("\\") || name === "." || name === "..") {
43
+ throw new Error("invalid folder name");
44
+ }
45
+ const abs = path.join(parent, name);
46
+ await fsp.mkdir(abs, { recursive: false });
47
+ return abs;
48
+ }
@@ -0,0 +1,52 @@
1
+ import fs from "node:fs";
2
+ import fsp from "node:fs/promises";
3
+ import os from "node:os";
4
+ import path from "node:path";
5
+ const DIR = path.join(os.homedir(), ".webcode");
6
+ const FILE = path.join(DIR, "config.json");
7
+ const RECENT_MAX = 10;
8
+ const DEFAULT = { recent: [], onboarded: false };
9
+ /** Load persisted config, tolerating a missing/corrupt file. */
10
+ export async function load() {
11
+ try {
12
+ const raw = await fsp.readFile(FILE, "utf8");
13
+ const parsed = JSON.parse(raw);
14
+ return {
15
+ recent: Array.isArray(parsed.recent) ? parsed.recent.filter((p) => typeof p === "string") : [],
16
+ onboarded: parsed.onboarded === true,
17
+ };
18
+ }
19
+ catch {
20
+ return { ...DEFAULT };
21
+ }
22
+ }
23
+ async function save(cfg) {
24
+ await fsp.mkdir(DIR, { recursive: true });
25
+ await fsp.writeFile(FILE, JSON.stringify(cfg, null, 2) + "\n", "utf8");
26
+ }
27
+ /** Prepend a workspace to the recent list (deduped, most-recent-first, capped). */
28
+ export async function addRecent(root) {
29
+ const cfg = await load();
30
+ cfg.recent = [root, ...cfg.recent.filter((p) => p !== root)].slice(0, RECENT_MAX);
31
+ await save(cfg);
32
+ return cfg;
33
+ }
34
+ export async function setOnboarded() {
35
+ const cfg = await load();
36
+ cfg.onboarded = true;
37
+ await save(cfg);
38
+ return cfg;
39
+ }
40
+ /** Pure helper (exported for tests): compute the next recent list. */
41
+ export function nextRecent(recent, root) {
42
+ return [root, ...recent.filter((p) => p !== root)].slice(0, RECENT_MAX);
43
+ }
44
+ /** Synchronous best-effort load for startup paths that can't await. */
45
+ export function loadSync() {
46
+ try {
47
+ return { ...DEFAULT, ...JSON.parse(fs.readFileSync(FILE, "utf8")) };
48
+ }
49
+ catch {
50
+ return { ...DEFAULT };
51
+ }
52
+ }
@@ -0,0 +1,87 @@
1
+ import { execFile, spawn } from "node:child_process";
2
+ import { promisify } from "node:util";
3
+ import fsp from "node:fs/promises";
4
+ import path from "node:path";
5
+ import { toRel } from "./safe-path.js";
6
+ const execFileAsync = promisify(execFile);
7
+ const IGNORE = new Set([".git", "node_modules", "dist", ".next", ".cache"]);
8
+ let rgAvailable = null;
9
+ async function hasRg() {
10
+ if (rgAvailable === null) {
11
+ rgAvailable = await execFileAsync("rg", ["--version"], { timeout: 2000 })
12
+ .then(() => true)
13
+ .catch(() => false);
14
+ }
15
+ return rgAvailable;
16
+ }
17
+ /**
18
+ * Flat list of workspace files for the ⌘K palette. Prefers `rg --files`
19
+ * (honors .gitignore, fast); falls back to a bounded manual walk.
20
+ */
21
+ export async function listFiles(root, limit = 5000) {
22
+ if (await hasRg())
23
+ return rgFiles(root, limit);
24
+ return walkFiles(root, limit);
25
+ }
26
+ function rgFiles(root, limit) {
27
+ return new Promise((resolve) => {
28
+ const proc = spawn("rg", ["--files", "--hidden", "-g", "!.git"], {
29
+ cwd: root,
30
+ stdio: ["ignore", "pipe", "ignore"],
31
+ });
32
+ const files = [];
33
+ let buf = "";
34
+ let truncated = false;
35
+ proc.stdout.on("data", (chunk) => {
36
+ buf += chunk.toString("utf8");
37
+ let nl;
38
+ while ((nl = buf.indexOf("\n")) >= 0) {
39
+ const line = buf.slice(0, nl);
40
+ buf = buf.slice(nl + 1);
41
+ if (line)
42
+ files.push(line);
43
+ if (files.length >= limit) {
44
+ truncated = true;
45
+ proc.kill("SIGTERM");
46
+ resolve({ files, truncated });
47
+ return;
48
+ }
49
+ }
50
+ });
51
+ proc.on("close", () => resolve({ files, truncated }));
52
+ proc.on("error", () => resolve({ files, truncated }));
53
+ });
54
+ }
55
+ async function walkFiles(root, limit) {
56
+ const files = [];
57
+ let truncated = false;
58
+ const stack = [root];
59
+ while (stack.length > 0 && files.length < limit) {
60
+ const dir = stack.pop();
61
+ let entries;
62
+ try {
63
+ entries = await fsp.readdir(dir, { withFileTypes: true });
64
+ }
65
+ catch {
66
+ continue;
67
+ }
68
+ for (const e of entries) {
69
+ if (e.name.startsWith(".") && e.name !== ".webcode")
70
+ continue;
71
+ if (IGNORE.has(e.name))
72
+ continue;
73
+ const full = path.join(dir, e.name);
74
+ if (e.isDirectory())
75
+ stack.push(full);
76
+ else if (e.isFile()) {
77
+ files.push(toRel(root, full));
78
+ if (files.length >= limit) {
79
+ truncated = true;
80
+ break;
81
+ }
82
+ }
83
+ }
84
+ }
85
+ files.sort();
86
+ return { files, truncated };
87
+ }
@@ -0,0 +1,16 @@
1
+ /**
2
+ * Binary WS framing: 1 opcode byte followed by the payload
3
+ * (raw bytes for I/O frames, UTF-8 JSON for control frames).
4
+ */
5
+ export function encodeFrame(op, payload = "") {
6
+ const body = typeof payload === "string" ? Buffer.from(payload, "utf8") : payload;
7
+ const buf = Buffer.allocUnsafe(1 + body.length);
8
+ buf[0] = op;
9
+ buf.set(body, 1);
10
+ return buf;
11
+ }
12
+ export function decodeFrame(data) {
13
+ if (data.length === 0)
14
+ throw new Error("empty frame");
15
+ return { op: data[0], payload: data.subarray(1) };
16
+ }
@@ -0,0 +1,226 @@
1
+ import fs from "node:fs";
2
+ import fsp from "node:fs/promises";
3
+ import path from "node:path";
4
+ import { spawn } from "node:child_process";
5
+ import { Readable } from "node:stream";
6
+ import { Hono } from "hono";
7
+ import { ZipArchive } from "archiver";
8
+ import { PathError, resolveSafe, toRel } from "./safe-path.js";
9
+ const STAT_CONCURRENCY = 64;
10
+ const MIME = {
11
+ ".txt": "text/plain; charset=utf-8",
12
+ ".md": "text/markdown; charset=utf-8",
13
+ ".json": "application/json",
14
+ ".js": "text/javascript",
15
+ ".ts": "text/plain; charset=utf-8",
16
+ ".html": "text/html; charset=utf-8",
17
+ ".css": "text/css",
18
+ ".png": "image/png",
19
+ ".jpg": "image/jpeg",
20
+ ".jpeg": "image/jpeg",
21
+ ".gif": "image/gif",
22
+ ".svg": "image/svg+xml",
23
+ ".webp": "image/webp",
24
+ ".pdf": "application/pdf",
25
+ ".zip": "application/zip",
26
+ ".mp4": "video/mp4",
27
+ ".m4v": "video/mp4",
28
+ ".webm": "video/webm",
29
+ ".mov": "video/quicktime",
30
+ ".mp3": "audio/mpeg",
31
+ ".wav": "audio/wav",
32
+ ".ogg": "audio/ogg",
33
+ ".m4a": "audio/mp4",
34
+ ".aac": "audio/aac",
35
+ ".flac": "audio/flac",
36
+ ".avif": "image/avif",
37
+ ".bmp": "image/bmp",
38
+ ".ico": "image/x-icon",
39
+ ".woff2": "font/woff2",
40
+ };
41
+ function contentType(file) {
42
+ return MIME[path.extname(file).toLowerCase()] ?? "application/octet-stream";
43
+ }
44
+ function kindOf(dirent) {
45
+ if (dirent.isSymbolicLink())
46
+ return "symlink";
47
+ if (dirent.isDirectory())
48
+ return "dir";
49
+ if (dirent.isFile())
50
+ return "file";
51
+ return "other";
52
+ }
53
+ async function mapLimit(items, limit, fn) {
54
+ const results = new Array(items.length);
55
+ let next = 0;
56
+ const workers = Array.from({ length: Math.min(limit, items.length) }, async () => {
57
+ for (;;) {
58
+ const i = next++;
59
+ if (i >= items.length)
60
+ return;
61
+ results[i] = await fn(items[i]);
62
+ }
63
+ });
64
+ await Promise.all(workers);
65
+ return results;
66
+ }
67
+ export function createFsApi(getRoot) {
68
+ const app = new Hono();
69
+ // resolve the (mutable) workspace root once per request, so handlers below
70
+ // keep referencing a stable `root` while the daemon can re-root at runtime.
71
+ let root = getRoot();
72
+ app.use("*", async (_c, next) => {
73
+ root = getRoot();
74
+ await next();
75
+ });
76
+ app.onError((err, c) => {
77
+ if (err instanceof PathError)
78
+ return c.json({ error: err.message }, 403);
79
+ const code = err.code;
80
+ if (code === "ENOENT")
81
+ return c.json({ error: "not found" }, 404);
82
+ if (code === "EEXIST")
83
+ return c.json({ error: "already exists" }, 409);
84
+ console.error("[fs-api]", err);
85
+ return c.json({ error: "internal error" }, 500);
86
+ });
87
+ app.get("/list", async (c) => {
88
+ const abs = await resolveSafe(root, c.req.query("path") ?? "");
89
+ const dirents = await fsp.readdir(abs, { withFileTypes: true });
90
+ const entries = await mapLimit(dirents, STAT_CONCURRENCY, async (d) => {
91
+ const full = path.join(abs, d.name);
92
+ const kind = kindOf(d);
93
+ let size = 0;
94
+ let mtimeMs = 0;
95
+ let targetKind;
96
+ try {
97
+ const lst = await fsp.lstat(full);
98
+ size = lst.size;
99
+ mtimeMs = lst.mtimeMs;
100
+ if (kind === "symlink") {
101
+ // Only report a resolvable target if it stays inside the workspace.
102
+ await resolveSafe(root, toRel(root, full));
103
+ const st = await fsp.stat(full);
104
+ targetKind = st.isDirectory() ? "dir" : st.isFile() ? "file" : "other";
105
+ }
106
+ }
107
+ catch {
108
+ // broken symlink / vanished entry — report what we know
109
+ }
110
+ return { name: d.name, kind, targetKind, size, mtimeMs };
111
+ });
112
+ entries.sort((a, b) => {
113
+ const aDir = a.kind === "dir" || a.targetKind === "dir" ? 0 : 1;
114
+ const bDir = b.kind === "dir" || b.targetKind === "dir" ? 0 : 1;
115
+ return aDir - bDir || a.name.localeCompare(b.name);
116
+ });
117
+ const body = { path: toRel(root, abs), entries };
118
+ return c.json(body);
119
+ });
120
+ app.post("/mkdir", async (c) => {
121
+ const body = await c.req.json();
122
+ const abs = await resolveSafe(root, body.path ?? "");
123
+ if (abs === root)
124
+ return c.json({ error: "invalid path" }, 400);
125
+ await fsp.mkdir(abs, { recursive: true });
126
+ return c.json({ ok: true });
127
+ });
128
+ // Save an editor buffer to disk.
129
+ app.post("/write", async (c) => {
130
+ const body = await c.req.json();
131
+ const abs = await resolveSafe(root, body.path ?? "");
132
+ if (abs === root)
133
+ return c.json({ error: "invalid path" }, 400);
134
+ if (typeof body.content !== "string")
135
+ return c.json({ error: "content required" }, 400);
136
+ await fsp.mkdir(path.dirname(abs), { recursive: true });
137
+ await fsp.writeFile(abs, body.content, "utf8");
138
+ return c.json({ ok: true });
139
+ });
140
+ app.post("/rename", async (c) => {
141
+ const body = await c.req.json();
142
+ const from = await resolveSafe(root, body.from ?? "");
143
+ const to = await resolveSafe(root, body.to ?? "");
144
+ if (from === root || to === root)
145
+ return c.json({ error: "invalid path" }, 400);
146
+ await fsp.rename(from, to);
147
+ return c.json({ ok: true });
148
+ });
149
+ app.post("/delete", async (c) => {
150
+ const body = await c.req.json();
151
+ if (!Array.isArray(body.paths) || body.paths.length === 0) {
152
+ return c.json({ error: "paths required" }, 400);
153
+ }
154
+ for (const p of body.paths) {
155
+ const abs = await resolveSafe(root, p);
156
+ if (abs === root)
157
+ return c.json({ error: "cannot delete workspace root" }, 400);
158
+ await fsp.rm(abs, { recursive: true, force: true });
159
+ }
160
+ return c.json({ ok: true });
161
+ });
162
+ app.get("/download", async (c) => {
163
+ const abs = await resolveSafe(root, c.req.query("path") ?? "");
164
+ const st = await fsp.stat(abs);
165
+ const name = path.basename(abs) || "workspace";
166
+ if (st.isDirectory()) {
167
+ // store-only zip: local transfer, no point burning CPU on deflate
168
+ const archive = new ZipArchive({ store: true });
169
+ archive.on("error", (err) => console.error("[zip]", err));
170
+ archive.directory(abs, name);
171
+ void archive.finalize();
172
+ return new Response(Readable.toWeb(archive), {
173
+ headers: {
174
+ "Content-Type": "application/zip",
175
+ "Content-Disposition": `attachment; filename*=UTF-8''${encodeURIComponent(name)}.zip`,
176
+ },
177
+ });
178
+ }
179
+ const range = c.req.header("range");
180
+ const headers = {
181
+ "Content-Type": contentType(abs),
182
+ "Accept-Ranges": "bytes",
183
+ "Content-Disposition": c.req.query("inline") === "1"
184
+ ? "inline"
185
+ : `attachment; filename*=UTF-8''${encodeURIComponent(name)}`,
186
+ };
187
+ const match = range ? /^bytes=(\d*)-(\d*)$/.exec(range) : null;
188
+ if (match && (match[1] || match[2])) {
189
+ const start = match[1] ? Number(match[1]) : st.size - Number(match[2]);
190
+ const end = match[1] && match[2] ? Math.min(Number(match[2]), st.size - 1) : st.size - 1;
191
+ if (Number.isNaN(start) || start < 0 || start > end) {
192
+ return c.body(null, 416, { "Content-Range": `bytes */${st.size}` });
193
+ }
194
+ headers["Content-Range"] = `bytes ${start}-${end}/${st.size}`;
195
+ headers["Content-Length"] = String(end - start + 1);
196
+ const stream = Readable.toWeb(fs.createReadStream(abs, { start, end }));
197
+ return new Response(stream, { status: 206, headers });
198
+ }
199
+ headers["Content-Length"] = String(st.size);
200
+ return new Response(Readable.toWeb(fs.createReadStream(abs)), { headers });
201
+ });
202
+ // Open with the OS default app, or reveal in the system file manager —
203
+ // the local-native replacement for download/upload.
204
+ app.post("/open", async (c) => {
205
+ const body = await c.req.json();
206
+ const abs = await resolveSafe(root, body.path ?? "");
207
+ await fsp.access(abs); // 404 if missing
208
+ let cmd;
209
+ let args;
210
+ if (process.platform === "darwin") {
211
+ cmd = "open";
212
+ args = body.reveal ? ["-R", abs] : [abs];
213
+ }
214
+ else if (process.platform === "win32") {
215
+ cmd = "explorer";
216
+ args = body.reveal ? [`/select,${abs}`] : [abs];
217
+ }
218
+ else {
219
+ cmd = "xdg-open";
220
+ args = [body.reveal ? path.dirname(abs) : abs];
221
+ }
222
+ spawn(cmd, args, { stdio: "ignore", detached: true }).unref();
223
+ return c.json({ ok: true });
224
+ });
225
+ return app;
226
+ }
@@ -0,0 +1,73 @@
1
+ import { execFile } from "node:child_process";
2
+ import { promisify } from "node:util";
3
+ const execFileAsync = promisify(execFile);
4
+ async function git(root, args) {
5
+ const { stdout } = await execFileAsync("git", args, {
6
+ cwd: root,
7
+ maxBuffer: 16 * 1024 * 1024,
8
+ timeout: 10_000,
9
+ });
10
+ return stdout;
11
+ }
12
+ async function isRepo(root) {
13
+ try {
14
+ const out = await git(root, ["rev-parse", "--is-inside-work-tree"]);
15
+ return out.trim() === "true";
16
+ }
17
+ catch {
18
+ return false;
19
+ }
20
+ }
21
+ export async function status(root) {
22
+ if (!(await isRepo(root)))
23
+ return { repo: false, branch: "", entries: [] };
24
+ let branch = "";
25
+ try {
26
+ branch = (await git(root, ["rev-parse", "--abbrev-ref", "HEAD"])).trim();
27
+ }
28
+ catch {
29
+ branch = "(detached)";
30
+ }
31
+ // -z gives NUL-separated records; renames emit an extra NUL with the old path
32
+ const raw = await git(root, ["status", "--porcelain=v1", "-z", "--untracked-files=all"]);
33
+ const entries = [];
34
+ const parts = raw.split("\0");
35
+ for (let i = 0; i < parts.length; i++) {
36
+ const rec = parts[i];
37
+ if (!rec || rec.length < 3)
38
+ continue;
39
+ const index = rec[0];
40
+ const worktree = rec[1];
41
+ const path = rec.slice(3);
42
+ // a rename/copy ("R"/"C") consumes the following NUL field (old path)
43
+ if (index === "R" || index === "C")
44
+ i++;
45
+ entries.push({ path, index, worktree });
46
+ }
47
+ entries.sort((a, b) => a.path.localeCompare(b.path));
48
+ return { repo: true, branch, entries };
49
+ }
50
+ /** HEAD/index content of a path for the diff editor's left side. */
51
+ export async function diffOriginal(root, path) {
52
+ // staged content if present, else HEAD; empty for untracked/new files
53
+ for (const ref of [`:${path}`, `HEAD:${path}`]) {
54
+ try {
55
+ return await git(root, ["show", ref]);
56
+ }
57
+ catch {
58
+ // try next
59
+ }
60
+ }
61
+ return "";
62
+ }
63
+ export async function stage(root, paths) {
64
+ if (paths.length)
65
+ await git(root, ["add", "--", ...paths]);
66
+ }
67
+ export async function unstage(root, paths) {
68
+ if (paths.length)
69
+ await git(root, ["reset", "-q", "HEAD", "--", ...paths]);
70
+ }
71
+ export async function commit(root, message) {
72
+ await git(root, ["commit", "-m", message]);
73
+ }
@@ -0,0 +1,45 @@
1
+ import fsp from "node:fs/promises";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ const MAX = 500;
5
+ /**
6
+ * Read the user's shell history (zsh extended `: <ts>:<dur>;<cmd>` or plain),
7
+ * deduped, most-recent-first. Best-effort — returns [] if no file is found.
8
+ */
9
+ export async function shellHistory() {
10
+ const home = os.homedir();
11
+ const candidates = [
12
+ process.env.HISTFILE,
13
+ path.join(home, ".zsh_history"),
14
+ path.join(home, ".bash_history"),
15
+ ].filter(Boolean);
16
+ for (const file of candidates) {
17
+ let raw;
18
+ try {
19
+ raw = await fsp.readFile(file, "utf8");
20
+ }
21
+ catch {
22
+ continue;
23
+ }
24
+ const cmds = [];
25
+ for (const line of raw.split("\n")) {
26
+ if (!line.trim())
27
+ continue;
28
+ // zsh extended: ": 1700000000:0;the command"
29
+ const m = /^: \d+:\d+;(.*)$/.exec(line);
30
+ cmds.push(m ? m[1] : line);
31
+ }
32
+ // most-recent-first, dedup keeping first occurrence
33
+ const seen = new Set();
34
+ const out = [];
35
+ for (let i = cmds.length - 1; i >= 0 && out.length < MAX; i--) {
36
+ const c = cmds[i].trim();
37
+ if (c && !seen.has(c)) {
38
+ seen.add(c);
39
+ out.push(c);
40
+ }
41
+ }
42
+ return out;
43
+ }
44
+ return [];
45
+ }