specrails-desktop 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (455) hide show
  1. package/.claude/commands/specrails/batch-implement.md +287 -0
  2. package/.claude/commands/specrails/compat-check.md +271 -0
  3. package/.claude/commands/specrails/doctor.md +62 -0
  4. package/.claude/commands/specrails/enrich.md +1635 -0
  5. package/.claude/commands/specrails/explore-spec.md +173 -0
  6. package/.claude/commands/specrails/health-check.md +527 -0
  7. package/.claude/commands/specrails/implement.md +1457 -0
  8. package/.claude/commands/specrails/memory-inspect.md +259 -0
  9. package/.claude/commands/specrails/opsx-diff.md +419 -0
  10. package/.claude/commands/specrails/propose-spec.md +102 -0
  11. package/.claude/commands/specrails/reconfig.md +89 -0
  12. package/.claude/commands/specrails/refactor-recommender.md +212 -0
  13. package/.claude/commands/specrails/retry.md +363 -0
  14. package/.claude/commands/specrails/telemetry.md +552 -0
  15. package/.claude/commands/specrails/why.md +96 -0
  16. package/LICENSE +21 -0
  17. package/README.md +290 -0
  18. package/cli/dist/specrails-desktop.js +1098 -0
  19. package/client/dist/assets/ActivityFeedPage-Gy4x8dBt.js +1 -0
  20. package/client/dist/assets/AgentsPage-CPgu--Fb.js +86 -0
  21. package/client/dist/assets/AnalyticsPage-B5sJEee2.js +1 -0
  22. package/client/dist/assets/BarChart-7IMQ8HY1.js +33 -0
  23. package/client/dist/assets/CodePage-CBdFvbwe.js +2 -0
  24. package/client/dist/assets/DesktopAnalyticsPage-w0rdTq4w.js +1 -0
  25. package/client/dist/assets/DocsDialog-BZUYM7wm.js +11 -0
  26. package/client/dist/assets/DocsPage-9QglWl46.js +11 -0
  27. package/client/dist/assets/ExportDropdown-BLZFXtNi.js +1 -0
  28. package/client/dist/assets/IntegrationsPage-BxBE4y99.js +3 -0
  29. package/client/dist/assets/JobDetailPage-DydWx_5S.js +16 -0
  30. package/client/dist/assets/JobsPage-20ibw0IO.js +1 -0
  31. package/client/dist/assets/abap-Bw6f2wDG.js +1 -0
  32. package/client/dist/assets/activity-BEIp_Y1A.js +1 -0
  33. package/client/dist/assets/activity-BdrPln96.js +1 -0
  34. package/client/dist/assets/activity-CpkRS8Sx.js +1 -0
  35. package/client/dist/assets/activity-DKCpESPt.js +1 -0
  36. package/client/dist/assets/activity-DOUVEjJi.js +1 -0
  37. package/client/dist/assets/activity-DRwkql_y.js +1 -0
  38. package/client/dist/assets/activity-DcDQ7tjw.js +1 -0
  39. package/client/dist/assets/activity-Dv6H7wEr.js +1 -0
  40. package/client/dist/assets/addon-image-3WCl5Vhd.js +1 -0
  41. package/client/dist/assets/addon-ligatures-C5OdliKs.js +2 -0
  42. package/client/dist/assets/addon-webgl-BbX6pSjl.js +44 -0
  43. package/client/dist/assets/addspec-B5yl4Loj.js +1 -0
  44. package/client/dist/assets/addspec-BEeF5-zc.js +1 -0
  45. package/client/dist/assets/addspec-D33ocMxf.js +1 -0
  46. package/client/dist/assets/addspec-DFswZ0jK.js +1 -0
  47. package/client/dist/assets/addspec-DRE-jZv7.js +1 -0
  48. package/client/dist/assets/addspec-DVZ15Jp8.js +1 -0
  49. package/client/dist/assets/addspec-Fkv91Opc.js +1 -0
  50. package/client/dist/assets/addspec-GWm4ffKl.js +1 -0
  51. package/client/dist/assets/agents-1nCDWRmP.js +1 -0
  52. package/client/dist/assets/agents-Bm9rPqnt.js +1 -0
  53. package/client/dist/assets/agents-CMxtJMLD.js +1 -0
  54. package/client/dist/assets/agents-DK-Dlc0i.js +1 -0
  55. package/client/dist/assets/agents-Q6Ldfpxx.js +1 -0
  56. package/client/dist/assets/agents-TeOSy-ax.js +1 -0
  57. package/client/dist/assets/agents-iTqjRajS.js +1 -0
  58. package/client/dist/assets/agents-s87sMGzL.js +1 -0
  59. package/client/dist/assets/agentstudio-B6Wb59E7.js +1 -0
  60. package/client/dist/assets/agentstudio-BADhZ41e.js +1 -0
  61. package/client/dist/assets/agentstudio-BSnWLR63.js +1 -0
  62. package/client/dist/assets/agentstudio-BdidyBzZ.js +1 -0
  63. package/client/dist/assets/agentstudio-CxlUllqI.js +1 -0
  64. package/client/dist/assets/agentstudio-D3I62TLJ.js +1 -0
  65. package/client/dist/assets/agentstudio-DuH9TogZ.js +1 -0
  66. package/client/dist/assets/agentstudio-Kw88_dUF.js +1 -0
  67. package/client/dist/assets/aiedit-BWxHGsYA.js +1 -0
  68. package/client/dist/assets/aiedit-D2ji6Qy0.js +1 -0
  69. package/client/dist/assets/aiedit-DAhZTvtk.js +1 -0
  70. package/client/dist/assets/aiedit-DJMny-D5.js +1 -0
  71. package/client/dist/assets/aiedit-DOcxERkU.js +1 -0
  72. package/client/dist/assets/aiedit-DvrcbwGv.js +1 -0
  73. package/client/dist/assets/aiedit-TTwzL1TS.js +1 -0
  74. package/client/dist/assets/aiedit-WBSjT_C1.js +1 -0
  75. package/client/dist/assets/analytics-BIdr0YfL.js +1 -0
  76. package/client/dist/assets/analytics-C6EzgtdE.js +1 -0
  77. package/client/dist/assets/analytics-C9Zc-rkM.js +1 -0
  78. package/client/dist/assets/analytics-CVx3YOc0.js +1 -0
  79. package/client/dist/assets/analytics-CYj0tfj7.js +1 -0
  80. package/client/dist/assets/analytics-CnY4kNG3.js +1 -0
  81. package/client/dist/assets/analytics-CrPCZRJ-.js +1 -0
  82. package/client/dist/assets/analytics-DMCto-TF.js +1 -0
  83. package/client/dist/assets/apex-Cw8_REBo.js +1 -0
  84. package/client/dist/assets/atom-one-dark-B-oHczHB.css +1 -0
  85. package/client/dist/assets/attachments-BIsSSnHJ.js +1 -0
  86. package/client/dist/assets/attachments-BW4L3l2L.js +1 -0
  87. package/client/dist/assets/attachments-Bcf6BG6V.js +1 -0
  88. package/client/dist/assets/attachments-Bke8sCU4.js +1 -0
  89. package/client/dist/assets/attachments-COcrGRFz.js +1 -0
  90. package/client/dist/assets/attachments-DYHGA2Dj.js +1 -0
  91. package/client/dist/assets/attachments-Dd92KpUH.js +1 -0
  92. package/client/dist/assets/attachments-DzdU6DV6.js +1 -0
  93. package/client/dist/assets/azcli-Cz6HAoOw.js +1 -0
  94. package/client/dist/assets/bat-CcJ-xyqL.js +1 -0
  95. package/client/dist/assets/bicep-z1WDCKYz.js +2 -0
  96. package/client/dist/assets/browser-5ErDlJoR.js +1 -0
  97. package/client/dist/assets/browser-Bc-YdlVg.js +1 -0
  98. package/client/dist/assets/browser-BlYF4OOq.js +1 -0
  99. package/client/dist/assets/browser-CT-ReZGt.js +1 -0
  100. package/client/dist/assets/browser-DGITz3fC.js +1 -0
  101. package/client/dist/assets/browser-JsAIGCEW.js +1 -0
  102. package/client/dist/assets/browser-M5-rbPlw.js +1 -0
  103. package/client/dist/assets/browser-Qya9cARy.js +1 -0
  104. package/client/dist/assets/cameligo-BRewOpfa.js +1 -0
  105. package/client/dist/assets/chat-BEGuC03z.js +1 -0
  106. package/client/dist/assets/chat-BEW60P_u.js +1 -0
  107. package/client/dist/assets/chat-BQNMD0PL.js +1 -0
  108. package/client/dist/assets/chat-BsbNGPW9.js +1 -0
  109. package/client/dist/assets/chat-CboQguCi.js +1 -0
  110. package/client/dist/assets/chat-DRCa9pOt.js +1 -0
  111. package/client/dist/assets/chat-DwUm6W9z.js +1 -0
  112. package/client/dist/assets/chat-yoXwguQu.js +1 -0
  113. package/client/dist/assets/chunk-CilyBKbf.js +1 -0
  114. package/client/dist/assets/clojure-DBjRWN6g.js +1 -0
  115. package/client/dist/assets/clsx-DnqN-uhr.js +1 -0
  116. package/client/dist/assets/code-AL1rVIMb.js +1 -0
  117. package/client/dist/assets/code-C0BKpkht.js +1 -0
  118. package/client/dist/assets/code-C0FTS3ew.js +1 -0
  119. package/client/dist/assets/code-CPcHxzxw.js +1 -0
  120. package/client/dist/assets/code-D3ryDniw.js +1 -0
  121. package/client/dist/assets/code-D3zVVQTj.js +1 -0
  122. package/client/dist/assets/code-PCmfS3dn.js +1 -0
  123. package/client/dist/assets/code-exI0G5Wd.js +1 -0
  124. package/client/dist/assets/codicon-ngg6Pgfi.ttf +0 -0
  125. package/client/dist/assets/coffee-Cfk_XHGR.js +1 -0
  126. package/client/dist/assets/commands-B772IyDa.js +1 -0
  127. package/client/dist/assets/commands-BDDp6xFG.js +1 -0
  128. package/client/dist/assets/commands-CJxCry-o.js +1 -0
  129. package/client/dist/assets/commands-CfgY-_of.js +1 -0
  130. package/client/dist/assets/commands-DLrvnPNg.js +1 -0
  131. package/client/dist/assets/commands-IXMOKBYt.js +1 -0
  132. package/client/dist/assets/commands-UD1NzmwX.js +1 -0
  133. package/client/dist/assets/commands-sqrqsxyE.js +1 -0
  134. package/client/dist/assets/common-DCr6VzJ7.js +1 -0
  135. package/client/dist/assets/common-Dard9UNH.js +1 -0
  136. package/client/dist/assets/common-DeDELLZJ.js +1 -0
  137. package/client/dist/assets/common-DltqHaAe.js +1 -0
  138. package/client/dist/assets/common-Dmm1GhdD.js +1 -0
  139. package/client/dist/assets/common-DnjcgkPH.js +1 -0
  140. package/client/dist/assets/common-GbpxfPG8.js +1 -0
  141. package/client/dist/assets/common-wA36jmj1.js +1 -0
  142. package/client/dist/assets/cpp-BVob6BaP.js +1 -0
  143. package/client/dist/assets/csharp-C4fbRuOu.js +1 -0
  144. package/client/dist/assets/csp-DthFP_vT.js +1 -0
  145. package/client/dist/assets/css-CGMH0hcW.js +3 -0
  146. package/client/dist/assets/css.worker-Wv5dxAWO.js +89 -0
  147. package/client/dist/assets/cssMode-Cc6ozl-J.js +1 -0
  148. package/client/dist/assets/cypher-Pnf68BRV.js +1 -0
  149. package/client/dist/assets/dart-PMMOtxZX.js +1 -0
  150. package/client/dist/assets/dashboard-B4ixDVk8.js +1 -0
  151. package/client/dist/assets/dashboard-BZBADHSj.js +1 -0
  152. package/client/dist/assets/dashboard-C1MfeUHs.js +1 -0
  153. package/client/dist/assets/dashboard-C7SK6xu5.js +1 -0
  154. package/client/dist/assets/dashboard-CB6Le1yN.js +1 -0
  155. package/client/dist/assets/dashboard-CoTpMOBM.js +1 -0
  156. package/client/dist/assets/dashboard-Duo4DDCW.js +1 -0
  157. package/client/dist/assets/dashboard-I19DXBxw.js +1 -0
  158. package/client/dist/assets/dist-js-BY-Fv_fg.js +1 -0
  159. package/client/dist/assets/dist-js-Bakc4uxT.js +1 -0
  160. package/client/dist/assets/dockerfile-di1nsJCc.js +1 -0
  161. package/client/dist/assets/ecl-D_WVcB5M.js +1 -0
  162. package/client/dist/assets/editor-Br_kD0ds.css +1 -0
  163. package/client/dist/assets/editor.api2-XLGzZfbc.js +872 -0
  164. package/client/dist/assets/editor.main-CfXxHimg.js +6 -0
  165. package/client/dist/assets/editor.worker-Bd9IXS8d.js +26 -0
  166. package/client/dist/assets/elixir-OAdJEMOn.js +1 -0
  167. package/client/dist/assets/explore-4mFpnrKU.js +1 -0
  168. package/client/dist/assets/explore-A8Ltoblq.js +1 -0
  169. package/client/dist/assets/explore-B9A3iN2W.js +1 -0
  170. package/client/dist/assets/explore-BV5Xxlsn.js +1 -0
  171. package/client/dist/assets/explore-BrBJvfjP.js +1 -0
  172. package/client/dist/assets/explore-C3FSE42C.js +1 -0
  173. package/client/dist/assets/explore-D2EFgt8J.js +1 -0
  174. package/client/dist/assets/explore-hFc3HFcp.js +1 -0
  175. package/client/dist/assets/flow9-D3QEZjgn.js +1 -0
  176. package/client/dist/assets/format-command-CwGuwzGA.js +1 -0
  177. package/client/dist/assets/freemarker2-DP7J1gG3.js +3 -0
  178. package/client/dist/assets/fsharp-BF0k_8N8.js +1 -0
  179. package/client/dist/assets/go-BAQO5Jsz.js +1 -0
  180. package/client/dist/assets/graphql-hdFVFkiV.js +1 -0
  181. package/client/dist/assets/handlebars-BjRlucw6.js +1 -0
  182. package/client/dist/assets/hcl-DWnl1o-X.js +1 -0
  183. package/client/dist/assets/html-OumBQJ-U.js +1 -0
  184. package/client/dist/assets/html.worker-CQP8QQsS.js +502 -0
  185. package/client/dist/assets/htmlMode-CStc3zXM.js +1 -0
  186. package/client/dist/assets/index-CimDRRi7.css +2 -0
  187. package/client/dist/assets/index-XGZaKl_u.js +142 -0
  188. package/client/dist/assets/ini-CB-6OVu3.js +1 -0
  189. package/client/dist/assets/integrations-C3p12Ms6.js +1 -0
  190. package/client/dist/assets/integrations-Cr6hH7XR.js +1 -0
  191. package/client/dist/assets/integrations-Cublz3m6.js +1 -0
  192. package/client/dist/assets/integrations-D28q1kF6.js +1 -0
  193. package/client/dist/assets/integrations-DRdbki5W.js +1 -0
  194. package/client/dist/assets/integrations-DaC4SzzL.js +1 -0
  195. package/client/dist/assets/integrations-DmQYCUvN.js +1 -0
  196. package/client/dist/assets/integrations-HIlUxXVs.js +1 -0
  197. package/client/dist/assets/java-d1CmfiHX.js +1 -0
  198. package/client/dist/assets/javascript-CMk--e7g.js +1 -0
  199. package/client/dist/assets/jobs-BE1siB0M.js +1 -0
  200. package/client/dist/assets/jobs-BHcQ_Faf.js +1 -0
  201. package/client/dist/assets/jobs-CFfc2dNX.js +1 -0
  202. package/client/dist/assets/jobs-CSi5n8X_.js +1 -0
  203. package/client/dist/assets/jobs-Dc3X86PY.js +1 -0
  204. package/client/dist/assets/jobs-De5tASex.js +1 -0
  205. package/client/dist/assets/jobs-DsoXEdo7.js +1 -0
  206. package/client/dist/assets/jobs-Wl-ApPMb.js +1 -0
  207. package/client/dist/assets/json.worker-DzV-CpCQ.js +58 -0
  208. package/client/dist/assets/jsonMode-C2h3ZcjZ.js +7 -0
  209. package/client/dist/assets/julia-Bgv08lKa.js +1 -0
  210. package/client/dist/assets/kotlin-u98kaVTf.js +1 -0
  211. package/client/dist/assets/less-CjYwpgg5.js +2 -0
  212. package/client/dist/assets/lexon-YTjaAFBB.js +1 -0
  213. package/client/dist/assets/lib-CPxTMOAq.js +7 -0
  214. package/client/dist/assets/liquid-mI3KJrBE.js +1 -0
  215. package/client/dist/assets/lspLanguageFeatures-DU09ggWi.js +4 -0
  216. package/client/dist/assets/lua-BzmkWv27.js +1 -0
  217. package/client/dist/assets/m3-CFwk9fw0.js +1 -0
  218. package/client/dist/assets/markdown-CR5iMpSZ.js +1 -0
  219. package/client/dist/assets/mdx-C41VDTR_.js +1 -0
  220. package/client/dist/assets/mips-CcEalc17.js +1 -0
  221. package/client/dist/assets/monaco.contribution-CPObAXMC.js +2 -0
  222. package/client/dist/assets/msdax-BQbkawnr.js +1 -0
  223. package/client/dist/assets/mysql-GTlaaW_P.js +1 -0
  224. package/client/dist/assets/nav-0fwkrgHt.js +1 -0
  225. package/client/dist/assets/nav-BEL3MTwK.js +1 -0
  226. package/client/dist/assets/nav-B_G-TJDW.js +1 -0
  227. package/client/dist/assets/nav-C2YXcbZS.js +1 -0
  228. package/client/dist/assets/nav-ClzOE4mA.js +1 -0
  229. package/client/dist/assets/nav-CtYwmMgu.js +1 -0
  230. package/client/dist/assets/nav-D2bOGSEg.js +1 -0
  231. package/client/dist/assets/nav-iH1V5j6o.js +1 -0
  232. package/client/dist/assets/objective-c-Byu1T5if.js +1 -0
  233. package/client/dist/assets/pascal-BrfzBfRm.js +1 -0
  234. package/client/dist/assets/pascaligo-BXXKFUeo.js +1 -0
  235. package/client/dist/assets/perl-B3OikKq-.js +1 -0
  236. package/client/dist/assets/pgsql-CTsa0Acc.js +1 -0
  237. package/client/dist/assets/php-DiQh3FUW.js +1 -0
  238. package/client/dist/assets/pla-92uH8Fzm.js +1 -0
  239. package/client/dist/assets/postiats-BbeWkKUr.js +1 -0
  240. package/client/dist/assets/powerquery-DgDMzpsm.js +1 -0
  241. package/client/dist/assets/powershell-BfdUUzaG.js +1 -0
  242. package/client/dist/assets/preload-helper-DSXbuxSR.js +1 -0
  243. package/client/dist/assets/protobuf-BojW2ftW.js +2 -0
  244. package/client/dist/assets/pug-BxqTg3IU.js +1 -0
  245. package/client/dist/assets/python-Y27rKQtk.js +1 -0
  246. package/client/dist/assets/qsharp-BX_A-MW9.js +1 -0
  247. package/client/dist/assets/r-D9BMnxvJ.js +1 -0
  248. package/client/dist/assets/razor-Cd5-q9Bp.js +1 -0
  249. package/client/dist/assets/redis-5cJqEQJJ.js +1 -0
  250. package/client/dist/assets/redshift-d8BBqiwb.js +1 -0
  251. package/client/dist/assets/restructuredtext-C8a6yIcZ.js +1 -0
  252. package/client/dist/assets/ruby-egeh-6KX.js +1 -0
  253. package/client/dist/assets/rust-a3r9IInB.js +1 -0
  254. package/client/dist/assets/sb-y8iRIDei.js +1 -0
  255. package/client/dist/assets/scala-BPDK2AmK.js +1 -0
  256. package/client/dist/assets/scheme-BIWUEoOs.js +1 -0
  257. package/client/dist/assets/scss-CA-PSzwg.js +3 -0
  258. package/client/dist/assets/settings-55oDcbSh.js +1 -0
  259. package/client/dist/assets/settings-Bd4Tq1RB.js +1 -0
  260. package/client/dist/assets/settings-CCSM-Fhn.js +1 -0
  261. package/client/dist/assets/settings-D3e_bDoW.js +1 -0
  262. package/client/dist/assets/settings-DKbTkbn7.js +1 -0
  263. package/client/dist/assets/settings-Dxpo6_w7.js +1 -0
  264. package/client/dist/assets/settings-bt84e3Aa.js +1 -0
  265. package/client/dist/assets/settings-nu68QukM.js +1 -0
  266. package/client/dist/assets/setup-BMqwfbW9.js +1 -0
  267. package/client/dist/assets/setup-Bb5LcG28.js +1 -0
  268. package/client/dist/assets/setup-BeEx2_da.js +1 -0
  269. package/client/dist/assets/setup-CCCrB53Q.js +1 -0
  270. package/client/dist/assets/setup-CJA0ATmd.js +1 -0
  271. package/client/dist/assets/setup-CeiDbZcb.js +1 -0
  272. package/client/dist/assets/setup-Cus7TApA.js +1 -0
  273. package/client/dist/assets/setup-D9qOs2Xo.js +1 -0
  274. package/client/dist/assets/shell--LiT1Bja.js +1 -0
  275. package/client/dist/assets/solidity-DdqZccZg.js +1 -0
  276. package/client/dist/assets/sophia-S6-YxNG_.js +1 -0
  277. package/client/dist/assets/sparql-BSf5kMp2.js +1 -0
  278. package/client/dist/assets/specs-BFfu3u-a.js +1 -0
  279. package/client/dist/assets/specs-B__C8-8a.js +1 -0
  280. package/client/dist/assets/specs-CZ1PsXsC.js +1 -0
  281. package/client/dist/assets/specs-D2FzlLn9.js +1 -0
  282. package/client/dist/assets/specs-DaUTrNF9.js +1 -0
  283. package/client/dist/assets/specs-Dyc5hYeE.js +1 -0
  284. package/client/dist/assets/specs-cKEh2LXt.js +1 -0
  285. package/client/dist/assets/specs-k0PyLDVt.js +1 -0
  286. package/client/dist/assets/sql-D7KgjR8G.js +1 -0
  287. package/client/dist/assets/st-BnoDa-Ml.js +1 -0
  288. package/client/dist/assets/swift-DEUHTkUX.js +1 -0
  289. package/client/dist/assets/systemverilog-Tqb_KPnW.js +1 -0
  290. package/client/dist/assets/tcl-BmBFS2qq.js +1 -0
  291. package/client/dist/assets/terminal-80yDMgMF.js +1 -0
  292. package/client/dist/assets/terminal-Bje4ziIa.js +1 -0
  293. package/client/dist/assets/terminal-C2WYcFHF.js +1 -0
  294. package/client/dist/assets/terminal-CSONJOex.js +1 -0
  295. package/client/dist/assets/terminal-DEqzGtcr.js +1 -0
  296. package/client/dist/assets/terminal-DeWzh6ys.js +1 -0
  297. package/client/dist/assets/terminal-YOlsJCQj.js +1 -0
  298. package/client/dist/assets/terminal-lkZYR4wJ.js +1 -0
  299. package/client/dist/assets/tickets-CB7N30gm.js +1 -0
  300. package/client/dist/assets/tickets-CF2PYelu.js +1 -0
  301. package/client/dist/assets/tickets-DNOANUXr.js +1 -0
  302. package/client/dist/assets/tickets-DU1aqsbr.js +1 -0
  303. package/client/dist/assets/tickets-DYvafSaY.js +1 -0
  304. package/client/dist/assets/tickets-DlpC_iTg.js +1 -0
  305. package/client/dist/assets/tickets-DucYgtdl.js +1 -0
  306. package/client/dist/assets/tickets-clefmXLv.js +1 -0
  307. package/client/dist/assets/ts.worker-METxwbDZ.js +67719 -0
  308. package/client/dist/assets/tsMode-B0y_xEci.js +11 -0
  309. package/client/dist/assets/twig-BQV8igWC.js +1 -0
  310. package/client/dist/assets/typescript-BzK0OgwW.js +1 -0
  311. package/client/dist/assets/typespec-DlFroUGY.js +1 -0
  312. package/client/dist/assets/useProjectCache-DSaiGFjV.js +1 -0
  313. package/client/dist/assets/vb-BlrJpIMX.js +1 -0
  314. package/client/dist/assets/wgsl-BWgIc6FZ.js +298 -0
  315. package/client/dist/assets/workers-rt--R2Qy.js +1 -0
  316. package/client/dist/assets/xml-eX9QXAmI.js +1 -0
  317. package/client/dist/assets/yaml-fcsNkpOt.js +1 -0
  318. package/client/dist/index.html +246 -0
  319. package/docs/README.md +54 -0
  320. package/docs/cli.md +198 -0
  321. package/docs/codex.md +210 -0
  322. package/docs/creating-specs.md +197 -0
  323. package/docs/customizing.md +197 -0
  324. package/docs/getting-started.md +140 -0
  325. package/docs/internals/README.md +25 -0
  326. package/docs/internals/adding-a-provider.md +238 -0
  327. package/docs/internals/api-reference.md +634 -0
  328. package/docs/internals/architecture.md +332 -0
  329. package/docs/internals/configuration.md +172 -0
  330. package/docs/internals/openspec-workflow.md +282 -0
  331. package/docs/internals/operations-runbook.md +198 -0
  332. package/docs/internals/profiles.md +152 -0
  333. package/docs/platforms/macos.md +130 -0
  334. package/docs/platforms/windows.md +81 -0
  335. package/docs/running-pipelines.md +240 -0
  336. package/docs/terminal.md +138 -0
  337. package/docs/tracking-cost.md +155 -0
  338. package/package.json +82 -0
  339. package/server/dist/agent-generator.js +232 -0
  340. package/server/dist/agent-refine-db.js +124 -0
  341. package/server/dist/agent-refine-manager.js +526 -0
  342. package/server/dist/ai-invocations.js +111 -0
  343. package/server/dist/attachment-manager.js +299 -0
  344. package/server/dist/auth.js +207 -0
  345. package/server/dist/binary-probe.js +35 -0
  346. package/server/dist/browser-capture-manager.js +576 -0
  347. package/server/dist/browser-capture-types.js +28 -0
  348. package/server/dist/browser-network.js +149 -0
  349. package/server/dist/browser-playwright.js +888 -0
  350. package/server/dist/build-dirs.js +44 -0
  351. package/server/dist/changes-reader.js +120 -0
  352. package/server/dist/chat-manager.js +1060 -0
  353. package/server/dist/chromium-resolver.js +311 -0
  354. package/server/dist/code-explorer-router.js +788 -0
  355. package/server/dist/codex-otel-bridge.js +235 -0
  356. package/server/dist/command-resolver.js +102 -0
  357. package/server/dist/config.js +306 -0
  358. package/server/dist/context-budget.js +113 -0
  359. package/server/dist/context-scope.js +279 -0
  360. package/server/dist/contract-refine-runner.js +521 -0
  361. package/server/dist/core-compat.js +207 -0
  362. package/server/dist/core-package.js +14 -0
  363. package/server/dist/db.js +1034 -0
  364. package/server/dist/desktop-analytics.js +156 -0
  365. package/server/dist/desktop-db.js +456 -0
  366. package/server/dist/desktop-router.js +735 -0
  367. package/server/dist/docs-router.js +207 -0
  368. package/server/dist/explore-contract-refine.js +421 -0
  369. package/server/dist/explore-cwd-manager.js +242 -0
  370. package/server/dist/explore-draft-title.js +47 -0
  371. package/server/dist/explore-smash.js +450 -0
  372. package/server/dist/feature-flags.js +17 -0
  373. package/server/dist/file-provenance.js +382 -0
  374. package/server/dist/file-summary-generator.js +221 -0
  375. package/server/dist/file-summary-manager.js +689 -0
  376. package/server/dist/hooks.js +102 -0
  377. package/server/dist/ids.js +7 -0
  378. package/server/dist/index.js +586 -0
  379. package/server/dist/metrics.js +136 -0
  380. package/server/dist/mobile/index.js +16 -0
  381. package/server/dist/mobile/mobile-admin-router.js +84 -0
  382. package/server/dist/mobile/mobile-auth.js +67 -0
  383. package/server/dist/mobile/mobile-devices.js +80 -0
  384. package/server/dist/mobile/mobile-event-bus.js +39 -0
  385. package/server/dist/mobile/mobile-gateway.js +285 -0
  386. package/server/dist/mobile/mobile-mdns.js +81 -0
  387. package/server/dist/mobile/mobile-pairing.js +179 -0
  388. package/server/dist/mobile/mobile-redact.js +53 -0
  389. package/server/dist/mobile/mobile-router.js +411 -0
  390. package/server/dist/mobile/mobile-tls.js +86 -0
  391. package/server/dist/mobile/mobile-types.js +9 -0
  392. package/server/dist/mobile/mobile-ws.js +275 -0
  393. package/server/dist/path-resolver.js +298 -0
  394. package/server/dist/plugin-manager.js +617 -0
  395. package/server/dist/plugins/claude-approval.js +179 -0
  396. package/server/dist/plugins/claude-md-mutation.js +146 -0
  397. package/server/dist/plugins/codex-mcp.js +108 -0
  398. package/server/dist/plugins/contributors.js +72 -0
  399. package/server/dist/plugins/drift.js +58 -0
  400. package/server/dist/plugins/index.js +14 -0
  401. package/server/dist/plugins/json-mutation.js +120 -0
  402. package/server/dist/plugins/manager.js +32 -0
  403. package/server/dist/plugins/ownership.js +86 -0
  404. package/server/dist/plugins/paths.js +37 -0
  405. package/server/dist/plugins/prereq-installer.js +104 -0
  406. package/server/dist/plugins/rail-integration.js +79 -0
  407. package/server/dist/plugins/serena/index.js +13 -0
  408. package/server/dist/plugins/serena/install.js +91 -0
  409. package/server/dist/plugins/serena/instructions-content.js +21 -0
  410. package/server/dist/plugins/serena/manifest.js +111 -0
  411. package/server/dist/plugins/serena/verify.js +78 -0
  412. package/server/dist/plugins-router.js +215 -0
  413. package/server/dist/pricing.js +89 -0
  414. package/server/dist/profile-manager.js +310 -0
  415. package/server/dist/profiles-router.js +759 -0
  416. package/server/dist/project-registry.js +443 -0
  417. package/server/dist/project-router.js +4016 -0
  418. package/server/dist/proposal-manager.js +291 -0
  419. package/server/dist/provider-selection.js +69 -0
  420. package/server/dist/providers/claude-adapter.js +281 -0
  421. package/server/dist/providers/codex-adapter.js +264 -0
  422. package/server/dist/providers/index.js +23 -0
  423. package/server/dist/providers/registry.js +37 -0
  424. package/server/dist/providers/types.js +22 -0
  425. package/server/dist/queue-manager.js +1511 -0
  426. package/server/dist/rails-router.js +362 -0
  427. package/server/dist/rails-store.js +116 -0
  428. package/server/dist/result-event.js +106 -0
  429. package/server/dist/schemas/profile.v1.json +151 -0
  430. package/server/dist/setup-manager.js +1165 -0
  431. package/server/dist/setup-prerequisites.js +372 -0
  432. package/server/dist/smash-runner.js +663 -0
  433. package/server/dist/spec-draft-parser.js +133 -0
  434. package/server/dist/spec-launcher-manager.js +174 -0
  435. package/server/dist/spec-models.js +32 -0
  436. package/server/dist/specrails-tech-client.js +82 -0
  437. package/server/dist/spending.js +448 -0
  438. package/server/dist/telemetry-compactor.js +180 -0
  439. package/server/dist/telemetry-export.js +317 -0
  440. package/server/dist/telemetry-receiver.js +224 -0
  441. package/server/dist/terminal-manager.js +633 -0
  442. package/server/dist/terminal-marks-store.js +117 -0
  443. package/server/dist/terminal-osc-parser.js +159 -0
  444. package/server/dist/terminal-settings.js +282 -0
  445. package/server/dist/terminal-shell-integration.js +196 -0
  446. package/server/dist/ticket-broadcast.js +47 -0
  447. package/server/dist/ticket-store.js +397 -0
  448. package/server/dist/ticket-watcher.js +117 -0
  449. package/server/dist/types.js +10 -0
  450. package/server/dist/user-mcp-config.js +117 -0
  451. package/server/dist/util/cli-prompt.js +181 -0
  452. package/server/dist/util/secure-fs.js +50 -0
  453. package/server/dist/util/win-spawn.js +43 -0
  454. package/server/dist/webhook-manager.js +89 -0
  455. package/server/dist/ws-routing.js +47 -0
@@ -0,0 +1,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 });