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,411 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.MOBILE_ALLOWLIST = void 0;
|
|
4
|
+
exports.createMobileRouter = createMobileRouter;
|
|
5
|
+
const express_1 = require("express");
|
|
6
|
+
const auth_1 = require("../auth");
|
|
7
|
+
const mobile_redact_1 = require("./mobile-redact");
|
|
8
|
+
const mobile_auth_1 = require("./mobile-auth");
|
|
9
|
+
// The LAN-facing REST surface. Two parts:
|
|
10
|
+
// /pair/* — unauthenticated, locked-down pairing handshake.
|
|
11
|
+
// /v1/* — authenticated allow-list. Each route forwards, in-process, via a
|
|
12
|
+
// REAL loopback HTTP request to http://127.0.0.1:<desktopPort> with
|
|
13
|
+
// the master token injected server-side as `x-desktop-token` (it
|
|
14
|
+
// never leaves the box).
|
|
15
|
+
//
|
|
16
|
+
// Forwarding is PARAMETERISED: the internal path is rebuilt from Express route
|
|
17
|
+
// params (each a single URL segment — `..`/`/` can't appear), never from the raw
|
|
18
|
+
// request URL, so there is no path-traversal or SPA-catch-all bypass. Responses
|
|
19
|
+
// are deep-redacted before reaching the phone.
|
|
20
|
+
const PID_RE = /^[A-Za-z0-9_-]{1,64}$/;
|
|
21
|
+
const NUM_RE = /^\d{1,9}$/;
|
|
22
|
+
const JOBID_RE = /^[A-Za-z0-9_-]{1,64}$/;
|
|
23
|
+
const CONV_ID_RE = /^[A-Za-z0-9-]{1,64}$/;
|
|
24
|
+
function createMobileRouter(deps) {
|
|
25
|
+
const router = (0, express_1.Router)();
|
|
26
|
+
const internalBase = `http://127.0.0.1:${deps.desktopPort}`;
|
|
27
|
+
// Express 5 types a route param as `string | string[]`; coerce to a single
|
|
28
|
+
// segment (an array — which path-to-regexp never produces here — collapses to
|
|
29
|
+
// '' and fails the validators below).
|
|
30
|
+
const seg = (v) => (typeof v === 'string' ? v : '');
|
|
31
|
+
// ─── Pairing (unauthenticated, rate-limited inside PairingManager) ──────────
|
|
32
|
+
const pairRouter = (0, express_1.Router)();
|
|
33
|
+
pairRouter.post('/claim', (req, res) => {
|
|
34
|
+
const body = (req.body ?? {});
|
|
35
|
+
const secret = typeof body.secret === 'string' ? body.secret : '';
|
|
36
|
+
const deviceName = typeof body.deviceName === 'string' ? body.deviceName : 'Device';
|
|
37
|
+
const platform = body.platform === 'android' ? 'android' : 'ios';
|
|
38
|
+
const ip = req.socket?.remoteAddress ?? 'unknown';
|
|
39
|
+
if (!secret) {
|
|
40
|
+
res.status(400).json({ error: 'secret required' });
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
const result = deps.pairing.claim(secret, { name: deviceName, platform }, ip);
|
|
44
|
+
if (result.ok) {
|
|
45
|
+
res.json({ ok: true });
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
const code = result.reason === 'locked' ? 429 : result.reason === 'no-session' || result.reason === 'expired' ? 410 : 403;
|
|
49
|
+
res.status(code).json({ ok: false, reason: result.reason });
|
|
50
|
+
});
|
|
51
|
+
pairRouter.get('/status', (req, res) => {
|
|
52
|
+
const claimId = typeof req.query.claimId === 'string' ? req.query.claimId : '';
|
|
53
|
+
if (!claimId) {
|
|
54
|
+
res.status(400).json({ error: 'claimId required' });
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
res.json(deps.pairing.pollStatus(claimId));
|
|
58
|
+
});
|
|
59
|
+
router.use('/pair', pairRouter);
|
|
60
|
+
// ─── Authenticated allow-list (/v1) ─────────────────────────────────────────
|
|
61
|
+
const v1 = (0, express_1.Router)();
|
|
62
|
+
v1.use((0, mobile_auth_1.createMobileAuthMiddleware)({ db: deps.db, currentFingerprint: deps.currentFingerprint }));
|
|
63
|
+
/** Forward to the internal desktop API with the master token injected; redact
|
|
64
|
+
* the JSON response. `internalPath` is built from validated params only. */
|
|
65
|
+
async function forward(res, method, internalPath, query, body) {
|
|
66
|
+
const url = internalBase + internalPath + (query ? `?${query}` : '');
|
|
67
|
+
try {
|
|
68
|
+
const init = {
|
|
69
|
+
method,
|
|
70
|
+
headers: {
|
|
71
|
+
'x-desktop-token': (0, auth_1.loadOrGenerateToken)(),
|
|
72
|
+
...(body !== undefined ? { 'Content-Type': 'application/json' } : {}),
|
|
73
|
+
},
|
|
74
|
+
...(body !== undefined ? { body: JSON.stringify(body) } : {}),
|
|
75
|
+
};
|
|
76
|
+
const upstream = await fetch(url, init);
|
|
77
|
+
const text = await upstream.text();
|
|
78
|
+
let json;
|
|
79
|
+
try {
|
|
80
|
+
json = text ? JSON.parse(text) : {};
|
|
81
|
+
}
|
|
82
|
+
catch {
|
|
83
|
+
// Non-JSON upstream (should not happen on the allow-listed API) — never
|
|
84
|
+
// forward raw bytes; collapse to a status echo.
|
|
85
|
+
res.status(502).json({ error: 'Bad upstream response' });
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
res.status(upstream.status).json((0, mobile_redact_1.redact)(json));
|
|
89
|
+
}
|
|
90
|
+
catch {
|
|
91
|
+
res.status(502).json({ error: 'Server unreachable' });
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
function rawQuery(req) {
|
|
95
|
+
const i = req.originalUrl.indexOf('?');
|
|
96
|
+
return i >= 0 ? req.originalUrl.slice(i + 1) : '';
|
|
97
|
+
}
|
|
98
|
+
/** 400 unless every (value, regex) pair matches. Guards param injection. */
|
|
99
|
+
function validate(res, ...pairs) {
|
|
100
|
+
for (const [value, re] of pairs) {
|
|
101
|
+
if (!re.test(value)) {
|
|
102
|
+
res.status(400).json({ error: 'Invalid parameter' });
|
|
103
|
+
return false;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
return true;
|
|
107
|
+
}
|
|
108
|
+
// —— Reads ——
|
|
109
|
+
v1.get('/projects', (req, res) => {
|
|
110
|
+
// Exact GET /api/projects is served by the desktop router (mounted at /api).
|
|
111
|
+
void forward(res, 'GET', '/api/projects', '');
|
|
112
|
+
});
|
|
113
|
+
const projectReads = [
|
|
114
|
+
['/projects/:pid/tickets', '/tickets'],
|
|
115
|
+
['/projects/:pid/jobs', '/jobs'],
|
|
116
|
+
['/projects/:pid/queue', '/queue'],
|
|
117
|
+
['/projects/:pid/rails', '/rails'],
|
|
118
|
+
['/projects/:pid/activity', '/activity'],
|
|
119
|
+
['/projects/:pid/stats', '/stats'],
|
|
120
|
+
['/projects/:pid/state', '/state'],
|
|
121
|
+
['/projects/:pid/spending', '/spending'],
|
|
122
|
+
];
|
|
123
|
+
for (const [gw, internal] of projectReads) {
|
|
124
|
+
v1.get(gw, (req, res) => {
|
|
125
|
+
const pid = seg(req.params.pid);
|
|
126
|
+
if (!validate(res, [pid, PID_RE]))
|
|
127
|
+
return;
|
|
128
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}${internal}`, rawQuery(req));
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
v1.get('/projects/:pid/tickets/:tid', (req, res) => {
|
|
132
|
+
const pid = seg(req.params.pid), tid = seg(req.params.tid);
|
|
133
|
+
if (!validate(res, [pid, PID_RE], [tid, NUM_RE]))
|
|
134
|
+
return;
|
|
135
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/tickets/${encodeURIComponent(tid)}`, '');
|
|
136
|
+
});
|
|
137
|
+
v1.get('/projects/:pid/jobs/:jid', (req, res) => {
|
|
138
|
+
const pid = seg(req.params.pid), jid = seg(req.params.jid);
|
|
139
|
+
if (!validate(res, [pid, PID_RE], [jid, JOBID_RE]))
|
|
140
|
+
return;
|
|
141
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/jobs/${encodeURIComponent(jid)}`, '');
|
|
142
|
+
});
|
|
143
|
+
v1.get('/projects/:pid/tickets/:tid/spending-summary', (req, res) => {
|
|
144
|
+
const pid = seg(req.params.pid), tid = seg(req.params.tid);
|
|
145
|
+
if (!validate(res, [pid, PID_RE], [tid, NUM_RE]))
|
|
146
|
+
return;
|
|
147
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/tickets/${encodeURIComponent(tid)}/spending-summary`, '');
|
|
148
|
+
});
|
|
149
|
+
// —— Actions (bodies narrowed) ——
|
|
150
|
+
v1.patch('/projects/:pid/tickets/:tid', (req, res) => {
|
|
151
|
+
const pid = seg(req.params.pid), tid = seg(req.params.tid);
|
|
152
|
+
if (!validate(res, [pid, PID_RE], [tid, NUM_RE]))
|
|
153
|
+
return;
|
|
154
|
+
const b = (req.body ?? {});
|
|
155
|
+
const narrowed = {};
|
|
156
|
+
if (typeof b.status === 'string')
|
|
157
|
+
narrowed.status = b.status;
|
|
158
|
+
if (typeof b.priority === 'string')
|
|
159
|
+
narrowed.priority = b.priority;
|
|
160
|
+
if (typeof b.title === 'string')
|
|
161
|
+
narrowed.title = b.title;
|
|
162
|
+
void forward(res, 'PATCH', `/api/projects/${encodeURIComponent(pid)}/tickets/${encodeURIComponent(tid)}`, '', narrowed);
|
|
163
|
+
});
|
|
164
|
+
v1.delete('/projects/:pid/tickets/:tid', (req, res) => {
|
|
165
|
+
const pid = seg(req.params.pid), tid = seg(req.params.tid);
|
|
166
|
+
if (!validate(res, [pid, PID_RE], [tid, NUM_RE]))
|
|
167
|
+
return;
|
|
168
|
+
void forward(res, 'DELETE', `/api/projects/${encodeURIComponent(pid)}/tickets/${encodeURIComponent(tid)}`, '');
|
|
169
|
+
});
|
|
170
|
+
v1.put('/projects/:pid/rails/:i/tickets', (req, res) => {
|
|
171
|
+
const pid = seg(req.params.pid), i = seg(req.params.i);
|
|
172
|
+
if (!validate(res, [pid, PID_RE], [i, NUM_RE]))
|
|
173
|
+
return;
|
|
174
|
+
const b = (req.body ?? {});
|
|
175
|
+
const ticketIds = Array.isArray(b.ticketIds) ? b.ticketIds.filter((n) => typeof n === 'number') : [];
|
|
176
|
+
void forward(res, 'PUT', `/api/projects/${encodeURIComponent(pid)}/rails/${encodeURIComponent(i)}/tickets`, '', { ticketIds });
|
|
177
|
+
});
|
|
178
|
+
v1.post('/projects/:pid/rails/:i/launch', (req, res) => {
|
|
179
|
+
const pid = seg(req.params.pid), i = seg(req.params.i);
|
|
180
|
+
if (!validate(res, [pid, PID_RE], [i, NUM_RE]))
|
|
181
|
+
return;
|
|
182
|
+
const b = (req.body ?? {});
|
|
183
|
+
const narrowed = {};
|
|
184
|
+
if (typeof b.mode === 'string')
|
|
185
|
+
narrowed.mode = b.mode;
|
|
186
|
+
if (typeof b.profileName === 'string')
|
|
187
|
+
narrowed.profileName = b.profileName;
|
|
188
|
+
if (typeof b.aiEngine === 'string')
|
|
189
|
+
narrowed.aiEngine = b.aiEngine;
|
|
190
|
+
if (typeof b.model === 'string')
|
|
191
|
+
narrowed.model = b.model; // ultracode model
|
|
192
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/rails/${encodeURIComponent(i)}/launch`, '', narrowed);
|
|
193
|
+
});
|
|
194
|
+
v1.put('/projects/:pid/rails/:i/engine', (req, res) => {
|
|
195
|
+
const pid = seg(req.params.pid), i = seg(req.params.i);
|
|
196
|
+
if (!validate(res, [pid, PID_RE], [i, NUM_RE]))
|
|
197
|
+
return;
|
|
198
|
+
const b = (req.body ?? {});
|
|
199
|
+
// aiEngine: a provider string, or null to clear the override.
|
|
200
|
+
const aiEngine = typeof b.aiEngine === 'string' ? b.aiEngine : null;
|
|
201
|
+
void forward(res, 'PUT', `/api/projects/${encodeURIComponent(pid)}/rails/${encodeURIComponent(i)}/engine`, '', { aiEngine });
|
|
202
|
+
});
|
|
203
|
+
v1.put('/projects/:pid/rails/:i/name', (req, res) => {
|
|
204
|
+
const pid = seg(req.params.pid), i = seg(req.params.i);
|
|
205
|
+
if (!validate(res, [pid, PID_RE], [i, NUM_RE]))
|
|
206
|
+
return;
|
|
207
|
+
const b = (req.body ?? {});
|
|
208
|
+
// name: the display label, or null to clear back to the default "Rail N".
|
|
209
|
+
const name = typeof b.name === 'string' ? b.name : null;
|
|
210
|
+
void forward(res, 'PUT', `/api/projects/${encodeURIComponent(pid)}/rails/${encodeURIComponent(i)}/name`, '', { name });
|
|
211
|
+
});
|
|
212
|
+
v1.post('/projects/:pid/rails/:i/stop', (req, res) => {
|
|
213
|
+
const pid = seg(req.params.pid), i = seg(req.params.i);
|
|
214
|
+
if (!validate(res, [pid, PID_RE], [i, NUM_RE]))
|
|
215
|
+
return;
|
|
216
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/rails/${encodeURIComponent(i)}/stop`, '', {});
|
|
217
|
+
});
|
|
218
|
+
v1.delete('/projects/:pid/jobs/:jid', (req, res) => {
|
|
219
|
+
const pid = seg(req.params.pid), jid = seg(req.params.jid);
|
|
220
|
+
if (!validate(res, [pid, PID_RE], [jid, JOBID_RE]))
|
|
221
|
+
return;
|
|
222
|
+
void forward(res, 'DELETE', `/api/projects/${encodeURIComponent(pid)}/jobs/${encodeURIComponent(jid)}`, '');
|
|
223
|
+
});
|
|
224
|
+
v1.post('/projects/:pid/queue/pause', (req, res) => {
|
|
225
|
+
const pid = seg(req.params.pid);
|
|
226
|
+
if (!validate(res, [pid, PID_RE]))
|
|
227
|
+
return;
|
|
228
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/queue/pause`, '', {});
|
|
229
|
+
});
|
|
230
|
+
v1.post('/projects/:pid/queue/resume', (req, res) => {
|
|
231
|
+
const pid = seg(req.params.pid);
|
|
232
|
+
if (!validate(res, [pid, PID_RE]))
|
|
233
|
+
return;
|
|
234
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/queue/resume`, '', {});
|
|
235
|
+
});
|
|
236
|
+
// —— Spec capture on the go ——
|
|
237
|
+
v1.post('/projects/:pid/tickets/generate-spec', (req, res) => {
|
|
238
|
+
const pid = seg(req.params.pid);
|
|
239
|
+
if (!validate(res, [pid, PID_RE]))
|
|
240
|
+
return;
|
|
241
|
+
const b = (req.body ?? {});
|
|
242
|
+
const narrowed = {};
|
|
243
|
+
// The server's generate-spec expects `idea` — the app sends `prompt`.
|
|
244
|
+
if (typeof b.prompt === 'string')
|
|
245
|
+
narrowed.idea = b.prompt;
|
|
246
|
+
if (typeof b.model === 'string')
|
|
247
|
+
narrowed.model = b.model;
|
|
248
|
+
if (typeof b.aiEngine === 'string')
|
|
249
|
+
narrowed.aiEngine = b.aiEngine;
|
|
250
|
+
if (typeof b.contractRefine === 'boolean')
|
|
251
|
+
narrowed.contractRefine = b.contractRefine;
|
|
252
|
+
// Forward the context scope so the server injects .specrails/local-tickets.json
|
|
253
|
+
// (specrails:true) and dedups against existing specs — without this the AI has
|
|
254
|
+
// no context and re-creates specs that already exist.
|
|
255
|
+
const scope = narrowScope(b.contextScope);
|
|
256
|
+
if (scope)
|
|
257
|
+
narrowed.contextScope = scope;
|
|
258
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/tickets/generate-spec`, '', narrowed);
|
|
259
|
+
});
|
|
260
|
+
v1.post('/projects/:pid/tickets/from-prompt', (req, res) => {
|
|
261
|
+
const pid = seg(req.params.pid);
|
|
262
|
+
if (!validate(res, [pid, PID_RE]))
|
|
263
|
+
return;
|
|
264
|
+
const b = (req.body ?? {});
|
|
265
|
+
const narrowed = {};
|
|
266
|
+
// The server's from-prompt expects `description` — the app sends `prompt`.
|
|
267
|
+
if (typeof b.prompt === 'string')
|
|
268
|
+
narrowed.description = b.prompt;
|
|
269
|
+
if (typeof b.title === 'string')
|
|
270
|
+
narrowed.title = b.title;
|
|
271
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/tickets/from-prompt`, '', narrowed);
|
|
272
|
+
});
|
|
273
|
+
// —— Model catalog (for the Quick/Explore model picker) ——
|
|
274
|
+
v1.get('/projects/:pid/default-spec-model', (req, res) => {
|
|
275
|
+
const pid = seg(req.params.pid);
|
|
276
|
+
if (!validate(res, [pid, PID_RE]))
|
|
277
|
+
return;
|
|
278
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/default-spec-model`, rawQuery(req));
|
|
279
|
+
});
|
|
280
|
+
// —— Explore conversations (Add Spec → Explore: a conversational agent) ——
|
|
281
|
+
function narrowScope(raw) {
|
|
282
|
+
if (!raw || typeof raw !== 'object')
|
|
283
|
+
return undefined;
|
|
284
|
+
const r = raw;
|
|
285
|
+
const out = {};
|
|
286
|
+
for (const k of ['specrails', 'openspec', 'full', 'mcp', 'contractRefine', 'userMcp']) {
|
|
287
|
+
if (typeof r[k] === 'boolean')
|
|
288
|
+
out[k] = r[k];
|
|
289
|
+
}
|
|
290
|
+
return out;
|
|
291
|
+
}
|
|
292
|
+
v1.post('/projects/:pid/chat/conversations', (req, res) => {
|
|
293
|
+
const pid = seg(req.params.pid);
|
|
294
|
+
if (!validate(res, [pid, PID_RE]))
|
|
295
|
+
return;
|
|
296
|
+
const b = (req.body ?? {});
|
|
297
|
+
const narrowed = {};
|
|
298
|
+
narrowed.kind = b.kind === 'explore' ? 'explore' : 'sidebar';
|
|
299
|
+
if (typeof b.model === 'string')
|
|
300
|
+
narrowed.model = b.model;
|
|
301
|
+
if (typeof b.aiEngine === 'string')
|
|
302
|
+
narrowed.aiEngine = b.aiEngine;
|
|
303
|
+
const scope = narrowScope(b.contextScope);
|
|
304
|
+
if (scope)
|
|
305
|
+
narrowed.contextScope = scope;
|
|
306
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/chat/conversations`, '', narrowed);
|
|
307
|
+
});
|
|
308
|
+
v1.get('/projects/:pid/chat/conversations/:cid', (req, res) => {
|
|
309
|
+
const pid = seg(req.params.pid), cid = seg(req.params.cid);
|
|
310
|
+
if (!validate(res, [pid, PID_RE], [cid, CONV_ID_RE]))
|
|
311
|
+
return;
|
|
312
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/chat/conversations/${encodeURIComponent(cid)}`, '');
|
|
313
|
+
});
|
|
314
|
+
v1.get('/projects/:pid/chat/conversations/:cid/spec-draft', (req, res) => {
|
|
315
|
+
const pid = seg(req.params.pid), cid = seg(req.params.cid);
|
|
316
|
+
if (!validate(res, [pid, PID_RE], [cid, CONV_ID_RE]))
|
|
317
|
+
return;
|
|
318
|
+
void forward(res, 'GET', `/api/projects/${encodeURIComponent(pid)}/chat/conversations/${encodeURIComponent(cid)}/spec-draft`, '');
|
|
319
|
+
});
|
|
320
|
+
v1.post('/projects/:pid/chat/conversations/:cid/messages', (req, res) => {
|
|
321
|
+
const pid = seg(req.params.pid), cid = seg(req.params.cid);
|
|
322
|
+
if (!validate(res, [pid, PID_RE], [cid, CONV_ID_RE]))
|
|
323
|
+
return;
|
|
324
|
+
const b = (req.body ?? {});
|
|
325
|
+
const narrowed = {};
|
|
326
|
+
if (typeof b.text === 'string')
|
|
327
|
+
narrowed.text = b.text;
|
|
328
|
+
narrowed.lightweight = b.lightweight === false ? false : true;
|
|
329
|
+
if (typeof b.maxTurns === 'number')
|
|
330
|
+
narrowed.maxTurns = b.maxTurns;
|
|
331
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/chat/conversations/${encodeURIComponent(cid)}/messages`, '', narrowed);
|
|
332
|
+
});
|
|
333
|
+
v1.delete('/projects/:pid/chat/conversations/:cid/messages/stream', (req, res) => {
|
|
334
|
+
const pid = seg(req.params.pid), cid = seg(req.params.cid);
|
|
335
|
+
if (!validate(res, [pid, PID_RE], [cid, CONV_ID_RE]))
|
|
336
|
+
return;
|
|
337
|
+
void forward(res, 'DELETE', `/api/projects/${encodeURIComponent(pid)}/chat/conversations/${encodeURIComponent(cid)}/messages/stream`, '');
|
|
338
|
+
});
|
|
339
|
+
for (const action of ['minimize', 'restore']) {
|
|
340
|
+
v1.post(`/projects/:pid/chat/conversations/:cid/${action}`, (req, res) => {
|
|
341
|
+
const pid = seg(req.params.pid), cid = seg(req.params.cid);
|
|
342
|
+
if (!validate(res, [pid, PID_RE], [cid, CONV_ID_RE]))
|
|
343
|
+
return;
|
|
344
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/chat/conversations/${encodeURIComponent(cid)}/${action}`, '', {});
|
|
345
|
+
});
|
|
346
|
+
}
|
|
347
|
+
// —— Commit an Explore conversation to a ticket ——
|
|
348
|
+
v1.post('/projects/:pid/tickets/from-draft', (req, res) => {
|
|
349
|
+
const pid = seg(req.params.pid);
|
|
350
|
+
if (!validate(res, [pid, PID_RE]))
|
|
351
|
+
return;
|
|
352
|
+
const b = (req.body ?? {});
|
|
353
|
+
const narrowed = {};
|
|
354
|
+
if (typeof b.title === 'string')
|
|
355
|
+
narrowed.title = b.title;
|
|
356
|
+
if (typeof b.conversationId === 'string')
|
|
357
|
+
narrowed.conversationId = b.conversationId;
|
|
358
|
+
if (typeof b.description === 'string')
|
|
359
|
+
narrowed.description = b.description;
|
|
360
|
+
if (typeof b.priority === 'string')
|
|
361
|
+
narrowed.priority = b.priority;
|
|
362
|
+
if (Array.isArray(b.labels))
|
|
363
|
+
narrowed.labels = b.labels.filter((x) => typeof x === 'string');
|
|
364
|
+
if (Array.isArray(b.acceptanceCriteria))
|
|
365
|
+
narrowed.acceptanceCriteria = b.acceptanceCriteria.filter((x) => typeof x === 'string');
|
|
366
|
+
void forward(res, 'POST', `/api/projects/${encodeURIComponent(pid)}/tickets/from-draft`, '', narrowed);
|
|
367
|
+
});
|
|
368
|
+
router.use('/v1', v1);
|
|
369
|
+
// Any unmatched gateway path → 404 JSON (NEVER falls through to a SPA handler;
|
|
370
|
+
// the gateway serves JSON/WS only, no static client).
|
|
371
|
+
router.use((_req, res) => {
|
|
372
|
+
res.status(404).json({ error: 'Not found' });
|
|
373
|
+
});
|
|
374
|
+
return router;
|
|
375
|
+
}
|
|
376
|
+
/** The allow-list, exported for the CI drift test (asserts each entry resolves
|
|
377
|
+
* against a real internal route, and that no traversal param is accepted). */
|
|
378
|
+
exports.MOBILE_ALLOWLIST = [
|
|
379
|
+
{ method: 'GET', path: '/v1/projects' },
|
|
380
|
+
{ method: 'GET', path: '/v1/projects/:pid/tickets' },
|
|
381
|
+
{ method: 'GET', path: '/v1/projects/:pid/tickets/:tid' },
|
|
382
|
+
{ method: 'GET', path: '/v1/projects/:pid/jobs' },
|
|
383
|
+
{ method: 'GET', path: '/v1/projects/:pid/jobs/:jid' },
|
|
384
|
+
{ method: 'GET', path: '/v1/projects/:pid/queue' },
|
|
385
|
+
{ method: 'GET', path: '/v1/projects/:pid/rails' },
|
|
386
|
+
{ method: 'GET', path: '/v1/projects/:pid/activity' },
|
|
387
|
+
{ method: 'GET', path: '/v1/projects/:pid/stats' },
|
|
388
|
+
{ method: 'GET', path: '/v1/projects/:pid/state' },
|
|
389
|
+
{ method: 'GET', path: '/v1/projects/:pid/spending' },
|
|
390
|
+
{ method: 'GET', path: '/v1/projects/:pid/tickets/:tid/spending-summary' },
|
|
391
|
+
{ method: 'PATCH', path: '/v1/projects/:pid/tickets/:tid' },
|
|
392
|
+
{ method: 'DELETE', path: '/v1/projects/:pid/tickets/:tid' },
|
|
393
|
+
{ method: 'PUT', path: '/v1/projects/:pid/rails/:i/tickets' },
|
|
394
|
+
{ method: 'POST', path: '/v1/projects/:pid/rails/:i/launch' },
|
|
395
|
+
{ method: 'PUT', path: '/v1/projects/:pid/rails/:i/engine' },
|
|
396
|
+
{ method: 'POST', path: '/v1/projects/:pid/rails/:i/stop' },
|
|
397
|
+
{ method: 'DELETE', path: '/v1/projects/:pid/jobs/:jid' },
|
|
398
|
+
{ method: 'POST', path: '/v1/projects/:pid/queue/pause' },
|
|
399
|
+
{ method: 'POST', path: '/v1/projects/:pid/queue/resume' },
|
|
400
|
+
{ method: 'POST', path: '/v1/projects/:pid/tickets/generate-spec' },
|
|
401
|
+
{ method: 'POST', path: '/v1/projects/:pid/tickets/from-prompt' },
|
|
402
|
+
{ method: 'GET', path: '/v1/projects/:pid/default-spec-model' },
|
|
403
|
+
{ method: 'POST', path: '/v1/projects/:pid/chat/conversations' },
|
|
404
|
+
{ method: 'GET', path: '/v1/projects/:pid/chat/conversations/:cid' },
|
|
405
|
+
{ method: 'GET', path: '/v1/projects/:pid/chat/conversations/:cid/spec-draft' },
|
|
406
|
+
{ method: 'POST', path: '/v1/projects/:pid/chat/conversations/:cid/messages' },
|
|
407
|
+
{ method: 'DELETE', path: '/v1/projects/:pid/chat/conversations/:cid/messages/stream' },
|
|
408
|
+
{ method: 'POST', path: '/v1/projects/:pid/chat/conversations/:cid/minimize' },
|
|
409
|
+
{ method: 'POST', path: '/v1/projects/:pid/chat/conversations/:cid/restore' },
|
|
410
|
+
{ method: 'POST', path: '/v1/projects/:pid/tickets/from-draft' },
|
|
411
|
+
];
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.mobileDir = mobileDir;
|
|
7
|
+
exports.fingerprintOf = fingerprintOf;
|
|
8
|
+
exports.loadOrCreateCert = loadOrCreateCert;
|
|
9
|
+
exports.rotateCert = rotateCert;
|
|
10
|
+
const fs_1 = __importDefault(require("fs"));
|
|
11
|
+
const path_1 = __importDefault(require("path"));
|
|
12
|
+
const os_1 = __importDefault(require("os"));
|
|
13
|
+
const crypto_1 = require("crypto");
|
|
14
|
+
const selfsigned_1 = require("selfsigned");
|
|
15
|
+
function mobileDir() {
|
|
16
|
+
return path_1.default.join(os_1.default.homedir(), '.specrails', 'mobile');
|
|
17
|
+
}
|
|
18
|
+
function certPath(dir) {
|
|
19
|
+
return path_1.default.join(dir, 'tls-cert.pem');
|
|
20
|
+
}
|
|
21
|
+
function keyPath(dir) {
|
|
22
|
+
return path_1.default.join(dir, 'tls-key.pem');
|
|
23
|
+
}
|
|
24
|
+
/** sha256(DER) of a PEM cert, hex lowercase — the value put in the QR and pinned
|
|
25
|
+
* by the phone (Dart: `sha256.convert(x509.der)`). */
|
|
26
|
+
function fingerprintOf(certPem) {
|
|
27
|
+
const der = new crypto_1.X509Certificate(certPem).raw;
|
|
28
|
+
return (0, crypto_1.createHash)('sha256').update(der).digest('hex');
|
|
29
|
+
}
|
|
30
|
+
async function generatePair() {
|
|
31
|
+
// The phone pins sha256(cert DER), never the CN, so renaming the CN only
|
|
32
|
+
// affects freshly generated certs (which always require a fresh QR pairing).
|
|
33
|
+
const attrs = [{ name: 'commonName', value: 'specrails-desktop-mobile' }];
|
|
34
|
+
const notAfter = new Date();
|
|
35
|
+
notAfter.setFullYear(notAfter.getFullYear() + 10);
|
|
36
|
+
const pems = await (0, selfsigned_1.generate)(attrs, {
|
|
37
|
+
keyType: 'ec',
|
|
38
|
+
curve: 'P-256',
|
|
39
|
+
algorithm: 'sha256',
|
|
40
|
+
notAfterDate: notAfter,
|
|
41
|
+
extensions: [
|
|
42
|
+
{ name: 'basicConstraints', cA: true },
|
|
43
|
+
{
|
|
44
|
+
name: 'subjectAltName',
|
|
45
|
+
altNames: [
|
|
46
|
+
{ type: 2, value: 'localhost' },
|
|
47
|
+
{ type: 7, ip: '127.0.0.1' },
|
|
48
|
+
],
|
|
49
|
+
},
|
|
50
|
+
],
|
|
51
|
+
});
|
|
52
|
+
return { certPem: pems.cert, keyPem: pems.private };
|
|
53
|
+
}
|
|
54
|
+
function writePair(dir, certPem, keyPem) {
|
|
55
|
+
fs_1.default.mkdirSync(dir, { recursive: true });
|
|
56
|
+
try {
|
|
57
|
+
fs_1.default.chmodSync(dir, 0o700);
|
|
58
|
+
}
|
|
59
|
+
catch { /* best-effort on platforms without chmod */ }
|
|
60
|
+
fs_1.default.writeFileSync(certPath(dir), certPem, { encoding: 'utf-8', mode: 0o600 });
|
|
61
|
+
fs_1.default.writeFileSync(keyPath(dir), keyPem, { encoding: 'utf-8', mode: 0o600 });
|
|
62
|
+
}
|
|
63
|
+
/** Load the existing gateway cert, generating + persisting one on first use. */
|
|
64
|
+
async function loadOrCreateCert(dir = mobileDir()) {
|
|
65
|
+
try {
|
|
66
|
+
const certPem = fs_1.default.readFileSync(certPath(dir), 'utf-8');
|
|
67
|
+
const keyPem = fs_1.default.readFileSync(keyPath(dir), 'utf-8');
|
|
68
|
+
if (certPem.includes('BEGIN CERTIFICATE') && keyPem.includes('PRIVATE KEY')) {
|
|
69
|
+
return { certPem, keyPem, fingerprint: fingerprintOf(certPem) };
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
catch {
|
|
73
|
+
// Fall through to generate.
|
|
74
|
+
}
|
|
75
|
+
const { certPem, keyPem } = await generatePair();
|
|
76
|
+
writePair(dir, certPem, keyPem);
|
|
77
|
+
return { certPem, keyPem, fingerprint: fingerprintOf(certPem) };
|
|
78
|
+
}
|
|
79
|
+
/** Generate a brand-new cert (rotation). Caller is responsible for revoking all
|
|
80
|
+
* devices afterwards — a rotated cert no longer matches any stored fingerprint,
|
|
81
|
+
* which is the point: "Reset mobile identity" invalidates every paired device. */
|
|
82
|
+
async function rotateCert(dir = mobileDir()) {
|
|
83
|
+
const { certPem, keyPem } = await generatePair();
|
|
84
|
+
writePair(dir, certPem, keyPem);
|
|
85
|
+
return { certPem, keyPem, fingerprint: fingerprintOf(certPem) };
|
|
86
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Shared types for the Mobile Gateway (server/mobile/*).
|
|
3
|
+
//
|
|
4
|
+
// The gateway is a second HTTPS+WSS listener in the SAME Node process as the
|
|
5
|
+
// main server, default port 4202, OFF by default. It pairs phones/tablets by QR +
|
|
6
|
+
// desktop-approval and exposes a deny-by-default allow-list of the existing API,
|
|
7
|
+
// redacted, over a per-device token. The main server at 127.0.0.1:4200 is never
|
|
8
|
+
// itself exposed. See docs/mobile.md.
|
|
9
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|