specrails-desktop 2.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 (455) hide show
  1. package/.claude/commands/specrails/batch-implement.md +287 -0
  2. package/.claude/commands/specrails/compat-check.md +271 -0
  3. package/.claude/commands/specrails/doctor.md +62 -0
  4. package/.claude/commands/specrails/enrich.md +1635 -0
  5. package/.claude/commands/specrails/explore-spec.md +173 -0
  6. package/.claude/commands/specrails/health-check.md +527 -0
  7. package/.claude/commands/specrails/implement.md +1457 -0
  8. package/.claude/commands/specrails/memory-inspect.md +259 -0
  9. package/.claude/commands/specrails/opsx-diff.md +419 -0
  10. package/.claude/commands/specrails/propose-spec.md +102 -0
  11. package/.claude/commands/specrails/reconfig.md +89 -0
  12. package/.claude/commands/specrails/refactor-recommender.md +212 -0
  13. package/.claude/commands/specrails/retry.md +363 -0
  14. package/.claude/commands/specrails/telemetry.md +552 -0
  15. package/.claude/commands/specrails/why.md +96 -0
  16. package/LICENSE +21 -0
  17. package/README.md +290 -0
  18. package/cli/dist/specrails-desktop.js +1098 -0
  19. package/client/dist/assets/ActivityFeedPage-Gy4x8dBt.js +1 -0
  20. package/client/dist/assets/AgentsPage-CPgu--Fb.js +86 -0
  21. package/client/dist/assets/AnalyticsPage-B5sJEee2.js +1 -0
  22. package/client/dist/assets/BarChart-7IMQ8HY1.js +33 -0
  23. package/client/dist/assets/CodePage-CBdFvbwe.js +2 -0
  24. package/client/dist/assets/DesktopAnalyticsPage-w0rdTq4w.js +1 -0
  25. package/client/dist/assets/DocsDialog-BZUYM7wm.js +11 -0
  26. package/client/dist/assets/DocsPage-9QglWl46.js +11 -0
  27. package/client/dist/assets/ExportDropdown-BLZFXtNi.js +1 -0
  28. package/client/dist/assets/IntegrationsPage-BxBE4y99.js +3 -0
  29. package/client/dist/assets/JobDetailPage-DydWx_5S.js +16 -0
  30. package/client/dist/assets/JobsPage-20ibw0IO.js +1 -0
  31. package/client/dist/assets/abap-Bw6f2wDG.js +1 -0
  32. package/client/dist/assets/activity-BEIp_Y1A.js +1 -0
  33. package/client/dist/assets/activity-BdrPln96.js +1 -0
  34. package/client/dist/assets/activity-CpkRS8Sx.js +1 -0
  35. package/client/dist/assets/activity-DKCpESPt.js +1 -0
  36. package/client/dist/assets/activity-DOUVEjJi.js +1 -0
  37. package/client/dist/assets/activity-DRwkql_y.js +1 -0
  38. package/client/dist/assets/activity-DcDQ7tjw.js +1 -0
  39. package/client/dist/assets/activity-Dv6H7wEr.js +1 -0
  40. package/client/dist/assets/addon-image-3WCl5Vhd.js +1 -0
  41. package/client/dist/assets/addon-ligatures-C5OdliKs.js +2 -0
  42. package/client/dist/assets/addon-webgl-BbX6pSjl.js +44 -0
  43. package/client/dist/assets/addspec-B5yl4Loj.js +1 -0
  44. package/client/dist/assets/addspec-BEeF5-zc.js +1 -0
  45. package/client/dist/assets/addspec-D33ocMxf.js +1 -0
  46. package/client/dist/assets/addspec-DFswZ0jK.js +1 -0
  47. package/client/dist/assets/addspec-DRE-jZv7.js +1 -0
  48. package/client/dist/assets/addspec-DVZ15Jp8.js +1 -0
  49. package/client/dist/assets/addspec-Fkv91Opc.js +1 -0
  50. package/client/dist/assets/addspec-GWm4ffKl.js +1 -0
  51. package/client/dist/assets/agents-1nCDWRmP.js +1 -0
  52. package/client/dist/assets/agents-Bm9rPqnt.js +1 -0
  53. package/client/dist/assets/agents-CMxtJMLD.js +1 -0
  54. package/client/dist/assets/agents-DK-Dlc0i.js +1 -0
  55. package/client/dist/assets/agents-Q6Ldfpxx.js +1 -0
  56. package/client/dist/assets/agents-TeOSy-ax.js +1 -0
  57. package/client/dist/assets/agents-iTqjRajS.js +1 -0
  58. package/client/dist/assets/agents-s87sMGzL.js +1 -0
  59. package/client/dist/assets/agentstudio-B6Wb59E7.js +1 -0
  60. package/client/dist/assets/agentstudio-BADhZ41e.js +1 -0
  61. package/client/dist/assets/agentstudio-BSnWLR63.js +1 -0
  62. package/client/dist/assets/agentstudio-BdidyBzZ.js +1 -0
  63. package/client/dist/assets/agentstudio-CxlUllqI.js +1 -0
  64. package/client/dist/assets/agentstudio-D3I62TLJ.js +1 -0
  65. package/client/dist/assets/agentstudio-DuH9TogZ.js +1 -0
  66. package/client/dist/assets/agentstudio-Kw88_dUF.js +1 -0
  67. package/client/dist/assets/aiedit-BWxHGsYA.js +1 -0
  68. package/client/dist/assets/aiedit-D2ji6Qy0.js +1 -0
  69. package/client/dist/assets/aiedit-DAhZTvtk.js +1 -0
  70. package/client/dist/assets/aiedit-DJMny-D5.js +1 -0
  71. package/client/dist/assets/aiedit-DOcxERkU.js +1 -0
  72. package/client/dist/assets/aiedit-DvrcbwGv.js +1 -0
  73. package/client/dist/assets/aiedit-TTwzL1TS.js +1 -0
  74. package/client/dist/assets/aiedit-WBSjT_C1.js +1 -0
  75. package/client/dist/assets/analytics-BIdr0YfL.js +1 -0
  76. package/client/dist/assets/analytics-C6EzgtdE.js +1 -0
  77. package/client/dist/assets/analytics-C9Zc-rkM.js +1 -0
  78. package/client/dist/assets/analytics-CVx3YOc0.js +1 -0
  79. package/client/dist/assets/analytics-CYj0tfj7.js +1 -0
  80. package/client/dist/assets/analytics-CnY4kNG3.js +1 -0
  81. package/client/dist/assets/analytics-CrPCZRJ-.js +1 -0
  82. package/client/dist/assets/analytics-DMCto-TF.js +1 -0
  83. package/client/dist/assets/apex-Cw8_REBo.js +1 -0
  84. package/client/dist/assets/atom-one-dark-B-oHczHB.css +1 -0
  85. package/client/dist/assets/attachments-BIsSSnHJ.js +1 -0
  86. package/client/dist/assets/attachments-BW4L3l2L.js +1 -0
  87. package/client/dist/assets/attachments-Bcf6BG6V.js +1 -0
  88. package/client/dist/assets/attachments-Bke8sCU4.js +1 -0
  89. package/client/dist/assets/attachments-COcrGRFz.js +1 -0
  90. package/client/dist/assets/attachments-DYHGA2Dj.js +1 -0
  91. package/client/dist/assets/attachments-Dd92KpUH.js +1 -0
  92. package/client/dist/assets/attachments-DzdU6DV6.js +1 -0
  93. package/client/dist/assets/azcli-Cz6HAoOw.js +1 -0
  94. package/client/dist/assets/bat-CcJ-xyqL.js +1 -0
  95. package/client/dist/assets/bicep-z1WDCKYz.js +2 -0
  96. package/client/dist/assets/browser-5ErDlJoR.js +1 -0
  97. package/client/dist/assets/browser-Bc-YdlVg.js +1 -0
  98. package/client/dist/assets/browser-BlYF4OOq.js +1 -0
  99. package/client/dist/assets/browser-CT-ReZGt.js +1 -0
  100. package/client/dist/assets/browser-DGITz3fC.js +1 -0
  101. package/client/dist/assets/browser-JsAIGCEW.js +1 -0
  102. package/client/dist/assets/browser-M5-rbPlw.js +1 -0
  103. package/client/dist/assets/browser-Qya9cARy.js +1 -0
  104. package/client/dist/assets/cameligo-BRewOpfa.js +1 -0
  105. package/client/dist/assets/chat-BEGuC03z.js +1 -0
  106. package/client/dist/assets/chat-BEW60P_u.js +1 -0
  107. package/client/dist/assets/chat-BQNMD0PL.js +1 -0
  108. package/client/dist/assets/chat-BsbNGPW9.js +1 -0
  109. package/client/dist/assets/chat-CboQguCi.js +1 -0
  110. package/client/dist/assets/chat-DRCa9pOt.js +1 -0
  111. package/client/dist/assets/chat-DwUm6W9z.js +1 -0
  112. package/client/dist/assets/chat-yoXwguQu.js +1 -0
  113. package/client/dist/assets/chunk-CilyBKbf.js +1 -0
  114. package/client/dist/assets/clojure-DBjRWN6g.js +1 -0
  115. package/client/dist/assets/clsx-DnqN-uhr.js +1 -0
  116. package/client/dist/assets/code-AL1rVIMb.js +1 -0
  117. package/client/dist/assets/code-C0BKpkht.js +1 -0
  118. package/client/dist/assets/code-C0FTS3ew.js +1 -0
  119. package/client/dist/assets/code-CPcHxzxw.js +1 -0
  120. package/client/dist/assets/code-D3ryDniw.js +1 -0
  121. package/client/dist/assets/code-D3zVVQTj.js +1 -0
  122. package/client/dist/assets/code-PCmfS3dn.js +1 -0
  123. package/client/dist/assets/code-exI0G5Wd.js +1 -0
  124. package/client/dist/assets/codicon-ngg6Pgfi.ttf +0 -0
  125. package/client/dist/assets/coffee-Cfk_XHGR.js +1 -0
  126. package/client/dist/assets/commands-B772IyDa.js +1 -0
  127. package/client/dist/assets/commands-BDDp6xFG.js +1 -0
  128. package/client/dist/assets/commands-CJxCry-o.js +1 -0
  129. package/client/dist/assets/commands-CfgY-_of.js +1 -0
  130. package/client/dist/assets/commands-DLrvnPNg.js +1 -0
  131. package/client/dist/assets/commands-IXMOKBYt.js +1 -0
  132. package/client/dist/assets/commands-UD1NzmwX.js +1 -0
  133. package/client/dist/assets/commands-sqrqsxyE.js +1 -0
  134. package/client/dist/assets/common-DCr6VzJ7.js +1 -0
  135. package/client/dist/assets/common-Dard9UNH.js +1 -0
  136. package/client/dist/assets/common-DeDELLZJ.js +1 -0
  137. package/client/dist/assets/common-DltqHaAe.js +1 -0
  138. package/client/dist/assets/common-Dmm1GhdD.js +1 -0
  139. package/client/dist/assets/common-DnjcgkPH.js +1 -0
  140. package/client/dist/assets/common-GbpxfPG8.js +1 -0
  141. package/client/dist/assets/common-wA36jmj1.js +1 -0
  142. package/client/dist/assets/cpp-BVob6BaP.js +1 -0
  143. package/client/dist/assets/csharp-C4fbRuOu.js +1 -0
  144. package/client/dist/assets/csp-DthFP_vT.js +1 -0
  145. package/client/dist/assets/css-CGMH0hcW.js +3 -0
  146. package/client/dist/assets/css.worker-Wv5dxAWO.js +89 -0
  147. package/client/dist/assets/cssMode-Cc6ozl-J.js +1 -0
  148. package/client/dist/assets/cypher-Pnf68BRV.js +1 -0
  149. package/client/dist/assets/dart-PMMOtxZX.js +1 -0
  150. package/client/dist/assets/dashboard-B4ixDVk8.js +1 -0
  151. package/client/dist/assets/dashboard-BZBADHSj.js +1 -0
  152. package/client/dist/assets/dashboard-C1MfeUHs.js +1 -0
  153. package/client/dist/assets/dashboard-C7SK6xu5.js +1 -0
  154. package/client/dist/assets/dashboard-CB6Le1yN.js +1 -0
  155. package/client/dist/assets/dashboard-CoTpMOBM.js +1 -0
  156. package/client/dist/assets/dashboard-Duo4DDCW.js +1 -0
  157. package/client/dist/assets/dashboard-I19DXBxw.js +1 -0
  158. package/client/dist/assets/dist-js-BY-Fv_fg.js +1 -0
  159. package/client/dist/assets/dist-js-Bakc4uxT.js +1 -0
  160. package/client/dist/assets/dockerfile-di1nsJCc.js +1 -0
  161. package/client/dist/assets/ecl-D_WVcB5M.js +1 -0
  162. package/client/dist/assets/editor-Br_kD0ds.css +1 -0
  163. package/client/dist/assets/editor.api2-XLGzZfbc.js +872 -0
  164. package/client/dist/assets/editor.main-CfXxHimg.js +6 -0
  165. package/client/dist/assets/editor.worker-Bd9IXS8d.js +26 -0
  166. package/client/dist/assets/elixir-OAdJEMOn.js +1 -0
  167. package/client/dist/assets/explore-4mFpnrKU.js +1 -0
  168. package/client/dist/assets/explore-A8Ltoblq.js +1 -0
  169. package/client/dist/assets/explore-B9A3iN2W.js +1 -0
  170. package/client/dist/assets/explore-BV5Xxlsn.js +1 -0
  171. package/client/dist/assets/explore-BrBJvfjP.js +1 -0
  172. package/client/dist/assets/explore-C3FSE42C.js +1 -0
  173. package/client/dist/assets/explore-D2EFgt8J.js +1 -0
  174. package/client/dist/assets/explore-hFc3HFcp.js +1 -0
  175. package/client/dist/assets/flow9-D3QEZjgn.js +1 -0
  176. package/client/dist/assets/format-command-CwGuwzGA.js +1 -0
  177. package/client/dist/assets/freemarker2-DP7J1gG3.js +3 -0
  178. package/client/dist/assets/fsharp-BF0k_8N8.js +1 -0
  179. package/client/dist/assets/go-BAQO5Jsz.js +1 -0
  180. package/client/dist/assets/graphql-hdFVFkiV.js +1 -0
  181. package/client/dist/assets/handlebars-BjRlucw6.js +1 -0
  182. package/client/dist/assets/hcl-DWnl1o-X.js +1 -0
  183. package/client/dist/assets/html-OumBQJ-U.js +1 -0
  184. package/client/dist/assets/html.worker-CQP8QQsS.js +502 -0
  185. package/client/dist/assets/htmlMode-CStc3zXM.js +1 -0
  186. package/client/dist/assets/index-CimDRRi7.css +2 -0
  187. package/client/dist/assets/index-XGZaKl_u.js +142 -0
  188. package/client/dist/assets/ini-CB-6OVu3.js +1 -0
  189. package/client/dist/assets/integrations-C3p12Ms6.js +1 -0
  190. package/client/dist/assets/integrations-Cr6hH7XR.js +1 -0
  191. package/client/dist/assets/integrations-Cublz3m6.js +1 -0
  192. package/client/dist/assets/integrations-D28q1kF6.js +1 -0
  193. package/client/dist/assets/integrations-DRdbki5W.js +1 -0
  194. package/client/dist/assets/integrations-DaC4SzzL.js +1 -0
  195. package/client/dist/assets/integrations-DmQYCUvN.js +1 -0
  196. package/client/dist/assets/integrations-HIlUxXVs.js +1 -0
  197. package/client/dist/assets/java-d1CmfiHX.js +1 -0
  198. package/client/dist/assets/javascript-CMk--e7g.js +1 -0
  199. package/client/dist/assets/jobs-BE1siB0M.js +1 -0
  200. package/client/dist/assets/jobs-BHcQ_Faf.js +1 -0
  201. package/client/dist/assets/jobs-CFfc2dNX.js +1 -0
  202. package/client/dist/assets/jobs-CSi5n8X_.js +1 -0
  203. package/client/dist/assets/jobs-Dc3X86PY.js +1 -0
  204. package/client/dist/assets/jobs-De5tASex.js +1 -0
  205. package/client/dist/assets/jobs-DsoXEdo7.js +1 -0
  206. package/client/dist/assets/jobs-Wl-ApPMb.js +1 -0
  207. package/client/dist/assets/json.worker-DzV-CpCQ.js +58 -0
  208. package/client/dist/assets/jsonMode-C2h3ZcjZ.js +7 -0
  209. package/client/dist/assets/julia-Bgv08lKa.js +1 -0
  210. package/client/dist/assets/kotlin-u98kaVTf.js +1 -0
  211. package/client/dist/assets/less-CjYwpgg5.js +2 -0
  212. package/client/dist/assets/lexon-YTjaAFBB.js +1 -0
  213. package/client/dist/assets/lib-CPxTMOAq.js +7 -0
  214. package/client/dist/assets/liquid-mI3KJrBE.js +1 -0
  215. package/client/dist/assets/lspLanguageFeatures-DU09ggWi.js +4 -0
  216. package/client/dist/assets/lua-BzmkWv27.js +1 -0
  217. package/client/dist/assets/m3-CFwk9fw0.js +1 -0
  218. package/client/dist/assets/markdown-CR5iMpSZ.js +1 -0
  219. package/client/dist/assets/mdx-C41VDTR_.js +1 -0
  220. package/client/dist/assets/mips-CcEalc17.js +1 -0
  221. package/client/dist/assets/monaco.contribution-CPObAXMC.js +2 -0
  222. package/client/dist/assets/msdax-BQbkawnr.js +1 -0
  223. package/client/dist/assets/mysql-GTlaaW_P.js +1 -0
  224. package/client/dist/assets/nav-0fwkrgHt.js +1 -0
  225. package/client/dist/assets/nav-BEL3MTwK.js +1 -0
  226. package/client/dist/assets/nav-B_G-TJDW.js +1 -0
  227. package/client/dist/assets/nav-C2YXcbZS.js +1 -0
  228. package/client/dist/assets/nav-ClzOE4mA.js +1 -0
  229. package/client/dist/assets/nav-CtYwmMgu.js +1 -0
  230. package/client/dist/assets/nav-D2bOGSEg.js +1 -0
  231. package/client/dist/assets/nav-iH1V5j6o.js +1 -0
  232. package/client/dist/assets/objective-c-Byu1T5if.js +1 -0
  233. package/client/dist/assets/pascal-BrfzBfRm.js +1 -0
  234. package/client/dist/assets/pascaligo-BXXKFUeo.js +1 -0
  235. package/client/dist/assets/perl-B3OikKq-.js +1 -0
  236. package/client/dist/assets/pgsql-CTsa0Acc.js +1 -0
  237. package/client/dist/assets/php-DiQh3FUW.js +1 -0
  238. package/client/dist/assets/pla-92uH8Fzm.js +1 -0
  239. package/client/dist/assets/postiats-BbeWkKUr.js +1 -0
  240. package/client/dist/assets/powerquery-DgDMzpsm.js +1 -0
  241. package/client/dist/assets/powershell-BfdUUzaG.js +1 -0
  242. package/client/dist/assets/preload-helper-DSXbuxSR.js +1 -0
  243. package/client/dist/assets/protobuf-BojW2ftW.js +2 -0
  244. package/client/dist/assets/pug-BxqTg3IU.js +1 -0
  245. package/client/dist/assets/python-Y27rKQtk.js +1 -0
  246. package/client/dist/assets/qsharp-BX_A-MW9.js +1 -0
  247. package/client/dist/assets/r-D9BMnxvJ.js +1 -0
  248. package/client/dist/assets/razor-Cd5-q9Bp.js +1 -0
  249. package/client/dist/assets/redis-5cJqEQJJ.js +1 -0
  250. package/client/dist/assets/redshift-d8BBqiwb.js +1 -0
  251. package/client/dist/assets/restructuredtext-C8a6yIcZ.js +1 -0
  252. package/client/dist/assets/ruby-egeh-6KX.js +1 -0
  253. package/client/dist/assets/rust-a3r9IInB.js +1 -0
  254. package/client/dist/assets/sb-y8iRIDei.js +1 -0
  255. package/client/dist/assets/scala-BPDK2AmK.js +1 -0
  256. package/client/dist/assets/scheme-BIWUEoOs.js +1 -0
  257. package/client/dist/assets/scss-CA-PSzwg.js +3 -0
  258. package/client/dist/assets/settings-55oDcbSh.js +1 -0
  259. package/client/dist/assets/settings-Bd4Tq1RB.js +1 -0
  260. package/client/dist/assets/settings-CCSM-Fhn.js +1 -0
  261. package/client/dist/assets/settings-D3e_bDoW.js +1 -0
  262. package/client/dist/assets/settings-DKbTkbn7.js +1 -0
  263. package/client/dist/assets/settings-Dxpo6_w7.js +1 -0
  264. package/client/dist/assets/settings-bt84e3Aa.js +1 -0
  265. package/client/dist/assets/settings-nu68QukM.js +1 -0
  266. package/client/dist/assets/setup-BMqwfbW9.js +1 -0
  267. package/client/dist/assets/setup-Bb5LcG28.js +1 -0
  268. package/client/dist/assets/setup-BeEx2_da.js +1 -0
  269. package/client/dist/assets/setup-CCCrB53Q.js +1 -0
  270. package/client/dist/assets/setup-CJA0ATmd.js +1 -0
  271. package/client/dist/assets/setup-CeiDbZcb.js +1 -0
  272. package/client/dist/assets/setup-Cus7TApA.js +1 -0
  273. package/client/dist/assets/setup-D9qOs2Xo.js +1 -0
  274. package/client/dist/assets/shell--LiT1Bja.js +1 -0
  275. package/client/dist/assets/solidity-DdqZccZg.js +1 -0
  276. package/client/dist/assets/sophia-S6-YxNG_.js +1 -0
  277. package/client/dist/assets/sparql-BSf5kMp2.js +1 -0
  278. package/client/dist/assets/specs-BFfu3u-a.js +1 -0
  279. package/client/dist/assets/specs-B__C8-8a.js +1 -0
  280. package/client/dist/assets/specs-CZ1PsXsC.js +1 -0
  281. package/client/dist/assets/specs-D2FzlLn9.js +1 -0
  282. package/client/dist/assets/specs-DaUTrNF9.js +1 -0
  283. package/client/dist/assets/specs-Dyc5hYeE.js +1 -0
  284. package/client/dist/assets/specs-cKEh2LXt.js +1 -0
  285. package/client/dist/assets/specs-k0PyLDVt.js +1 -0
  286. package/client/dist/assets/sql-D7KgjR8G.js +1 -0
  287. package/client/dist/assets/st-BnoDa-Ml.js +1 -0
  288. package/client/dist/assets/swift-DEUHTkUX.js +1 -0
  289. package/client/dist/assets/systemverilog-Tqb_KPnW.js +1 -0
  290. package/client/dist/assets/tcl-BmBFS2qq.js +1 -0
  291. package/client/dist/assets/terminal-80yDMgMF.js +1 -0
  292. package/client/dist/assets/terminal-Bje4ziIa.js +1 -0
  293. package/client/dist/assets/terminal-C2WYcFHF.js +1 -0
  294. package/client/dist/assets/terminal-CSONJOex.js +1 -0
  295. package/client/dist/assets/terminal-DEqzGtcr.js +1 -0
  296. package/client/dist/assets/terminal-DeWzh6ys.js +1 -0
  297. package/client/dist/assets/terminal-YOlsJCQj.js +1 -0
  298. package/client/dist/assets/terminal-lkZYR4wJ.js +1 -0
  299. package/client/dist/assets/tickets-CB7N30gm.js +1 -0
  300. package/client/dist/assets/tickets-CF2PYelu.js +1 -0
  301. package/client/dist/assets/tickets-DNOANUXr.js +1 -0
  302. package/client/dist/assets/tickets-DU1aqsbr.js +1 -0
  303. package/client/dist/assets/tickets-DYvafSaY.js +1 -0
  304. package/client/dist/assets/tickets-DlpC_iTg.js +1 -0
  305. package/client/dist/assets/tickets-DucYgtdl.js +1 -0
  306. package/client/dist/assets/tickets-clefmXLv.js +1 -0
  307. package/client/dist/assets/ts.worker-METxwbDZ.js +67719 -0
  308. package/client/dist/assets/tsMode-B0y_xEci.js +11 -0
  309. package/client/dist/assets/twig-BQV8igWC.js +1 -0
  310. package/client/dist/assets/typescript-BzK0OgwW.js +1 -0
  311. package/client/dist/assets/typespec-DlFroUGY.js +1 -0
  312. package/client/dist/assets/useProjectCache-DSaiGFjV.js +1 -0
  313. package/client/dist/assets/vb-BlrJpIMX.js +1 -0
  314. package/client/dist/assets/wgsl-BWgIc6FZ.js +298 -0
  315. package/client/dist/assets/workers-rt--R2Qy.js +1 -0
  316. package/client/dist/assets/xml-eX9QXAmI.js +1 -0
  317. package/client/dist/assets/yaml-fcsNkpOt.js +1 -0
  318. package/client/dist/index.html +246 -0
  319. package/docs/README.md +54 -0
  320. package/docs/cli.md +198 -0
  321. package/docs/codex.md +210 -0
  322. package/docs/creating-specs.md +197 -0
  323. package/docs/customizing.md +197 -0
  324. package/docs/getting-started.md +140 -0
  325. package/docs/internals/README.md +25 -0
  326. package/docs/internals/adding-a-provider.md +238 -0
  327. package/docs/internals/api-reference.md +634 -0
  328. package/docs/internals/architecture.md +332 -0
  329. package/docs/internals/configuration.md +172 -0
  330. package/docs/internals/openspec-workflow.md +282 -0
  331. package/docs/internals/operations-runbook.md +198 -0
  332. package/docs/internals/profiles.md +152 -0
  333. package/docs/platforms/macos.md +130 -0
  334. package/docs/platforms/windows.md +81 -0
  335. package/docs/running-pipelines.md +240 -0
  336. package/docs/terminal.md +138 -0
  337. package/docs/tracking-cost.md +155 -0
  338. package/package.json +82 -0
  339. package/server/dist/agent-generator.js +232 -0
  340. package/server/dist/agent-refine-db.js +124 -0
  341. package/server/dist/agent-refine-manager.js +526 -0
  342. package/server/dist/ai-invocations.js +111 -0
  343. package/server/dist/attachment-manager.js +299 -0
  344. package/server/dist/auth.js +207 -0
  345. package/server/dist/binary-probe.js +35 -0
  346. package/server/dist/browser-capture-manager.js +576 -0
  347. package/server/dist/browser-capture-types.js +28 -0
  348. package/server/dist/browser-network.js +149 -0
  349. package/server/dist/browser-playwright.js +888 -0
  350. package/server/dist/build-dirs.js +44 -0
  351. package/server/dist/changes-reader.js +120 -0
  352. package/server/dist/chat-manager.js +1060 -0
  353. package/server/dist/chromium-resolver.js +311 -0
  354. package/server/dist/code-explorer-router.js +788 -0
  355. package/server/dist/codex-otel-bridge.js +235 -0
  356. package/server/dist/command-resolver.js +102 -0
  357. package/server/dist/config.js +306 -0
  358. package/server/dist/context-budget.js +113 -0
  359. package/server/dist/context-scope.js +279 -0
  360. package/server/dist/contract-refine-runner.js +521 -0
  361. package/server/dist/core-compat.js +207 -0
  362. package/server/dist/core-package.js +14 -0
  363. package/server/dist/db.js +1034 -0
  364. package/server/dist/desktop-analytics.js +156 -0
  365. package/server/dist/desktop-db.js +456 -0
  366. package/server/dist/desktop-router.js +735 -0
  367. package/server/dist/docs-router.js +207 -0
  368. package/server/dist/explore-contract-refine.js +421 -0
  369. package/server/dist/explore-cwd-manager.js +242 -0
  370. package/server/dist/explore-draft-title.js +47 -0
  371. package/server/dist/explore-smash.js +450 -0
  372. package/server/dist/feature-flags.js +17 -0
  373. package/server/dist/file-provenance.js +382 -0
  374. package/server/dist/file-summary-generator.js +221 -0
  375. package/server/dist/file-summary-manager.js +689 -0
  376. package/server/dist/hooks.js +102 -0
  377. package/server/dist/ids.js +7 -0
  378. package/server/dist/index.js +586 -0
  379. package/server/dist/metrics.js +136 -0
  380. package/server/dist/mobile/index.js +16 -0
  381. package/server/dist/mobile/mobile-admin-router.js +84 -0
  382. package/server/dist/mobile/mobile-auth.js +67 -0
  383. package/server/dist/mobile/mobile-devices.js +80 -0
  384. package/server/dist/mobile/mobile-event-bus.js +39 -0
  385. package/server/dist/mobile/mobile-gateway.js +285 -0
  386. package/server/dist/mobile/mobile-mdns.js +81 -0
  387. package/server/dist/mobile/mobile-pairing.js +179 -0
  388. package/server/dist/mobile/mobile-redact.js +53 -0
  389. package/server/dist/mobile/mobile-router.js +411 -0
  390. package/server/dist/mobile/mobile-tls.js +86 -0
  391. package/server/dist/mobile/mobile-types.js +9 -0
  392. package/server/dist/mobile/mobile-ws.js +275 -0
  393. package/server/dist/path-resolver.js +298 -0
  394. package/server/dist/plugin-manager.js +617 -0
  395. package/server/dist/plugins/claude-approval.js +179 -0
  396. package/server/dist/plugins/claude-md-mutation.js +146 -0
  397. package/server/dist/plugins/codex-mcp.js +108 -0
  398. package/server/dist/plugins/contributors.js +72 -0
  399. package/server/dist/plugins/drift.js +58 -0
  400. package/server/dist/plugins/index.js +14 -0
  401. package/server/dist/plugins/json-mutation.js +120 -0
  402. package/server/dist/plugins/manager.js +32 -0
  403. package/server/dist/plugins/ownership.js +86 -0
  404. package/server/dist/plugins/paths.js +37 -0
  405. package/server/dist/plugins/prereq-installer.js +104 -0
  406. package/server/dist/plugins/rail-integration.js +79 -0
  407. package/server/dist/plugins/serena/index.js +13 -0
  408. package/server/dist/plugins/serena/install.js +91 -0
  409. package/server/dist/plugins/serena/instructions-content.js +21 -0
  410. package/server/dist/plugins/serena/manifest.js +111 -0
  411. package/server/dist/plugins/serena/verify.js +78 -0
  412. package/server/dist/plugins-router.js +215 -0
  413. package/server/dist/pricing.js +89 -0
  414. package/server/dist/profile-manager.js +310 -0
  415. package/server/dist/profiles-router.js +759 -0
  416. package/server/dist/project-registry.js +443 -0
  417. package/server/dist/project-router.js +4016 -0
  418. package/server/dist/proposal-manager.js +291 -0
  419. package/server/dist/provider-selection.js +69 -0
  420. package/server/dist/providers/claude-adapter.js +281 -0
  421. package/server/dist/providers/codex-adapter.js +264 -0
  422. package/server/dist/providers/index.js +23 -0
  423. package/server/dist/providers/registry.js +37 -0
  424. package/server/dist/providers/types.js +22 -0
  425. package/server/dist/queue-manager.js +1511 -0
  426. package/server/dist/rails-router.js +362 -0
  427. package/server/dist/rails-store.js +116 -0
  428. package/server/dist/result-event.js +106 -0
  429. package/server/dist/schemas/profile.v1.json +151 -0
  430. package/server/dist/setup-manager.js +1165 -0
  431. package/server/dist/setup-prerequisites.js +372 -0
  432. package/server/dist/smash-runner.js +663 -0
  433. package/server/dist/spec-draft-parser.js +133 -0
  434. package/server/dist/spec-launcher-manager.js +174 -0
  435. package/server/dist/spec-models.js +32 -0
  436. package/server/dist/specrails-tech-client.js +82 -0
  437. package/server/dist/spending.js +448 -0
  438. package/server/dist/telemetry-compactor.js +180 -0
  439. package/server/dist/telemetry-export.js +317 -0
  440. package/server/dist/telemetry-receiver.js +224 -0
  441. package/server/dist/terminal-manager.js +633 -0
  442. package/server/dist/terminal-marks-store.js +117 -0
  443. package/server/dist/terminal-osc-parser.js +159 -0
  444. package/server/dist/terminal-settings.js +282 -0
  445. package/server/dist/terminal-shell-integration.js +196 -0
  446. package/server/dist/ticket-broadcast.js +47 -0
  447. package/server/dist/ticket-store.js +397 -0
  448. package/server/dist/ticket-watcher.js +117 -0
  449. package/server/dist/types.js +10 -0
  450. package/server/dist/user-mcp-config.js +117 -0
  451. package/server/dist/util/cli-prompt.js +181 -0
  452. package/server/dist/util/secure-fs.js +50 -0
  453. package/server/dist/util/win-spawn.js +43 -0
  454. package/server/dist/webhook-manager.js +89 -0
  455. package/server/dist/ws-routing.js +47 -0
@@ -0,0 +1,1098 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ /**
4
+ * specrails-desktop — specrails CLI bridge
5
+ *
6
+ * Routes commands to the manager when running, or falls back to invoking
7
+ * claude directly when the manager is not reachable.
8
+ *
9
+ * Usage:
10
+ * specrails-desktop implement #42 → /specrails:implement #42 (via manager or direct)
11
+ * specrails-desktop "any raw prompt" → raw prompt (no /specrails: prefix)
12
+ * specrails-desktop --status → print manager state
13
+ * specrails-desktop --jobs → print job history table
14
+ * specrails-desktop --port 5000 <command> → use port 5000 instead of 4200
15
+ * specrails-desktop --help → print usage and exit 0
16
+ */
17
+ var __importDefault = (this && this.__importDefault) || function (mod) {
18
+ return (mod && mod.__esModule) ? mod : { "default": mod };
19
+ };
20
+ Object.defineProperty(exports, "__esModule", { value: true });
21
+ exports._internal = exports.KNOWN_VERBS = void 0;
22
+ exports.parseArgs = parseArgs;
23
+ exports.getVersion = getVersion;
24
+ exports.detectWebManager = detectWebManager;
25
+ exports.formatDuration = formatDuration;
26
+ exports.formatTokens = formatTokens;
27
+ exports.printSummary = printSummary;
28
+ const http_1 = __importDefault(require("http"));
29
+ const net_1 = __importDefault(require("net"));
30
+ const child_process_1 = require("child_process");
31
+ const child_process_2 = require("child_process");
32
+ const readline_1 = require("readline");
33
+ const ws_1 = __importDefault(require("ws"));
34
+ const path_1 = __importDefault(require("path"));
35
+ const os_1 = __importDefault(require("os"));
36
+ const fs_1 = __importDefault(require("fs"));
37
+ // ---------------------------------------------------------------------------
38
+ // Constants
39
+ // ---------------------------------------------------------------------------
40
+ const DEFAULT_PORT = 4200;
41
+ const DETECTION_TIMEOUT_MS = 500;
42
+ exports.KNOWN_VERBS = new Set([
43
+ 'implement',
44
+ 'batch-implement',
45
+ 'why',
46
+ 'get-backlog-specs',
47
+ 'auto-propose-backlog-specs',
48
+ 'propose-spec',
49
+ 'refactor-recommender',
50
+ 'health-check',
51
+ 'compat-check',
52
+ 'enrich',
53
+ ]);
54
+ const EXIT_PATTERN = /\[process exited with code (\d+)/;
55
+ // ---------------------------------------------------------------------------
56
+ // ANSI helpers
57
+ // ---------------------------------------------------------------------------
58
+ const isTTY = process.stdout.isTTY === true;
59
+ function ansi(code, text) {
60
+ if (!isTTY)
61
+ return text;
62
+ return `\x1b[${code}m${text}\x1b[0m`;
63
+ }
64
+ const dim = (t) => ansi('2', t);
65
+ const red = (t) => ansi('31', t);
66
+ const bold = (t) => ansi('1', t);
67
+ const dimCyan = (t) => ansi('2;36', t);
68
+ function cliPrefix() {
69
+ return dim('[specrails-desktop]');
70
+ }
71
+ function cliLog(msg) {
72
+ process.stdout.write(`${cliPrefix()} ${msg}\n`);
73
+ }
74
+ function cliError(msg) {
75
+ process.stderr.write(`${cliPrefix()} ${red(`error: ${msg}`)}\n`);
76
+ }
77
+ function cliWarn(msg) {
78
+ process.stderr.write(`${cliPrefix()} ${dim(`warning: ${msg}`)}\n`);
79
+ }
80
+ function parseArgs(argv) {
81
+ // argv is process.argv.slice(2)
82
+ let port = DEFAULT_PORT;
83
+ let projectOverride;
84
+ const args = [...argv];
85
+ // Extract --port <n> and --project <name|path> from any position
86
+ for (let i = 0; i < args.length; i++) {
87
+ if (args[i] === '--port' && i + 1 < args.length) {
88
+ const parsed = parseInt(args[i + 1], 10);
89
+ if (!isNaN(parsed)) {
90
+ port = parsed;
91
+ }
92
+ args.splice(i, 2);
93
+ i--;
94
+ }
95
+ else if (args[i] === '--project' && i + 1 < args.length) {
96
+ projectOverride = args[i + 1];
97
+ args.splice(i, 2);
98
+ i--;
99
+ }
100
+ }
101
+ if (args[0] === '--version' || args[0] === '-v') {
102
+ return { mode: 'version' };
103
+ }
104
+ if (args.length === 0 || args[0] === '--help' || args[0] === '-h') {
105
+ return { mode: 'help' };
106
+ }
107
+ if (args[0] === '--status') {
108
+ return { mode: 'status', port };
109
+ }
110
+ if (args[0] === '--jobs') {
111
+ return { mode: 'jobs', port };
112
+ }
113
+ if (args[0] === 'desktop') {
114
+ return { mode: 'desktop', subArgs: args.slice(1), port };
115
+ }
116
+ // Allow server-management subcommands directly without the 'desktop' prefix:
117
+ // specrails-desktop start → specrails-desktop desktop start
118
+ // specrails-desktop stop → specrails-desktop desktop stop
119
+ // specrails-desktop add → specrails-desktop desktop add
120
+ // etc.
121
+ const DESKTOP_SUBCOMMANDS = new Set(['start', 'stop', 'status', 'add', 'remove', 'list']);
122
+ if (DESKTOP_SUBCOMMANDS.has(args[0])) {
123
+ return { mode: 'desktop', subArgs: args, port };
124
+ }
125
+ const first = args[0];
126
+ // Slash-prefixed command: pass through unchanged
127
+ if (first.startsWith('/')) {
128
+ const resolved = args.join(' ');
129
+ return { mode: 'raw', resolved, port, projectOverride };
130
+ }
131
+ // Known verb: inject /specrails: prefix
132
+ if (exports.KNOWN_VERBS.has(first)) {
133
+ const resolved = `/specrails:${args.join(' ')}`;
134
+ return { mode: 'command', resolved, port, projectOverride };
135
+ }
136
+ // Unknown first token: treat as raw prompt
137
+ const resolved = args.join(' ');
138
+ return { mode: 'raw', resolved, port, projectOverride };
139
+ }
140
+ function getVersion() {
141
+ for (const rel of ['../package.json', '../../package.json']) {
142
+ try {
143
+ const pkgPath = path_1.default.join(__dirname, rel);
144
+ const pkg = JSON.parse(fs_1.default.readFileSync(pkgPath, 'utf8'));
145
+ if (typeof pkg.version === 'string')
146
+ return pkg.version;
147
+ }
148
+ catch {
149
+ // try next
150
+ }
151
+ }
152
+ return 'unknown';
153
+ }
154
+ function printVersion() {
155
+ process.stdout.write(`specrails-desktop v${getVersion()}\n`);
156
+ }
157
+ function printHelp() {
158
+ const version = getVersion();
159
+ process.stdout.write(`
160
+ ${bold(`specrails-desktop v${version}`)} — specrails CLI bridge
161
+
162
+ ${bold('Project Required:')}
163
+ Every command runs in the context of a project registered for the current
164
+ directory. Register your project once before running any commands:
165
+
166
+ ${dim('# Register your project (run once per project):')}
167
+ specrails-desktop add .
168
+
169
+ ${dim('# Then run commands from that directory:')}
170
+ specrails-desktop implement #42
171
+
172
+ ${bold('Usage:')}
173
+ specrails-desktop implement #42 Run a known specrails verb (prepends /specrails:)
174
+ specrails-desktop batch-implement #40 #41 Batch implementation across issues
175
+ specrails-desktop why Explain recent changes
176
+ specrails-desktop get-backlog-specs View prioritized spec backlog
177
+ specrails-desktop auto-propose-backlog-specs Generate new spec ideas
178
+ specrails-desktop propose-spec Explore an idea and produce a spec
179
+ specrails-desktop refactor-recommender Find refactoring opportunities
180
+ specrails-desktop health-check Run codebase health check
181
+ specrails-desktop compat-check Check for breaking API changes
182
+ specrails-desktop "any raw prompt" Pass a raw prompt directly to claude
183
+ specrails-desktop --status Print manager status and exit
184
+ specrails-desktop --jobs Print recent job history and exit
185
+ specrails-desktop start|stop|add|remove|list Manage the server
186
+ specrails-desktop --project <name|path> Override project (default: current directory)
187
+ specrails-desktop --port <n> Override default port (${DEFAULT_PORT})
188
+ specrails-desktop --version, -v Print version and exit
189
+ specrails-desktop --help, -h Show this help text
190
+
191
+ ${bold('Execution paths:')}
192
+ Manager running → POST /api/spawn + stream logs via WebSocket
193
+ Manager not running → spawn claude directly with stream-json output
194
+ `.trimStart());
195
+ }
196
+ function detectWebManager(port) {
197
+ const baseUrl = `http://127.0.0.1:${port}`;
198
+ return new Promise((resolve) => {
199
+ const timer = setTimeout(() => {
200
+ req.destroy();
201
+ resolve({ running: false, baseUrl });
202
+ }, DETECTION_TIMEOUT_MS);
203
+ const req = http_1.default.get(`${baseUrl}/api/health`, { timeout: DETECTION_TIMEOUT_MS }, (res) => {
204
+ clearTimeout(timer);
205
+ res.resume(); // drain the response
206
+ if (res.statusCode !== undefined && res.statusCode >= 200 && res.statusCode < 300) {
207
+ resolve({ running: true, baseUrl });
208
+ }
209
+ else {
210
+ resolve({ running: false, baseUrl });
211
+ }
212
+ });
213
+ req.on('error', () => {
214
+ clearTimeout(timer);
215
+ resolve({ running: false, baseUrl });
216
+ });
217
+ req.on('timeout', () => {
218
+ req.destroy();
219
+ clearTimeout(timer);
220
+ resolve({ running: false, baseUrl });
221
+ });
222
+ });
223
+ }
224
+ // ---------------------------------------------------------------------------
225
+ // HTTP helpers
226
+ // ---------------------------------------------------------------------------
227
+ function loadDesktopToken() {
228
+ const candidates = [
229
+ path_1.default.join(os_1.default.homedir(), '.specrails', 'desktop.token'),
230
+ // Legacy fallback: pre-rename installs keep the token at hub.token until the
231
+ // server migrates it on next startup. Compat only — do not extend.
232
+ path_1.default.join(os_1.default.homedir(), '.specrails', 'hub.token'),
233
+ ];
234
+ for (const tokenPath of candidates) {
235
+ try {
236
+ const t = fs_1.default.readFileSync(tokenPath, 'utf-8').trim();
237
+ if (t.length >= 32)
238
+ return t;
239
+ }
240
+ catch {
241
+ // try next candidate
242
+ }
243
+ }
244
+ return null;
245
+ }
246
+ function httpGet(url) {
247
+ return new Promise((resolve, reject) => {
248
+ const token = loadDesktopToken();
249
+ const parsed = new URL(url);
250
+ const headers = {};
251
+ if (token)
252
+ headers['Authorization'] = `Bearer ${token}`;
253
+ const options = {
254
+ hostname: parsed.hostname,
255
+ port: parsed.port,
256
+ path: parsed.pathname + parsed.search,
257
+ headers,
258
+ };
259
+ const req = http_1.default.get(options, (res) => {
260
+ let body = '';
261
+ res.on('data', (chunk) => { body += chunk; });
262
+ res.on('end', () => resolve({ status: res.statusCode ?? 0, body }));
263
+ });
264
+ req.on('error', reject);
265
+ });
266
+ }
267
+ function httpPost(url, payload) {
268
+ return new Promise((resolve, reject) => {
269
+ const data = JSON.stringify(payload);
270
+ const urlObj = new URL(url);
271
+ const token = loadDesktopToken();
272
+ const headers = {
273
+ 'Content-Type': 'application/json',
274
+ 'Content-Length': Buffer.byteLength(data),
275
+ };
276
+ if (token)
277
+ headers['Authorization'] = `Bearer ${token}`;
278
+ const options = {
279
+ hostname: urlObj.hostname,
280
+ port: urlObj.port,
281
+ path: urlObj.pathname,
282
+ method: 'POST',
283
+ headers,
284
+ };
285
+ const req = http_1.default.request(options, (res) => {
286
+ let body = '';
287
+ res.on('data', (chunk) => { body += chunk; });
288
+ res.on('end', () => resolve({ status: res.statusCode ?? 0, body }));
289
+ });
290
+ req.on('error', reject);
291
+ req.write(data);
292
+ req.end();
293
+ });
294
+ }
295
+ // ---------------------------------------------------------------------------
296
+ // Duration formatting
297
+ // ---------------------------------------------------------------------------
298
+ function formatDuration(ms) {
299
+ const totalSeconds = Math.floor(ms / 1000);
300
+ if (totalSeconds < 60) {
301
+ return `${totalSeconds}s`;
302
+ }
303
+ const minutes = Math.floor(totalSeconds / 60);
304
+ const seconds = totalSeconds % 60;
305
+ return `${minutes}m ${seconds}s`;
306
+ }
307
+ // ---------------------------------------------------------------------------
308
+ // Token formatting
309
+ // ---------------------------------------------------------------------------
310
+ function formatTokens(n) {
311
+ return new Intl.NumberFormat('en-US', { useGrouping: true })
312
+ .format(n)
313
+ .replace(/,/g, ' ');
314
+ }
315
+ function printSummary(data) {
316
+ const doneLabel = isTTY ? bold('[specrails-desktop] done') : '[specrails-desktop] done';
317
+ const durationPart = `duration: ${formatDuration(data.durationMs)}`;
318
+ const costPart = data.costUsd != null ? ` cost: $${data.costUsd.toFixed(2)}` : '';
319
+ const tokenPart = data.totalTokens != null ? ` tokens: ${formatTokens(data.totalTokens)}` : '';
320
+ const exitPart = ` exit: ${data.exitCode}`;
321
+ process.stdout.write(`${doneLabel} ${durationPart}${costPart}${tokenPart}${exitPart}\n`);
322
+ }
323
+ async function resolveProjectFromCwd(baseUrl, projectOverride) {
324
+ try {
325
+ // --project flag: resolve by path (absolute/relative) or by name
326
+ if (projectOverride) {
327
+ const isPathLike = projectOverride.startsWith('/') || projectOverride.startsWith('.');
328
+ if (isPathLike) {
329
+ const res = await httpGet(`${baseUrl}/api/resolve?path=${encodeURIComponent(projectOverride)}`);
330
+ if (res.status === 200) {
331
+ const data = JSON.parse(res.body);
332
+ return data.project ?? null;
333
+ }
334
+ }
335
+ else {
336
+ // Resolve by name: fetch all projects and match
337
+ const res = await httpGet(`${baseUrl}/api/projects`);
338
+ if (res.status === 200) {
339
+ const data = JSON.parse(res.body);
340
+ const match = (data.projects ?? []).find((p) => p.name.toLowerCase() === projectOverride.toLowerCase());
341
+ return match ?? null;
342
+ }
343
+ }
344
+ return null;
345
+ }
346
+ // Default: resolve from CWD
347
+ const cwd = process.cwd();
348
+ const res = await httpGet(`${baseUrl}/api/resolve?path=${encodeURIComponent(cwd)}`);
349
+ if (res.status === 200) {
350
+ const data = JSON.parse(res.body);
351
+ return data.project ?? null;
352
+ }
353
+ }
354
+ catch {
355
+ // Resolve endpoint not available — not in Super mode
356
+ }
357
+ return null;
358
+ }
359
+ async function runViaWebManager(command, baseUrl, projectOverride) {
360
+ // Detect Super mode: check if /api/state is reachable
361
+ let spawnUrl = `${baseUrl}/api/spawn`;
362
+ let jobApiBase = `${baseUrl}/api`;
363
+ try {
364
+ const superCheck = await httpGet(`${baseUrl}/api/state`);
365
+ if (superCheck.status === 200) {
366
+ // Super mode: resolve project from CWD or --project override
367
+ const project = await resolveProjectFromCwd(baseUrl, projectOverride);
368
+ if (!project) {
369
+ const hint = projectOverride
370
+ ? `no project found matching: ${projectOverride}`
371
+ : `no project registered for the current directory.\n Run: specrails-desktop add ${process.cwd()}`;
372
+ cliError(`server is running but ${hint}`);
373
+ return 1;
374
+ }
375
+ spawnUrl = `${baseUrl}/api/projects/${project.id}/spawn`;
376
+ jobApiBase = `${baseUrl}/api/projects/${project.id}`;
377
+ cliLog(`project: ${project.name}`);
378
+ }
379
+ }
380
+ catch {
381
+ // Single-project mode — use default paths
382
+ }
383
+ // Spawn the job
384
+ let spawnRes;
385
+ try {
386
+ spawnRes = await httpPost(spawnUrl, { command });
387
+ }
388
+ catch (err) {
389
+ cliError('failed to connect to manager');
390
+ return 1;
391
+ }
392
+ if (spawnRes.status === 409) {
393
+ cliError('manager is busy (another job is running)');
394
+ return 1;
395
+ }
396
+ if (spawnRes.status >= 400) {
397
+ let errMsg = `spawn failed with HTTP ${spawnRes.status}`;
398
+ try {
399
+ const parsed = JSON.parse(spawnRes.body);
400
+ if (parsed.error)
401
+ errMsg = parsed.error;
402
+ }
403
+ catch { /* use default */ }
404
+ cliError(errMsg);
405
+ return 1;
406
+ }
407
+ let processId;
408
+ try {
409
+ const parsed = JSON.parse(spawnRes.body);
410
+ // Server returns jobId; processId is the legacy field name used in LogMessage
411
+ processId = (parsed.jobId ?? parsed.processId) ?? '';
412
+ if (!processId)
413
+ throw new Error('missing jobId');
414
+ }
415
+ catch {
416
+ cliError('invalid response from /api/spawn');
417
+ return 1;
418
+ }
419
+ const startTime = Date.now();
420
+ // Connect WebSocket and stream logs
421
+ const wsUrl = baseUrl.replace(/^http/, 'ws');
422
+ const token = loadDesktopToken();
423
+ let exitCode = 1;
424
+ let resolved = false;
425
+ await new Promise((resolve) => {
426
+ const ws = new ws_1.default(wsUrl, {
427
+ headers: token ? { Authorization: `Bearer ${token}` } : undefined,
428
+ });
429
+ ws.on('message', (data) => {
430
+ let msg;
431
+ try {
432
+ msg = JSON.parse(data.toString());
433
+ }
434
+ catch {
435
+ return;
436
+ }
437
+ if (msg.type === 'init') {
438
+ // Replay only log lines from our processId
439
+ const initMsg = msg;
440
+ for (const logLine of initMsg.logBuffer) {
441
+ if (logLine.processId === processId) {
442
+ handleLogLine(logLine);
443
+ }
444
+ }
445
+ return;
446
+ }
447
+ if (msg.type === 'log') {
448
+ const logMsg = msg;
449
+ if (logMsg.processId !== processId)
450
+ return;
451
+ handleLogLine(logMsg);
452
+ return;
453
+ }
454
+ if (msg.type === 'phase') {
455
+ const phaseMsg = msg;
456
+ process.stdout.write(` ${dimCyan(`→ [${phaseMsg.phase}] ${phaseMsg.state}`)}\n`);
457
+ return;
458
+ }
459
+ });
460
+ function handleLogLine(logMsg) {
461
+ if (resolved)
462
+ return;
463
+ // Check for exit signal
464
+ const match = EXIT_PATTERN.exec(logMsg.line);
465
+ if (match) {
466
+ exitCode = parseInt(match[1], 10);
467
+ resolved = true;
468
+ ws.close();
469
+ resolve();
470
+ return;
471
+ }
472
+ // Print to appropriate stream, preserving ANSI
473
+ if (logMsg.source === 'stderr') {
474
+ process.stderr.write(`${logMsg.line}\n`);
475
+ }
476
+ else {
477
+ process.stdout.write(`${logMsg.line}\n`);
478
+ }
479
+ }
480
+ ws.on('close', () => {
481
+ if (!resolved) {
482
+ cliWarn('lost connection to manager');
483
+ resolved = true;
484
+ resolve();
485
+ }
486
+ });
487
+ ws.on('error', (err) => {
488
+ if (!resolved) {
489
+ cliWarn(`WebSocket error: ${err.message}`);
490
+ resolved = true;
491
+ resolve();
492
+ }
493
+ });
494
+ });
495
+ const durationMs = Date.now() - startTime;
496
+ // Fetch job metadata for cost/tokens
497
+ let costUsd;
498
+ let totalTokens;
499
+ try {
500
+ const jobRes = await httpGet(`${jobApiBase}/jobs/${processId}`);
501
+ if (jobRes.status === 200) {
502
+ const parsed = JSON.parse(jobRes.body);
503
+ if (parsed.job) {
504
+ if (parsed.job.total_cost_usd != null)
505
+ costUsd = parsed.job.total_cost_usd;
506
+ const tokensIn = parsed.job.tokens_in ?? 0;
507
+ const tokensOut = parsed.job.tokens_out ?? 0;
508
+ if (parsed.job.tokens_in != null || parsed.job.tokens_out != null) {
509
+ totalTokens = tokensIn + tokensOut;
510
+ }
511
+ // Prefer server-side duration when available
512
+ if (parsed.job.duration_ms != null) {
513
+ printSummary({ durationMs: parsed.job.duration_ms, costUsd, totalTokens, exitCode });
514
+ return exitCode;
515
+ }
516
+ }
517
+ }
518
+ }
519
+ catch { /* fall through to duration-only summary */ }
520
+ printSummary({ durationMs, costUsd, totalTokens, exitCode });
521
+ return exitCode;
522
+ }
523
+ async function runDirect(command) {
524
+ const startTime = Date.now();
525
+ const args = [
526
+ '--dangerously-skip-permissions',
527
+ '-p',
528
+ ...command.trim().split(/\s+/),
529
+ '--output-format', 'stream-json',
530
+ '--verbose',
531
+ ];
532
+ let child;
533
+ try {
534
+ child = (0, child_process_1.spawn)('claude', args, {
535
+ env: process.env,
536
+ shell: false,
537
+ });
538
+ }
539
+ catch (err) {
540
+ const code = err.code;
541
+ if (code === 'ENOENT') {
542
+ cliError('claude binary not found');
543
+ }
544
+ else {
545
+ cliError(`failed to spawn claude: ${err.message}`);
546
+ }
547
+ return 1;
548
+ }
549
+ let resultData;
550
+ // Stderr: pass through unchanged
551
+ child.stderr?.pipe(process.stderr);
552
+ // Stdout: parse NDJSON line by line
553
+ const rl = (0, readline_1.createInterface)({ input: child.stdout, crlfDelay: Infinity });
554
+ rl.on('line', (line) => {
555
+ if (!line.trim())
556
+ return;
557
+ let parsed = null;
558
+ try {
559
+ parsed = JSON.parse(line);
560
+ }
561
+ catch {
562
+ // Non-JSON line: print as-is
563
+ process.stdout.write(`${line}\n`);
564
+ return;
565
+ }
566
+ if (parsed.type === 'text') {
567
+ const content = parsed.content ?? '';
568
+ if (content)
569
+ process.stdout.write(`${content}\n`);
570
+ }
571
+ else if (parsed.type === 'result') {
572
+ resultData = parsed;
573
+ }
574
+ // All other types: silently ignore
575
+ });
576
+ const exitCode = await new Promise((resolve) => {
577
+ child.on('close', (code) => {
578
+ resolve(code ?? 1);
579
+ });
580
+ child.on('error', (err) => {
581
+ if (err.code === 'ENOENT') {
582
+ cliError('claude binary not found');
583
+ }
584
+ else {
585
+ cliError(`claude process error: ${err.message}`);
586
+ }
587
+ resolve(1);
588
+ });
589
+ });
590
+ const durationMs = Date.now() - startTime;
591
+ let costUsd;
592
+ let totalTokens;
593
+ if (resultData) {
594
+ if (resultData.cost_usd != null)
595
+ costUsd = resultData.cost_usd;
596
+ const tokensIn = resultData.input_tokens ?? 0;
597
+ const tokensOut = resultData.output_tokens ?? 0;
598
+ if (resultData.input_tokens != null || resultData.output_tokens != null) {
599
+ totalTokens = tokensIn + tokensOut;
600
+ }
601
+ }
602
+ printSummary({ durationMs, costUsd, totalTokens, exitCode });
603
+ return exitCode;
604
+ }
605
+ // ---------------------------------------------------------------------------
606
+ // --status handler
607
+ // ---------------------------------------------------------------------------
608
+ async function handleStatus(port) {
609
+ const baseUrl = `http://127.0.0.1:${port}`;
610
+ const detection = await detectWebManager(port);
611
+ if (!detection.running) {
612
+ process.stdout.write(`manager: not running (${baseUrl})\n`);
613
+ return 1;
614
+ }
615
+ try {
616
+ const healthRes = await httpGet(`${baseUrl}/api/health`);
617
+ if (healthRes.status !== 200) {
618
+ process.stdout.write(`manager: not running (${baseUrl})\n`);
619
+ return 1;
620
+ }
621
+ const health = JSON.parse(healthRes.body);
622
+ const version = health.version ? ` (v${health.version})` : '';
623
+ process.stdout.write(`manager: running${version}\n`);
624
+ process.stdout.write(`mode: ${health.mode ?? 'unknown'}\n`);
625
+ if (health.projects !== undefined) {
626
+ process.stdout.write(`projects: ${health.projects}\n`);
627
+ }
628
+ // Legacy mode: fetch additional per-project details from /api/state
629
+ if (health.mode !== 'super') {
630
+ const stateRes = await httpGet(`${baseUrl}/api/state`);
631
+ if (stateRes.status === 200) {
632
+ const state = JSON.parse(stateRes.body);
633
+ process.stdout.write(`project: ${state.projectName ?? 'unknown'}\n`);
634
+ process.stdout.write(`busy: ${state.busy ? 'true' : 'false'}\n`);
635
+ if (state.phases) {
636
+ const phaseStr = Object.entries(state.phases)
637
+ .map(([phase, st]) => `${phase}=${st}`)
638
+ .join(' ');
639
+ process.stdout.write(`phases: ${phaseStr}\n`);
640
+ }
641
+ }
642
+ }
643
+ return 0;
644
+ }
645
+ catch {
646
+ process.stdout.write(`manager: not running (${baseUrl})\n`);
647
+ return 1;
648
+ }
649
+ }
650
+ function formatJobDuration(ms) {
651
+ if (ms == null)
652
+ return '-';
653
+ return formatDuration(ms);
654
+ }
655
+ function formatJobStarted(isoStr) {
656
+ try {
657
+ const d = new Date(isoStr);
658
+ const year = d.getFullYear();
659
+ const month = String(d.getMonth() + 1).padStart(2, '0');
660
+ const day = String(d.getDate()).padStart(2, '0');
661
+ const hour = String(d.getHours()).padStart(2, '0');
662
+ const min = String(d.getMinutes()).padStart(2, '0');
663
+ return `${year}-${month}-${day} ${hour}:${min}`;
664
+ }
665
+ catch {
666
+ return isoStr.slice(0, 16);
667
+ }
668
+ }
669
+ async function handleJobs(port) {
670
+ const baseUrl = `http://127.0.0.1:${port}`;
671
+ const detection = await detectWebManager(port);
672
+ if (!detection.running) {
673
+ cliError(`manager is not running (${baseUrl})`);
674
+ return 1;
675
+ }
676
+ let res;
677
+ try {
678
+ res = await httpGet(`${baseUrl}/api/jobs`);
679
+ }
680
+ catch {
681
+ cliError('failed to fetch job list');
682
+ return 1;
683
+ }
684
+ if (res.status === 501 || res.status === 404) {
685
+ cliLog('jobs history requires manager with SQLite persistence (#57)');
686
+ return 1;
687
+ }
688
+ if (res.status !== 200) {
689
+ cliError(`unexpected response from /api/jobs: HTTP ${res.status}`);
690
+ return 1;
691
+ }
692
+ let data;
693
+ try {
694
+ data = JSON.parse(res.body);
695
+ }
696
+ catch {
697
+ cliError('invalid response from /api/jobs');
698
+ return 1;
699
+ }
700
+ if (!data.jobs || data.jobs.length === 0) {
701
+ cliLog('no jobs recorded yet');
702
+ return 0;
703
+ }
704
+ // Column widths
705
+ const idW = 8;
706
+ const cmdW = 30;
707
+ const startW = 18;
708
+ const durW = 8;
709
+ const exitW = 4;
710
+ const header = [
711
+ 'ID'.padEnd(idW),
712
+ 'COMMAND'.padEnd(cmdW),
713
+ 'STARTED'.padEnd(startW),
714
+ 'DURATION'.padEnd(durW),
715
+ 'EXIT'.padEnd(exitW),
716
+ ].join(' ');
717
+ process.stdout.write(`${bold(header)}\n`);
718
+ for (const job of data.jobs) {
719
+ const idCell = job.id.slice(0, idW).padEnd(idW);
720
+ const cmdCell = job.command.slice(0, cmdW).padEnd(cmdW);
721
+ const startCell = formatJobStarted(job.started_at).padEnd(startW);
722
+ const durCell = formatJobDuration(job.duration_ms).padEnd(durW);
723
+ const exitCell = (job.exit_code ?? '-').toString().padEnd(exitW);
724
+ process.stdout.write(`${idCell} ${cmdCell} ${startCell} ${durCell} ${exitCell}\n`);
725
+ }
726
+ return 0;
727
+ }
728
+ // ---------------------------------------------------------------------------
729
+ // Server-management subcommand group
730
+ // ---------------------------------------------------------------------------
731
+ const DESKTOP_PID_FILE = path_1.default.join(os_1.default.homedir(), '.specrails', 'manager.pid');
732
+ const DESKTOP_LOG_FILE = path_1.default.join(os_1.default.homedir(), '.specrails', 'desktop.log');
733
+ function isPortInUse(port) {
734
+ return new Promise((resolve) => {
735
+ const srv = net_1.default.createServer();
736
+ srv.once('error', (err) => {
737
+ resolve(err.code === 'EADDRINUSE');
738
+ });
739
+ srv.once('listening', () => {
740
+ srv.close();
741
+ resolve(false);
742
+ });
743
+ srv.listen(port, '127.0.0.1');
744
+ });
745
+ }
746
+ function readPid() {
747
+ try {
748
+ const raw = fs_1.default.readFileSync(DESKTOP_PID_FILE, 'utf-8').trim();
749
+ const pid = parseInt(raw, 10);
750
+ return isNaN(pid) ? null : pid;
751
+ }
752
+ catch {
753
+ return null;
754
+ }
755
+ }
756
+ function isProcessRunning(pid) {
757
+ try {
758
+ process.kill(pid, 0);
759
+ return true;
760
+ }
761
+ catch {
762
+ return false;
763
+ }
764
+ }
765
+ function desktopServerPath() {
766
+ // __dirname differs by runtime:
767
+ // compiled (npm install): <root>/cli/dist/ → need ../../server/dist/index.js
768
+ // tsx dev: <root>/cli/ → need ../server/dist/index.js
769
+ // Try both, compiled path first.
770
+ const fromDist = path_1.default.resolve(__dirname, '..', '..', 'server', 'dist', 'index.js');
771
+ const fromSrc = path_1.default.resolve(__dirname, '..', 'server', 'dist', 'index.js');
772
+ const devTs = path_1.default.resolve(__dirname, '..', 'server', 'index.ts');
773
+ if (fs_1.default.existsSync(fromDist))
774
+ return fromDist;
775
+ if (fs_1.default.existsSync(fromSrc))
776
+ return fromSrc;
777
+ if (fs_1.default.existsSync(devTs))
778
+ return devTs;
779
+ return fromDist;
780
+ }
781
+ async function desktopStart(port) {
782
+ const pid = readPid();
783
+ if (pid !== null && isProcessRunning(pid)) {
784
+ cliLog(`server already running (pid ${pid}) on port ${port}`);
785
+ return 0;
786
+ }
787
+ // Check if port is already in use by another process
788
+ const portBusy = await isPortInUse(port);
789
+ if (portBusy) {
790
+ cliError(`port ${port} is already in use by another process`);
791
+ cliError(`if a previous server is stale, run: specrails-desktop stop`);
792
+ cliError(`or use a different port: specrails-desktop --port <port> start`);
793
+ return 1;
794
+ }
795
+ const serverPath = desktopServerPath();
796
+ const isTs = serverPath.endsWith('.ts');
797
+ const args = isTs
798
+ ? ['tsx', serverPath, '--port', String(port)]
799
+ : ['node', serverPath, '--port', String(port)];
800
+ // Ensure log dir exists and open log file for server output
801
+ try {
802
+ fs_1.default.mkdirSync(path_1.default.dirname(DESKTOP_LOG_FILE), { recursive: true });
803
+ }
804
+ catch { /* ignore */ }
805
+ let logFd;
806
+ try {
807
+ logFd = fs_1.default.openSync(DESKTOP_LOG_FILE, 'a');
808
+ }
809
+ catch { /* ignore — fall back to silent */ }
810
+ const stdio = [
811
+ 'ignore',
812
+ logFd ?? 'ignore',
813
+ logFd ?? 'ignore',
814
+ ];
815
+ const child = (0, child_process_2.spawn)(args[0], args.slice(1), {
816
+ detached: true,
817
+ stdio,
818
+ env: { ...process.env },
819
+ });
820
+ if (logFd !== undefined) {
821
+ try {
822
+ fs_1.default.closeSync(logFd);
823
+ }
824
+ catch { /* ignore */ }
825
+ }
826
+ child.unref();
827
+ // Poll until the server is ready (up to 15 seconds, checking every 300ms)
828
+ const pollTimeoutMs = 15000;
829
+ const pollIntervalMs = 300;
830
+ const startPoll = Date.now();
831
+ while (Date.now() - startPoll < pollTimeoutMs) {
832
+ await new Promise((resolve) => setTimeout(resolve, pollIntervalMs));
833
+ const detection = await detectWebManager(port);
834
+ if (detection.running) {
835
+ cliLog(`server started on http://127.0.0.1:${port}`);
836
+ return 0;
837
+ }
838
+ }
839
+ cliError(`server failed to start — logs: ${DESKTOP_LOG_FILE}`);
840
+ return 1;
841
+ }
842
+ async function desktopStop() {
843
+ const pid = readPid();
844
+ if (pid === null) {
845
+ cliLog('server is not running (no pid file)');
846
+ return 0;
847
+ }
848
+ if (!isProcessRunning(pid)) {
849
+ cliLog('server is not running (stale pid file)');
850
+ try {
851
+ fs_1.default.unlinkSync(DESKTOP_PID_FILE);
852
+ }
853
+ catch { /* ignore */ }
854
+ return 0;
855
+ }
856
+ try {
857
+ process.kill(pid, 'SIGTERM');
858
+ cliLog(`server stopped (pid ${pid})`);
859
+ return 0;
860
+ }
861
+ catch (err) {
862
+ cliError(`failed to stop server: ${err.message}`);
863
+ return 1;
864
+ }
865
+ }
866
+ async function desktopStatus(port) {
867
+ const pid = readPid();
868
+ const detection = await detectWebManager(port);
869
+ if (!detection.running) {
870
+ process.stdout.write(`server: not running\n`);
871
+ return 1;
872
+ }
873
+ try {
874
+ const res = await httpGet(`${detection.baseUrl}/api/state`);
875
+ const state = JSON.parse(res.body);
876
+ process.stdout.write(`server: running (pid ${pid ?? '?'}) on ${detection.baseUrl}\n`);
877
+ process.stdout.write(`projects: ${state.projectCount ?? 0}\n`);
878
+ if (state.projects) {
879
+ for (const p of state.projects) {
880
+ process.stdout.write(` - ${p.name}\n`);
881
+ }
882
+ }
883
+ return 0;
884
+ }
885
+ catch {
886
+ process.stdout.write(`server: running on ${detection.baseUrl}\n`);
887
+ return 0;
888
+ }
889
+ }
890
+ async function desktopAdd(projectPath, port) {
891
+ const detection = await detectWebManager(port);
892
+ if (!detection.running) {
893
+ cliError('server is not running. Start it first with: specrails-desktop start');
894
+ return 1;
895
+ }
896
+ try {
897
+ const res = await httpPost(`${detection.baseUrl}/api/projects`, {
898
+ path: path_1.default.resolve(projectPath),
899
+ });
900
+ if (res.status === 201) {
901
+ const data = JSON.parse(res.body);
902
+ cliLog(`added project: ${data.project?.name ?? projectPath}`);
903
+ return 0;
904
+ }
905
+ else if (res.status === 409) {
906
+ cliLog('project already registered');
907
+ return 0;
908
+ }
909
+ else {
910
+ let errMsg = `HTTP ${res.status}`;
911
+ try {
912
+ errMsg = JSON.parse(res.body).error ?? errMsg;
913
+ }
914
+ catch { /* use default */ }
915
+ cliError(`failed to add project: ${errMsg}`);
916
+ return 1;
917
+ }
918
+ }
919
+ catch (err) {
920
+ cliError(`failed to connect to server: ${err.message}`);
921
+ return 1;
922
+ }
923
+ }
924
+ async function desktopRemove(projectId, port) {
925
+ const detection = await detectWebManager(port);
926
+ if (!detection.running) {
927
+ cliError('server is not running');
928
+ return 1;
929
+ }
930
+ try {
931
+ const deleteRes = await new Promise((resolve, reject) => {
932
+ const urlObj = new URL(`${detection.baseUrl}/api/projects/${projectId}`);
933
+ const token = loadDesktopToken();
934
+ const headers = {};
935
+ if (token)
936
+ headers['Authorization'] = `Bearer ${token}`;
937
+ const options = {
938
+ hostname: urlObj.hostname,
939
+ port: urlObj.port,
940
+ path: urlObj.pathname,
941
+ method: 'DELETE',
942
+ headers,
943
+ };
944
+ const req = http_1.default.request(options, (res) => {
945
+ let body = '';
946
+ res.on('data', (chunk) => { body += chunk; });
947
+ res.on('end', () => resolve({ status: res.statusCode ?? 0, body }));
948
+ });
949
+ req.on('error', reject);
950
+ req.end();
951
+ });
952
+ if (deleteRes.status === 200) {
953
+ cliLog(`project removed`);
954
+ return 0;
955
+ }
956
+ else {
957
+ cliError(`failed to remove project: HTTP ${deleteRes.status}`);
958
+ return 1;
959
+ }
960
+ }
961
+ catch (err) {
962
+ cliError(`failed to connect to server: ${err.message}`);
963
+ return 1;
964
+ }
965
+ }
966
+ async function desktopList(port) {
967
+ const detection = await detectWebManager(port);
968
+ if (!detection.running) {
969
+ cliError('server is not running');
970
+ return 1;
971
+ }
972
+ try {
973
+ const res = await httpGet(`${detection.baseUrl}/api/projects`);
974
+ const data = JSON.parse(res.body);
975
+ if (!data.projects || data.projects.length === 0) {
976
+ cliLog('no projects registered');
977
+ return 0;
978
+ }
979
+ const idW = 36;
980
+ const nameW = 24;
981
+ process.stdout.write(`${bold('ID'.padEnd(idW))} ${bold('NAME'.padEnd(nameW))} ${bold('PATH')}\n`);
982
+ for (const p of data.projects) {
983
+ process.stdout.write(`${p.id.padEnd(idW)} ${p.name.padEnd(nameW)} ${p.path}\n`);
984
+ }
985
+ return 0;
986
+ }
987
+ catch (err) {
988
+ cliError(`failed to fetch projects: ${err.message}`);
989
+ return 1;
990
+ }
991
+ }
992
+ async function handleDesktop(subArgs, port) {
993
+ const sub = subArgs[0];
994
+ if (!sub || sub === 'help' || sub === '--help' || sub === '-h') {
995
+ process.stdout.write(`
996
+ ${bold('specrails-desktop')} — server management
997
+
998
+ ${bold('Usage:')}
999
+ specrails-desktop start Start the Specrails server
1000
+ specrails-desktop stop Stop the Specrails server
1001
+ specrails-desktop status Show server status and registered projects
1002
+ specrails-desktop add <path> Register a project by path
1003
+ specrails-desktop remove <id> Unregister a project by ID
1004
+ specrails-desktop list List all registered projects
1005
+ `.trimStart());
1006
+ return 0;
1007
+ }
1008
+ if (sub === 'start') {
1009
+ return desktopStart(port);
1010
+ }
1011
+ if (sub === 'stop') {
1012
+ return desktopStop();
1013
+ }
1014
+ if (sub === 'status') {
1015
+ return desktopStatus(port);
1016
+ }
1017
+ if (sub === 'add') {
1018
+ const projectPath = subArgs[1];
1019
+ if (!projectPath) {
1020
+ cliError('usage: specrails-desktop add <path>');
1021
+ return 1;
1022
+ }
1023
+ return desktopAdd(projectPath, port);
1024
+ }
1025
+ if (sub === 'remove') {
1026
+ const projectId = subArgs[1];
1027
+ if (!projectId) {
1028
+ cliError('usage: specrails-desktop remove <id>');
1029
+ return 1;
1030
+ }
1031
+ return desktopRemove(projectId, port);
1032
+ }
1033
+ if (sub === 'list') {
1034
+ return desktopList(port);
1035
+ }
1036
+ cliError(`unknown subcommand: ${sub}`);
1037
+ return 1;
1038
+ }
1039
+ // ---------------------------------------------------------------------------
1040
+ // Main entry point
1041
+ // ---------------------------------------------------------------------------
1042
+ async function main() {
1043
+ const argv = process.argv.slice(2);
1044
+ const parsed = parseArgs(argv);
1045
+ if (parsed.mode === 'version') {
1046
+ printVersion();
1047
+ process.exit(0);
1048
+ }
1049
+ if (parsed.mode === 'help') {
1050
+ printHelp();
1051
+ process.exit(0);
1052
+ }
1053
+ if (parsed.mode === 'status') {
1054
+ const code = await handleStatus(parsed.port);
1055
+ process.exit(code);
1056
+ }
1057
+ if (parsed.mode === 'jobs') {
1058
+ const code = await handleJobs(parsed.port);
1059
+ process.exit(code);
1060
+ }
1061
+ if (parsed.mode === 'desktop') {
1062
+ const code = await handleDesktop(parsed.subArgs, parsed.port);
1063
+ process.exit(code);
1064
+ }
1065
+ // Command or raw: resolve command string
1066
+ const command = parsed.resolved;
1067
+ const port = parsed.port;
1068
+ cliLog(`running: ${command}`);
1069
+ const detection = await detectWebManager(port);
1070
+ let exitCode;
1071
+ const projectOverride = (parsed.mode === 'command' || parsed.mode === 'raw') ? parsed.projectOverride : undefined;
1072
+ if (detection.running) {
1073
+ cliLog(`routing via manager at ${detection.baseUrl}`);
1074
+ exitCode = await runViaWebManager(command, detection.baseUrl, projectOverride);
1075
+ }
1076
+ else {
1077
+ cliLog('manager not running — invoking claude directly');
1078
+ exitCode = await runDirect(command);
1079
+ }
1080
+ process.exit(exitCode);
1081
+ }
1082
+ // Only run main() when this file is executed directly (not when imported in tests)
1083
+ if (require.main === module) {
1084
+ main().catch((err) => {
1085
+ cliError(err.message ?? String(err));
1086
+ process.exit(1);
1087
+ });
1088
+ }
1089
+ // ---------------------------------------------------------------------------
1090
+ // Test-only exports — not part of the public API
1091
+ // ---------------------------------------------------------------------------
1092
+ exports._internal = {
1093
+ ansi, dim, red, bold, dimCyan, cliPrefix, cliLog, cliError, cliWarn,
1094
+ httpGet, httpPost, formatJobDuration, formatJobStarted, printVersion, printHelp,
1095
+ handleStatus, handleJobs, handleDesktop, desktopStart, desktopStop, desktopStatus, desktopAdd, desktopRemove, desktopList, desktopServerPath,
1096
+ resolveProjectFromCwd, runViaWebManager, runDirect, isPortInUse, readPid, isProcessRunning, main,
1097
+ isTTY, DESKTOP_PID_FILE, DESKTOP_LOG_FILE, EXIT_PATTERN, DEFAULT_PORT, DETECTION_TIMEOUT_MS,
1098
+ };