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.
- package/.claude/commands/specrails/batch-implement.md +287 -0
- package/.claude/commands/specrails/compat-check.md +271 -0
- package/.claude/commands/specrails/doctor.md +62 -0
- package/.claude/commands/specrails/enrich.md +1635 -0
- package/.claude/commands/specrails/explore-spec.md +173 -0
- package/.claude/commands/specrails/health-check.md +527 -0
- package/.claude/commands/specrails/implement.md +1457 -0
- package/.claude/commands/specrails/memory-inspect.md +259 -0
- package/.claude/commands/specrails/opsx-diff.md +419 -0
- package/.claude/commands/specrails/propose-spec.md +102 -0
- package/.claude/commands/specrails/reconfig.md +89 -0
- package/.claude/commands/specrails/refactor-recommender.md +212 -0
- package/.claude/commands/specrails/retry.md +363 -0
- package/.claude/commands/specrails/telemetry.md +552 -0
- package/.claude/commands/specrails/why.md +96 -0
- package/LICENSE +21 -0
- package/README.md +290 -0
- package/cli/dist/specrails-desktop.js +1098 -0
- package/client/dist/assets/ActivityFeedPage-Gy4x8dBt.js +1 -0
- package/client/dist/assets/AgentsPage-CPgu--Fb.js +86 -0
- package/client/dist/assets/AnalyticsPage-B5sJEee2.js +1 -0
- package/client/dist/assets/BarChart-7IMQ8HY1.js +33 -0
- package/client/dist/assets/CodePage-CBdFvbwe.js +2 -0
- package/client/dist/assets/DesktopAnalyticsPage-w0rdTq4w.js +1 -0
- package/client/dist/assets/DocsDialog-BZUYM7wm.js +11 -0
- package/client/dist/assets/DocsPage-9QglWl46.js +11 -0
- package/client/dist/assets/ExportDropdown-BLZFXtNi.js +1 -0
- package/client/dist/assets/IntegrationsPage-BxBE4y99.js +3 -0
- package/client/dist/assets/JobDetailPage-DydWx_5S.js +16 -0
- package/client/dist/assets/JobsPage-20ibw0IO.js +1 -0
- package/client/dist/assets/abap-Bw6f2wDG.js +1 -0
- package/client/dist/assets/activity-BEIp_Y1A.js +1 -0
- package/client/dist/assets/activity-BdrPln96.js +1 -0
- package/client/dist/assets/activity-CpkRS8Sx.js +1 -0
- package/client/dist/assets/activity-DKCpESPt.js +1 -0
- package/client/dist/assets/activity-DOUVEjJi.js +1 -0
- package/client/dist/assets/activity-DRwkql_y.js +1 -0
- package/client/dist/assets/activity-DcDQ7tjw.js +1 -0
- package/client/dist/assets/activity-Dv6H7wEr.js +1 -0
- package/client/dist/assets/addon-image-3WCl5Vhd.js +1 -0
- package/client/dist/assets/addon-ligatures-C5OdliKs.js +2 -0
- package/client/dist/assets/addon-webgl-BbX6pSjl.js +44 -0
- package/client/dist/assets/addspec-B5yl4Loj.js +1 -0
- package/client/dist/assets/addspec-BEeF5-zc.js +1 -0
- package/client/dist/assets/addspec-D33ocMxf.js +1 -0
- package/client/dist/assets/addspec-DFswZ0jK.js +1 -0
- package/client/dist/assets/addspec-DRE-jZv7.js +1 -0
- package/client/dist/assets/addspec-DVZ15Jp8.js +1 -0
- package/client/dist/assets/addspec-Fkv91Opc.js +1 -0
- package/client/dist/assets/addspec-GWm4ffKl.js +1 -0
- package/client/dist/assets/agents-1nCDWRmP.js +1 -0
- package/client/dist/assets/agents-Bm9rPqnt.js +1 -0
- package/client/dist/assets/agents-CMxtJMLD.js +1 -0
- package/client/dist/assets/agents-DK-Dlc0i.js +1 -0
- package/client/dist/assets/agents-Q6Ldfpxx.js +1 -0
- package/client/dist/assets/agents-TeOSy-ax.js +1 -0
- package/client/dist/assets/agents-iTqjRajS.js +1 -0
- package/client/dist/assets/agents-s87sMGzL.js +1 -0
- package/client/dist/assets/agentstudio-B6Wb59E7.js +1 -0
- package/client/dist/assets/agentstudio-BADhZ41e.js +1 -0
- package/client/dist/assets/agentstudio-BSnWLR63.js +1 -0
- package/client/dist/assets/agentstudio-BdidyBzZ.js +1 -0
- package/client/dist/assets/agentstudio-CxlUllqI.js +1 -0
- package/client/dist/assets/agentstudio-D3I62TLJ.js +1 -0
- package/client/dist/assets/agentstudio-DuH9TogZ.js +1 -0
- package/client/dist/assets/agentstudio-Kw88_dUF.js +1 -0
- package/client/dist/assets/aiedit-BWxHGsYA.js +1 -0
- package/client/dist/assets/aiedit-D2ji6Qy0.js +1 -0
- package/client/dist/assets/aiedit-DAhZTvtk.js +1 -0
- package/client/dist/assets/aiedit-DJMny-D5.js +1 -0
- package/client/dist/assets/aiedit-DOcxERkU.js +1 -0
- package/client/dist/assets/aiedit-DvrcbwGv.js +1 -0
- package/client/dist/assets/aiedit-TTwzL1TS.js +1 -0
- package/client/dist/assets/aiedit-WBSjT_C1.js +1 -0
- package/client/dist/assets/analytics-BIdr0YfL.js +1 -0
- package/client/dist/assets/analytics-C6EzgtdE.js +1 -0
- package/client/dist/assets/analytics-C9Zc-rkM.js +1 -0
- package/client/dist/assets/analytics-CVx3YOc0.js +1 -0
- package/client/dist/assets/analytics-CYj0tfj7.js +1 -0
- package/client/dist/assets/analytics-CnY4kNG3.js +1 -0
- package/client/dist/assets/analytics-CrPCZRJ-.js +1 -0
- package/client/dist/assets/analytics-DMCto-TF.js +1 -0
- package/client/dist/assets/apex-Cw8_REBo.js +1 -0
- package/client/dist/assets/atom-one-dark-B-oHczHB.css +1 -0
- package/client/dist/assets/attachments-BIsSSnHJ.js +1 -0
- package/client/dist/assets/attachments-BW4L3l2L.js +1 -0
- package/client/dist/assets/attachments-Bcf6BG6V.js +1 -0
- package/client/dist/assets/attachments-Bke8sCU4.js +1 -0
- package/client/dist/assets/attachments-COcrGRFz.js +1 -0
- package/client/dist/assets/attachments-DYHGA2Dj.js +1 -0
- package/client/dist/assets/attachments-Dd92KpUH.js +1 -0
- package/client/dist/assets/attachments-DzdU6DV6.js +1 -0
- package/client/dist/assets/azcli-Cz6HAoOw.js +1 -0
- package/client/dist/assets/bat-CcJ-xyqL.js +1 -0
- package/client/dist/assets/bicep-z1WDCKYz.js +2 -0
- package/client/dist/assets/browser-5ErDlJoR.js +1 -0
- package/client/dist/assets/browser-Bc-YdlVg.js +1 -0
- package/client/dist/assets/browser-BlYF4OOq.js +1 -0
- package/client/dist/assets/browser-CT-ReZGt.js +1 -0
- package/client/dist/assets/browser-DGITz3fC.js +1 -0
- package/client/dist/assets/browser-JsAIGCEW.js +1 -0
- package/client/dist/assets/browser-M5-rbPlw.js +1 -0
- package/client/dist/assets/browser-Qya9cARy.js +1 -0
- package/client/dist/assets/cameligo-BRewOpfa.js +1 -0
- package/client/dist/assets/chat-BEGuC03z.js +1 -0
- package/client/dist/assets/chat-BEW60P_u.js +1 -0
- package/client/dist/assets/chat-BQNMD0PL.js +1 -0
- package/client/dist/assets/chat-BsbNGPW9.js +1 -0
- package/client/dist/assets/chat-CboQguCi.js +1 -0
- package/client/dist/assets/chat-DRCa9pOt.js +1 -0
- package/client/dist/assets/chat-DwUm6W9z.js +1 -0
- package/client/dist/assets/chat-yoXwguQu.js +1 -0
- package/client/dist/assets/chunk-CilyBKbf.js +1 -0
- package/client/dist/assets/clojure-DBjRWN6g.js +1 -0
- package/client/dist/assets/clsx-DnqN-uhr.js +1 -0
- package/client/dist/assets/code-AL1rVIMb.js +1 -0
- package/client/dist/assets/code-C0BKpkht.js +1 -0
- package/client/dist/assets/code-C0FTS3ew.js +1 -0
- package/client/dist/assets/code-CPcHxzxw.js +1 -0
- package/client/dist/assets/code-D3ryDniw.js +1 -0
- package/client/dist/assets/code-D3zVVQTj.js +1 -0
- package/client/dist/assets/code-PCmfS3dn.js +1 -0
- package/client/dist/assets/code-exI0G5Wd.js +1 -0
- package/client/dist/assets/codicon-ngg6Pgfi.ttf +0 -0
- package/client/dist/assets/coffee-Cfk_XHGR.js +1 -0
- package/client/dist/assets/commands-B772IyDa.js +1 -0
- package/client/dist/assets/commands-BDDp6xFG.js +1 -0
- package/client/dist/assets/commands-CJxCry-o.js +1 -0
- package/client/dist/assets/commands-CfgY-_of.js +1 -0
- package/client/dist/assets/commands-DLrvnPNg.js +1 -0
- package/client/dist/assets/commands-IXMOKBYt.js +1 -0
- package/client/dist/assets/commands-UD1NzmwX.js +1 -0
- package/client/dist/assets/commands-sqrqsxyE.js +1 -0
- package/client/dist/assets/common-DCr6VzJ7.js +1 -0
- package/client/dist/assets/common-Dard9UNH.js +1 -0
- package/client/dist/assets/common-DeDELLZJ.js +1 -0
- package/client/dist/assets/common-DltqHaAe.js +1 -0
- package/client/dist/assets/common-Dmm1GhdD.js +1 -0
- package/client/dist/assets/common-DnjcgkPH.js +1 -0
- package/client/dist/assets/common-GbpxfPG8.js +1 -0
- package/client/dist/assets/common-wA36jmj1.js +1 -0
- package/client/dist/assets/cpp-BVob6BaP.js +1 -0
- package/client/dist/assets/csharp-C4fbRuOu.js +1 -0
- package/client/dist/assets/csp-DthFP_vT.js +1 -0
- package/client/dist/assets/css-CGMH0hcW.js +3 -0
- package/client/dist/assets/css.worker-Wv5dxAWO.js +89 -0
- package/client/dist/assets/cssMode-Cc6ozl-J.js +1 -0
- package/client/dist/assets/cypher-Pnf68BRV.js +1 -0
- package/client/dist/assets/dart-PMMOtxZX.js +1 -0
- package/client/dist/assets/dashboard-B4ixDVk8.js +1 -0
- package/client/dist/assets/dashboard-BZBADHSj.js +1 -0
- package/client/dist/assets/dashboard-C1MfeUHs.js +1 -0
- package/client/dist/assets/dashboard-C7SK6xu5.js +1 -0
- package/client/dist/assets/dashboard-CB6Le1yN.js +1 -0
- package/client/dist/assets/dashboard-CoTpMOBM.js +1 -0
- package/client/dist/assets/dashboard-Duo4DDCW.js +1 -0
- package/client/dist/assets/dashboard-I19DXBxw.js +1 -0
- package/client/dist/assets/dist-js-BY-Fv_fg.js +1 -0
- package/client/dist/assets/dist-js-Bakc4uxT.js +1 -0
- package/client/dist/assets/dockerfile-di1nsJCc.js +1 -0
- package/client/dist/assets/ecl-D_WVcB5M.js +1 -0
- package/client/dist/assets/editor-Br_kD0ds.css +1 -0
- package/client/dist/assets/editor.api2-XLGzZfbc.js +872 -0
- package/client/dist/assets/editor.main-CfXxHimg.js +6 -0
- package/client/dist/assets/editor.worker-Bd9IXS8d.js +26 -0
- package/client/dist/assets/elixir-OAdJEMOn.js +1 -0
- package/client/dist/assets/explore-4mFpnrKU.js +1 -0
- package/client/dist/assets/explore-A8Ltoblq.js +1 -0
- package/client/dist/assets/explore-B9A3iN2W.js +1 -0
- package/client/dist/assets/explore-BV5Xxlsn.js +1 -0
- package/client/dist/assets/explore-BrBJvfjP.js +1 -0
- package/client/dist/assets/explore-C3FSE42C.js +1 -0
- package/client/dist/assets/explore-D2EFgt8J.js +1 -0
- package/client/dist/assets/explore-hFc3HFcp.js +1 -0
- package/client/dist/assets/flow9-D3QEZjgn.js +1 -0
- package/client/dist/assets/format-command-CwGuwzGA.js +1 -0
- package/client/dist/assets/freemarker2-DP7J1gG3.js +3 -0
- package/client/dist/assets/fsharp-BF0k_8N8.js +1 -0
- package/client/dist/assets/go-BAQO5Jsz.js +1 -0
- package/client/dist/assets/graphql-hdFVFkiV.js +1 -0
- package/client/dist/assets/handlebars-BjRlucw6.js +1 -0
- package/client/dist/assets/hcl-DWnl1o-X.js +1 -0
- package/client/dist/assets/html-OumBQJ-U.js +1 -0
- package/client/dist/assets/html.worker-CQP8QQsS.js +502 -0
- package/client/dist/assets/htmlMode-CStc3zXM.js +1 -0
- package/client/dist/assets/index-CimDRRi7.css +2 -0
- package/client/dist/assets/index-XGZaKl_u.js +142 -0
- package/client/dist/assets/ini-CB-6OVu3.js +1 -0
- package/client/dist/assets/integrations-C3p12Ms6.js +1 -0
- package/client/dist/assets/integrations-Cr6hH7XR.js +1 -0
- package/client/dist/assets/integrations-Cublz3m6.js +1 -0
- package/client/dist/assets/integrations-D28q1kF6.js +1 -0
- package/client/dist/assets/integrations-DRdbki5W.js +1 -0
- package/client/dist/assets/integrations-DaC4SzzL.js +1 -0
- package/client/dist/assets/integrations-DmQYCUvN.js +1 -0
- package/client/dist/assets/integrations-HIlUxXVs.js +1 -0
- package/client/dist/assets/java-d1CmfiHX.js +1 -0
- package/client/dist/assets/javascript-CMk--e7g.js +1 -0
- package/client/dist/assets/jobs-BE1siB0M.js +1 -0
- package/client/dist/assets/jobs-BHcQ_Faf.js +1 -0
- package/client/dist/assets/jobs-CFfc2dNX.js +1 -0
- package/client/dist/assets/jobs-CSi5n8X_.js +1 -0
- package/client/dist/assets/jobs-Dc3X86PY.js +1 -0
- package/client/dist/assets/jobs-De5tASex.js +1 -0
- package/client/dist/assets/jobs-DsoXEdo7.js +1 -0
- package/client/dist/assets/jobs-Wl-ApPMb.js +1 -0
- package/client/dist/assets/json.worker-DzV-CpCQ.js +58 -0
- package/client/dist/assets/jsonMode-C2h3ZcjZ.js +7 -0
- package/client/dist/assets/julia-Bgv08lKa.js +1 -0
- package/client/dist/assets/kotlin-u98kaVTf.js +1 -0
- package/client/dist/assets/less-CjYwpgg5.js +2 -0
- package/client/dist/assets/lexon-YTjaAFBB.js +1 -0
- package/client/dist/assets/lib-CPxTMOAq.js +7 -0
- package/client/dist/assets/liquid-mI3KJrBE.js +1 -0
- package/client/dist/assets/lspLanguageFeatures-DU09ggWi.js +4 -0
- package/client/dist/assets/lua-BzmkWv27.js +1 -0
- package/client/dist/assets/m3-CFwk9fw0.js +1 -0
- package/client/dist/assets/markdown-CR5iMpSZ.js +1 -0
- package/client/dist/assets/mdx-C41VDTR_.js +1 -0
- package/client/dist/assets/mips-CcEalc17.js +1 -0
- package/client/dist/assets/monaco.contribution-CPObAXMC.js +2 -0
- package/client/dist/assets/msdax-BQbkawnr.js +1 -0
- package/client/dist/assets/mysql-GTlaaW_P.js +1 -0
- package/client/dist/assets/nav-0fwkrgHt.js +1 -0
- package/client/dist/assets/nav-BEL3MTwK.js +1 -0
- package/client/dist/assets/nav-B_G-TJDW.js +1 -0
- package/client/dist/assets/nav-C2YXcbZS.js +1 -0
- package/client/dist/assets/nav-ClzOE4mA.js +1 -0
- package/client/dist/assets/nav-CtYwmMgu.js +1 -0
- package/client/dist/assets/nav-D2bOGSEg.js +1 -0
- package/client/dist/assets/nav-iH1V5j6o.js +1 -0
- package/client/dist/assets/objective-c-Byu1T5if.js +1 -0
- package/client/dist/assets/pascal-BrfzBfRm.js +1 -0
- package/client/dist/assets/pascaligo-BXXKFUeo.js +1 -0
- package/client/dist/assets/perl-B3OikKq-.js +1 -0
- package/client/dist/assets/pgsql-CTsa0Acc.js +1 -0
- package/client/dist/assets/php-DiQh3FUW.js +1 -0
- package/client/dist/assets/pla-92uH8Fzm.js +1 -0
- package/client/dist/assets/postiats-BbeWkKUr.js +1 -0
- package/client/dist/assets/powerquery-DgDMzpsm.js +1 -0
- package/client/dist/assets/powershell-BfdUUzaG.js +1 -0
- package/client/dist/assets/preload-helper-DSXbuxSR.js +1 -0
- package/client/dist/assets/protobuf-BojW2ftW.js +2 -0
- package/client/dist/assets/pug-BxqTg3IU.js +1 -0
- package/client/dist/assets/python-Y27rKQtk.js +1 -0
- package/client/dist/assets/qsharp-BX_A-MW9.js +1 -0
- package/client/dist/assets/r-D9BMnxvJ.js +1 -0
- package/client/dist/assets/razor-Cd5-q9Bp.js +1 -0
- package/client/dist/assets/redis-5cJqEQJJ.js +1 -0
- package/client/dist/assets/redshift-d8BBqiwb.js +1 -0
- package/client/dist/assets/restructuredtext-C8a6yIcZ.js +1 -0
- package/client/dist/assets/ruby-egeh-6KX.js +1 -0
- package/client/dist/assets/rust-a3r9IInB.js +1 -0
- package/client/dist/assets/sb-y8iRIDei.js +1 -0
- package/client/dist/assets/scala-BPDK2AmK.js +1 -0
- package/client/dist/assets/scheme-BIWUEoOs.js +1 -0
- package/client/dist/assets/scss-CA-PSzwg.js +3 -0
- package/client/dist/assets/settings-55oDcbSh.js +1 -0
- package/client/dist/assets/settings-Bd4Tq1RB.js +1 -0
- package/client/dist/assets/settings-CCSM-Fhn.js +1 -0
- package/client/dist/assets/settings-D3e_bDoW.js +1 -0
- package/client/dist/assets/settings-DKbTkbn7.js +1 -0
- package/client/dist/assets/settings-Dxpo6_w7.js +1 -0
- package/client/dist/assets/settings-bt84e3Aa.js +1 -0
- package/client/dist/assets/settings-nu68QukM.js +1 -0
- package/client/dist/assets/setup-BMqwfbW9.js +1 -0
- package/client/dist/assets/setup-Bb5LcG28.js +1 -0
- package/client/dist/assets/setup-BeEx2_da.js +1 -0
- package/client/dist/assets/setup-CCCrB53Q.js +1 -0
- package/client/dist/assets/setup-CJA0ATmd.js +1 -0
- package/client/dist/assets/setup-CeiDbZcb.js +1 -0
- package/client/dist/assets/setup-Cus7TApA.js +1 -0
- package/client/dist/assets/setup-D9qOs2Xo.js +1 -0
- package/client/dist/assets/shell--LiT1Bja.js +1 -0
- package/client/dist/assets/solidity-DdqZccZg.js +1 -0
- package/client/dist/assets/sophia-S6-YxNG_.js +1 -0
- package/client/dist/assets/sparql-BSf5kMp2.js +1 -0
- package/client/dist/assets/specs-BFfu3u-a.js +1 -0
- package/client/dist/assets/specs-B__C8-8a.js +1 -0
- package/client/dist/assets/specs-CZ1PsXsC.js +1 -0
- package/client/dist/assets/specs-D2FzlLn9.js +1 -0
- package/client/dist/assets/specs-DaUTrNF9.js +1 -0
- package/client/dist/assets/specs-Dyc5hYeE.js +1 -0
- package/client/dist/assets/specs-cKEh2LXt.js +1 -0
- package/client/dist/assets/specs-k0PyLDVt.js +1 -0
- package/client/dist/assets/sql-D7KgjR8G.js +1 -0
- package/client/dist/assets/st-BnoDa-Ml.js +1 -0
- package/client/dist/assets/swift-DEUHTkUX.js +1 -0
- package/client/dist/assets/systemverilog-Tqb_KPnW.js +1 -0
- package/client/dist/assets/tcl-BmBFS2qq.js +1 -0
- package/client/dist/assets/terminal-80yDMgMF.js +1 -0
- package/client/dist/assets/terminal-Bje4ziIa.js +1 -0
- package/client/dist/assets/terminal-C2WYcFHF.js +1 -0
- package/client/dist/assets/terminal-CSONJOex.js +1 -0
- package/client/dist/assets/terminal-DEqzGtcr.js +1 -0
- package/client/dist/assets/terminal-DeWzh6ys.js +1 -0
- package/client/dist/assets/terminal-YOlsJCQj.js +1 -0
- package/client/dist/assets/terminal-lkZYR4wJ.js +1 -0
- package/client/dist/assets/tickets-CB7N30gm.js +1 -0
- package/client/dist/assets/tickets-CF2PYelu.js +1 -0
- package/client/dist/assets/tickets-DNOANUXr.js +1 -0
- package/client/dist/assets/tickets-DU1aqsbr.js +1 -0
- package/client/dist/assets/tickets-DYvafSaY.js +1 -0
- package/client/dist/assets/tickets-DlpC_iTg.js +1 -0
- package/client/dist/assets/tickets-DucYgtdl.js +1 -0
- package/client/dist/assets/tickets-clefmXLv.js +1 -0
- package/client/dist/assets/ts.worker-METxwbDZ.js +67719 -0
- package/client/dist/assets/tsMode-B0y_xEci.js +11 -0
- package/client/dist/assets/twig-BQV8igWC.js +1 -0
- package/client/dist/assets/typescript-BzK0OgwW.js +1 -0
- package/client/dist/assets/typespec-DlFroUGY.js +1 -0
- package/client/dist/assets/useProjectCache-DSaiGFjV.js +1 -0
- package/client/dist/assets/vb-BlrJpIMX.js +1 -0
- package/client/dist/assets/wgsl-BWgIc6FZ.js +298 -0
- package/client/dist/assets/workers-rt--R2Qy.js +1 -0
- package/client/dist/assets/xml-eX9QXAmI.js +1 -0
- package/client/dist/assets/yaml-fcsNkpOt.js +1 -0
- package/client/dist/index.html +246 -0
- package/docs/README.md +54 -0
- package/docs/cli.md +198 -0
- package/docs/codex.md +210 -0
- package/docs/creating-specs.md +197 -0
- package/docs/customizing.md +197 -0
- package/docs/getting-started.md +140 -0
- package/docs/internals/README.md +25 -0
- package/docs/internals/adding-a-provider.md +238 -0
- package/docs/internals/api-reference.md +634 -0
- package/docs/internals/architecture.md +332 -0
- package/docs/internals/configuration.md +172 -0
- package/docs/internals/openspec-workflow.md +282 -0
- package/docs/internals/operations-runbook.md +198 -0
- package/docs/internals/profiles.md +152 -0
- package/docs/platforms/macos.md +130 -0
- package/docs/platforms/windows.md +81 -0
- package/docs/running-pipelines.md +240 -0
- package/docs/terminal.md +138 -0
- package/docs/tracking-cost.md +155 -0
- package/package.json +82 -0
- package/server/dist/agent-generator.js +232 -0
- package/server/dist/agent-refine-db.js +124 -0
- package/server/dist/agent-refine-manager.js +526 -0
- package/server/dist/ai-invocations.js +111 -0
- package/server/dist/attachment-manager.js +299 -0
- package/server/dist/auth.js +207 -0
- package/server/dist/binary-probe.js +35 -0
- package/server/dist/browser-capture-manager.js +576 -0
- package/server/dist/browser-capture-types.js +28 -0
- package/server/dist/browser-network.js +149 -0
- package/server/dist/browser-playwright.js +888 -0
- package/server/dist/build-dirs.js +44 -0
- package/server/dist/changes-reader.js +120 -0
- package/server/dist/chat-manager.js +1060 -0
- package/server/dist/chromium-resolver.js +311 -0
- package/server/dist/code-explorer-router.js +788 -0
- package/server/dist/codex-otel-bridge.js +235 -0
- package/server/dist/command-resolver.js +102 -0
- package/server/dist/config.js +306 -0
- package/server/dist/context-budget.js +113 -0
- package/server/dist/context-scope.js +279 -0
- package/server/dist/contract-refine-runner.js +521 -0
- package/server/dist/core-compat.js +207 -0
- package/server/dist/core-package.js +14 -0
- package/server/dist/db.js +1034 -0
- package/server/dist/desktop-analytics.js +156 -0
- package/server/dist/desktop-db.js +456 -0
- package/server/dist/desktop-router.js +735 -0
- package/server/dist/docs-router.js +207 -0
- package/server/dist/explore-contract-refine.js +421 -0
- package/server/dist/explore-cwd-manager.js +242 -0
- package/server/dist/explore-draft-title.js +47 -0
- package/server/dist/explore-smash.js +450 -0
- package/server/dist/feature-flags.js +17 -0
- package/server/dist/file-provenance.js +382 -0
- package/server/dist/file-summary-generator.js +221 -0
- package/server/dist/file-summary-manager.js +689 -0
- package/server/dist/hooks.js +102 -0
- package/server/dist/ids.js +7 -0
- package/server/dist/index.js +586 -0
- package/server/dist/metrics.js +136 -0
- package/server/dist/mobile/index.js +16 -0
- package/server/dist/mobile/mobile-admin-router.js +84 -0
- package/server/dist/mobile/mobile-auth.js +67 -0
- package/server/dist/mobile/mobile-devices.js +80 -0
- package/server/dist/mobile/mobile-event-bus.js +39 -0
- package/server/dist/mobile/mobile-gateway.js +285 -0
- package/server/dist/mobile/mobile-mdns.js +81 -0
- package/server/dist/mobile/mobile-pairing.js +179 -0
- package/server/dist/mobile/mobile-redact.js +53 -0
- package/server/dist/mobile/mobile-router.js +411 -0
- package/server/dist/mobile/mobile-tls.js +86 -0
- package/server/dist/mobile/mobile-types.js +9 -0
- package/server/dist/mobile/mobile-ws.js +275 -0
- package/server/dist/path-resolver.js +298 -0
- package/server/dist/plugin-manager.js +617 -0
- package/server/dist/plugins/claude-approval.js +179 -0
- package/server/dist/plugins/claude-md-mutation.js +146 -0
- package/server/dist/plugins/codex-mcp.js +108 -0
- package/server/dist/plugins/contributors.js +72 -0
- package/server/dist/plugins/drift.js +58 -0
- package/server/dist/plugins/index.js +14 -0
- package/server/dist/plugins/json-mutation.js +120 -0
- package/server/dist/plugins/manager.js +32 -0
- package/server/dist/plugins/ownership.js +86 -0
- package/server/dist/plugins/paths.js +37 -0
- package/server/dist/plugins/prereq-installer.js +104 -0
- package/server/dist/plugins/rail-integration.js +79 -0
- package/server/dist/plugins/serena/index.js +13 -0
- package/server/dist/plugins/serena/install.js +91 -0
- package/server/dist/plugins/serena/instructions-content.js +21 -0
- package/server/dist/plugins/serena/manifest.js +111 -0
- package/server/dist/plugins/serena/verify.js +78 -0
- package/server/dist/plugins-router.js +215 -0
- package/server/dist/pricing.js +89 -0
- package/server/dist/profile-manager.js +310 -0
- package/server/dist/profiles-router.js +759 -0
- package/server/dist/project-registry.js +443 -0
- package/server/dist/project-router.js +4016 -0
- package/server/dist/proposal-manager.js +291 -0
- package/server/dist/provider-selection.js +69 -0
- package/server/dist/providers/claude-adapter.js +281 -0
- package/server/dist/providers/codex-adapter.js +264 -0
- package/server/dist/providers/index.js +23 -0
- package/server/dist/providers/registry.js +37 -0
- package/server/dist/providers/types.js +22 -0
- package/server/dist/queue-manager.js +1511 -0
- package/server/dist/rails-router.js +362 -0
- package/server/dist/rails-store.js +116 -0
- package/server/dist/result-event.js +106 -0
- package/server/dist/schemas/profile.v1.json +151 -0
- package/server/dist/setup-manager.js +1165 -0
- package/server/dist/setup-prerequisites.js +372 -0
- package/server/dist/smash-runner.js +663 -0
- package/server/dist/spec-draft-parser.js +133 -0
- package/server/dist/spec-launcher-manager.js +174 -0
- package/server/dist/spec-models.js +32 -0
- package/server/dist/specrails-tech-client.js +82 -0
- package/server/dist/spending.js +448 -0
- package/server/dist/telemetry-compactor.js +180 -0
- package/server/dist/telemetry-export.js +317 -0
- package/server/dist/telemetry-receiver.js +224 -0
- package/server/dist/terminal-manager.js +633 -0
- package/server/dist/terminal-marks-store.js +117 -0
- package/server/dist/terminal-osc-parser.js +159 -0
- package/server/dist/terminal-settings.js +282 -0
- package/server/dist/terminal-shell-integration.js +196 -0
- package/server/dist/ticket-broadcast.js +47 -0
- package/server/dist/ticket-store.js +397 -0
- package/server/dist/ticket-watcher.js +117 -0
- package/server/dist/types.js +10 -0
- package/server/dist/user-mcp-config.js +117 -0
- package/server/dist/util/cli-prompt.js +181 -0
- package/server/dist/util/secure-fs.js +50 -0
- package/server/dist/util/win-spawn.js +43 -0
- package/server/dist/webhook-manager.js +89 -0
- 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
|
+
};
|