hammoc 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 (838) hide show
  1. package/LICENSE +661 -0
  2. package/README.md +272 -0
  3. package/bin/hammoc.js +59 -0
  4. package/package.json +86 -0
  5. package/packages/client/dist/assets/abap-DsBKuouk.js +1 -0
  6. package/packages/client/dist/assets/actionscript-3-D_z4Izcz.js +1 -0
  7. package/packages/client/dist/assets/ada-727ZlQH0.js +1 -0
  8. package/packages/client/dist/assets/andromeeda-C3khCPGq.js +1 -0
  9. package/packages/client/dist/assets/angular-html-LfdN0zeE.js +1 -0
  10. package/packages/client/dist/assets/angular-ts-CKsD7JZE.js +1 -0
  11. package/packages/client/dist/assets/apache-Dn00JSTd.js +1 -0
  12. package/packages/client/dist/assets/apex-COJ4H7py.js +1 -0
  13. package/packages/client/dist/assets/apl-BBq3IX1j.js +1 -0
  14. package/packages/client/dist/assets/applescript-Bu5BbsvL.js +1 -0
  15. package/packages/client/dist/assets/ara-7O62HKoU.js +1 -0
  16. package/packages/client/dist/assets/asciidoc-BPT9niGB.js +1 -0
  17. package/packages/client/dist/assets/asm-Dhn9LcZ4.js +1 -0
  18. package/packages/client/dist/assets/astro-CqkE3fuf.js +1 -0
  19. package/packages/client/dist/assets/aurora-x-D-2ljcwZ.js +1 -0
  20. package/packages/client/dist/assets/awk-eg146-Ew.js +1 -0
  21. package/packages/client/dist/assets/ayu-dark-Cv9koXgw.js +1 -0
  22. package/packages/client/dist/assets/ballerina-Du268qiB.js +1 -0
  23. package/packages/client/dist/assets/bat-fje9CFhw.js +1 -0
  24. package/packages/client/dist/assets/beancount-BwXTMy5W.js +1 -0
  25. package/packages/client/dist/assets/berry-3xVqZejG.js +1 -0
  26. package/packages/client/dist/assets/bibtex-xW4inM5L.js +1 -0
  27. package/packages/client/dist/assets/bicep-DHo0CJ0O.js +1 -0
  28. package/packages/client/dist/assets/blade-a8OxSdnT.js +1 -0
  29. package/packages/client/dist/assets/bsl-Dgyn0ogV.js +1 -0
  30. package/packages/client/dist/assets/c-C3t2pwGQ.js +1 -0
  31. package/packages/client/dist/assets/cadence-DNquZEk8.js +1 -0
  32. package/packages/client/dist/assets/cairo--RitsXJZ.js +1 -0
  33. package/packages/client/dist/assets/catppuccin-frappe-CD_QflpE.js +1 -0
  34. package/packages/client/dist/assets/catppuccin-latte-DRW-0cLl.js +1 -0
  35. package/packages/client/dist/assets/catppuccin-macchiato-C-_shW-Y.js +1 -0
  36. package/packages/client/dist/assets/catppuccin-mocha-LGGdnPYs.js +1 -0
  37. package/packages/client/dist/assets/clarity-BHOwM8T6.js +1 -0
  38. package/packages/client/dist/assets/clojure-DxSadP1t.js +1 -0
  39. package/packages/client/dist/assets/cmake-DbXoA79R.js +1 -0
  40. package/packages/client/dist/assets/cobol-PTqiYgYu.js +1 -0
  41. package/packages/client/dist/assets/codeowners-Bp6g37R7.js +1 -0
  42. package/packages/client/dist/assets/codeql-sacFqUAJ.js +1 -0
  43. package/packages/client/dist/assets/coffee-dyiR41kL.js +1 -0
  44. package/packages/client/dist/assets/common-lisp-C7gG9l05.js +1 -0
  45. package/packages/client/dist/assets/coq-Dsg_Bt_b.js +1 -0
  46. package/packages/client/dist/assets/cpp-BksuvNSY.js +1 -0
  47. package/packages/client/dist/assets/crystal-DtDmRg-F.js +1 -0
  48. package/packages/client/dist/assets/csharp-D9R-vmeu.js +1 -0
  49. package/packages/client/dist/assets/css-BPhBrDlE.js +1 -0
  50. package/packages/client/dist/assets/csv-B0qRVHPH.js +1 -0
  51. package/packages/client/dist/assets/cue-DtFQj3wx.js +1 -0
  52. package/packages/client/dist/assets/cypher-m2LEI-9-.js +1 -0
  53. package/packages/client/dist/assets/d-BoXegm-a.js +1 -0
  54. package/packages/client/dist/assets/dark-plus-C3mMm8J8.js +1 -0
  55. package/packages/client/dist/assets/dart-B9wLZaAG.js +1 -0
  56. package/packages/client/dist/assets/dax-ClGRhx96.js +1 -0
  57. package/packages/client/dist/assets/desktop-DEIpsLCJ.js +1 -0
  58. package/packages/client/dist/assets/diff-BgYniUM_.js +1 -0
  59. package/packages/client/dist/assets/docker-COcR7UxN.js +1 -0
  60. package/packages/client/dist/assets/dotenv-BjQB5zDj.js +1 -0
  61. package/packages/client/dist/assets/dracula-BzJJZx-M.js +1 -0
  62. package/packages/client/dist/assets/dracula-soft-BXkSAIEj.js +1 -0
  63. package/packages/client/dist/assets/dream-maker-C-nORZOA.js +1 -0
  64. package/packages/client/dist/assets/edge-D5gP-w-T.js +1 -0
  65. package/packages/client/dist/assets/elixir-CLiX3zqd.js +1 -0
  66. package/packages/client/dist/assets/elm-CmHSxxaM.js +1 -0
  67. package/packages/client/dist/assets/emacs-lisp-BX77sIaO.js +1 -0
  68. package/packages/client/dist/assets/erb-BYTLMnw6.js +1 -0
  69. package/packages/client/dist/assets/erlang-B-DoSBHF.js +1 -0
  70. package/packages/client/dist/assets/everforest-dark-BgDCqdQA.js +1 -0
  71. package/packages/client/dist/assets/everforest-light-C8M2exoo.js +1 -0
  72. package/packages/client/dist/assets/fennel-bCA53EVm.js +1 -0
  73. package/packages/client/dist/assets/fish-w-ucz2PV.js +1 -0
  74. package/packages/client/dist/assets/fluent-Dayu4EKP.js +1 -0
  75. package/packages/client/dist/assets/fortran-fixed-form-TqA4NnZg.js +1 -0
  76. package/packages/client/dist/assets/fortran-free-form-DKXYxT9g.js +1 -0
  77. package/packages/client/dist/assets/fsharp-XplgxFYe.js +1 -0
  78. package/packages/client/dist/assets/gdresource-BHYsBjWJ.js +1 -0
  79. package/packages/client/dist/assets/gdscript-DfxzS6Rs.js +1 -0
  80. package/packages/client/dist/assets/gdshader-SKMF96pI.js +1 -0
  81. package/packages/client/dist/assets/genie-ajMbGru0.js +1 -0
  82. package/packages/client/dist/assets/gherkin--30QC5Em.js +1 -0
  83. package/packages/client/dist/assets/git-commit-i4q6IMui.js +1 -0
  84. package/packages/client/dist/assets/git-rebase-B-v9cOL2.js +1 -0
  85. package/packages/client/dist/assets/github-dark-DHJKELXO.js +1 -0
  86. package/packages/client/dist/assets/github-dark-default-Cuk6v7N8.js +1 -0
  87. package/packages/client/dist/assets/github-dark-dimmed-DH5Ifo-i.js +1 -0
  88. package/packages/client/dist/assets/github-dark-high-contrast-E3gJ1_iC.js +1 -0
  89. package/packages/client/dist/assets/github-light-DAi9KRSo.js +1 -0
  90. package/packages/client/dist/assets/github-light-default-D7oLnXFd.js +1 -0
  91. package/packages/client/dist/assets/github-light-high-contrast-BfjtVDDH.js +1 -0
  92. package/packages/client/dist/assets/gleam-B430Bg39.js +1 -0
  93. package/packages/client/dist/assets/glimmer-js-D-cwc0-E.js +1 -0
  94. package/packages/client/dist/assets/glimmer-ts-pgjy16dm.js +1 -0
  95. package/packages/client/dist/assets/glsl-DBO2IWDn.js +1 -0
  96. package/packages/client/dist/assets/gnuplot-CM8KxXT1.js +1 -0
  97. package/packages/client/dist/assets/go-B1SYOhNW.js +1 -0
  98. package/packages/client/dist/assets/graphql-cDcHW_If.js +1 -0
  99. package/packages/client/dist/assets/groovy-DkBy-JyN.js +1 -0
  100. package/packages/client/dist/assets/hack-D1yCygmZ.js +1 -0
  101. package/packages/client/dist/assets/haml-B2EZWmdv.js +1 -0
  102. package/packages/client/dist/assets/handlebars-BQGss363.js +1 -0
  103. package/packages/client/dist/assets/haskell-BILxekzW.js +1 -0
  104. package/packages/client/dist/assets/haxe-C5wWYbrZ.js +1 -0
  105. package/packages/client/dist/assets/hcl-HzYwdGDm.js +1 -0
  106. package/packages/client/dist/assets/hjson-T-Tgc4AT.js +1 -0
  107. package/packages/client/dist/assets/hlsl-ifBTmRxC.js +1 -0
  108. package/packages/client/dist/assets/houston-DnULxvSX.js +1 -0
  109. package/packages/client/dist/assets/html-C2L_23MC.js +1 -0
  110. package/packages/client/dist/assets/html-derivative-CSfWNPLT.js +1 -0
  111. package/packages/client/dist/assets/http-FRrOvY1W.js +1 -0
  112. package/packages/client/dist/assets/hxml-TIA70rKU.js +1 -0
  113. package/packages/client/dist/assets/hy-BMj5Y0dO.js +1 -0
  114. package/packages/client/dist/assets/imba-bv_oIlVt.js +1 -0
  115. package/packages/client/dist/assets/index-Bzql1gnR.js +2 -0
  116. package/packages/client/dist/assets/index-DjQDxRju.css +32 -0
  117. package/packages/client/dist/assets/index-q3VZc6RP.js +1297 -0
  118. package/packages/client/dist/assets/ini-BjABl1g7.js +1 -0
  119. package/packages/client/dist/assets/java-xI-RfyKK.js +1 -0
  120. package/packages/client/dist/assets/javascript-ySlJ1b_l.js +1 -0
  121. package/packages/client/dist/assets/jinja-DGy0s7-h.js +1 -0
  122. package/packages/client/dist/assets/jison-BqZprYcd.js +1 -0
  123. package/packages/client/dist/assets/json-BQoSv7ci.js +1 -0
  124. package/packages/client/dist/assets/json5-w8dY5SsB.js +1 -0
  125. package/packages/client/dist/assets/jsonc-TU54ms6u.js +1 -0
  126. package/packages/client/dist/assets/jsonl-DREVFZK8.js +1 -0
  127. package/packages/client/dist/assets/jsonnet-BfivnA6A.js +1 -0
  128. package/packages/client/dist/assets/jssm-P4WzXJd0.js +1 -0
  129. package/packages/client/dist/assets/jsx-BAng5TT0.js +1 -0
  130. package/packages/client/dist/assets/julia-BBuGR-5E.js +1 -0
  131. package/packages/client/dist/assets/kanagawa-dragon-CkXjmgJE.js +1 -0
  132. package/packages/client/dist/assets/kanagawa-lotus-CfQXZHmo.js +1 -0
  133. package/packages/client/dist/assets/kanagawa-wave-DWedfzmr.js +1 -0
  134. package/packages/client/dist/assets/kotlin-B5lbUyaz.js +1 -0
  135. package/packages/client/dist/assets/kusto-mebxcVVE.js +1 -0
  136. package/packages/client/dist/assets/laserwave-DUszq2jm.js +1 -0
  137. package/packages/client/dist/assets/latex-C-cWTeAZ.js +1 -0
  138. package/packages/client/dist/assets/lean-XBlWyCtg.js +1 -0
  139. package/packages/client/dist/assets/less-BfCpw3nA.js +1 -0
  140. package/packages/client/dist/assets/light-plus-B7mTdjB0.js +1 -0
  141. package/packages/client/dist/assets/liquid-D3W5UaiH.js +1 -0
  142. package/packages/client/dist/assets/log-Cc5clBb7.js +1 -0
  143. package/packages/client/dist/assets/logo-IuBKFhSY.js +1 -0
  144. package/packages/client/dist/assets/lua-CvWAzNxB.js +1 -0
  145. package/packages/client/dist/assets/luau-Du5NY7AG.js +1 -0
  146. package/packages/client/dist/assets/make-Bvotw-X0.js +1 -0
  147. package/packages/client/dist/assets/markdown-UIAJJxZW.js +1 -0
  148. package/packages/client/dist/assets/marko-z0MBrx5-.js +1 -0
  149. package/packages/client/dist/assets/material-theme-D5KoaKCx.js +1 -0
  150. package/packages/client/dist/assets/material-theme-darker-BfHTSMKl.js +1 -0
  151. package/packages/client/dist/assets/material-theme-lighter-B0m2ddpp.js +1 -0
  152. package/packages/client/dist/assets/material-theme-ocean-CyktbL80.js +1 -0
  153. package/packages/client/dist/assets/material-theme-palenight-Csfq5Kiy.js +1 -0
  154. package/packages/client/dist/assets/matlab-D9-PGadD.js +1 -0
  155. package/packages/client/dist/assets/mdc-DB_EDNY_.js +1 -0
  156. package/packages/client/dist/assets/mdx-sdHcTMYB.js +1 -0
  157. package/packages/client/dist/assets/mermaid-Ci6OQyBP.js +1 -0
  158. package/packages/client/dist/assets/min-dark-CafNBF8u.js +1 -0
  159. package/packages/client/dist/assets/min-light-CTRr51gU.js +1 -0
  160. package/packages/client/dist/assets/mipsasm-BC5c_5Pe.js +1 -0
  161. package/packages/client/dist/assets/mojo-Tz6hzZYG.js +1 -0
  162. package/packages/client/dist/assets/monokai-D4h5O-jR.js +1 -0
  163. package/packages/client/dist/assets/move-DB_GagMm.js +1 -0
  164. package/packages/client/dist/assets/narrat-DLbgOhZU.js +1 -0
  165. package/packages/client/dist/assets/nextflow-B0XVJmRM.js +1 -0
  166. package/packages/client/dist/assets/nginx-D_VnBJ67.js +1 -0
  167. package/packages/client/dist/assets/night-owl-C39BiMTA.js +1 -0
  168. package/packages/client/dist/assets/nim-ZlGxZxc3.js +1 -0
  169. package/packages/client/dist/assets/nix-shcSOmrb.js +1 -0
  170. package/packages/client/dist/assets/nord-Ddv68eIx.js +1 -0
  171. package/packages/client/dist/assets/nushell-D4Tzg5kh.js +1 -0
  172. package/packages/client/dist/assets/objective-c-Deuh7S70.js +1 -0
  173. package/packages/client/dist/assets/objective-cpp-BUEGK8hf.js +1 -0
  174. package/packages/client/dist/assets/ocaml-BNioltXt.js +1 -0
  175. package/packages/client/dist/assets/one-dark-pro-GBQ2dnAY.js +1 -0
  176. package/packages/client/dist/assets/one-light-PoHY5YXO.js +1 -0
  177. package/packages/client/dist/assets/pascal-JqZropPD.js +1 -0
  178. package/packages/client/dist/assets/perl-CHQXSrWU.js +1 -0
  179. package/packages/client/dist/assets/php-B5ebYQev.js +1 -0
  180. package/packages/client/dist/assets/plastic-3e1v2bzS.js +1 -0
  181. package/packages/client/dist/assets/plsql-LKU2TuZ1.js +1 -0
  182. package/packages/client/dist/assets/po-BFLt1xDp.js +1 -0
  183. package/packages/client/dist/assets/poimandres-CS3Unz2-.js +1 -0
  184. package/packages/client/dist/assets/polar-DKykz6zU.js +1 -0
  185. package/packages/client/dist/assets/postcss-B3ZDOciz.js +1 -0
  186. package/packages/client/dist/assets/powerquery-CSHBycmS.js +1 -0
  187. package/packages/client/dist/assets/powershell-BIEUsx6d.js +1 -0
  188. package/packages/client/dist/assets/prisma-B48N-Iqd.js +1 -0
  189. package/packages/client/dist/assets/prolog-BY-TUvya.js +1 -0
  190. package/packages/client/dist/assets/proto-zocC4JxJ.js +1 -0
  191. package/packages/client/dist/assets/pug-CM9l7STV.js +1 -0
  192. package/packages/client/dist/assets/puppet-Cza_XSSt.js +1 -0
  193. package/packages/client/dist/assets/purescript-Bg-kzb6g.js +1 -0
  194. package/packages/client/dist/assets/python-DhUJRlN_.js +1 -0
  195. package/packages/client/dist/assets/qml-D8XfuvdV.js +1 -0
  196. package/packages/client/dist/assets/qmldir-C8lEn-DE.js +1 -0
  197. package/packages/client/dist/assets/qss-DhMKtDLN.js +1 -0
  198. package/packages/client/dist/assets/r-CwjWoCRV.js +1 -0
  199. package/packages/client/dist/assets/racket-CzouJOBO.js +1 -0
  200. package/packages/client/dist/assets/raku-B1bQXN8T.js +1 -0
  201. package/packages/client/dist/assets/razor-CNLDkMZG.js +1 -0
  202. package/packages/client/dist/assets/red-bN70gL4F.js +1 -0
  203. package/packages/client/dist/assets/reg-5LuOXUq_.js +1 -0
  204. package/packages/client/dist/assets/regexp-DWJ3fJO_.js +1 -0
  205. package/packages/client/dist/assets/rel-DJlmqQ1C.js +1 -0
  206. package/packages/client/dist/assets/riscv-QhoSD0DR.js +1 -0
  207. package/packages/client/dist/assets/rose-pine-CmCqftbK.js +1 -0
  208. package/packages/client/dist/assets/rose-pine-dawn-Ds-gbosJ.js +1 -0
  209. package/packages/client/dist/assets/rose-pine-moon-CjDtw9vr.js +1 -0
  210. package/packages/client/dist/assets/rst-4NLicBqY.js +1 -0
  211. package/packages/client/dist/assets/ruby-DeZ3UC14.js +1 -0
  212. package/packages/client/dist/assets/rust-Be6lgOlo.js +1 -0
  213. package/packages/client/dist/assets/sas-BmTFh92c.js +1 -0
  214. package/packages/client/dist/assets/sass-BJ4Li9vH.js +1 -0
  215. package/packages/client/dist/assets/scala-DQVVAn-B.js +1 -0
  216. package/packages/client/dist/assets/scheme-BJGe-b2p.js +1 -0
  217. package/packages/client/dist/assets/scss-C31hgJw-.js +1 -0
  218. package/packages/client/dist/assets/sdbl-BLhTXw86.js +1 -0
  219. package/packages/client/dist/assets/shaderlab-B7qAK45m.js +1 -0
  220. package/packages/client/dist/assets/shellscript-atvbtKCR.js +1 -0
  221. package/packages/client/dist/assets/shellsession-C_rIy8kc.js +1 -0
  222. package/packages/client/dist/assets/slack-dark-BthQWCQV.js +1 -0
  223. package/packages/client/dist/assets/slack-ochin-DqwNpetd.js +1 -0
  224. package/packages/client/dist/assets/smalltalk-DkLiglaE.js +1 -0
  225. package/packages/client/dist/assets/snazzy-light-Bw305WKR.js +1 -0
  226. package/packages/client/dist/assets/solarized-dark-DXbdFlpD.js +1 -0
  227. package/packages/client/dist/assets/solarized-light-L9t79GZl.js +1 -0
  228. package/packages/client/dist/assets/solidity-C1w2a3ep.js +1 -0
  229. package/packages/client/dist/assets/soy-C-lX7w71.js +1 -0
  230. package/packages/client/dist/assets/sparql-bYkjHRlG.js +1 -0
  231. package/packages/client/dist/assets/splunk-Cf8iN4DR.js +1 -0
  232. package/packages/client/dist/assets/sql-COK4E0Yg.js +1 -0
  233. package/packages/client/dist/assets/ssh-config-BknIz3MU.js +1 -0
  234. package/packages/client/dist/assets/stata-DorPZHa4.js +1 -0
  235. package/packages/client/dist/assets/stylus-BeQkCIfX.js +1 -0
  236. package/packages/client/dist/assets/svelte-MSaWC3Je.js +1 -0
  237. package/packages/client/dist/assets/swift-BSxZ-RaX.js +1 -0
  238. package/packages/client/dist/assets/synthwave-84-CbfX1IO0.js +1 -0
  239. package/packages/client/dist/assets/system-verilog-C7L56vO4.js +1 -0
  240. package/packages/client/dist/assets/systemd-CUnW07Te.js +1 -0
  241. package/packages/client/dist/assets/talonscript-C1XDQQGZ.js +1 -0
  242. package/packages/client/dist/assets/tasl-CQjiPCtT.js +1 -0
  243. package/packages/client/dist/assets/tcl-DQ1-QYvQ.js +1 -0
  244. package/packages/client/dist/assets/templ-dwX3ZSMB.js +1 -0
  245. package/packages/client/dist/assets/terraform-BbSNqyBO.js +1 -0
  246. package/packages/client/dist/assets/tex-rYs2v40G.js +1 -0
  247. package/packages/client/dist/assets/tokyo-night-DBQeEorK.js +1 -0
  248. package/packages/client/dist/assets/toml-CB2ApiWb.js +1 -0
  249. package/packages/client/dist/assets/ts-tags-CipyTH0X.js +1 -0
  250. package/packages/client/dist/assets/tsv-B_m7g4N7.js +1 -0
  251. package/packages/client/dist/assets/tsx-B6W0miNI.js +1 -0
  252. package/packages/client/dist/assets/turtle-BMR_PYu6.js +1 -0
  253. package/packages/client/dist/assets/twig-NC5TFiHP.js +1 -0
  254. package/packages/client/dist/assets/typescript-Dj6nwHGl.js +1 -0
  255. package/packages/client/dist/assets/typespec-BpWG_bgh.js +1 -0
  256. package/packages/client/dist/assets/typst-BVUVsWT6.js +1 -0
  257. package/packages/client/dist/assets/v-CAQ2eGtk.js +1 -0
  258. package/packages/client/dist/assets/vala-BFOHcciG.js +1 -0
  259. package/packages/client/dist/assets/vb-CdO5JTpU.js +1 -0
  260. package/packages/client/dist/assets/verilog-CJaU5se_.js +1 -0
  261. package/packages/client/dist/assets/vesper-BEBZ7ncR.js +1 -0
  262. package/packages/client/dist/assets/vhdl-DYoNaHQp.js +1 -0
  263. package/packages/client/dist/assets/viml-m4uW47V2.js +1 -0
  264. package/packages/client/dist/assets/vitesse-black-Bkuqu6BP.js +1 -0
  265. package/packages/client/dist/assets/vitesse-dark-D0r3Knsf.js +1 -0
  266. package/packages/client/dist/assets/vitesse-light-CVO1_9PV.js +1 -0
  267. package/packages/client/dist/assets/vue-BuYVFjOK.js +1 -0
  268. package/packages/client/dist/assets/vue-html-xdeiXROB.js +1 -0
  269. package/packages/client/dist/assets/vyper-nyqBNV6O.js +1 -0
  270. package/packages/client/dist/assets/wasm-C6j12Q_x.js +1 -0
  271. package/packages/client/dist/assets/wasm-CG6Dc4jp.js +1 -0
  272. package/packages/client/dist/assets/wenyan-7A4Fjokl.js +1 -0
  273. package/packages/client/dist/assets/wgsl-CB0Krxn9.js +1 -0
  274. package/packages/client/dist/assets/wikitext-DCE3LsBG.js +1 -0
  275. package/packages/client/dist/assets/wolfram-C3FkfJm5.js +1 -0
  276. package/packages/client/dist/assets/xml-e3z08dGr.js +1 -0
  277. package/packages/client/dist/assets/xsl-Dd0NUgwM.js +1 -0
  278. package/packages/client/dist/assets/yaml-CVw76BM1.js +1 -0
  279. package/packages/client/dist/assets/zenscript-HnGAYVZD.js +1 -0
  280. package/packages/client/dist/assets/zig-BVz_zdnA.js +1 -0
  281. package/packages/client/dist/index.html +13 -0
  282. package/packages/server/dist/__tests__/i18n.test.d.ts +5 -0
  283. package/packages/server/dist/__tests__/i18n.test.d.ts.map +1 -0
  284. package/packages/server/dist/__tests__/i18n.test.js +31 -0
  285. package/packages/server/dist/__tests__/i18n.test.js.map +1 -0
  286. package/packages/server/dist/app.d.ts +15 -0
  287. package/packages/server/dist/app.d.ts.map +1 -0
  288. package/packages/server/dist/app.js +120 -0
  289. package/packages/server/dist/app.js.map +1 -0
  290. package/packages/server/dist/cli/passwordSetup.d.ts +5 -0
  291. package/packages/server/dist/cli/passwordSetup.d.ts.map +1 -0
  292. package/packages/server/dist/cli/passwordSetup.js +74 -0
  293. package/packages/server/dist/cli/passwordSetup.js.map +1 -0
  294. package/packages/server/dist/config/index.d.ts +81 -0
  295. package/packages/server/dist/config/index.d.ts.map +1 -0
  296. package/packages/server/dist/config/index.js +107 -0
  297. package/packages/server/dist/config/index.js.map +1 -0
  298. package/packages/server/dist/controllers/__tests__/authController.test.d.ts +6 -0
  299. package/packages/server/dist/controllers/__tests__/authController.test.d.ts.map +1 -0
  300. package/packages/server/dist/controllers/__tests__/authController.test.js +198 -0
  301. package/packages/server/dist/controllers/__tests__/authController.test.js.map +1 -0
  302. package/packages/server/dist/controllers/__tests__/bmadStatusController.test.d.ts +6 -0
  303. package/packages/server/dist/controllers/__tests__/bmadStatusController.test.d.ts.map +1 -0
  304. package/packages/server/dist/controllers/__tests__/bmadStatusController.test.js +121 -0
  305. package/packages/server/dist/controllers/__tests__/bmadStatusController.test.js.map +1 -0
  306. package/packages/server/dist/controllers/__tests__/commandController.test.d.ts +6 -0
  307. package/packages/server/dist/controllers/__tests__/commandController.test.d.ts.map +1 -0
  308. package/packages/server/dist/controllers/__tests__/commandController.test.js +73 -0
  309. package/packages/server/dist/controllers/__tests__/commandController.test.js.map +1 -0
  310. package/packages/server/dist/controllers/__tests__/dashboardController.test.d.ts +6 -0
  311. package/packages/server/dist/controllers/__tests__/dashboardController.test.d.ts.map +1 -0
  312. package/packages/server/dist/controllers/__tests__/dashboardController.test.js +63 -0
  313. package/packages/server/dist/controllers/__tests__/dashboardController.test.js.map +1 -0
  314. package/packages/server/dist/controllers/__tests__/gitController.test.d.ts +6 -0
  315. package/packages/server/dist/controllers/__tests__/gitController.test.d.ts.map +1 -0
  316. package/packages/server/dist/controllers/__tests__/gitController.test.js +585 -0
  317. package/packages/server/dist/controllers/__tests__/gitController.test.js.map +1 -0
  318. package/packages/server/dist/controllers/__tests__/projectController.test.d.ts +6 -0
  319. package/packages/server/dist/controllers/__tests__/projectController.test.d.ts.map +1 -0
  320. package/packages/server/dist/controllers/__tests__/projectController.test.js +175 -0
  321. package/packages/server/dist/controllers/__tests__/projectController.test.js.map +1 -0
  322. package/packages/server/dist/controllers/__tests__/queueController.test.d.ts +6 -0
  323. package/packages/server/dist/controllers/__tests__/queueController.test.d.ts.map +1 -0
  324. package/packages/server/dist/controllers/__tests__/queueController.test.js +115 -0
  325. package/packages/server/dist/controllers/__tests__/queueController.test.js.map +1 -0
  326. package/packages/server/dist/controllers/__tests__/queueTemplateController.test.d.ts +6 -0
  327. package/packages/server/dist/controllers/__tests__/queueTemplateController.test.d.ts.map +1 -0
  328. package/packages/server/dist/controllers/__tests__/queueTemplateController.test.js +167 -0
  329. package/packages/server/dist/controllers/__tests__/queueTemplateController.test.js.map +1 -0
  330. package/packages/server/dist/controllers/__tests__/sessionController.test.d.ts +6 -0
  331. package/packages/server/dist/controllers/__tests__/sessionController.test.d.ts.map +1 -0
  332. package/packages/server/dist/controllers/__tests__/sessionController.test.js +113 -0
  333. package/packages/server/dist/controllers/__tests__/sessionController.test.js.map +1 -0
  334. package/packages/server/dist/controllers/authController.d.ts +30 -0
  335. package/packages/server/dist/controllers/authController.d.ts.map +1 -0
  336. package/packages/server/dist/controllers/authController.js +209 -0
  337. package/packages/server/dist/controllers/authController.js.map +1 -0
  338. package/packages/server/dist/controllers/bmadStatusController.d.ts +5 -0
  339. package/packages/server/dist/controllers/bmadStatusController.d.ts.map +1 -0
  340. package/packages/server/dist/controllers/bmadStatusController.js +53 -0
  341. package/packages/server/dist/controllers/bmadStatusController.js.map +1 -0
  342. package/packages/server/dist/controllers/boardController.d.ts +16 -0
  343. package/packages/server/dist/controllers/boardController.d.ts.map +1 -0
  344. package/packages/server/dist/controllers/boardController.js +295 -0
  345. package/packages/server/dist/controllers/boardController.js.map +1 -0
  346. package/packages/server/dist/controllers/cliController.d.ts +7 -0
  347. package/packages/server/dist/controllers/cliController.d.ts.map +1 -0
  348. package/packages/server/dist/controllers/cliController.js +27 -0
  349. package/packages/server/dist/controllers/cliController.js.map +1 -0
  350. package/packages/server/dist/controllers/commandController.d.ts +14 -0
  351. package/packages/server/dist/controllers/commandController.d.ts.map +1 -0
  352. package/packages/server/dist/controllers/commandController.js +31 -0
  353. package/packages/server/dist/controllers/commandController.js.map +1 -0
  354. package/packages/server/dist/controllers/dashboardController.d.ts +10 -0
  355. package/packages/server/dist/controllers/dashboardController.d.ts.map +1 -0
  356. package/packages/server/dist/controllers/dashboardController.js +23 -0
  357. package/packages/server/dist/controllers/dashboardController.js.map +1 -0
  358. package/packages/server/dist/controllers/fileSystemController.d.ts +54 -0
  359. package/packages/server/dist/controllers/fileSystemController.d.ts.map +1 -0
  360. package/packages/server/dist/controllers/fileSystemController.js +435 -0
  361. package/packages/server/dist/controllers/fileSystemController.js.map +1 -0
  362. package/packages/server/dist/controllers/gitController.d.ts +57 -0
  363. package/packages/server/dist/controllers/gitController.d.ts.map +1 -0
  364. package/packages/server/dist/controllers/gitController.js +494 -0
  365. package/packages/server/dist/controllers/gitController.js.map +1 -0
  366. package/packages/server/dist/controllers/projectController.d.ts +57 -0
  367. package/packages/server/dist/controllers/projectController.d.ts.map +1 -0
  368. package/packages/server/dist/controllers/projectController.js +358 -0
  369. package/packages/server/dist/controllers/projectController.js.map +1 -0
  370. package/packages/server/dist/controllers/queueController.d.ts +15 -0
  371. package/packages/server/dist/controllers/queueController.d.ts.map +1 -0
  372. package/packages/server/dist/controllers/queueController.js +131 -0
  373. package/packages/server/dist/controllers/queueController.js.map +1 -0
  374. package/packages/server/dist/controllers/queueTemplateController.d.ts +15 -0
  375. package/packages/server/dist/controllers/queueTemplateController.d.ts.map +1 -0
  376. package/packages/server/dist/controllers/queueTemplateController.js +187 -0
  377. package/packages/server/dist/controllers/queueTemplateController.js.map +1 -0
  378. package/packages/server/dist/controllers/serverController.d.ts +14 -0
  379. package/packages/server/dist/controllers/serverController.d.ts.map +1 -0
  380. package/packages/server/dist/controllers/serverController.js +164 -0
  381. package/packages/server/dist/controllers/serverController.js.map +1 -0
  382. package/packages/server/dist/controllers/sessionController.d.ts +39 -0
  383. package/packages/server/dist/controllers/sessionController.d.ts.map +1 -0
  384. package/packages/server/dist/controllers/sessionController.js +230 -0
  385. package/packages/server/dist/controllers/sessionController.js.map +1 -0
  386. package/packages/server/dist/dev.d.ts +2 -0
  387. package/packages/server/dist/dev.d.ts.map +1 -0
  388. package/packages/server/dist/dev.js +8 -0
  389. package/packages/server/dist/dev.js.map +1 -0
  390. package/packages/server/dist/handlers/__tests__/websocket.auth.test.d.ts +7 -0
  391. package/packages/server/dist/handlers/__tests__/websocket.auth.test.d.ts.map +1 -0
  392. package/packages/server/dist/handlers/__tests__/websocket.auth.test.js +125 -0
  393. package/packages/server/dist/handlers/__tests__/websocket.auth.test.js.map +1 -0
  394. package/packages/server/dist/handlers/__tests__/websocket.test.d.ts +2 -0
  395. package/packages/server/dist/handlers/__tests__/websocket.test.d.ts.map +1 -0
  396. package/packages/server/dist/handlers/__tests__/websocket.test.js +1316 -0
  397. package/packages/server/dist/handlers/__tests__/websocket.test.js.map +1 -0
  398. package/packages/server/dist/handlers/streamCallbacks.d.ts +52 -0
  399. package/packages/server/dist/handlers/streamCallbacks.d.ts.map +1 -0
  400. package/packages/server/dist/handlers/streamCallbacks.js +107 -0
  401. package/packages/server/dist/handlers/streamCallbacks.js.map +1 -0
  402. package/packages/server/dist/handlers/websocket.d.ts +82 -0
  403. package/packages/server/dist/handlers/websocket.d.ts.map +1 -0
  404. package/packages/server/dist/handlers/websocket.js +1041 -0
  405. package/packages/server/dist/handlers/websocket.js.map +1 -0
  406. package/packages/server/dist/i18n.d.ts +7 -0
  407. package/packages/server/dist/i18n.d.ts.map +1 -0
  408. package/packages/server/dist/i18n.js +32 -0
  409. package/packages/server/dist/i18n.js.map +1 -0
  410. package/packages/server/dist/index.d.ts +2 -0
  411. package/packages/server/dist/index.d.ts.map +1 -0
  412. package/packages/server/dist/index.js +114 -0
  413. package/packages/server/dist/index.js.map +1 -0
  414. package/packages/server/dist/locales/en/server.json +303 -0
  415. package/packages/server/dist/locales/es/server.json +303 -0
  416. package/packages/server/dist/locales/ja/server.json +303 -0
  417. package/packages/server/dist/locales/ko/server.json +303 -0
  418. package/packages/server/dist/locales/pt/server.json +303 -0
  419. package/packages/server/dist/locales/zh-CN/server.json +303 -0
  420. package/packages/server/dist/middleware/__tests__/auth.test.d.ts +6 -0
  421. package/packages/server/dist/middleware/__tests__/auth.test.d.ts.map +1 -0
  422. package/packages/server/dist/middleware/__tests__/auth.test.js +146 -0
  423. package/packages/server/dist/middleware/__tests__/auth.test.js.map +1 -0
  424. package/packages/server/dist/middleware/__tests__/i18n.test.d.ts +5 -0
  425. package/packages/server/dist/middleware/__tests__/i18n.test.d.ts.map +1 -0
  426. package/packages/server/dist/middleware/__tests__/i18n.test.js +96 -0
  427. package/packages/server/dist/middleware/__tests__/i18n.test.js.map +1 -0
  428. package/packages/server/dist/middleware/__tests__/pathGuard.test.d.ts +6 -0
  429. package/packages/server/dist/middleware/__tests__/pathGuard.test.d.ts.map +1 -0
  430. package/packages/server/dist/middleware/__tests__/pathGuard.test.js +73 -0
  431. package/packages/server/dist/middleware/__tests__/pathGuard.test.js.map +1 -0
  432. package/packages/server/dist/middleware/__tests__/session.test.d.ts +6 -0
  433. package/packages/server/dist/middleware/__tests__/session.test.d.ts.map +1 -0
  434. package/packages/server/dist/middleware/__tests__/session.test.js +33 -0
  435. package/packages/server/dist/middleware/__tests__/session.test.js.map +1 -0
  436. package/packages/server/dist/middleware/auth.d.ts +18 -0
  437. package/packages/server/dist/middleware/auth.d.ts.map +1 -0
  438. package/packages/server/dist/middleware/auth.js +52 -0
  439. package/packages/server/dist/middleware/auth.js.map +1 -0
  440. package/packages/server/dist/middleware/i18n.d.ts +17 -0
  441. package/packages/server/dist/middleware/i18n.d.ts.map +1 -0
  442. package/packages/server/dist/middleware/i18n.js +42 -0
  443. package/packages/server/dist/middleware/i18n.js.map +1 -0
  444. package/packages/server/dist/middleware/pathGuard.d.ts +15 -0
  445. package/packages/server/dist/middleware/pathGuard.d.ts.map +1 -0
  446. package/packages/server/dist/middleware/pathGuard.js +41 -0
  447. package/packages/server/dist/middleware/pathGuard.js.map +1 -0
  448. package/packages/server/dist/middleware/session.d.ts +18 -0
  449. package/packages/server/dist/middleware/session.d.ts.map +1 -0
  450. package/packages/server/dist/middleware/session.js +35 -0
  451. package/packages/server/dist/middleware/session.js.map +1 -0
  452. package/packages/server/dist/routes/__tests__/auth.integration.test.d.ts +6 -0
  453. package/packages/server/dist/routes/__tests__/auth.integration.test.d.ts.map +1 -0
  454. package/packages/server/dist/routes/__tests__/auth.integration.test.js +111 -0
  455. package/packages/server/dist/routes/__tests__/auth.integration.test.js.map +1 -0
  456. package/packages/server/dist/routes/__tests__/auth.test.d.ts +6 -0
  457. package/packages/server/dist/routes/__tests__/auth.test.d.ts.map +1 -0
  458. package/packages/server/dist/routes/__tests__/auth.test.js +155 -0
  459. package/packages/server/dist/routes/__tests__/auth.test.js.map +1 -0
  460. package/packages/server/dist/routes/__tests__/board.test.d.ts +2 -0
  461. package/packages/server/dist/routes/__tests__/board.test.d.ts.map +1 -0
  462. package/packages/server/dist/routes/__tests__/board.test.js +201 -0
  463. package/packages/server/dist/routes/__tests__/board.test.js.map +1 -0
  464. package/packages/server/dist/routes/__tests__/cli.test.d.ts +2 -0
  465. package/packages/server/dist/routes/__tests__/cli.test.d.ts.map +1 -0
  466. package/packages/server/dist/routes/__tests__/cli.test.js +121 -0
  467. package/packages/server/dist/routes/__tests__/cli.test.js.map +1 -0
  468. package/packages/server/dist/routes/__tests__/commands.test.d.ts +6 -0
  469. package/packages/server/dist/routes/__tests__/commands.test.d.ts.map +1 -0
  470. package/packages/server/dist/routes/__tests__/commands.test.js +74 -0
  471. package/packages/server/dist/routes/__tests__/commands.test.js.map +1 -0
  472. package/packages/server/dist/routes/__tests__/projects.integration.test.d.ts +9 -0
  473. package/packages/server/dist/routes/__tests__/projects.integration.test.d.ts.map +1 -0
  474. package/packages/server/dist/routes/__tests__/projects.integration.test.js +517 -0
  475. package/packages/server/dist/routes/__tests__/projects.integration.test.js.map +1 -0
  476. package/packages/server/dist/routes/__tests__/projects.test.d.ts +7 -0
  477. package/packages/server/dist/routes/__tests__/projects.test.d.ts.map +1 -0
  478. package/packages/server/dist/routes/__tests__/projects.test.js +361 -0
  479. package/packages/server/dist/routes/__tests__/projects.test.js.map +1 -0
  480. package/packages/server/dist/routes/__tests__/sessions.integration.test.d.ts +8 -0
  481. package/packages/server/dist/routes/__tests__/sessions.integration.test.d.ts.map +1 -0
  482. package/packages/server/dist/routes/__tests__/sessions.integration.test.js +287 -0
  483. package/packages/server/dist/routes/__tests__/sessions.integration.test.js.map +1 -0
  484. package/packages/server/dist/routes/__tests__/sessions.test.d.ts +6 -0
  485. package/packages/server/dist/routes/__tests__/sessions.test.d.ts.map +1 -0
  486. package/packages/server/dist/routes/__tests__/sessions.test.js +261 -0
  487. package/packages/server/dist/routes/__tests__/sessions.test.js.map +1 -0
  488. package/packages/server/dist/routes/auth.d.ts +8 -0
  489. package/packages/server/dist/routes/auth.d.ts.map +1 -0
  490. package/packages/server/dist/routes/auth.js +18 -0
  491. package/packages/server/dist/routes/auth.js.map +1 -0
  492. package/packages/server/dist/routes/bmadStatus.d.ts +3 -0
  493. package/packages/server/dist/routes/bmadStatus.d.ts.map +1 -0
  494. package/packages/server/dist/routes/bmadStatus.js +7 -0
  495. package/packages/server/dist/routes/bmadStatus.js.map +1 -0
  496. package/packages/server/dist/routes/board.d.ts +3 -0
  497. package/packages/server/dist/routes/board.d.ts.map +1 -0
  498. package/packages/server/dist/routes/board.js +16 -0
  499. package/packages/server/dist/routes/board.js.map +1 -0
  500. package/packages/server/dist/routes/cli.d.ts +3 -0
  501. package/packages/server/dist/routes/cli.d.ts.map +1 -0
  502. package/packages/server/dist/routes/cli.js +10 -0
  503. package/packages/server/dist/routes/cli.js.map +1 -0
  504. package/packages/server/dist/routes/commands.d.ts +8 -0
  505. package/packages/server/dist/routes/commands.d.ts.map +1 -0
  506. package/packages/server/dist/routes/commands.js +12 -0
  507. package/packages/server/dist/routes/commands.js.map +1 -0
  508. package/packages/server/dist/routes/dashboard.d.ts +3 -0
  509. package/packages/server/dist/routes/dashboard.d.ts.map +1 -0
  510. package/packages/server/dist/routes/dashboard.js +7 -0
  511. package/packages/server/dist/routes/dashboard.js.map +1 -0
  512. package/packages/server/dist/routes/debug.d.ts +6 -0
  513. package/packages/server/dist/routes/debug.d.ts.map +1 -0
  514. package/packages/server/dist/routes/debug.js +59 -0
  515. package/packages/server/dist/routes/debug.js.map +1 -0
  516. package/packages/server/dist/routes/fileSystem.d.ts +8 -0
  517. package/packages/server/dist/routes/fileSystem.d.ts.map +1 -0
  518. package/packages/server/dist/routes/fileSystem.js +24 -0
  519. package/packages/server/dist/routes/fileSystem.js.map +1 -0
  520. package/packages/server/dist/routes/git.d.ts +8 -0
  521. package/packages/server/dist/routes/git.d.ts.map +1 -0
  522. package/packages/server/dist/routes/git.js +24 -0
  523. package/packages/server/dist/routes/git.js.map +1 -0
  524. package/packages/server/dist/routes/preferences.d.ts +7 -0
  525. package/packages/server/dist/routes/preferences.d.ts.map +1 -0
  526. package/packages/server/dist/routes/preferences.js +112 -0
  527. package/packages/server/dist/routes/preferences.js.map +1 -0
  528. package/packages/server/dist/routes/projects.d.ts +9 -0
  529. package/packages/server/dist/routes/projects.d.ts.map +1 -0
  530. package/packages/server/dist/routes/projects.js +84 -0
  531. package/packages/server/dist/routes/projects.js.map +1 -0
  532. package/packages/server/dist/routes/queue.d.ts +7 -0
  533. package/packages/server/dist/routes/queue.d.ts.map +1 -0
  534. package/packages/server/dist/routes/queue.js +22 -0
  535. package/packages/server/dist/routes/queue.js.map +1 -0
  536. package/packages/server/dist/routes/server.d.ts +3 -0
  537. package/packages/server/dist/routes/server.d.ts.map +1 -0
  538. package/packages/server/dist/routes/server.js +10 -0
  539. package/packages/server/dist/routes/server.js.map +1 -0
  540. package/packages/server/dist/routes/sessions.d.ts +8 -0
  541. package/packages/server/dist/routes/sessions.d.ts.map +1 -0
  542. package/packages/server/dist/routes/sessions.js +47 -0
  543. package/packages/server/dist/routes/sessions.js.map +1 -0
  544. package/packages/server/dist/services/__tests__/authConfigService.test.d.ts +2 -0
  545. package/packages/server/dist/services/__tests__/authConfigService.test.d.ts.map +1 -0
  546. package/packages/server/dist/services/__tests__/authConfigService.test.js +274 -0
  547. package/packages/server/dist/services/__tests__/authConfigService.test.js.map +1 -0
  548. package/packages/server/dist/services/__tests__/bmadStatusService.test.d.ts +6 -0
  549. package/packages/server/dist/services/__tests__/bmadStatusService.test.d.ts.map +1 -0
  550. package/packages/server/dist/services/__tests__/bmadStatusService.test.js +456 -0
  551. package/packages/server/dist/services/__tests__/bmadStatusService.test.js.map +1 -0
  552. package/packages/server/dist/services/__tests__/chatService.integration.test.d.ts +32 -0
  553. package/packages/server/dist/services/__tests__/chatService.integration.test.d.ts.map +1 -0
  554. package/packages/server/dist/services/__tests__/chatService.integration.test.js +97 -0
  555. package/packages/server/dist/services/__tests__/chatService.integration.test.js.map +1 -0
  556. package/packages/server/dist/services/__tests__/chatService.test.d.ts +2 -0
  557. package/packages/server/dist/services/__tests__/chatService.test.d.ts.map +1 -0
  558. package/packages/server/dist/services/__tests__/chatService.test.js +517 -0
  559. package/packages/server/dist/services/__tests__/chatService.test.js.map +1 -0
  560. package/packages/server/dist/services/__tests__/cliService.test.d.ts +2 -0
  561. package/packages/server/dist/services/__tests__/cliService.test.d.ts.map +1 -0
  562. package/packages/server/dist/services/__tests__/cliService.test.js +172 -0
  563. package/packages/server/dist/services/__tests__/cliService.test.js.map +1 -0
  564. package/packages/server/dist/services/__tests__/commandService.test.d.ts +6 -0
  565. package/packages/server/dist/services/__tests__/commandService.test.d.ts.map +1 -0
  566. package/packages/server/dist/services/__tests__/commandService.test.js +525 -0
  567. package/packages/server/dist/services/__tests__/commandService.test.js.map +1 -0
  568. package/packages/server/dist/services/__tests__/dashboardService.test.d.ts +6 -0
  569. package/packages/server/dist/services/__tests__/dashboardService.test.d.ts.map +1 -0
  570. package/packages/server/dist/services/__tests__/dashboardService.test.js +248 -0
  571. package/packages/server/dist/services/__tests__/dashboardService.test.js.map +1 -0
  572. package/packages/server/dist/services/__tests__/fileSystemService.test.d.ts +7 -0
  573. package/packages/server/dist/services/__tests__/fileSystemService.test.d.ts.map +1 -0
  574. package/packages/server/dist/services/__tests__/fileSystemService.test.js +348 -0
  575. package/packages/server/dist/services/__tests__/fileSystemService.test.js.map +1 -0
  576. package/packages/server/dist/services/__tests__/gitService.test.d.ts +6 -0
  577. package/packages/server/dist/services/__tests__/gitService.test.d.ts.map +1 -0
  578. package/packages/server/dist/services/__tests__/gitService.test.js +340 -0
  579. package/packages/server/dist/services/__tests__/gitService.test.js.map +1 -0
  580. package/packages/server/dist/services/__tests__/historyParser.test.d.ts +2 -0
  581. package/packages/server/dist/services/__tests__/historyParser.test.d.ts.map +1 -0
  582. package/packages/server/dist/services/__tests__/historyParser.test.js +471 -0
  583. package/packages/server/dist/services/__tests__/historyParser.test.js.map +1 -0
  584. package/packages/server/dist/services/__tests__/issueService.test.d.ts +2 -0
  585. package/packages/server/dist/services/__tests__/issueService.test.d.ts.map +1 -0
  586. package/packages/server/dist/services/__tests__/issueService.test.js +382 -0
  587. package/packages/server/dist/services/__tests__/issueService.test.js.map +1 -0
  588. package/packages/server/dist/services/__tests__/notificationService.test.d.ts +6 -0
  589. package/packages/server/dist/services/__tests__/notificationService.test.d.ts.map +1 -0
  590. package/packages/server/dist/services/__tests__/notificationService.test.js +131 -0
  591. package/packages/server/dist/services/__tests__/notificationService.test.js.map +1 -0
  592. package/packages/server/dist/services/__tests__/preferencesService.telegram.test.d.ts +6 -0
  593. package/packages/server/dist/services/__tests__/preferencesService.telegram.test.d.ts.map +1 -0
  594. package/packages/server/dist/services/__tests__/preferencesService.telegram.test.js +101 -0
  595. package/packages/server/dist/services/__tests__/preferencesService.telegram.test.js.map +1 -0
  596. package/packages/server/dist/services/__tests__/preferencesService.test.d.ts +6 -0
  597. package/packages/server/dist/services/__tests__/preferencesService.test.d.ts.map +1 -0
  598. package/packages/server/dist/services/__tests__/preferencesService.test.js +68 -0
  599. package/packages/server/dist/services/__tests__/preferencesService.test.js.map +1 -0
  600. package/packages/server/dist/services/__tests__/projectService.settings.test.d.ts +6 -0
  601. package/packages/server/dist/services/__tests__/projectService.settings.test.d.ts.map +1 -0
  602. package/packages/server/dist/services/__tests__/projectService.settings.test.js +101 -0
  603. package/packages/server/dist/services/__tests__/projectService.settings.test.js.map +1 -0
  604. package/packages/server/dist/services/__tests__/projectService.test.d.ts +7 -0
  605. package/packages/server/dist/services/__tests__/projectService.test.d.ts.map +1 -0
  606. package/packages/server/dist/services/__tests__/projectService.test.js +618 -0
  607. package/packages/server/dist/services/__tests__/projectService.test.js.map +1 -0
  608. package/packages/server/dist/services/__tests__/ptyService.test.d.ts +6 -0
  609. package/packages/server/dist/services/__tests__/ptyService.test.d.ts.map +1 -0
  610. package/packages/server/dist/services/__tests__/ptyService.test.js +289 -0
  611. package/packages/server/dist/services/__tests__/ptyService.test.js.map +1 -0
  612. package/packages/server/dist/services/__tests__/queueService.test.d.ts +6 -0
  613. package/packages/server/dist/services/__tests__/queueService.test.d.ts.map +1 -0
  614. package/packages/server/dist/services/__tests__/queueService.test.js +477 -0
  615. package/packages/server/dist/services/__tests__/queueService.test.js.map +1 -0
  616. package/packages/server/dist/services/__tests__/queueTemplateService.test.d.ts +6 -0
  617. package/packages/server/dist/services/__tests__/queueTemplateService.test.d.ts.map +1 -0
  618. package/packages/server/dist/services/__tests__/queueTemplateService.test.js +129 -0
  619. package/packages/server/dist/services/__tests__/queueTemplateService.test.js.map +1 -0
  620. package/packages/server/dist/services/__tests__/rateLimiter.test.d.ts +6 -0
  621. package/packages/server/dist/services/__tests__/rateLimiter.test.d.ts.map +1 -0
  622. package/packages/server/dist/services/__tests__/rateLimiter.test.js +123 -0
  623. package/packages/server/dist/services/__tests__/rateLimiter.test.js.map +1 -0
  624. package/packages/server/dist/services/__tests__/sessionService.test.d.ts +2 -0
  625. package/packages/server/dist/services/__tests__/sessionService.test.d.ts.map +1 -0
  626. package/packages/server/dist/services/__tests__/sessionService.test.js +349 -0
  627. package/packages/server/dist/services/__tests__/sessionService.test.js.map +1 -0
  628. package/packages/server/dist/services/__tests__/streamHandler.test.d.ts +2 -0
  629. package/packages/server/dist/services/__tests__/streamHandler.test.d.ts.map +1 -0
  630. package/packages/server/dist/services/__tests__/streamHandler.test.js +654 -0
  631. package/packages/server/dist/services/__tests__/streamHandler.test.js.map +1 -0
  632. package/packages/server/dist/services/authConfigService.d.ts +56 -0
  633. package/packages/server/dist/services/authConfigService.d.ts.map +1 -0
  634. package/packages/server/dist/services/authConfigService.js +186 -0
  635. package/packages/server/dist/services/authConfigService.js.map +1 -0
  636. package/packages/server/dist/services/bmadStatusService.d.ts +56 -0
  637. package/packages/server/dist/services/bmadStatusService.d.ts.map +1 -0
  638. package/packages/server/dist/services/bmadStatusService.js +438 -0
  639. package/packages/server/dist/services/bmadStatusService.js.map +1 -0
  640. package/packages/server/dist/services/chatService.d.ts +78 -0
  641. package/packages/server/dist/services/chatService.d.ts.map +1 -0
  642. package/packages/server/dist/services/chatService.js +335 -0
  643. package/packages/server/dist/services/chatService.js.map +1 -0
  644. package/packages/server/dist/services/cliService.d.ts +29 -0
  645. package/packages/server/dist/services/cliService.d.ts.map +1 -0
  646. package/packages/server/dist/services/cliService.js +101 -0
  647. package/packages/server/dist/services/cliService.js.map +1 -0
  648. package/packages/server/dist/services/commandService.d.ts +84 -0
  649. package/packages/server/dist/services/commandService.d.ts.map +1 -0
  650. package/packages/server/dist/services/commandService.js +322 -0
  651. package/packages/server/dist/services/commandService.js.map +1 -0
  652. package/packages/server/dist/services/dashboardService.d.ts +23 -0
  653. package/packages/server/dist/services/dashboardService.d.ts.map +1 -0
  654. package/packages/server/dist/services/dashboardService.js +81 -0
  655. package/packages/server/dist/services/dashboardService.js.map +1 -0
  656. package/packages/server/dist/services/fileSystemService.d.ts +105 -0
  657. package/packages/server/dist/services/fileSystemService.d.ts.map +1 -0
  658. package/packages/server/dist/services/fileSystemService.js +498 -0
  659. package/packages/server/dist/services/fileSystemService.js.map +1 -0
  660. package/packages/server/dist/services/gitService.d.ts +25 -0
  661. package/packages/server/dist/services/gitService.d.ts.map +1 -0
  662. package/packages/server/dist/services/gitService.js +230 -0
  663. package/packages/server/dist/services/gitService.js.map +1 -0
  664. package/packages/server/dist/services/historyParser.d.ts +45 -0
  665. package/packages/server/dist/services/historyParser.d.ts.map +1 -0
  666. package/packages/server/dist/services/historyParser.js +315 -0
  667. package/packages/server/dist/services/historyParser.js.map +1 -0
  668. package/packages/server/dist/services/issueService.d.ts +79 -0
  669. package/packages/server/dist/services/issueService.d.ts.map +1 -0
  670. package/packages/server/dist/services/issueService.js +708 -0
  671. package/packages/server/dist/services/issueService.js.map +1 -0
  672. package/packages/server/dist/services/notificationService.d.ts +78 -0
  673. package/packages/server/dist/services/notificationService.d.ts.map +1 -0
  674. package/packages/server/dist/services/notificationService.js +246 -0
  675. package/packages/server/dist/services/notificationService.js.map +1 -0
  676. package/packages/server/dist/services/preferencesService.d.ts +35 -0
  677. package/packages/server/dist/services/preferencesService.d.ts.map +1 -0
  678. package/packages/server/dist/services/preferencesService.js +121 -0
  679. package/packages/server/dist/services/preferencesService.js.map +1 -0
  680. package/packages/server/dist/services/projectService.d.ts +210 -0
  681. package/packages/server/dist/services/projectService.d.ts.map +1 -0
  682. package/packages/server/dist/services/projectService.js +892 -0
  683. package/packages/server/dist/services/projectService.js.map +1 -0
  684. package/packages/server/dist/services/ptyService.d.ts +83 -0
  685. package/packages/server/dist/services/ptyService.d.ts.map +1 -0
  686. package/packages/server/dist/services/ptyService.js +214 -0
  687. package/packages/server/dist/services/ptyService.js.map +1 -0
  688. package/packages/server/dist/services/queueService.d.ts +93 -0
  689. package/packages/server/dist/services/queueService.d.ts.map +1 -0
  690. package/packages/server/dist/services/queueService.js +598 -0
  691. package/packages/server/dist/services/queueService.js.map +1 -0
  692. package/packages/server/dist/services/queueTemplateService.d.ts +12 -0
  693. package/packages/server/dist/services/queueTemplateService.d.ts.map +1 -0
  694. package/packages/server/dist/services/queueTemplateService.js +73 -0
  695. package/packages/server/dist/services/queueTemplateService.js.map +1 -0
  696. package/packages/server/dist/services/rateLimitProbeService.d.ts +56 -0
  697. package/packages/server/dist/services/rateLimitProbeService.d.ts.map +1 -0
  698. package/packages/server/dist/services/rateLimitProbeService.js +227 -0
  699. package/packages/server/dist/services/rateLimitProbeService.js.map +1 -0
  700. package/packages/server/dist/services/rateLimiter.d.ts +32 -0
  701. package/packages/server/dist/services/rateLimiter.d.ts.map +1 -0
  702. package/packages/server/dist/services/rateLimiter.js +60 -0
  703. package/packages/server/dist/services/rateLimiter.js.map +1 -0
  704. package/packages/server/dist/services/sessionService.d.ts +152 -0
  705. package/packages/server/dist/services/sessionService.d.ts.map +1 -0
  706. package/packages/server/dist/services/sessionService.js +672 -0
  707. package/packages/server/dist/services/sessionService.js.map +1 -0
  708. package/packages/server/dist/services/streamHandler.d.ts +145 -0
  709. package/packages/server/dist/services/streamHandler.d.ts.map +1 -0
  710. package/packages/server/dist/services/streamHandler.js +753 -0
  711. package/packages/server/dist/services/streamHandler.js.map +1 -0
  712. package/packages/server/dist/utils/__tests__/networkUtils.test.d.ts +6 -0
  713. package/packages/server/dist/utils/__tests__/networkUtils.test.d.ts.map +1 -0
  714. package/packages/server/dist/utils/__tests__/networkUtils.test.js +106 -0
  715. package/packages/server/dist/utils/__tests__/networkUtils.test.js.map +1 -0
  716. package/packages/server/dist/utils/__tests__/pathUtils.test.d.ts +6 -0
  717. package/packages/server/dist/utils/__tests__/pathUtils.test.d.ts.map +1 -0
  718. package/packages/server/dist/utils/__tests__/pathUtils.test.js +111 -0
  719. package/packages/server/dist/utils/__tests__/pathUtils.test.js.map +1 -0
  720. package/packages/server/dist/utils/errors.d.ts +79 -0
  721. package/packages/server/dist/utils/errors.d.ts.map +1 -0
  722. package/packages/server/dist/utils/errors.js +167 -0
  723. package/packages/server/dist/utils/errors.js.map +1 -0
  724. package/packages/server/dist/utils/logger.d.ts +28 -0
  725. package/packages/server/dist/utils/logger.d.ts.map +1 -0
  726. package/packages/server/dist/utils/logger.js +87 -0
  727. package/packages/server/dist/utils/logger.js.map +1 -0
  728. package/packages/server/dist/utils/networkUtils.d.ts +26 -0
  729. package/packages/server/dist/utils/networkUtils.d.ts.map +1 -0
  730. package/packages/server/dist/utils/networkUtils.js +70 -0
  731. package/packages/server/dist/utils/networkUtils.js.map +1 -0
  732. package/packages/server/dist/utils/pathUtils.d.ts +30 -0
  733. package/packages/server/dist/utils/pathUtils.d.ts.map +1 -0
  734. package/packages/server/dist/utils/pathUtils.js +81 -0
  735. package/packages/server/dist/utils/pathUtils.js.map +1 -0
  736. package/packages/server/package.json +48 -0
  737. package/packages/shared/dist/constants/errorCodes.d.ts +31 -0
  738. package/packages/shared/dist/constants/errorCodes.d.ts.map +1 -0
  739. package/packages/shared/dist/constants/errorCodes.js +33 -0
  740. package/packages/shared/dist/constants/errorCodes.js.map +1 -0
  741. package/packages/shared/dist/index.d.ts +26 -0
  742. package/packages/shared/dist/index.d.ts.map +1 -0
  743. package/packages/shared/dist/index.js +48 -0
  744. package/packages/shared/dist/index.js.map +1 -0
  745. package/packages/shared/dist/types/auth.d.ts +159 -0
  746. package/packages/shared/dist/types/auth.d.ts.map +1 -0
  747. package/packages/shared/dist/types/auth.js +74 -0
  748. package/packages/shared/dist/types/auth.js.map +1 -0
  749. package/packages/shared/dist/types/bmadStatus.d.ts +97 -0
  750. package/packages/shared/dist/types/bmadStatus.d.ts.map +1 -0
  751. package/packages/shared/dist/types/bmadStatus.js +18 -0
  752. package/packages/shared/dist/types/bmadStatus.js.map +1 -0
  753. package/packages/shared/dist/types/board.d.ts +68 -0
  754. package/packages/shared/dist/types/board.d.ts.map +1 -0
  755. package/packages/shared/dist/types/board.js +108 -0
  756. package/packages/shared/dist/types/board.js.map +1 -0
  757. package/packages/shared/dist/types/cli.d.ts +34 -0
  758. package/packages/shared/dist/types/cli.d.ts.map +1 -0
  759. package/packages/shared/dist/types/cli.js +10 -0
  760. package/packages/shared/dist/types/cli.js.map +1 -0
  761. package/packages/shared/dist/types/command.d.ts +39 -0
  762. package/packages/shared/dist/types/command.d.ts.map +1 -0
  763. package/packages/shared/dist/types/command.js +6 -0
  764. package/packages/shared/dist/types/command.js.map +1 -0
  765. package/packages/shared/dist/types/dashboard.d.ts +15 -0
  766. package/packages/shared/dist/types/dashboard.d.ts.map +1 -0
  767. package/packages/shared/dist/types/dashboard.js +3 -0
  768. package/packages/shared/dist/types/dashboard.js.map +1 -0
  769. package/packages/shared/dist/types/fileSystem.d.ts +175 -0
  770. package/packages/shared/dist/types/fileSystem.d.ts.map +1 -0
  771. package/packages/shared/dist/types/fileSystem.js +62 -0
  772. package/packages/shared/dist/types/fileSystem.js.map +1 -0
  773. package/packages/shared/dist/types/git.d.ts +81 -0
  774. package/packages/shared/dist/types/git.d.ts.map +1 -0
  775. package/packages/shared/dist/types/git.js +34 -0
  776. package/packages/shared/dist/types/git.js.map +1 -0
  777. package/packages/shared/dist/types/history.d.ts +103 -0
  778. package/packages/shared/dist/types/history.d.ts.map +1 -0
  779. package/packages/shared/dist/types/history.js +6 -0
  780. package/packages/shared/dist/types/history.js.map +1 -0
  781. package/packages/shared/dist/types/logger.d.ts +18 -0
  782. package/packages/shared/dist/types/logger.d.ts.map +1 -0
  783. package/packages/shared/dist/types/logger.js +29 -0
  784. package/packages/shared/dist/types/logger.js.map +1 -0
  785. package/packages/shared/dist/types/message.d.ts +34 -0
  786. package/packages/shared/dist/types/message.d.ts.map +1 -0
  787. package/packages/shared/dist/types/message.js +14 -0
  788. package/packages/shared/dist/types/message.js.map +1 -0
  789. package/packages/shared/dist/types/preferences.d.ts +95 -0
  790. package/packages/shared/dist/types/preferences.d.ts.map +1 -0
  791. package/packages/shared/dist/types/preferences.js +15 -0
  792. package/packages/shared/dist/types/preferences.js.map +1 -0
  793. package/packages/shared/dist/types/project.d.ts +205 -0
  794. package/packages/shared/dist/types/project.d.ts.map +1 -0
  795. package/packages/shared/dist/types/project.js +64 -0
  796. package/packages/shared/dist/types/project.js.map +1 -0
  797. package/packages/shared/dist/types/queue.d.ts +94 -0
  798. package/packages/shared/dist/types/queue.d.ts.map +1 -0
  799. package/packages/shared/dist/types/queue.js +2 -0
  800. package/packages/shared/dist/types/queue.js.map +1 -0
  801. package/packages/shared/dist/types/sdk.d.ts +156 -0
  802. package/packages/shared/dist/types/sdk.d.ts.map +1 -0
  803. package/packages/shared/dist/types/sdk.js +13 -0
  804. package/packages/shared/dist/types/sdk.js.map +1 -0
  805. package/packages/shared/dist/types/session.d.ts +149 -0
  806. package/packages/shared/dist/types/session.d.ts.map +1 -0
  807. package/packages/shared/dist/types/session.js +51 -0
  808. package/packages/shared/dist/types/session.js.map +1 -0
  809. package/packages/shared/dist/types/streaming.d.ts +269 -0
  810. package/packages/shared/dist/types/streaming.d.ts.map +1 -0
  811. package/packages/shared/dist/types/streaming.js +43 -0
  812. package/packages/shared/dist/types/streaming.js.map +1 -0
  813. package/packages/shared/dist/types/terminal.d.ts +93 -0
  814. package/packages/shared/dist/types/terminal.d.ts.map +1 -0
  815. package/packages/shared/dist/types/terminal.js +38 -0
  816. package/packages/shared/dist/types/terminal.js.map +1 -0
  817. package/packages/shared/dist/types/websocket.d.ts +196 -0
  818. package/packages/shared/dist/types/websocket.d.ts.map +1 -0
  819. package/packages/shared/dist/types/websocket.js +7 -0
  820. package/packages/shared/dist/types/websocket.js.map +1 -0
  821. package/packages/shared/dist/utils/__tests__/queueParser.test.d.ts +2 -0
  822. package/packages/shared/dist/utils/__tests__/queueParser.test.d.ts.map +1 -0
  823. package/packages/shared/dist/utils/__tests__/queueParser.test.js +246 -0
  824. package/packages/shared/dist/utils/__tests__/queueParser.test.js.map +1 -0
  825. package/packages/shared/dist/utils/__tests__/queueTemplateUtils.test.d.ts +6 -0
  826. package/packages/shared/dist/utils/__tests__/queueTemplateUtils.test.d.ts.map +1 -0
  827. package/packages/shared/dist/utils/__tests__/queueTemplateUtils.test.js +138 -0
  828. package/packages/shared/dist/utils/__tests__/queueTemplateUtils.test.js.map +1 -0
  829. package/packages/shared/dist/utils/queueParser.d.ts +3 -0
  830. package/packages/shared/dist/utils/queueParser.d.ts.map +1 -0
  831. package/packages/shared/dist/utils/queueParser.js +165 -0
  832. package/packages/shared/dist/utils/queueParser.js.map +1 -0
  833. package/packages/shared/dist/utils/queueTemplateUtils.d.ts +17 -0
  834. package/packages/shared/dist/utils/queueTemplateUtils.d.ts.map +1 -0
  835. package/packages/shared/dist/utils/queueTemplateUtils.js +65 -0
  836. package/packages/shared/dist/utils/queueTemplateUtils.js.map +1 -0
  837. package/packages/shared/package.json +23 -0
  838. package/scripts/postinstall.cjs +23 -0
@@ -0,0 +1,1316 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { createServer } from 'http';
3
+ import { Server as SocketIOServer } from 'socket.io';
4
+ import { io as ioc } from 'socket.io-client';
5
+ import { existsSync } from 'fs';
6
+ import { initializeWebSocket, getIO, getConnectedClientsCount, } from '../websocket.js';
7
+ import { ERROR_CODES } from '@bmad-studio/shared';
8
+ // Shared mock state — accessible via vi.hoisted() for vi.mock factories
9
+ const { mockState } = vi.hoisted(() => {
10
+ const mockState = {
11
+ sendImpl: vi.fn().mockResolvedValue({}),
12
+ lastCtorArgs: undefined,
13
+ };
14
+ return { mockState };
15
+ });
16
+ // Mock fs module used by logger and other imports
17
+ vi.mock('fs', () => ({
18
+ existsSync: vi.fn(),
19
+ mkdirSync: vi.fn(),
20
+ }));
21
+ // Mock session middleware (Story 2.5 - WebSocket auth)
22
+ vi.mock('../../middleware/session.js', () => ({
23
+ createSessionMiddleware: vi.fn().mockResolvedValue(
24
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
25
+ (req, _res, next) => {
26
+ // Simulate authenticated session for tests
27
+ req.session = { authenticated: true };
28
+ next();
29
+ }),
30
+ }));
31
+ // Mock ChatService - real class that captures constructor args and delegates to mockState
32
+ vi.mock('../../services/chatService.js', () => ({
33
+ ChatService: class MockChatService {
34
+ constructor(...args) { mockState.lastCtorArgs = args[0]; }
35
+ sendMessageWithCallbacks(...args) { return mockState.sendImpl(...args); }
36
+ setPermissionMode() { }
37
+ },
38
+ }));
39
+ // Mock SessionService - must be a real class (used with `new`)
40
+ vi.mock('../../services/sessionService.js', () => ({
41
+ SessionService: class MockSessionService {
42
+ saveSessionId = vi.fn().mockResolvedValue(undefined);
43
+ getSessionId = vi.fn().mockResolvedValue(null);
44
+ listSessions = vi.fn().mockResolvedValue([
45
+ {
46
+ sessionId: 'session-123',
47
+ projectSlug: 'test-project',
48
+ firstPrompt: 'Create a component',
49
+ messageCount: 5,
50
+ created: new Date('2026-01-30T10:00:00Z'),
51
+ modified: new Date('2026-01-30T11:00:00Z'),
52
+ },
53
+ ]);
54
+ },
55
+ }));
56
+ // Mock preferencesService (Story 10.2)
57
+ vi.mock('../../services/preferencesService.js', () => ({
58
+ preferencesService: {
59
+ getEffectivePreferences: vi.fn().mockResolvedValue({
60
+ theme: 'dark',
61
+ defaultModel: '',
62
+ permissionMode: 'default',
63
+ chatTimeoutMs: 300000,
64
+ }),
65
+ getTerminalEnabled: vi.fn().mockResolvedValue(true),
66
+ },
67
+ }));
68
+ // Mock config (Story 4.6)
69
+ vi.mock('../../config/index.js', () => ({
70
+ config: {
71
+ chat: {
72
+ timeoutMs: 300000, // 5 minutes default
73
+ },
74
+ websocket: {
75
+ cors: {
76
+ origin: 'http://localhost:5173',
77
+ methods: ['GET', 'POST'],
78
+ credentials: true,
79
+ },
80
+ },
81
+ telegram: {
82
+ botToken: '',
83
+ chatId: '',
84
+ enabled: false,
85
+ },
86
+ terminal: {
87
+ enabled: true,
88
+ shellTimeout: 30000,
89
+ maxSessions: 10,
90
+ },
91
+ },
92
+ }));
93
+ // Mock notificationService (Story 10.4)
94
+ vi.mock('../../services/notificationService.js', () => ({
95
+ notificationService: {
96
+ notifyInputRequired: vi.fn().mockResolvedValue(undefined),
97
+ notifyComplete: vi.fn().mockResolvedValue(undefined),
98
+ notifyError: vi.fn().mockResolvedValue(undefined),
99
+ notifyQueueStart: vi.fn().mockResolvedValue(undefined),
100
+ notifyQueueComplete: vi.fn().mockResolvedValue(undefined),
101
+ notifyQueueError: vi.fn().mockResolvedValue(undefined),
102
+ notifyQueueInputRequired: vi.fn().mockResolvedValue(undefined),
103
+ reload: vi.fn().mockResolvedValue(undefined),
104
+ },
105
+ }));
106
+ // Mock queueController (Story 15.2)
107
+ vi.mock('../../controllers/queueController.js', () => ({
108
+ getOrCreateQueueService: vi.fn().mockReturnValue({
109
+ isRunning: false,
110
+ start: vi.fn().mockResolvedValue(undefined),
111
+ pause: vi.fn().mockResolvedValue(undefined),
112
+ resume: vi.fn().mockResolvedValue(undefined),
113
+ abort: vi.fn().mockResolvedValue(undefined),
114
+ }),
115
+ getQueueInstances: vi.fn().mockReturnValue(new Map()),
116
+ }));
117
+ // Mock utils/networkUtils
118
+ vi.mock('../../utils/networkUtils.js', () => ({
119
+ isLocalIP: vi.fn().mockReturnValue(true),
120
+ extractClientIP: vi.fn().mockReturnValue('127.0.0.1'),
121
+ }));
122
+ // Mock utils/errors — parseSDKError passes through the original error
123
+ // so that `instanceof` checks (e.g. AbortedError) still work in the handler
124
+ vi.mock('../../utils/errors.js', () => ({
125
+ parseSDKError: vi.fn().mockImplementation((err) => err),
126
+ AbortedError: class AbortedError extends Error {
127
+ constructor(message) { super(message ?? 'Aborted'); this.name = 'AbortedError'; }
128
+ },
129
+ }));
130
+ // streamCallbacks: use real implementation (callbacks drive the socket events tests verify)
131
+ // Mock rateLimitProbeService
132
+ vi.mock('../../services/rateLimitProbeService.js', () => ({
133
+ rateLimitProbeService: {
134
+ startPolling: vi.fn(),
135
+ stopPolling: vi.fn(),
136
+ getCachedResult: vi.fn().mockReturnValue(null),
137
+ getApiHealth: vi.fn().mockReturnValue(null),
138
+ },
139
+ }));
140
+ // Mock ptyService
141
+ vi.mock('../../services/ptyService.js', () => ({
142
+ ptyService: {
143
+ createSession: vi.fn().mockReturnValue({ terminalId: 'term-1', shell: 'bash' }),
144
+ getSession: vi.fn(),
145
+ closeSession: vi.fn(),
146
+ scheduleCleanup: vi.fn(),
147
+ cancelCleanup: vi.fn(),
148
+ onData: vi.fn(),
149
+ onExit: vi.fn(),
150
+ writeInput: vi.fn(),
151
+ resize: vi.fn(),
152
+ },
153
+ }));
154
+ // Mock projectService
155
+ vi.mock('../../services/projectService.js', () => ({
156
+ projectService: {
157
+ resolveProjectPath: vi.fn().mockResolvedValue('/mock/project/path'),
158
+ findProjectByPath: vi.fn().mockResolvedValue({ projectSlug: 'test-project' }),
159
+ },
160
+ }));
161
+ // Mock dashboardService (Story 20.1)
162
+ const mockGetProjectStatus = vi.fn().mockResolvedValue({
163
+ projectSlug: 'test-project',
164
+ activeSessionCount: 0,
165
+ totalSessionCount: 0,
166
+ queueStatus: 'idle',
167
+ terminalCount: 0,
168
+ });
169
+ vi.mock('../../services/dashboardService.js', () => ({
170
+ dashboardService: {
171
+ getProjectStatus: (...args) => mockGetProjectStatus(...args),
172
+ },
173
+ }));
174
+ // Mock logger
175
+ vi.mock('../../utils/logger.js', () => ({
176
+ createLogger: vi.fn().mockReturnValue({
177
+ info: vi.fn(),
178
+ warn: vi.fn(),
179
+ error: vi.fn(),
180
+ debug: vi.fn(),
181
+ verbose: vi.fn(),
182
+ }),
183
+ }));
184
+ describe('WebSocket Handler', () => {
185
+ let httpServer;
186
+ let ioServer;
187
+ let clientSocket;
188
+ const TEST_PORT = 3001;
189
+ beforeEach(async () => {
190
+ // Reset shared ChatService mock state
191
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
192
+ mockState.lastCtorArgs = undefined;
193
+ // Create HTTP server
194
+ httpServer = createServer();
195
+ // Initialize WebSocket (now async for session middleware - Story 2.5)
196
+ ioServer = await initializeWebSocket(httpServer);
197
+ // Start server
198
+ await new Promise((resolve) => {
199
+ httpServer.listen(TEST_PORT, () => resolve());
200
+ });
201
+ });
202
+ afterEach(async () => {
203
+ // Cleanup client socket
204
+ if (clientSocket?.connected) {
205
+ clientSocket.disconnect();
206
+ }
207
+ // Close server
208
+ await new Promise((resolve) => {
209
+ ioServer.close(() => {
210
+ httpServer.close(() => resolve());
211
+ });
212
+ });
213
+ });
214
+ describe('initializeWebSocket', () => {
215
+ it('should return a Socket.io server instance', () => {
216
+ expect(ioServer).toBeInstanceOf(SocketIOServer);
217
+ });
218
+ it('should configure CORS for localhost:5173', () => {
219
+ const opts = ioServer.engine.opts;
220
+ expect(opts.cors).toBeDefined();
221
+ // Type assertion for CorsOptions object
222
+ const corsOpts = opts.cors;
223
+ expect(corsOpts.origin).toBe('http://localhost:5173');
224
+ expect(corsOpts.methods).toContain('GET');
225
+ expect(corsOpts.methods).toContain('POST');
226
+ expect(corsOpts.credentials).toBe(true);
227
+ });
228
+ });
229
+ describe('getIO', () => {
230
+ it('should return the initialized Socket.io instance', () => {
231
+ const io = getIO();
232
+ expect(io).toBe(ioServer);
233
+ });
234
+ });
235
+ describe('Client connection', () => {
236
+ it('should accept client connection', async () => {
237
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
238
+ transports: ['websocket'],
239
+ });
240
+ await new Promise((resolve) => {
241
+ clientSocket.on('connect', () => {
242
+ expect(clientSocket.connected).toBe(true);
243
+ resolve();
244
+ });
245
+ });
246
+ });
247
+ it('should increment connected clients count on connection', async () => {
248
+ const initialCount = getConnectedClientsCount();
249
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
250
+ transports: ['websocket'],
251
+ });
252
+ await new Promise((resolve) => {
253
+ clientSocket.on('connect', () => {
254
+ expect(getConnectedClientsCount()).toBe(initialCount + 1);
255
+ resolve();
256
+ });
257
+ });
258
+ });
259
+ it('should decrement connected clients count on disconnect', async () => {
260
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
261
+ transports: ['websocket'],
262
+ });
263
+ await new Promise((resolve) => {
264
+ clientSocket.on('connect', () => resolve());
265
+ });
266
+ const countBeforeDisconnect = getConnectedClientsCount();
267
+ clientSocket.disconnect();
268
+ // Wait for disconnect to be processed
269
+ await new Promise((resolve) => setTimeout(resolve, 100));
270
+ expect(getConnectedClientsCount()).toBe(countBeforeDisconnect - 1);
271
+ });
272
+ it('should log connection and disconnection', async () => {
273
+ const { createLogger } = await import('../../utils/logger.js');
274
+ const mockLogger = vi.mocked(createLogger).mock.results[0]?.value;
275
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
276
+ transports: ['websocket'],
277
+ });
278
+ await new Promise((resolve) => {
279
+ clientSocket.on('connect', () => resolve());
280
+ });
281
+ expect(mockLogger.info).toHaveBeenCalledWith(expect.stringContaining('Client connected'));
282
+ clientSocket.disconnect();
283
+ // Wait for disconnect to be processed
284
+ await new Promise((resolve) => setTimeout(resolve, 100));
285
+ expect(mockLogger.info).toHaveBeenCalledWith(expect.stringContaining('Client disconnected'));
286
+ });
287
+ });
288
+ describe('Multiple clients', () => {
289
+ let clientSocket2;
290
+ afterEach(() => {
291
+ if (clientSocket2?.connected) {
292
+ clientSocket2.disconnect();
293
+ }
294
+ });
295
+ it('should track multiple client connections', async () => {
296
+ const initialCount = getConnectedClientsCount();
297
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
298
+ transports: ['websocket'],
299
+ });
300
+ clientSocket2 = ioc(`http://localhost:${TEST_PORT}`, {
301
+ transports: ['websocket'],
302
+ });
303
+ await Promise.all([
304
+ new Promise((resolve) => {
305
+ clientSocket.on('connect', () => resolve());
306
+ }),
307
+ new Promise((resolve) => {
308
+ clientSocket2.on('connect', () => resolve());
309
+ }),
310
+ ]);
311
+ expect(getConnectedClientsCount()).toBe(initialCount + 2);
312
+ });
313
+ });
314
+ describe('chat:send event handler', () => {
315
+ beforeEach(() => {
316
+ vi.clearAllMocks();
317
+ });
318
+ it('should emit error for invalid workingDirectory (non-existent)', async () => {
319
+ vi.mocked(existsSync).mockReturnValue(false);
320
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
321
+ transports: ['websocket'],
322
+ });
323
+ await new Promise((resolve) => {
324
+ clientSocket.on('connect', () => resolve());
325
+ });
326
+ const errorPromise = new Promise((resolve) => {
327
+ clientSocket.on('error', (error) => resolve(error));
328
+ });
329
+ clientSocket.emit('chat:send', {
330
+ sessionId: 'test-session',
331
+ content: 'Hello',
332
+ workingDirectory: '/non/existent/path',
333
+ });
334
+ const error = await errorPromise;
335
+ expect(error.code).toBe(ERROR_CODES.INVALID_WORKING_DIR);
336
+ expect(error.message).toBe('ws.error.projectPathNotFound');
337
+ });
338
+ it('should emit error for empty workingDirectory', async () => {
339
+ vi.mocked(existsSync).mockReturnValue(false);
340
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
341
+ transports: ['websocket'],
342
+ });
343
+ await new Promise((resolve) => {
344
+ clientSocket.on('connect', () => resolve());
345
+ });
346
+ const errorPromise = new Promise((resolve) => {
347
+ clientSocket.on('error', (error) => resolve(error));
348
+ });
349
+ clientSocket.emit('chat:send', {
350
+ sessionId: 'test-session',
351
+ content: 'Hello',
352
+ workingDirectory: '',
353
+ });
354
+ const error = await errorPromise;
355
+ expect(error.code).toBe(ERROR_CODES.INVALID_WORKING_DIR);
356
+ });
357
+ it('should call ChatService when workingDirectory is valid', async () => {
358
+ vi.mocked(existsSync).mockReturnValue(true);
359
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
360
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
361
+ transports: ['websocket'],
362
+ });
363
+ await new Promise((resolve) => {
364
+ clientSocket.on('connect', () => resolve());
365
+ });
366
+ clientSocket.emit('chat:send', {
367
+ content: 'Hello Claude',
368
+ workingDirectory: '/valid/path',
369
+ });
370
+ // Wait for the handler to process
371
+ await new Promise((resolve) => setTimeout(resolve, 100));
372
+ expect(mockState.lastCtorArgs).toEqual(expect.objectContaining({ workingDirectory: '/valid/path' }));
373
+ expect(mockState.sendImpl).toHaveBeenCalledWith('Hello Claude', expect.any(Object), expect.any(Object), expect.any(Function), expect.any(Function));
374
+ });
375
+ it('should pass resume option when sessionId and resume are provided', async () => {
376
+ vi.mocked(existsSync).mockReturnValue(true);
377
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
378
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
379
+ transports: ['websocket'],
380
+ });
381
+ await new Promise((resolve) => {
382
+ clientSocket.on('connect', () => resolve());
383
+ });
384
+ clientSocket.emit('chat:send', {
385
+ content: 'Continue our work',
386
+ workingDirectory: '/valid/path',
387
+ sessionId: 'resume-session-id',
388
+ resume: true,
389
+ });
390
+ // Wait for the handler to process
391
+ await new Promise((resolve) => setTimeout(resolve, 100));
392
+ // chatOptions includes resume and abortController
393
+ expect(mockState.sendImpl).toHaveBeenCalledWith('Continue our work', expect.any(Object), expect.objectContaining({ resume: 'resume-session-id', abortController: expect.any(AbortController) }), expect.any(Function), expect.any(Function));
394
+ });
395
+ });
396
+ describe('session:list event handler', () => {
397
+ beforeEach(() => {
398
+ vi.clearAllMocks();
399
+ });
400
+ it('should emit error for invalid projectPath', async () => {
401
+ vi.mocked(existsSync).mockReturnValue(false);
402
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
403
+ transports: ['websocket'],
404
+ });
405
+ await new Promise((resolve) => {
406
+ clientSocket.on('connect', () => resolve());
407
+ });
408
+ const errorPromise = new Promise((resolve) => {
409
+ clientSocket.on('error', (error) => resolve(error));
410
+ });
411
+ clientSocket.emit('session:list', {
412
+ projectPath: '/non/existent/path',
413
+ });
414
+ const error = await errorPromise;
415
+ expect(error.code).toBe(ERROR_CODES.INVALID_WORKING_DIR);
416
+ });
417
+ it('should emit session:list with sessions for valid projectPath', async () => {
418
+ vi.mocked(existsSync).mockReturnValue(true);
419
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
420
+ transports: ['websocket'],
421
+ });
422
+ await new Promise((resolve) => {
423
+ clientSocket.on('connect', () => resolve());
424
+ });
425
+ const sessionsPromise = new Promise((resolve) => {
426
+ clientSocket.on('session:list', (data) => resolve(data));
427
+ });
428
+ clientSocket.emit('session:list', {
429
+ projectPath: '/valid/path',
430
+ });
431
+ const result = await sessionsPromise;
432
+ expect(result.sessions).toBeDefined();
433
+ expect(result.sessions).toHaveLength(1);
434
+ expect(result.sessions[0]).toMatchObject({
435
+ sessionId: 'session-123',
436
+ firstPrompt: 'Create a component',
437
+ });
438
+ });
439
+ });
440
+ describe('WebSocket Authentication (Story 2.5)', () => {
441
+ it('should accept connection with authenticated session', async () => {
442
+ // Default mock has authenticated: true
443
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
444
+ transports: ['websocket'],
445
+ });
446
+ await new Promise((resolve) => {
447
+ clientSocket.on('connect', () => {
448
+ expect(clientSocket.connected).toBe(true);
449
+ resolve();
450
+ });
451
+ });
452
+ });
453
+ });
454
+ describe('Story 5.2: permissionMode handling', () => {
455
+ beforeEach(() => {
456
+ vi.clearAllMocks();
457
+ });
458
+ it('should pass permissionMode to ChatService constructor when provided', async () => {
459
+ vi.mocked(existsSync).mockReturnValue(true);
460
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
461
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
462
+ transports: ['websocket'],
463
+ });
464
+ await new Promise((resolve) => {
465
+ clientSocket.on('connect', () => resolve());
466
+ });
467
+ clientSocket.emit('chat:send', {
468
+ content: 'Hello Claude',
469
+ workingDirectory: '/valid/path',
470
+ permissionMode: 'plan',
471
+ });
472
+ // Wait for the handler to process
473
+ await new Promise((resolve) => setTimeout(resolve, 100));
474
+ expect(mockState.lastCtorArgs).toEqual(expect.objectContaining({ workingDirectory: '/valid/path', permissionMode: 'plan' }));
475
+ });
476
+ it('should pass undefined permissionMode when not provided (defaults to default)', async () => {
477
+ vi.mocked(existsSync).mockReturnValue(true);
478
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
479
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
480
+ transports: ['websocket'],
481
+ });
482
+ await new Promise((resolve) => {
483
+ clientSocket.on('connect', () => resolve());
484
+ });
485
+ clientSocket.emit('chat:send', {
486
+ content: 'Hello Claude',
487
+ workingDirectory: '/valid/path',
488
+ });
489
+ // Wait for the handler to process
490
+ await new Promise((resolve) => setTimeout(resolve, 100));
491
+ expect(mockState.lastCtorArgs).toEqual(expect.objectContaining({ workingDirectory: '/valid/path' }));
492
+ });
493
+ it('should pass acceptEdits permissionMode to ChatService', async () => {
494
+ vi.mocked(existsSync).mockReturnValue(true);
495
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
496
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
497
+ transports: ['websocket'],
498
+ });
499
+ await new Promise((resolve) => {
500
+ clientSocket.on('connect', () => resolve());
501
+ });
502
+ clientSocket.emit('chat:send', {
503
+ content: 'Auto edit',
504
+ workingDirectory: '/valid/path',
505
+ permissionMode: 'acceptEdits',
506
+ });
507
+ // Wait for the handler to process
508
+ await new Promise((resolve) => setTimeout(resolve, 100));
509
+ expect(mockState.lastCtorArgs).toEqual(expect.objectContaining({ workingDirectory: '/valid/path', permissionMode: 'acceptEdits' }));
510
+ });
511
+ });
512
+ describe('Story 5.5: Image validation', () => {
513
+ beforeEach(() => {
514
+ vi.clearAllMocks();
515
+ });
516
+ it('should reject images exceeding 10MB', async () => {
517
+ vi.mocked(existsSync).mockReturnValue(true);
518
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
519
+ transports: ['websocket'],
520
+ });
521
+ await new Promise((resolve) => {
522
+ clientSocket.on('connect', () => resolve());
523
+ });
524
+ const errorPromise = new Promise((resolve) => {
525
+ clientSocket.on('error', (error) => resolve(error));
526
+ });
527
+ // 10MB = 10 * 1024 * 1024 bytes; base64 length * 0.75 = original bytes
528
+ // So base64 length > 10MB / 0.75 ≈ 14,000,000 chars
529
+ const oversizedBase64 = 'A'.repeat(15_000_000);
530
+ clientSocket.emit('chat:send', {
531
+ content: 'Check this',
532
+ workingDirectory: '/valid/path',
533
+ images: [
534
+ { mimeType: 'image/png', data: oversizedBase64, name: 'large.png' },
535
+ ],
536
+ });
537
+ const error = await errorPromise;
538
+ expect(error.code).toBe(ERROR_CODES.VALIDATION_ERROR);
539
+ expect(error.message).toBe('ws.error.imageSizeExceeded');
540
+ });
541
+ it('should reject unsupported MIME types', async () => {
542
+ vi.mocked(existsSync).mockReturnValue(true);
543
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
544
+ transports: ['websocket'],
545
+ });
546
+ await new Promise((resolve) => {
547
+ clientSocket.on('connect', () => resolve());
548
+ });
549
+ const errorPromise = new Promise((resolve) => {
550
+ clientSocket.on('error', (error) => resolve(error));
551
+ });
552
+ clientSocket.emit('chat:send', {
553
+ content: 'Check this',
554
+ workingDirectory: '/valid/path',
555
+ images: [
556
+ { mimeType: 'image/svg+xml', data: 'PHN2Zz4=', name: 'icon.svg' },
557
+ ],
558
+ });
559
+ const error = await errorPromise;
560
+ expect(error.code).toBe(ERROR_CODES.VALIDATION_ERROR);
561
+ expect(error.message).toBe('ws.error.unsupportedImageFormat');
562
+ });
563
+ it('should reject more than 5 images', async () => {
564
+ vi.mocked(existsSync).mockReturnValue(true);
565
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
566
+ transports: ['websocket'],
567
+ });
568
+ await new Promise((resolve) => {
569
+ clientSocket.on('connect', () => resolve());
570
+ });
571
+ const errorPromise = new Promise((resolve) => {
572
+ clientSocket.on('error', (error) => resolve(error));
573
+ });
574
+ const sixImages = Array.from({ length: 6 }, (_, i) => ({
575
+ mimeType: 'image/png',
576
+ data: 'iVBORw0KGgo=',
577
+ name: `img${i}.png`,
578
+ }));
579
+ clientSocket.emit('chat:send', {
580
+ content: 'Check these',
581
+ workingDirectory: '/valid/path',
582
+ images: sixImages,
583
+ });
584
+ const error = await errorPromise;
585
+ expect(error.code).toBe(ERROR_CODES.VALIDATION_ERROR);
586
+ expect(error.message).toBe('ws.error.maxImages');
587
+ });
588
+ });
589
+ describe('Story 4.6: Error Handling', () => {
590
+ beforeEach(() => {
591
+ vi.clearAllMocks();
592
+ });
593
+ it('should pass abortController in chatOptions', async () => {
594
+ vi.mocked(existsSync).mockReturnValue(true);
595
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
596
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
597
+ transports: ['websocket'],
598
+ });
599
+ await new Promise((resolve) => {
600
+ clientSocket.on('connect', () => resolve());
601
+ });
602
+ clientSocket.emit('chat:send', {
603
+ content: 'Hello Claude',
604
+ workingDirectory: '/valid/path',
605
+ });
606
+ // Wait for the handler to process
607
+ await new Promise((resolve) => setTimeout(resolve, 100));
608
+ expect(mockState.sendImpl).toHaveBeenCalledWith('Hello Claude', expect.any(Object), expect.objectContaining({
609
+ abortController: expect.any(AbortController),
610
+ }), expect.any(Function), expect.any(Function));
611
+ });
612
+ it('should emit SESSION_NOT_FOUND error when session resume fails with session not found message', async () => {
613
+ vi.mocked(existsSync).mockReturnValue(true);
614
+ mockState.sendImpl = vi.fn().mockRejectedValue(new Error('Session not found: invalid-session-id'));
615
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
616
+ transports: ['websocket'],
617
+ });
618
+ await new Promise((resolve) => {
619
+ clientSocket.on('connect', () => resolve());
620
+ });
621
+ const errorPromise = new Promise((resolve) => {
622
+ clientSocket.on('error', (error) => resolve(error));
623
+ });
624
+ clientSocket.emit('chat:send', {
625
+ content: 'Continue our work',
626
+ workingDirectory: '/valid/path',
627
+ sessionId: 'invalid-session-id',
628
+ resume: true,
629
+ });
630
+ const error = await errorPromise;
631
+ expect(error.code).toBe(ERROR_CODES.SESSION_NOT_FOUND);
632
+ expect(error.message).toBe('ws.error.sessionNotFound');
633
+ });
634
+ it('should emit CHAT_ERROR for non-session errors during resume', async () => {
635
+ vi.mocked(existsSync).mockReturnValue(true);
636
+ mockState.sendImpl = vi.fn().mockRejectedValue(new Error('Network connection failed'));
637
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
638
+ transports: ['websocket'],
639
+ });
640
+ await new Promise((resolve) => {
641
+ clientSocket.on('connect', () => resolve());
642
+ });
643
+ const errorPromise = new Promise((resolve) => {
644
+ clientSocket.on('error', (error) => resolve(error));
645
+ });
646
+ clientSocket.emit('chat:send', {
647
+ content: 'Continue our work',
648
+ workingDirectory: '/valid/path',
649
+ sessionId: 'valid-session-id',
650
+ resume: true,
651
+ });
652
+ const error = await errorPromise;
653
+ expect(error.code).toBe(ERROR_CODES.CHAT_ERROR);
654
+ });
655
+ it('should emit CHAT_ERROR when SDK throws error', async () => {
656
+ vi.mocked(existsSync).mockReturnValue(true);
657
+ mockState.sendImpl = vi.fn().mockRejectedValue(new Error('SDK internal error'));
658
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
659
+ transports: ['websocket'],
660
+ });
661
+ await new Promise((resolve) => {
662
+ clientSocket.on('connect', () => resolve());
663
+ });
664
+ const errorPromise = new Promise((resolve) => {
665
+ clientSocket.on('error', (error) => resolve(error));
666
+ });
667
+ clientSocket.emit('chat:send', {
668
+ content: 'Hello',
669
+ workingDirectory: '/valid/path',
670
+ });
671
+ const error = await errorPromise;
672
+ expect(error.code).toBe(ERROR_CODES.CHAT_ERROR);
673
+ });
674
+ it('should emit TIMEOUT_ERROR when chat response times out', async () => {
675
+ vi.mocked(existsSync).mockReturnValue(true);
676
+ // Import AbortedError for simulation
677
+ const { AbortedError } = await import('../../utils/errors.js');
678
+ // Simulate timeout by throwing AbortedError
679
+ mockState.sendImpl = vi.fn().mockRejectedValue(new AbortedError());
680
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
681
+ transports: ['websocket'],
682
+ });
683
+ await new Promise((resolve) => {
684
+ clientSocket.on('connect', () => resolve());
685
+ });
686
+ const errorPromise = new Promise((resolve) => {
687
+ clientSocket.on('error', (error) => resolve(error));
688
+ });
689
+ clientSocket.emit('chat:send', {
690
+ content: 'Hello',
691
+ workingDirectory: '/valid/path',
692
+ });
693
+ const error = await errorPromise;
694
+ expect(error.code).toBe(ERROR_CODES.TIMEOUT_ERROR);
695
+ expect(error.message).toBe('ws.error.timeout');
696
+ });
697
+ it('should call abortController.abort() when timeout occurs', async () => {
698
+ vi.mocked(existsSync).mockReturnValue(true);
699
+ // Track if abortController was passed and its state
700
+ let capturedAbortController = null;
701
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, _callbacks, options) => {
702
+ capturedAbortController = options.abortController || null;
703
+ // Simulate long-running operation that checks abort signal
704
+ return new Promise((resolve, reject) => {
705
+ if (capturedAbortController) {
706
+ capturedAbortController.signal.addEventListener('abort', () => {
707
+ reject(new Error('Aborted'));
708
+ });
709
+ }
710
+ // Don't resolve - let timeout trigger
711
+ });
712
+ });
713
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
714
+ transports: ['websocket'],
715
+ });
716
+ await new Promise((resolve) => {
717
+ clientSocket.on('connect', () => resolve());
718
+ });
719
+ clientSocket.emit('chat:send', {
720
+ content: 'Hello',
721
+ workingDirectory: '/valid/path',
722
+ });
723
+ // Wait for handler to start processing
724
+ await new Promise((resolve) => setTimeout(resolve, 50));
725
+ // Verify abortController was passed
726
+ expect(capturedAbortController).toBeInstanceOf(AbortController);
727
+ });
728
+ it('should use CHAT_TIMEOUT_MS from config for timeout value', async () => {
729
+ // This test verifies that config.chat.timeoutMs is used
730
+ // The mock config sets it to 300000 (5 minutes)
731
+ vi.mocked(existsSync).mockReturnValue(true);
732
+ mockState.sendImpl = vi.fn().mockResolvedValue({});
733
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
734
+ transports: ['websocket'],
735
+ });
736
+ await new Promise((resolve) => {
737
+ clientSocket.on('connect', () => resolve());
738
+ });
739
+ clientSocket.emit('chat:send', {
740
+ content: 'Hello',
741
+ workingDirectory: '/valid/path',
742
+ });
743
+ // Wait for handler to process
744
+ await new Promise((resolve) => setTimeout(resolve, 100));
745
+ // Verify ChatService was called (timeout didn't trigger immediately)
746
+ // This implicitly tests that config.chat.timeoutMs (300000ms) is being used
747
+ // If timeout was 0 or very small, the call would abort immediately
748
+ expect(mockState.sendImpl).toHaveBeenCalled();
749
+ });
750
+ });
751
+ describe('Story 4.6: Session Events', () => {
752
+ beforeEach(() => {
753
+ vi.clearAllMocks();
754
+ });
755
+ it('should emit session:created for new session', async () => {
756
+ vi.mocked(existsSync).mockReturnValue(true);
757
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
758
+ // Simulate session init callback
759
+ callbacks.onSessionInit?.('new-session-123');
760
+ return {};
761
+ });
762
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
763
+ transports: ['websocket'],
764
+ });
765
+ await new Promise((resolve) => {
766
+ clientSocket.on('connect', () => resolve());
767
+ });
768
+ const sessionPromise = new Promise((resolve) => {
769
+ clientSocket.on('session:created', (data) => resolve(data));
770
+ });
771
+ clientSocket.emit('chat:send', {
772
+ content: 'Hello Claude',
773
+ workingDirectory: '/valid/path',
774
+ });
775
+ const session = await sessionPromise;
776
+ expect(session.sessionId).toBe('new-session-123');
777
+ });
778
+ it('should emit session:resumed for resumed session', async () => {
779
+ vi.mocked(existsSync).mockReturnValue(true);
780
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
781
+ // Simulate session init callback for resumed session
782
+ callbacks.onSessionInit?.('resumed-session-456');
783
+ return {};
784
+ });
785
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
786
+ transports: ['websocket'],
787
+ });
788
+ await new Promise((resolve) => {
789
+ clientSocket.on('connect', () => resolve());
790
+ });
791
+ const sessionPromise = new Promise((resolve) => {
792
+ clientSocket.on('session:resumed', (data) => resolve(data));
793
+ });
794
+ clientSocket.emit('chat:send', {
795
+ content: 'Continue our work',
796
+ workingDirectory: '/valid/path',
797
+ sessionId: 'resumed-session-456',
798
+ resume: true,
799
+ });
800
+ const session = await sessionPromise;
801
+ expect(session.sessionId).toBe('resumed-session-456');
802
+ });
803
+ });
804
+ describe('Story 4.6: Message Type Events', () => {
805
+ beforeEach(() => {
806
+ vi.clearAllMocks();
807
+ });
808
+ it('should emit message:chunk for text chunks', async () => {
809
+ vi.mocked(existsSync).mockReturnValue(true);
810
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
811
+ callbacks.onSessionInit?.('session-123');
812
+ callbacks.onTextChunk?.({
813
+ sessionId: 'session-123',
814
+ messageId: 'msg-1',
815
+ content: 'Hello',
816
+ done: false,
817
+ });
818
+ return {};
819
+ });
820
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
821
+ transports: ['websocket'],
822
+ });
823
+ await new Promise((resolve) => {
824
+ clientSocket.on('connect', () => resolve());
825
+ });
826
+ const chunkPromise = new Promise((resolve) => {
827
+ clientSocket.on('message:chunk', (data) => resolve(data));
828
+ });
829
+ clientSocket.emit('chat:send', {
830
+ content: 'Hello Claude',
831
+ workingDirectory: '/valid/path',
832
+ });
833
+ const chunk = await chunkPromise;
834
+ expect(chunk.content).toBe('Hello');
835
+ });
836
+ it('should emit tool:call for tool use', async () => {
837
+ vi.mocked(existsSync).mockReturnValue(true);
838
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
839
+ callbacks.onSessionInit?.('session-123');
840
+ callbacks.onToolUse?.({
841
+ id: 'tool-call-1',
842
+ name: 'Read',
843
+ input: { file_path: '/path/to/file.ts' },
844
+ });
845
+ return {};
846
+ });
847
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
848
+ transports: ['websocket'],
849
+ });
850
+ await new Promise((resolve) => {
851
+ clientSocket.on('connect', () => resolve());
852
+ });
853
+ const toolCallPromise = new Promise((resolve) => {
854
+ clientSocket.on('tool:call', (data) => resolve(data));
855
+ });
856
+ clientSocket.emit('chat:send', {
857
+ content: 'Read the file',
858
+ workingDirectory: '/valid/path',
859
+ });
860
+ const toolCall = await toolCallPromise;
861
+ expect(toolCall.id).toBe('tool-call-1');
862
+ expect(toolCall.name).toBe('Read');
863
+ });
864
+ it('should emit tool:result for tool results', async () => {
865
+ vi.mocked(existsSync).mockReturnValue(true);
866
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
867
+ callbacks.onSessionInit?.('session-123');
868
+ callbacks.onToolResult?.('tool-call-1', { success: true, output: 'file contents' });
869
+ return {};
870
+ });
871
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
872
+ transports: ['websocket'],
873
+ });
874
+ await new Promise((resolve) => {
875
+ clientSocket.on('connect', () => resolve());
876
+ });
877
+ const toolResultPromise = new Promise((resolve) => {
878
+ clientSocket.on('tool:result', (data) => resolve(data));
879
+ });
880
+ clientSocket.emit('chat:send', {
881
+ content: 'Read the file',
882
+ workingDirectory: '/valid/path',
883
+ });
884
+ const toolResult = await toolResultPromise;
885
+ expect(toolResult.toolCallId).toBe('tool-call-1');
886
+ expect(toolResult.result.success).toBe(true);
887
+ });
888
+ it('should emit message:complete for completed messages', async () => {
889
+ vi.mocked(existsSync).mockReturnValue(true);
890
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
891
+ callbacks.onSessionInit?.('session-123');
892
+ callbacks.onComplete?.({
893
+ id: 'response-1',
894
+ sessionId: 'session-123',
895
+ content: 'Task completed',
896
+ });
897
+ return {};
898
+ });
899
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
900
+ transports: ['websocket'],
901
+ });
902
+ await new Promise((resolve) => {
903
+ clientSocket.on('connect', () => resolve());
904
+ });
905
+ const completePromise = new Promise((resolve) => {
906
+ clientSocket.on('message:complete', (data) => resolve(data));
907
+ });
908
+ clientSocket.emit('chat:send', {
909
+ content: 'Complete the task',
910
+ workingDirectory: '/valid/path',
911
+ });
912
+ const message = await completePromise;
913
+ expect(message.id).toBe('response-1');
914
+ expect(message.role).toBe('assistant');
915
+ expect(message.content).toBe('Task completed');
916
+ });
917
+ });
918
+ describe('Story 5.6: context:usage event', () => {
919
+ beforeEach(() => {
920
+ vi.clearAllMocks();
921
+ });
922
+ it('should emit context:usage when response has usage data', async () => {
923
+ vi.mocked(existsSync).mockReturnValue(true);
924
+ const mockUsage = {
925
+ inputTokens: 150000,
926
+ outputTokens: 500,
927
+ cacheReadInputTokens: 80000,
928
+ cacheCreationInputTokens: 5000,
929
+ totalCostUSD: 0.05,
930
+ contextWindow: 200000,
931
+ };
932
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
933
+ callbacks.onSessionInit?.('session-123');
934
+ callbacks.onComplete?.({
935
+ id: 'response-1',
936
+ sessionId: 'session-123',
937
+ content: 'Done',
938
+ usage: mockUsage,
939
+ });
940
+ return {};
941
+ });
942
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
943
+ transports: ['websocket'],
944
+ });
945
+ await new Promise((resolve) => {
946
+ clientSocket.on('connect', () => resolve());
947
+ });
948
+ const usagePromise = new Promise((resolve) => {
949
+ clientSocket.on('context:usage', (data) => resolve(data));
950
+ });
951
+ clientSocket.emit('chat:send', {
952
+ content: 'Hello',
953
+ workingDirectory: '/valid/path',
954
+ });
955
+ const usage = await usagePromise;
956
+ expect(usage).toEqual(mockUsage);
957
+ });
958
+ it('should not emit context:usage when response has no usage data', async () => {
959
+ vi.mocked(existsSync).mockReturnValue(true);
960
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, callbacks) => {
961
+ callbacks.onSessionInit?.('session-123');
962
+ callbacks.onComplete?.({
963
+ id: 'response-1',
964
+ sessionId: 'session-123',
965
+ content: 'Done',
966
+ });
967
+ return {};
968
+ });
969
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
970
+ transports: ['websocket'],
971
+ });
972
+ await new Promise((resolve) => {
973
+ clientSocket.on('connect', () => resolve());
974
+ });
975
+ let usageReceived = false;
976
+ clientSocket.on('context:usage', () => {
977
+ usageReceived = true;
978
+ });
979
+ clientSocket.emit('chat:send', {
980
+ content: 'Hello',
981
+ workingDirectory: '/valid/path',
982
+ });
983
+ // Wait to ensure no context:usage event is emitted
984
+ await new Promise((resolve) => setTimeout(resolve, 200));
985
+ expect(usageReceived).toBe(false);
986
+ });
987
+ });
988
+ describe('Story 5.4: chat:abort event handler', () => {
989
+ beforeEach(() => {
990
+ vi.clearAllMocks();
991
+ });
992
+ it('should abort active request when chat:abort is received', async () => {
993
+ vi.mocked(existsSync).mockReturnValue(true);
994
+ let capturedAbortController = null;
995
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, _callbacks, options) => {
996
+ capturedAbortController = options.abortController || null;
997
+ // Simulate long-running operation
998
+ return new Promise((_resolve, reject) => {
999
+ if (capturedAbortController) {
1000
+ capturedAbortController.signal.addEventListener('abort', () => {
1001
+ reject(new Error('Aborted'));
1002
+ });
1003
+ }
1004
+ });
1005
+ });
1006
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1007
+ transports: ['websocket'],
1008
+ });
1009
+ await new Promise((resolve) => {
1010
+ clientSocket.on('connect', () => resolve());
1011
+ });
1012
+ // Start a chat request
1013
+ clientSocket.emit('chat:send', {
1014
+ content: 'Hello',
1015
+ workingDirectory: '/valid/path',
1016
+ });
1017
+ // Wait for handler to start processing
1018
+ await new Promise((resolve) => setTimeout(resolve, 50));
1019
+ // Emit abort
1020
+ clientSocket.emit('chat:abort');
1021
+ // Wait for abort to be processed
1022
+ await new Promise((resolve) => setTimeout(resolve, 100));
1023
+ // Verify the abort controller was aborted with 'user-abort' reason
1024
+ expect(capturedAbortController).toBeInstanceOf(AbortController);
1025
+ expect(capturedAbortController.signal.aborted).toBe(true);
1026
+ expect(capturedAbortController.signal.reason).toBe('user-abort');
1027
+ });
1028
+ it('should be no-op when no active request exists', async () => {
1029
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1030
+ transports: ['websocket'],
1031
+ });
1032
+ await new Promise((resolve) => {
1033
+ clientSocket.on('connect', () => resolve());
1034
+ });
1035
+ // Emit abort without any active chat:send — should not throw
1036
+ clientSocket.emit('chat:abort');
1037
+ // Wait to ensure no errors
1038
+ await new Promise((resolve) => setTimeout(resolve, 100));
1039
+ // If we reach here, no error was thrown — test passes
1040
+ expect(true).toBe(true);
1041
+ });
1042
+ it('should not emit error event on user-initiated abort', async () => {
1043
+ vi.mocked(existsSync).mockReturnValue(true);
1044
+ const { AbortedError } = await import('../../utils/errors.js');
1045
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, _callbacks, options) => {
1046
+ return new Promise((_resolve, reject) => {
1047
+ if (options.abortController) {
1048
+ options.abortController.signal.addEventListener('abort', () => {
1049
+ reject(new AbortedError());
1050
+ });
1051
+ }
1052
+ });
1053
+ });
1054
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1055
+ transports: ['websocket'],
1056
+ });
1057
+ await new Promise((resolve) => {
1058
+ clientSocket.on('connect', () => resolve());
1059
+ });
1060
+ const errors = [];
1061
+ clientSocket.on('error', (error) => errors.push(error));
1062
+ // Start a chat request
1063
+ clientSocket.emit('chat:send', {
1064
+ content: 'Hello',
1065
+ workingDirectory: '/valid/path',
1066
+ });
1067
+ // Wait for handler to start
1068
+ await new Promise((resolve) => setTimeout(resolve, 50));
1069
+ // User-initiated abort
1070
+ clientSocket.emit('chat:abort');
1071
+ // Wait for processing
1072
+ await new Promise((resolve) => setTimeout(resolve, 200));
1073
+ // Should NOT have received any error events
1074
+ expect(errors).toHaveLength(0);
1075
+ });
1076
+ it('should NOT abort active stream on disconnect (stream survives for reconnection)', async () => {
1077
+ vi.mocked(existsSync).mockReturnValue(true);
1078
+ let capturedAbortController = null;
1079
+ mockState.sendImpl = vi.fn().mockImplementation(async (_content, _callbacks, options) => {
1080
+ capturedAbortController = options.abortController || null;
1081
+ return new Promise((_resolve, reject) => {
1082
+ if (capturedAbortController) {
1083
+ capturedAbortController.signal.addEventListener('abort', () => {
1084
+ reject(new Error('Aborted'));
1085
+ });
1086
+ }
1087
+ });
1088
+ });
1089
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1090
+ transports: ['websocket'],
1091
+ });
1092
+ await new Promise((resolve) => {
1093
+ clientSocket.on('connect', () => resolve());
1094
+ });
1095
+ // Start a chat request
1096
+ clientSocket.emit('chat:send', {
1097
+ content: 'Hello',
1098
+ workingDirectory: '/valid/path',
1099
+ });
1100
+ // Wait for handler to start
1101
+ await new Promise((resolve) => setTimeout(resolve, 50));
1102
+ // Disconnect
1103
+ clientSocket.disconnect();
1104
+ // Wait for disconnect to be processed
1105
+ await new Promise((resolve) => setTimeout(resolve, 200));
1106
+ // Stream should NOT be aborted — it survives disconnects for reconnection support.
1107
+ // The activity timeout will eventually abort it if no client reconnects.
1108
+ expect(capturedAbortController).toBeInstanceOf(AbortController);
1109
+ expect(capturedAbortController.signal.aborted).toBe(false);
1110
+ });
1111
+ });
1112
+ // Story 20.1: Dashboard WebSocket events
1113
+ describe('Dashboard subscribe/unsubscribe', () => {
1114
+ it('should join dashboard room on dashboard:subscribe', async () => {
1115
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1116
+ transports: ['websocket'],
1117
+ });
1118
+ await new Promise((resolve) => {
1119
+ clientSocket.on('connect', () => resolve());
1120
+ });
1121
+ clientSocket.emit('dashboard:subscribe');
1122
+ await new Promise((resolve) => setTimeout(resolve, 50));
1123
+ // Verify by checking server-side room membership
1124
+ const rooms = ioServer.sockets.adapter.rooms.get('dashboard');
1125
+ expect(rooms).toBeDefined();
1126
+ expect(rooms.size).toBe(1);
1127
+ });
1128
+ it('should leave dashboard room on dashboard:unsubscribe', async () => {
1129
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1130
+ transports: ['websocket'],
1131
+ });
1132
+ await new Promise((resolve) => {
1133
+ clientSocket.on('connect', () => resolve());
1134
+ });
1135
+ clientSocket.emit('dashboard:subscribe');
1136
+ await new Promise((resolve) => setTimeout(resolve, 50));
1137
+ clientSocket.emit('dashboard:unsubscribe');
1138
+ await new Promise((resolve) => setTimeout(resolve, 50));
1139
+ const rooms = ioServer.sockets.adapter.rooms.get('dashboard');
1140
+ // Room should be empty or deleted
1141
+ expect(rooms?.size ?? 0).toBe(0);
1142
+ });
1143
+ it('should clean up dashboard room on disconnect', async () => {
1144
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1145
+ transports: ['websocket'],
1146
+ });
1147
+ await new Promise((resolve) => {
1148
+ clientSocket.on('connect', () => resolve());
1149
+ });
1150
+ clientSocket.emit('dashboard:subscribe');
1151
+ await new Promise((resolve) => setTimeout(resolve, 50));
1152
+ clientSocket.disconnect();
1153
+ await new Promise((resolve) => setTimeout(resolve, 100));
1154
+ const rooms = ioServer.sockets.adapter.rooms.get('dashboard');
1155
+ expect(rooms?.size ?? 0).toBe(0);
1156
+ });
1157
+ it('should emit dashboard:status-change only to dashboard room', async () => {
1158
+ // Client 1: subscribed to dashboard
1159
+ const client1 = ioc(`http://localhost:${TEST_PORT}`, {
1160
+ transports: ['websocket'],
1161
+ });
1162
+ // Client 2: NOT subscribed to dashboard
1163
+ const client2 = ioc(`http://localhost:${TEST_PORT}`, {
1164
+ transports: ['websocket'],
1165
+ });
1166
+ await Promise.all([
1167
+ new Promise((resolve) => client1.on('connect', () => resolve())),
1168
+ new Promise((resolve) => client2.on('connect', () => resolve())),
1169
+ ]);
1170
+ client1.emit('dashboard:subscribe');
1171
+ await new Promise((resolve) => setTimeout(resolve, 50));
1172
+ let client1Received = false;
1173
+ let client2Received = false;
1174
+ client1.on('dashboard:status-change', () => { client1Received = true; });
1175
+ client2.on('dashboard:status-change', () => { client2Received = true; });
1176
+ // Emit directly to dashboard room to test room isolation
1177
+ ioServer.to('dashboard').emit('dashboard:status-change', {
1178
+ projectSlug: 'test',
1179
+ status: {
1180
+ projectSlug: 'test',
1181
+ activeSessionCount: 0,
1182
+ totalSessionCount: 0,
1183
+ queueStatus: 'idle',
1184
+ terminalCount: 0,
1185
+ },
1186
+ });
1187
+ await new Promise((resolve) => setTimeout(resolve, 100));
1188
+ expect(client1Received).toBe(true);
1189
+ expect(client2Received).toBe(false);
1190
+ client1.disconnect();
1191
+ client2.disconnect();
1192
+ });
1193
+ it('should debounce rapid status changes (300ms)', async () => {
1194
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1195
+ transports: ['websocket'],
1196
+ });
1197
+ await new Promise((resolve) => {
1198
+ clientSocket.on('connect', () => resolve());
1199
+ });
1200
+ clientSocket.emit('dashboard:subscribe');
1201
+ await new Promise((resolve) => setTimeout(resolve, 50));
1202
+ let receivedCount = 0;
1203
+ clientSocket.on('dashboard:status-change', () => { receivedCount++; });
1204
+ // Import the trigger function indirectly by triggering terminal events
1205
+ // which internally call triggerDashboardStatusChange
1206
+ const { ptyService } = await import('../../services/ptyService.js');
1207
+ vi.mocked(ptyService.createSession).mockReturnValue({ terminalId: 'term-debounce-1', shell: 'bash' });
1208
+ vi.mocked(ptyService.onData).mockImplementation(() => { });
1209
+ vi.mocked(ptyService.onExit).mockImplementation(() => { });
1210
+ // Trigger 3 rapid terminal creates (each triggers dashboard status change)
1211
+ clientSocket.emit('terminal:create', { projectSlug: 'test-project' });
1212
+ clientSocket.emit('terminal:create', { projectSlug: 'test-project' });
1213
+ clientSocket.emit('terminal:create', { projectSlug: 'test-project' });
1214
+ // Wait less than debounce interval — should not have received yet
1215
+ await new Promise((resolve) => setTimeout(resolve, 100));
1216
+ expect(receivedCount).toBe(0);
1217
+ // Wait for debounce to fire (300ms + buffer)
1218
+ await new Promise((resolve) => setTimeout(resolve, 350));
1219
+ expect(receivedCount).toBe(1);
1220
+ });
1221
+ it('should debounce per-project independently', async () => {
1222
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1223
+ transports: ['websocket'],
1224
+ });
1225
+ await new Promise((resolve) => {
1226
+ clientSocket.on('connect', () => resolve());
1227
+ });
1228
+ clientSocket.emit('dashboard:subscribe');
1229
+ await new Promise((resolve) => setTimeout(resolve, 50));
1230
+ const receivedSlugs = [];
1231
+ clientSocket.on('dashboard:status-change', (data) => {
1232
+ receivedSlugs.push(data.projectSlug);
1233
+ });
1234
+ mockGetProjectStatus.mockImplementation(async (slug) => ({
1235
+ projectSlug: slug,
1236
+ activeSessionCount: 0,
1237
+ totalSessionCount: 0,
1238
+ queueStatus: 'idle',
1239
+ terminalCount: 0,
1240
+ }));
1241
+ const { ptyService } = await import('../../services/ptyService.js');
1242
+ vi.mocked(ptyService.createSession).mockReturnValue({ terminalId: 'term-iso-1', shell: 'bash' });
1243
+ vi.mocked(ptyService.onData).mockImplementation(() => { });
1244
+ vi.mocked(ptyService.onExit).mockImplementation(() => { });
1245
+ const { projectService } = await import('../../services/projectService.js');
1246
+ vi.mocked(projectService.resolveProjectPath).mockResolvedValue('/mock/path');
1247
+ // Trigger for project-A
1248
+ clientSocket.emit('terminal:create', { projectSlug: 'project-a' });
1249
+ // Trigger for project-B
1250
+ clientSocket.emit('terminal:create', { projectSlug: 'project-b' });
1251
+ // Wait for debounce to fire
1252
+ await new Promise((resolve) => setTimeout(resolve, 500));
1253
+ // Both projects should get separate broadcasts
1254
+ expect(receivedSlugs).toContain('project-a');
1255
+ expect(receivedSlugs).toContain('project-b');
1256
+ });
1257
+ it('should trigger dashboard status change on terminal:close', async () => {
1258
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1259
+ transports: ['websocket'],
1260
+ });
1261
+ await new Promise((resolve) => {
1262
+ clientSocket.on('connect', () => resolve());
1263
+ });
1264
+ clientSocket.emit('dashboard:subscribe');
1265
+ await new Promise((resolve) => setTimeout(resolve, 50));
1266
+ let received = false;
1267
+ clientSocket.on('dashboard:status-change', () => { received = true; });
1268
+ // Mock ptyService.getSession to return session data with projectSlug
1269
+ const { ptyService } = await import('../../services/ptyService.js');
1270
+ vi.mocked(ptyService.getSession).mockReturnValue({
1271
+ terminalId: 'term-close-1',
1272
+ projectSlug: 'test-project',
1273
+ shell: 'bash',
1274
+ createdAt: Date.now(),
1275
+ lastActivityAt: Date.now(),
1276
+ });
1277
+ clientSocket.emit('terminal:close', { terminalId: 'term-close-1' });
1278
+ // Wait for debounce
1279
+ await new Promise((resolve) => setTimeout(resolve, 500));
1280
+ expect(received).toBe(true);
1281
+ expect(ptyService.getSession).toHaveBeenCalledWith('term-close-1');
1282
+ });
1283
+ it('should trigger dashboard status change on terminal:exit (natural PTY exit)', async () => {
1284
+ clientSocket = ioc(`http://localhost:${TEST_PORT}`, {
1285
+ transports: ['websocket'],
1286
+ });
1287
+ await new Promise((resolve) => {
1288
+ clientSocket.on('connect', () => resolve());
1289
+ });
1290
+ clientSocket.emit('dashboard:subscribe');
1291
+ await new Promise((resolve) => setTimeout(resolve, 50));
1292
+ let received = false;
1293
+ clientSocket.on('dashboard:status-change', () => { received = true; });
1294
+ // Capture the onExit callback
1295
+ let capturedOnExit = null;
1296
+ const { ptyService } = await import('../../services/ptyService.js');
1297
+ vi.mocked(ptyService.createSession).mockReturnValue({ terminalId: 'term-exit-1', shell: 'bash' });
1298
+ vi.mocked(ptyService.onData).mockImplementation(() => { });
1299
+ vi.mocked(ptyService.onExit).mockImplementation((_id, cb) => {
1300
+ capturedOnExit = cb;
1301
+ });
1302
+ clientSocket.emit('terminal:create', { projectSlug: 'test-project' });
1303
+ await new Promise((resolve) => setTimeout(resolve, 100));
1304
+ // Reset received flag (terminal:create also triggers dashboard)
1305
+ received = false;
1306
+ mockGetProjectStatus.mockClear();
1307
+ // Simulate natural PTY exit
1308
+ expect(capturedOnExit).not.toBeNull();
1309
+ capturedOnExit(0);
1310
+ // Wait for debounce
1311
+ await new Promise((resolve) => setTimeout(resolve, 500));
1312
+ expect(received).toBe(true);
1313
+ });
1314
+ });
1315
+ });
1316
+ //# sourceMappingURL=websocket.test.js.map