specrails-desktop 2.2.1 → 2.4.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 (350) hide show
  1. package/client/dist/assets/ActivityFeedPage-DJJlZ3mF.js +1 -0
  2. package/client/dist/assets/AgentsPage-49JaEDjR.js +86 -0
  3. package/client/dist/assets/{AnalyticsPage-BD0paa75.js → AnalyticsPage-BUd3gWYC.js} +1 -1
  4. package/client/dist/assets/{BarChart-D8ZZRab3.js → BarChart-HDe_YoUD.js} +1 -1
  5. package/client/dist/assets/CodePage-CqPPND47.js +2 -0
  6. package/client/dist/assets/{DesktopAnalyticsPage-mwd8460_.js → DesktopAnalyticsPage-CgvmSvF0.js} +1 -1
  7. package/client/dist/assets/DocsDialog-hHFd3Ejs.js +11 -0
  8. package/client/dist/assets/DocsPage-B4R1aksg.js +11 -0
  9. package/client/dist/assets/{ExportDropdown-CLYmQhic.js → ExportDropdown-f4dwQjlT.js} +1 -1
  10. package/client/dist/assets/IntegrationsPage-CX2Ybxx0.js +3 -0
  11. package/client/dist/assets/JobDetailPage-DN2Jc8Ti.js +16 -0
  12. package/client/dist/assets/JobsPage-DmdpqijT.js +1 -0
  13. package/client/dist/assets/code-BwIz8agY.js +1 -0
  14. package/client/dist/assets/code-CD7yNSK0.js +1 -0
  15. package/client/dist/assets/code-CDFlxUFC.js +1 -0
  16. package/client/dist/assets/code-CY85RXZU.js +1 -0
  17. package/client/dist/assets/code-Cp3Fdng-.js +1 -0
  18. package/client/dist/assets/code-D24e1Crx.js +1 -0
  19. package/client/dist/assets/code-DtZBQTi9.js +1 -0
  20. package/client/dist/assets/code-nKa0fkm_.js +1 -0
  21. package/client/dist/assets/{cssMode-Cc6ozl-J.js → cssMode-DzNPAYFh.js} +1 -1
  22. package/client/dist/assets/{dist-js-D3MxtOYa.js → dist-js-COfIfLRE.js} +1 -1
  23. package/client/dist/assets/{dist-js-BOu_cXw3.js → dist-js-CvScGQU_.js} +1 -1
  24. package/client/dist/assets/{editor.main-CfXxHimg.js → editor.main-C7Rmw-hR.js} +2 -2
  25. package/client/dist/assets/{freemarker2-DP7J1gG3.js → freemarker2-Cszs4SVo.js} +1 -1
  26. package/client/dist/assets/{handlebars-BjRlucw6.js → handlebars-Dp7Lsuym.js} +1 -1
  27. package/client/dist/assets/{html-OumBQJ-U.js → html-BURidrEm.js} +1 -1
  28. package/client/dist/assets/{htmlMode-CStc3zXM.js → htmlMode--k5M7GjZ.js} +1 -1
  29. package/client/dist/assets/index-DBpvYrDK.css +2 -0
  30. package/client/dist/assets/index-DGIXKRHE.js +142 -0
  31. package/client/dist/assets/{integrations-Cublz3m6.js → integrations-2C7MkGT0.js} +1 -1
  32. package/client/dist/assets/{integrations-HIlUxXVs.js → integrations-BDC670cg.js} +1 -1
  33. package/client/dist/assets/integrations-BqUmRUef.js +1 -0
  34. package/client/dist/assets/{integrations-DmQYCUvN.js → integrations-C2jQtv-s.js} +1 -1
  35. package/client/dist/assets/{integrations-DRdbki5W.js → integrations-CB98NeH5.js} +1 -1
  36. package/client/dist/assets/{integrations-C3p12Ms6.js → integrations-CX4p_bij.js} +1 -1
  37. package/client/dist/assets/{integrations-DaC4SzzL.js → integrations-_SuVeQIG.js} +1 -1
  38. package/client/dist/assets/{integrations-Cr6hH7XR.js → integrations-eQPHAYsE.js} +1 -1
  39. package/client/dist/assets/{javascript-CMk--e7g.js → javascript-kJQz__44.js} +1 -1
  40. package/client/dist/assets/jira-C-ATCti0.js +1 -0
  41. package/client/dist/assets/jira-CmVfRM-b.js +1 -0
  42. package/client/dist/assets/jira-D7bkKAX8.js +1 -0
  43. package/client/dist/assets/jira-DKImM1YH.js +1 -0
  44. package/client/dist/assets/jira-DOw8bkIR.js +1 -0
  45. package/client/dist/assets/jira-DlA-wGp-.js +1 -0
  46. package/client/dist/assets/jira-Fob8EGxN.js +1 -0
  47. package/client/dist/assets/jira-xZA2lixb.js +1 -0
  48. package/client/dist/assets/{jsonMode-C2h3ZcjZ.js → jsonMode-v5JYPpnz.js} +1 -1
  49. package/client/dist/assets/{lib-DQ2hrj8m.js → lib-Bro9Z0gp.js} +1 -1
  50. package/client/dist/assets/{liquid-mI3KJrBE.js → liquid-Dl9I6gWt.js} +1 -1
  51. package/client/dist/assets/{lspLanguageFeatures-DU09ggWi.js → lspLanguageFeatures-CPlEe0NK.js} +1 -1
  52. package/client/dist/assets/{mdx-C41VDTR_.js → mdx-Byl7TtzQ.js} +1 -1
  53. package/client/dist/assets/{monaco.contribution-CPObAXMC.js → monaco.contribution-YMAkHQcQ.js} +2 -2
  54. package/client/dist/assets/{python-Y27rKQtk.js → python-jWQwT6j2.js} +1 -1
  55. package/client/dist/assets/{razor-Cd5-q9Bp.js → razor-BWS3sP-E.js} +1 -1
  56. package/client/dist/assets/setup-C0dzw8j4.js +1 -0
  57. package/client/dist/assets/setup-C1IA-9YS.js +1 -0
  58. package/client/dist/assets/setup-CpfjaNut.js +1 -0
  59. package/client/dist/assets/setup-D3rNZA9A.js +1 -0
  60. package/client/dist/assets/setup-UD2aanGs.js +1 -0
  61. package/client/dist/assets/setup-WP6WOYQh.js +1 -0
  62. package/client/dist/assets/setup-gzLG8T6F.js +1 -0
  63. package/client/dist/assets/setup-pjgmYHx6.js +1 -0
  64. package/client/dist/assets/specs-4lA_u79w.js +1 -0
  65. package/client/dist/assets/{specs-D2FzlLn9.js → specs-BHjxcjOf.js} +1 -1
  66. package/client/dist/assets/{specs-CZ1PsXsC.js → specs-CXNQzPk9.js} +1 -1
  67. package/client/dist/assets/{specs-Dyc5hYeE.js → specs-DFnc2Huj.js} +1 -1
  68. package/client/dist/assets/{specs-BFfu3u-a.js → specs-DZCLH2-l.js} +1 -1
  69. package/client/dist/assets/{specs-B__C8-8a.js → specs-DgmyAE3N.js} +1 -1
  70. package/client/dist/assets/{specs-DaUTrNF9.js → specs-DicWhvwi.js} +1 -1
  71. package/client/dist/assets/{specs-k0PyLDVt.js → specs-dkro6lSM.js} +1 -1
  72. package/client/dist/assets/{tsMode-B0y_xEci.js → tsMode-BbOGOuSV.js} +1 -1
  73. package/client/dist/assets/{typescript-BzK0OgwW.js → typescript-eBtFQJLs.js} +1 -1
  74. package/client/dist/assets/{useProjectCache-BxY4aTjd.js → useProjectCache-D9juBhsO.js} +1 -1
  75. package/client/dist/assets/{workers-rt--R2Qy.js → workers-BvicOoDf.js} +1 -1
  76. package/client/dist/assets/{xml-eX9QXAmI.js → xml-BJepAPyM.js} +1 -1
  77. package/client/dist/assets/{yaml-fcsNkpOt.js → yaml-DabgV-eA.js} +1 -1
  78. package/client/dist/index.html +13 -12
  79. package/docs/jira-integration-plan.md +321 -0
  80. package/package.json +1 -1
  81. package/server/dist/agent-refine-manager.js +128 -153
  82. package/server/dist/chat-manager.js +242 -0
  83. package/server/dist/code-explorer-router.js +78 -0
  84. package/server/dist/command-resolver.js +17 -0
  85. package/server/dist/contract-refine-runner.js +42 -10
  86. package/server/dist/db.js +86 -0
  87. package/server/dist/desktop-db.js +3 -0
  88. package/server/dist/explore-stdin-session.js +129 -0
  89. package/server/dist/feature-flags.js +11 -0
  90. package/server/dist/jira/jira-adf.js +113 -0
  91. package/server/dist/jira/jira-backlog-config.js +58 -0
  92. package/server/dist/jira/jira-client.js +279 -0
  93. package/server/dist/jira/jira-credential-store.js +103 -0
  94. package/server/dist/jira/jira-db.js +341 -0
  95. package/server/dist/jira/jira-issue-fields.js +428 -0
  96. package/server/dist/jira/jira-materializer.js +250 -0
  97. package/server/dist/jira/jira-status-resolver.js +211 -0
  98. package/server/dist/jira/jira-sync-manager.js +1014 -0
  99. package/server/dist/jira/types.js +9 -0
  100. package/server/dist/jira-router.js +304 -0
  101. package/server/dist/mobile/mobile-auth.js +16 -0
  102. package/server/dist/project-registry.js +43 -1
  103. package/server/dist/project-router-chat.js +218 -0
  104. package/server/dist/project-router-helpers.js +275 -0
  105. package/server/dist/project-router-jobs.js +389 -0
  106. package/server/dist/project-router-settings.js +312 -0
  107. package/server/dist/project-router-setup.js +456 -0
  108. package/server/dist/project-router-spending.js +320 -0
  109. package/server/dist/project-router-terminals.js +312 -0
  110. package/server/dist/project-router-tickets.js +1815 -0
  111. package/server/dist/project-router.js +31 -3950
  112. package/server/dist/providers/claude-adapter.js +23 -0
  113. package/server/dist/providers/codex-adapter.js +6 -0
  114. package/server/dist/rails-router.js +12 -0
  115. package/server/dist/spawn-lifecycle.js +117 -0
  116. package/client/dist/assets/ActivityFeedPage-BpjXuX2H.js +0 -1
  117. package/client/dist/assets/AgentsPage-D-7fDbTc.js +0 -86
  118. package/client/dist/assets/CodePage-B6q6CiYJ.js +0 -2
  119. package/client/dist/assets/DocsDialog-D_dyF2h9.js +0 -11
  120. package/client/dist/assets/DocsPage-C9-Ru8wE.js +0 -11
  121. package/client/dist/assets/IntegrationsPage-3WWtx9hi.js +0 -3
  122. package/client/dist/assets/JobDetailPage-DgN-79s-.js +0 -16
  123. package/client/dist/assets/JobsPage-Du8_w1ob.js +0 -1
  124. package/client/dist/assets/code-AL1rVIMb.js +0 -1
  125. package/client/dist/assets/code-C0BKpkht.js +0 -1
  126. package/client/dist/assets/code-C0FTS3ew.js +0 -1
  127. package/client/dist/assets/code-CPcHxzxw.js +0 -1
  128. package/client/dist/assets/code-D3ryDniw.js +0 -1
  129. package/client/dist/assets/code-D3zVVQTj.js +0 -1
  130. package/client/dist/assets/code-PCmfS3dn.js +0 -1
  131. package/client/dist/assets/code-exI0G5Wd.js +0 -1
  132. package/client/dist/assets/index-D17R4Cjc.css +0 -2
  133. package/client/dist/assets/index-D9G_K4L-.js +0 -142
  134. package/client/dist/assets/integrations-D28q1kF6.js +0 -1
  135. package/client/dist/assets/setup--FMCsnQS.js +0 -1
  136. package/client/dist/assets/setup-B19ZpBNi.js +0 -1
  137. package/client/dist/assets/setup-BZPmkjSN.js +0 -1
  138. package/client/dist/assets/setup-BqYA02rS.js +0 -1
  139. package/client/dist/assets/setup-ChKQDHN9.js +0 -1
  140. package/client/dist/assets/setup-D2n9jMfM.js +0 -1
  141. package/client/dist/assets/setup-P3r6YP1D.js +0 -1
  142. package/client/dist/assets/setup-fnfEbwlv.js +0 -1
  143. package/client/dist/assets/specs-cKEh2LXt.js +0 -1
  144. /package/client/dist/assets/{abap-Bw6f2wDG.js → abap-s65oMlhi.js} +0 -0
  145. /package/client/dist/assets/{activity-BdrPln96.js → activity-BqqwnH_h.js} +0 -0
  146. /package/client/dist/assets/{activity-BEIp_Y1A.js → activity-C8qqEIoP.js} +0 -0
  147. /package/client/dist/assets/{activity-CpkRS8Sx.js → activity-CZVM4nlJ.js} +0 -0
  148. /package/client/dist/assets/{activity-DOUVEjJi.js → activity-Cyy07Tgo.js} +0 -0
  149. /package/client/dist/assets/{activity-DRwkql_y.js → activity-DlbWCa4y.js} +0 -0
  150. /package/client/dist/assets/{activity-DKCpESPt.js → activity-Dwq0heud.js} +0 -0
  151. /package/client/dist/assets/{activity-DcDQ7tjw.js → activity-qFTcMyW9.js} +0 -0
  152. /package/client/dist/assets/{addon-image-3WCl5Vhd.js → addon-image-CpF0L0jM.js} +0 -0
  153. /package/client/dist/assets/{addon-ligatures-C5OdliKs.js → addon-ligatures-hXysGZrA.js} +0 -0
  154. /package/client/dist/assets/{addon-webgl-BbX6pSjl.js → addon-webgl-Cn1slavz.js} +0 -0
  155. /package/client/dist/assets/{addspec-D33ocMxf.js → addspec-B1FTtI2a.js} +0 -0
  156. /package/client/dist/assets/{addspec-DFswZ0jK.js → addspec-BCT9vm_c.js} +0 -0
  157. /package/client/dist/assets/{addspec-DVZ15Jp8.js → addspec-DeDOztDr.js} +0 -0
  158. /package/client/dist/assets/{addspec-Fkv91Opc.js → addspec-DpRgmfmx.js} +0 -0
  159. /package/client/dist/assets/{addspec-BEeF5-zc.js → addspec-Dw-0Dg-4.js} +0 -0
  160. /package/client/dist/assets/{addspec-B5yl4Loj.js → addspec-rp496P_F.js} +0 -0
  161. /package/client/dist/assets/{addspec-DRE-jZv7.js → addspec-v8j6A7CD.js} +0 -0
  162. /package/client/dist/assets/{agents-DK-Dlc0i.js → agents-23iPejcA.js} +0 -0
  163. /package/client/dist/assets/{agents-Q6Ldfpxx.js → agents-BDx1RXcl.js} +0 -0
  164. /package/client/dist/assets/{agents-TeOSy-ax.js → agents-BFr3kUhK.js} +0 -0
  165. /package/client/dist/assets/{agents-Bm9rPqnt.js → agents-B_1L9xRg.js} +0 -0
  166. /package/client/dist/assets/{agents-1nCDWRmP.js → agents-BlPnx-mz.js} +0 -0
  167. /package/client/dist/assets/{agents-iTqjRajS.js → agents-DcxZHzNr.js} +0 -0
  168. /package/client/dist/assets/{agents-s87sMGzL.js → agents-G3shOewU.js} +0 -0
  169. /package/client/dist/assets/{agentstudio-B6Wb59E7.js → agentstudio-B-CMAQqy.js} +0 -0
  170. /package/client/dist/assets/{agentstudio-D3I62TLJ.js → agentstudio-Bk1eZcv4.js} +0 -0
  171. /package/client/dist/assets/{agentstudio-DuH9TogZ.js → agentstudio-ChxNuGAu.js} +0 -0
  172. /package/client/dist/assets/{agentstudio-Kw88_dUF.js → agentstudio-DNlme601.js} +0 -0
  173. /package/client/dist/assets/{agentstudio-BdidyBzZ.js → agentstudio-DpP9caEE.js} +0 -0
  174. /package/client/dist/assets/{agentstudio-BSnWLR63.js → agentstudio-Y3G0ddJ2.js} +0 -0
  175. /package/client/dist/assets/{agentstudio-BADhZ41e.js → agentstudio-kk9RB7Se.js} +0 -0
  176. /package/client/dist/assets/{aiedit-DJMny-D5.js → aiedit-5ETerMK1.js} +0 -0
  177. /package/client/dist/assets/{aiedit-D2ji6Qy0.js → aiedit-BBCrOpHq.js} +0 -0
  178. /package/client/dist/assets/{aiedit-DAhZTvtk.js → aiedit-BMtcGYNE.js} +0 -0
  179. /package/client/dist/assets/{aiedit-DvrcbwGv.js → aiedit-D9ddlgkM.js} +0 -0
  180. /package/client/dist/assets/{aiedit-WBSjT_C1.js → aiedit-De0SOH6S.js} +0 -0
  181. /package/client/dist/assets/{aiedit-BWxHGsYA.js → aiedit-DrfzQroF.js} +0 -0
  182. /package/client/dist/assets/{aiedit-DOcxERkU.js → aiedit-fMltW101.js} +0 -0
  183. /package/client/dist/assets/{analytics-C9Zc-rkM.js → analytics-BeTyviO8.js} +0 -0
  184. /package/client/dist/assets/{analytics-CrPCZRJ-.js → analytics-C4eEO260.js} +0 -0
  185. /package/client/dist/assets/{analytics-CYj0tfj7.js → analytics-C67cIA1b.js} +0 -0
  186. /package/client/dist/assets/{analytics-C6EzgtdE.js → analytics-CAguvW28.js} +0 -0
  187. /package/client/dist/assets/{analytics-CVx3YOc0.js → analytics-DBtt8Mgk.js} +0 -0
  188. /package/client/dist/assets/{analytics-CnY4kNG3.js → analytics-DUPtODxX.js} +0 -0
  189. /package/client/dist/assets/{analytics-BIdr0YfL.js → analytics-YIpQvPAc.js} +0 -0
  190. /package/client/dist/assets/{apex-Cw8_REBo.js → apex-BLUBIldB.js} +0 -0
  191. /package/client/dist/assets/{attachments-DYHGA2Dj.js → attachments-CCWasu-P.js} +0 -0
  192. /package/client/dist/assets/{attachments-Dd92KpUH.js → attachments-CHaDUfjB.js} +0 -0
  193. /package/client/dist/assets/{attachments-DzdU6DV6.js → attachments-CVSAbGNl.js} +0 -0
  194. /package/client/dist/assets/{attachments-Bcf6BG6V.js → attachments-Chg5poG1.js} +0 -0
  195. /package/client/dist/assets/{attachments-BW4L3l2L.js → attachments-DazTVJbH.js} +0 -0
  196. /package/client/dist/assets/{attachments-COcrGRFz.js → attachments-Dn-JImAK.js} +0 -0
  197. /package/client/dist/assets/{attachments-Bke8sCU4.js → attachments-LDA9kp2X.js} +0 -0
  198. /package/client/dist/assets/{azcli-Cz6HAoOw.js → azcli-DuWxh9mO.js} +0 -0
  199. /package/client/dist/assets/{bat-CcJ-xyqL.js → bat-UKoTejQm.js} +0 -0
  200. /package/client/dist/assets/{bicep-z1WDCKYz.js → bicep-4sTT4B3D.js} +0 -0
  201. /package/client/dist/assets/{browser-DGITz3fC.js → browser-BDd1dbFa.js} +0 -0
  202. /package/client/dist/assets/{browser-JsAIGCEW.js → browser-BWSgbfdX.js} +0 -0
  203. /package/client/dist/assets/{browser-M5-rbPlw.js → browser-D2Y_UAKA.js} +0 -0
  204. /package/client/dist/assets/{browser-BlYF4OOq.js → browser-DH9SGVfM.js} +0 -0
  205. /package/client/dist/assets/{browser-Bc-YdlVg.js → browser-DWOVYMlg.js} +0 -0
  206. /package/client/dist/assets/{browser-CT-ReZGt.js → browser-Dxc_VIRK.js} +0 -0
  207. /package/client/dist/assets/{browser-5ErDlJoR.js → browser-lTQwcDCI.js} +0 -0
  208. /package/client/dist/assets/{cameligo-BRewOpfa.js → cameligo-CAAryRYO.js} +0 -0
  209. /package/client/dist/assets/{chat-DwUm6W9z.js → chat-BO9MvVID.js} +0 -0
  210. /package/client/dist/assets/{chat-BEGuC03z.js → chat-CPgmgZOj.js} +0 -0
  211. /package/client/dist/assets/{chat-CboQguCi.js → chat-CUrG1eVg.js} +0 -0
  212. /package/client/dist/assets/{chat-DRCa9pOt.js → chat-CvOOKB2s.js} +0 -0
  213. /package/client/dist/assets/{chat-BEW60P_u.js → chat-DIh3hr6y.js} +0 -0
  214. /package/client/dist/assets/{chat-yoXwguQu.js → chat-UVVZqA57.js} +0 -0
  215. /package/client/dist/assets/{chat-BQNMD0PL.js → chat-mPn3UlMl.js} +0 -0
  216. /package/client/dist/assets/{clojure-DBjRWN6g.js → clojure-BlMERO1w.js} +0 -0
  217. /package/client/dist/assets/{clsx-DnqN-uhr.js → clsx-CnH-HMk3.js} +0 -0
  218. /package/client/dist/assets/{coffee-Cfk_XHGR.js → coffee-Cj8D-Wl1.js} +0 -0
  219. /package/client/dist/assets/{commands-sqrqsxyE.js → commands-B-MVT-2F.js} +0 -0
  220. /package/client/dist/assets/{commands-UD1NzmwX.js → commands-B0yFTp7e.js} +0 -0
  221. /package/client/dist/assets/{commands-DLrvnPNg.js → commands-BR1kDkHQ.js} +0 -0
  222. /package/client/dist/assets/{commands-CJxCry-o.js → commands-Cb21pDlG.js} +0 -0
  223. /package/client/dist/assets/{commands-CfgY-_of.js → commands-DWgp-8W1.js} +0 -0
  224. /package/client/dist/assets/{commands-B772IyDa.js → commands-ddsl1V91.js} +0 -0
  225. /package/client/dist/assets/{commands-BDDp6xFG.js → commands-t4frzhB0.js} +0 -0
  226. /package/client/dist/assets/{common-Dmm1GhdD.js → common-5ilvMOcH.js} +0 -0
  227. /package/client/dist/assets/{common-DltqHaAe.js → common-B4sqsKp7.js} +0 -0
  228. /package/client/dist/assets/{common-GbpxfPG8.js → common-BKpVwUIf.js} +0 -0
  229. /package/client/dist/assets/{common-DeDELLZJ.js → common-BzEC3kJU.js} +0 -0
  230. /package/client/dist/assets/{common-DnjcgkPH.js → common-CALKUpYm.js} +0 -0
  231. /package/client/dist/assets/{common-Dard9UNH.js → common-CTEbWVZS.js} +0 -0
  232. /package/client/dist/assets/{common-DCr6VzJ7.js → common-DQiza2Xp.js} +0 -0
  233. /package/client/dist/assets/{cpp-BVob6BaP.js → cpp-BPfKnaj_.js} +0 -0
  234. /package/client/dist/assets/{csharp-C4fbRuOu.js → csharp-gX-x5uD6.js} +0 -0
  235. /package/client/dist/assets/{csp-DthFP_vT.js → csp-DKGVt8SM.js} +0 -0
  236. /package/client/dist/assets/{css-CGMH0hcW.js → css-CPMdnAVq.js} +0 -0
  237. /package/client/dist/assets/{cypher-Pnf68BRV.js → cypher-ClMDrj9S.js} +0 -0
  238. /package/client/dist/assets/{dart-PMMOtxZX.js → dart-C4zbzpVv.js} +0 -0
  239. /package/client/dist/assets/{dashboard-BZBADHSj.js → dashboard--Y6yzMlf.js} +0 -0
  240. /package/client/dist/assets/{dashboard-I19DXBxw.js → dashboard--a4-6oYE.js} +0 -0
  241. /package/client/dist/assets/{dashboard-CB6Le1yN.js → dashboard-BiJ3CDTG.js} +0 -0
  242. /package/client/dist/assets/{dashboard-B4ixDVk8.js → dashboard-CiXjk63Z.js} +0 -0
  243. /package/client/dist/assets/{dashboard-C1MfeUHs.js → dashboard-Cx5VjCea.js} +0 -0
  244. /package/client/dist/assets/{dashboard-C7SK6xu5.js → dashboard-D7jg25XR.js} +0 -0
  245. /package/client/dist/assets/{dashboard-CoTpMOBM.js → dashboard-DpGYK2s1.js} +0 -0
  246. /package/client/dist/assets/{dockerfile-di1nsJCc.js → dockerfile-D9xw73D1.js} +0 -0
  247. /package/client/dist/assets/{ecl-D_WVcB5M.js → ecl-gqO8tIR9.js} +0 -0
  248. /package/client/dist/assets/{editor.api2-XLGzZfbc.js → editor.api2-BPnIxMjz.js} +0 -0
  249. /package/client/dist/assets/{elixir-OAdJEMOn.js → elixir-DSAhVF3_.js} +0 -0
  250. /package/client/dist/assets/{explore-D2EFgt8J.js → explore-BE5UmlbD.js} +0 -0
  251. /package/client/dist/assets/{explore-BV5Xxlsn.js → explore-BmTaI8dX.js} +0 -0
  252. /package/client/dist/assets/{explore-A8Ltoblq.js → explore-CCwkqoWq.js} +0 -0
  253. /package/client/dist/assets/{explore-4mFpnrKU.js → explore-CMdEoPDx.js} +0 -0
  254. /package/client/dist/assets/{explore-C3FSE42C.js → explore-CtdCL4QU.js} +0 -0
  255. /package/client/dist/assets/{explore-B9A3iN2W.js → explore-DHjxSkqQ.js} +0 -0
  256. /package/client/dist/assets/{explore-BrBJvfjP.js → explore-__BeALjE.js} +0 -0
  257. /package/client/dist/assets/{flow9-D3QEZjgn.js → flow9-DeQCSPOd.js} +0 -0
  258. /package/client/dist/assets/{format-command-CwGuwzGA.js → format-command-2VNoNnMv.js} +0 -0
  259. /package/client/dist/assets/{fsharp-BF0k_8N8.js → fsharp-CEfaXL-S.js} +0 -0
  260. /package/client/dist/assets/{go-BAQO5Jsz.js → go-Xp1OkZCh.js} +0 -0
  261. /package/client/dist/assets/{graphql-hdFVFkiV.js → graphql-BwRXrUwe.js} +0 -0
  262. /package/client/dist/assets/{hcl-DWnl1o-X.js → hcl-u06DtVFk.js} +0 -0
  263. /package/client/dist/assets/{ini-CB-6OVu3.js → ini-AmeIpFND.js} +0 -0
  264. /package/client/dist/assets/{java-d1CmfiHX.js → java-CyDbRQjX.js} +0 -0
  265. /package/client/dist/assets/{jobs-DPPT6bV6.js → jobs-8viuHLDV.js} +0 -0
  266. /package/client/dist/assets/{jobs-3j3_npyo.js → jobs-AW2eB5D-.js} +0 -0
  267. /package/client/dist/assets/{jobs-2N3RXDAM.js → jobs-BSm89DL5.js} +0 -0
  268. /package/client/dist/assets/{jobs-BqEbCCxD.js → jobs-BZ3sQHjZ.js} +0 -0
  269. /package/client/dist/assets/{jobs-cHYInoau.js → jobs-Bd8AdOTb.js} +0 -0
  270. /package/client/dist/assets/{jobs-DRzjWI9u.js → jobs-CRtsq_u0.js} +0 -0
  271. /package/client/dist/assets/{jobs-2f6Hdc72.js → jobs-CSRwFQ6K.js} +0 -0
  272. /package/client/dist/assets/{jobs-vGfzIDQa.js → jobs-CbEl7WMI.js} +0 -0
  273. /package/client/dist/assets/{julia-Bgv08lKa.js → julia-BqialFRG.js} +0 -0
  274. /package/client/dist/assets/{kotlin-u98kaVTf.js → kotlin-Dzz8TWAt.js} +0 -0
  275. /package/client/dist/assets/{less-CjYwpgg5.js → less-DHRJD3TR.js} +0 -0
  276. /package/client/dist/assets/{lexon-YTjaAFBB.js → lexon-5Y3QgTmT.js} +0 -0
  277. /package/client/dist/assets/{lua-BzmkWv27.js → lua-sKvhfPn5.js} +0 -0
  278. /package/client/dist/assets/{m3-CFwk9fw0.js → m3-DWDVwkFG.js} +0 -0
  279. /package/client/dist/assets/{markdown-CR5iMpSZ.js → markdown-CD_aSBxW.js} +0 -0
  280. /package/client/dist/assets/{mips-CcEalc17.js → mips-687T03hg.js} +0 -0
  281. /package/client/dist/assets/{msdax-BQbkawnr.js → msdax-C1St-dIV.js} +0 -0
  282. /package/client/dist/assets/{mysql-GTlaaW_P.js → mysql-BG7r8oBS.js} +0 -0
  283. /package/client/dist/assets/{nav-C2YXcbZS.js → nav-B05EYB0b.js} +0 -0
  284. /package/client/dist/assets/{nav-D2bOGSEg.js → nav-BNGCq-0y.js} +0 -0
  285. /package/client/dist/assets/{nav-BEL3MTwK.js → nav-BRInPX8a.js} +0 -0
  286. /package/client/dist/assets/{nav-CtYwmMgu.js → nav-Bf87DRHD.js} +0 -0
  287. /package/client/dist/assets/{nav-iH1V5j6o.js → nav-BkVzzFpc.js} +0 -0
  288. /package/client/dist/assets/{nav-0fwkrgHt.js → nav-BzFLtS1W.js} +0 -0
  289. /package/client/dist/assets/{nav-ClzOE4mA.js → nav-DBDbQOYn.js} +0 -0
  290. /package/client/dist/assets/{nav-B_G-TJDW.js → nav-X9sVtUWC.js} +0 -0
  291. /package/client/dist/assets/{objective-c-Byu1T5if.js → objective-c-Ds1-m05L.js} +0 -0
  292. /package/client/dist/assets/{pascal-BrfzBfRm.js → pascal-BKK9FpIi.js} +0 -0
  293. /package/client/dist/assets/{pascaligo-BXXKFUeo.js → pascaligo-SRS3nwtO.js} +0 -0
  294. /package/client/dist/assets/{perl-B3OikKq-.js → perl-B2hTOlrF.js} +0 -0
  295. /package/client/dist/assets/{pgsql-CTsa0Acc.js → pgsql-DIQJYNpL.js} +0 -0
  296. /package/client/dist/assets/{php-DiQh3FUW.js → php-BEaVe8X2.js} +0 -0
  297. /package/client/dist/assets/{pla-92uH8Fzm.js → pla-oPLHpZ-Q.js} +0 -0
  298. /package/client/dist/assets/{postiats-BbeWkKUr.js → postiats-D_vzrAzD.js} +0 -0
  299. /package/client/dist/assets/{powerquery-DgDMzpsm.js → powerquery-BKG6w-FH.js} +0 -0
  300. /package/client/dist/assets/{powershell-BfdUUzaG.js → powershell-B3dLhDt4.js} +0 -0
  301. /package/client/dist/assets/{protobuf-BojW2ftW.js → protobuf-DC8SGjcl.js} +0 -0
  302. /package/client/dist/assets/{pug-BxqTg3IU.js → pug-D5E-4fI0.js} +0 -0
  303. /package/client/dist/assets/{qsharp-BX_A-MW9.js → qsharp-6vJAWv0x.js} +0 -0
  304. /package/client/dist/assets/{r-D9BMnxvJ.js → r-CDwsEcbM.js} +0 -0
  305. /package/client/dist/assets/{redis-5cJqEQJJ.js → redis-CuQbbESS.js} +0 -0
  306. /package/client/dist/assets/{redshift-d8BBqiwb.js → redshift-B9e1k-qI.js} +0 -0
  307. /package/client/dist/assets/{restructuredtext-C8a6yIcZ.js → restructuredtext-BiJ5IwaU.js} +0 -0
  308. /package/client/dist/assets/{ruby-egeh-6KX.js → ruby-B0UAHY9b.js} +0 -0
  309. /package/client/dist/assets/{rust-a3r9IInB.js → rust-Dg_spmFr.js} +0 -0
  310. /package/client/dist/assets/{sb-y8iRIDei.js → sb-DjU66I8Q.js} +0 -0
  311. /package/client/dist/assets/{scala-BPDK2AmK.js → scala-qvStIdfG.js} +0 -0
  312. /package/client/dist/assets/{scheme-BIWUEoOs.js → scheme-FstEk5Rh.js} +0 -0
  313. /package/client/dist/assets/{scss-CA-PSzwg.js → scss-w0U3rQLK.js} +0 -0
  314. /package/client/dist/assets/{settings-CTcwN9RE.js → settings-5tzo0Rn3.js} +0 -0
  315. /package/client/dist/assets/{settings-D_dujJZI.js → settings-BDAW3trC.js} +0 -0
  316. /package/client/dist/assets/{settings-Bg0A3zoS.js → settings-BEWv3VEu.js} +0 -0
  317. /package/client/dist/assets/{settings-BgPqg2nv.js → settings-BORg56um.js} +0 -0
  318. /package/client/dist/assets/{settings-BSze3_9q.js → settings-D3LurcR5.js} +0 -0
  319. /package/client/dist/assets/{settings-CSJ0ahZ8.js → settings-DcqWIEM6.js} +0 -0
  320. /package/client/dist/assets/{settings-DYIV89nV.js → settings-Dfz8QbZS.js} +0 -0
  321. /package/client/dist/assets/{settings-DDcfx_ca.js → settings-yMubjqYw.js} +0 -0
  322. /package/client/dist/assets/{shell--LiT1Bja.js → shell-DJ78wREd.js} +0 -0
  323. /package/client/dist/assets/{solidity-DdqZccZg.js → solidity-1aGIVsdX.js} +0 -0
  324. /package/client/dist/assets/{sophia-S6-YxNG_.js → sophia-40LqcGjB.js} +0 -0
  325. /package/client/dist/assets/{sparql-BSf5kMp2.js → sparql-Cz5dqG_g.js} +0 -0
  326. /package/client/dist/assets/{sql-D7KgjR8G.js → sql-64f62Ni4.js} +0 -0
  327. /package/client/dist/assets/{st-BnoDa-Ml.js → st-gJe2yG8J.js} +0 -0
  328. /package/client/dist/assets/{swift-DEUHTkUX.js → swift-C6ME22mv.js} +0 -0
  329. /package/client/dist/assets/{systemverilog-Tqb_KPnW.js → systemverilog-CEWz259w.js} +0 -0
  330. /package/client/dist/assets/{tcl-BmBFS2qq.js → tcl-CcLVIi3m.js} +0 -0
  331. /package/client/dist/assets/{terminal-Bje4ziIa.js → terminal-BYtreaaF.js} +0 -0
  332. /package/client/dist/assets/{terminal-CSONJOex.js → terminal-C0xx0SjA.js} +0 -0
  333. /package/client/dist/assets/{terminal-DeWzh6ys.js → terminal-CPpK58RC.js} +0 -0
  334. /package/client/dist/assets/{terminal-C2WYcFHF.js → terminal-CdxkpafL.js} +0 -0
  335. /package/client/dist/assets/{terminal-DEqzGtcr.js → terminal-Ciia0wh2.js} +0 -0
  336. /package/client/dist/assets/{terminal-80yDMgMF.js → terminal-DHIkiWcs.js} +0 -0
  337. /package/client/dist/assets/{terminal-lkZYR4wJ.js → terminal-DY42QANg.js} +0 -0
  338. /package/client/dist/assets/{terminal-YOlsJCQj.js → terminal-DoxtVdma.js} +0 -0
  339. /package/client/dist/assets/{tickets-DYvafSaY.js → tickets-0rM0lIXd.js} +0 -0
  340. /package/client/dist/assets/{tickets-DNOANUXr.js → tickets-1UIGf_oA.js} +0 -0
  341. /package/client/dist/assets/{tickets-DlpC_iTg.js → tickets-9kdPXInd.js} +0 -0
  342. /package/client/dist/assets/{tickets-CF2PYelu.js → tickets-C6pwZwt4.js} +0 -0
  343. /package/client/dist/assets/{tickets-CB7N30gm.js → tickets-DAjtxAVb.js} +0 -0
  344. /package/client/dist/assets/{tickets-DU1aqsbr.js → tickets-DNmXcAwu.js} +0 -0
  345. /package/client/dist/assets/{tickets-clefmXLv.js → tickets-n23kDqJT.js} +0 -0
  346. /package/client/dist/assets/{tickets-DucYgtdl.js → tickets-tGx5AR5b.js} +0 -0
  347. /package/client/dist/assets/{twig-BQV8igWC.js → twig-DvsO-WjW.js} +0 -0
  348. /package/client/dist/assets/{typespec-DlFroUGY.js → typespec-Brkt3IAA.js} +0 -0
  349. /package/client/dist/assets/{vb-BlrJpIMX.js → vb-r121Uzxt.js} +0 -0
  350. /package/client/dist/assets/{wgsl-BWgIc6FZ.js → wgsl-BRX8uYh4.js} +0 -0
@@ -0,0 +1,456 @@
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.registerSetupRoutes = registerSetupRoutes;
7
+ // Domain routes extracted from project-router.ts (setup).
8
+ // Registered on the shared router by createProjectRouter — behaviour-preserving.
9
+ const fs_1 = __importDefault(require("fs"));
10
+ const path_1 = __importDefault(require("path"));
11
+ const ids_1 = require("./ids");
12
+ const db_1 = require("./db");
13
+ const desktop_db_1 = require("./desktop-db");
14
+ const queue_manager_1 = require("./queue-manager");
15
+ const command_resolver_1 = require("./command-resolver");
16
+ const changes_reader_1 = require("./changes-reader");
17
+ const project_router_helpers_1 = require("./project-router-helpers");
18
+ function registerSetupRoutes(deps) {
19
+ const { router, registry, ctx, ticketPath } = deps;
20
+ // ─── Install-config route ─────────────────────────────────────────────────────
21
+ router.post('/:projectId/setup/install-config', (req, res) => {
22
+ const { project } = ctx(req);
23
+ const config = req.body ?? {};
24
+ if (typeof config !== 'object' || Array.isArray(config)) {
25
+ res.status(400).json({ error: 'Request body must be a config object' });
26
+ return;
27
+ }
28
+ const configDir = path_1.default.join(project.path, '.specrails');
29
+ const configPath = path_1.default.join(configDir, 'install-config.yaml');
30
+ try {
31
+ fs_1.default.mkdirSync(configDir, { recursive: true });
32
+ const yaml = (0, project_router_helpers_1.serializeInstallConfigYaml)(config);
33
+ fs_1.default.writeFileSync(configPath, yaml, 'utf-8');
34
+ res.json({ ok: true, path: configPath });
35
+ }
36
+ catch (err) {
37
+ res.status(500).json({ error: `Failed to write install-config.yaml: ${err}` });
38
+ }
39
+ });
40
+ // ─── Enrich routes (v3) + Setup aliases (v1/v2 backward compat) ──────────────
41
+ router.post('/:projectId/setup/install', (req, res) => {
42
+ const { project, setupManager } = ctx(req);
43
+ if (setupManager.isInstalling(project.id)) {
44
+ res.status(409).json({ error: 'Install already in progress' });
45
+ return;
46
+ }
47
+ res.status(202).json({ ok: true });
48
+ setupManager.startInstall(project.id, project.path);
49
+ });
50
+ router.post('/:projectId/enrich/start', (req, res) => {
51
+ const { project, setupManager } = ctx(req);
52
+ if (setupManager.isEnriching(project.id)) {
53
+ res.status(409).json({ error: 'Enrich already in progress' });
54
+ return;
55
+ }
56
+ res.status(202).json({ ok: true });
57
+ setupManager.startEnrich(project.id, project.path, project.provider);
58
+ });
59
+ // Legacy alias: /setup/start → /enrich/start
60
+ router.post('/:projectId/setup/start', (req, res) => {
61
+ const { project, setupManager } = ctx(req);
62
+ if (setupManager.isEnriching(project.id)) {
63
+ res.status(409).json({ error: 'Setup already in progress' });
64
+ return;
65
+ }
66
+ res.status(202).json({ ok: true });
67
+ setupManager.startEnrich(project.id, project.path, project.provider);
68
+ });
69
+ router.post('/:projectId/enrich/message', (req, res) => {
70
+ const { project, setupManager } = ctx(req);
71
+ const { sessionId, message } = req.body ?? {};
72
+ if (!sessionId || typeof sessionId !== 'string') {
73
+ res.status(400).json({ error: 'sessionId is required' });
74
+ return;
75
+ }
76
+ if (!message || typeof message !== 'string' || !message.trim()) {
77
+ res.status(400).json({ error: 'message is required' });
78
+ return;
79
+ }
80
+ if (setupManager.isEnriching(project.id)) {
81
+ res.status(409).json({ error: 'Enrich already in progress' });
82
+ return;
83
+ }
84
+ res.status(202).json({ ok: true });
85
+ setupManager.resumeEnrich(project.id, project.path, sessionId, message.trim(), project.provider);
86
+ });
87
+ // Legacy alias: /setup/message → /enrich/message
88
+ router.post('/:projectId/setup/message', (req, res) => {
89
+ const { project, setupManager } = ctx(req);
90
+ const { sessionId, message } = req.body ?? {};
91
+ if (!sessionId || typeof sessionId !== 'string') {
92
+ res.status(400).json({ error: 'sessionId is required' });
93
+ return;
94
+ }
95
+ if (!message || typeof message !== 'string' || !message.trim()) {
96
+ res.status(400).json({ error: 'message is required' });
97
+ return;
98
+ }
99
+ if (setupManager.isEnriching(project.id)) {
100
+ res.status(409).json({ error: 'Setup already in progress' });
101
+ return;
102
+ }
103
+ res.status(202).json({ ok: true });
104
+ setupManager.resumeEnrich(project.id, project.path, sessionId, message.trim(), project.provider);
105
+ });
106
+ router.get('/:projectId/setup/checkpoints', (req, res) => {
107
+ const { project, setupManager } = ctx(req);
108
+ const checkpoints = setupManager.getCheckpointStatus(project.id, project.path);
109
+ const savedSessionId = (0, desktop_db_1.getProjectSetupSession)(registry.desktopDb, project.id);
110
+ res.json({
111
+ checkpoints,
112
+ isInstalling: setupManager.isInstalling(project.id),
113
+ isSettingUp: setupManager.isEnriching(project.id),
114
+ isEnriching: setupManager.isEnriching(project.id),
115
+ tier: setupManager.getInstallTier(project.id) ?? null,
116
+ savedSessionId: savedSessionId ?? null,
117
+ logLines: setupManager.getInstallLog(project.id),
118
+ summary: setupManager.getSummary(project.path),
119
+ });
120
+ });
121
+ router.post('/:projectId/setup/abort', (req, res) => {
122
+ const { project, setupManager } = ctx(req);
123
+ setupManager.abort(project.id);
124
+ res.json({ ok: true });
125
+ });
126
+ // ─── Proposal routes ──────────────────────────────────────────────────────
127
+ router.get('/:projectId/propose', (req, res) => {
128
+ const limit = Math.min(parseInt(String(req.query.limit ?? '20'), 10) || 20, 100);
129
+ const offset = parseInt(String(req.query.offset ?? '0'), 10) || 0;
130
+ const result = (0, db_1.listProposals)(ctx(req).db, { limit, offset });
131
+ res.json(result);
132
+ });
133
+ router.post('/:projectId/propose', async (req, res) => {
134
+ const { idea } = req.body ?? {};
135
+ if (!idea || typeof idea !== 'string' || !idea.trim()) {
136
+ res.status(400).json({ error: 'idea is required' });
137
+ return;
138
+ }
139
+ // Pre-check: does the propose-feature command exist in this project?
140
+ const testCmd = `/specrails:propose-feature test`;
141
+ const resolved = (0, command_resolver_1.resolveCommand)(testCmd, ctx(req).project.path);
142
+ if (resolved === testCmd) {
143
+ res.status(400).json({ error: 'This project does not have the /specrails:propose-feature command installed. Run "npx specrails-core@latest" to update.' });
144
+ return;
145
+ }
146
+ const id = (0, ids_1.newId)();
147
+ (0, db_1.createProposal)(ctx(req).db, { id, idea: idea.trim() });
148
+ res.status(202).json({ proposalId: id });
149
+ ctx(req).proposalManager.startExploration(id, idea.trim()).catch((err) => {
150
+ console.error('[project-router] proposal startExploration error:', err);
151
+ });
152
+ });
153
+ router.get('/:projectId/propose/:id', (req, res) => {
154
+ const proposal = (0, db_1.getProposal)(ctx(req).db, req.params.id);
155
+ if (!proposal) {
156
+ res.status(404).json({ error: 'Proposal not found' });
157
+ return;
158
+ }
159
+ res.json({ proposal });
160
+ });
161
+ router.post('/:projectId/propose/:id/refine', async (req, res) => {
162
+ const proposal = (0, db_1.getProposal)(ctx(req).db, req.params.id);
163
+ if (!proposal) {
164
+ res.status(404).json({ error: 'Proposal not found' });
165
+ return;
166
+ }
167
+ const { feedback } = req.body ?? {};
168
+ if (!feedback || typeof feedback !== 'string' || !feedback.trim()) {
169
+ res.status(400).json({ error: 'feedback is required' });
170
+ return;
171
+ }
172
+ if (ctx(req).proposalManager.isActive(req.params.id)) {
173
+ res.status(409).json({ error: 'PROPOSAL_BUSY' });
174
+ return;
175
+ }
176
+ if (proposal.status !== 'review') {
177
+ res.status(409).json({ error: 'Proposal is not in review state' });
178
+ return;
179
+ }
180
+ res.status(202).json({ ok: true });
181
+ ctx(req).proposalManager.sendRefinement(req.params.id, feedback.trim()).catch((err) => {
182
+ console.error('[project-router] proposal sendRefinement error:', err);
183
+ });
184
+ });
185
+ router.post('/:projectId/propose/:id/create-issue', async (req, res) => {
186
+ const proposal = (0, db_1.getProposal)(ctx(req).db, req.params.id);
187
+ if (!proposal) {
188
+ res.status(404).json({ error: 'Proposal not found' });
189
+ return;
190
+ }
191
+ if (ctx(req).proposalManager.isActive(req.params.id)) {
192
+ res.status(409).json({ error: 'PROPOSAL_BUSY' });
193
+ return;
194
+ }
195
+ if (proposal.status !== 'review') {
196
+ res.status(409).json({ error: 'Proposal is not in review state' });
197
+ return;
198
+ }
199
+ res.status(202).json({ ok: true });
200
+ ctx(req).proposalManager.createIssue(req.params.id).catch((err) => {
201
+ console.error('[project-router] proposal createIssue error:', err);
202
+ });
203
+ });
204
+ router.delete('/:projectId/propose/:id', (req, res) => {
205
+ const proposal = (0, db_1.getProposal)(ctx(req).db, req.params.id);
206
+ if (!proposal) {
207
+ res.status(404).json({ error: 'Proposal not found' });
208
+ return;
209
+ }
210
+ ctx(req).proposalManager.cancel(req.params.id);
211
+ res.json({ ok: true });
212
+ });
213
+ // ─── Feature Funnel ─────────────────────────────────────────────────────────
214
+ router.get('/:projectId/changes', (req, res) => {
215
+ const { project, queueManager } = ctx(req);
216
+ const activeCommands = queueManager.getJobs()
217
+ .filter((j) => j.status === 'running' || j.status === 'queued')
218
+ .map((j) => j.command);
219
+ const changes = (0, changes_reader_1.readChanges)(project.path, activeCommands);
220
+ res.json({ changes });
221
+ });
222
+ // ─── Change Artifact Browser ─────────────────────────────────────────────────
223
+ const ALLOWED_ARTIFACTS = new Set(['proposal.md', 'design.md', 'tasks.md', 'delta-spec.md', 'context-bundle.md']);
224
+ router.get('/:projectId/changes/:changeId/artifacts/:artifact', (req, res) => {
225
+ const changeId = req.params.changeId;
226
+ const artifact = req.params.artifact;
227
+ if (!ALLOWED_ARTIFACTS.has(artifact)) {
228
+ res.status(400).json({ error: 'Invalid artifact name' });
229
+ return;
230
+ }
231
+ // Sanitize changeId to prevent path traversal
232
+ if (!/^[\w-]+$/.test(changeId)) {
233
+ res.status(400).json({ error: 'Invalid change ID' });
234
+ return;
235
+ }
236
+ const { project } = ctx(req);
237
+ const changesRoot = path_1.default.join(project.path, 'openspec', 'changes');
238
+ // Check active dir first, then archive
239
+ let filePath = path_1.default.join(changesRoot, changeId, artifact);
240
+ if (!fs_1.default.existsSync(filePath)) {
241
+ filePath = path_1.default.join(changesRoot, 'archive', changeId, artifact);
242
+ }
243
+ if (!fs_1.default.existsSync(filePath)) {
244
+ res.status(404).json({ error: 'Artifact not found' });
245
+ return;
246
+ }
247
+ try {
248
+ const content = fs_1.default.readFileSync(filePath, 'utf-8');
249
+ res.json({ content, artifact, changeId });
250
+ }
251
+ catch {
252
+ res.status(500).json({ error: 'Failed to read artifact' });
253
+ }
254
+ });
255
+ // ─── Spec Launcher ───────────────────────────────────────────────────────────
256
+ router.post('/:projectId/spec-launcher/start', (req, res) => {
257
+ const { description } = req.body ?? {};
258
+ if (!description || typeof description !== 'string' || !description.trim()) {
259
+ res.status(400).json({ error: 'description is required' });
260
+ return;
261
+ }
262
+ const launchId = (0, ids_1.newId)();
263
+ res.status(202).json({ launchId });
264
+ ctx(req).specLauncherManager.launch(launchId, description.trim()).catch((err) => {
265
+ console.error('[project-router] spec-launcher error:', err);
266
+ });
267
+ });
268
+ router.delete('/:projectId/spec-launcher/:launchId', (req, res) => {
269
+ const { specLauncherManager } = ctx(req);
270
+ if (!specLauncherManager.isActive(req.params.launchId)) {
271
+ res.status(404).json({ error: 'No active launch with that ID' });
272
+ return;
273
+ }
274
+ specLauncherManager.cancel(req.params.launchId);
275
+ res.json({ ok: true });
276
+ });
277
+ // ─── Job Templates ────────────────────────────────────────────────────────
278
+ function templateToPublic(row) {
279
+ if (!row)
280
+ return null;
281
+ return {
282
+ id: row.id,
283
+ name: row.name,
284
+ description: row.description,
285
+ commands: JSON.parse(row.commands),
286
+ created_at: row.created_at,
287
+ updated_at: row.updated_at,
288
+ };
289
+ }
290
+ router.get('/:projectId/templates', (req, res) => {
291
+ const rows = (0, db_1.listTemplates)(ctx(req).db);
292
+ const templates = rows.map((r) => templateToPublic(r));
293
+ res.json({ templates });
294
+ });
295
+ router.post('/:projectId/templates', (req, res) => {
296
+ const { name, description, commands } = req.body ?? {};
297
+ if (!name || typeof name !== 'string' || !name.trim()) {
298
+ res.status(400).json({ error: 'name is required' });
299
+ return;
300
+ }
301
+ if (!Array.isArray(commands) || commands.length === 0) {
302
+ res.status(400).json({ error: 'commands must be a non-empty array' });
303
+ return;
304
+ }
305
+ if (commands.some((c) => typeof c !== 'string' || !String(c).trim())) {
306
+ res.status(400).json({ error: 'each command must be a non-empty string' });
307
+ return;
308
+ }
309
+ const id = (0, ids_1.newId)();
310
+ try {
311
+ (0, db_1.createTemplate)(ctx(req).db, {
312
+ id,
313
+ name: name.trim(),
314
+ description: description && typeof description === 'string' ? description.trim() : undefined,
315
+ commands: commands.map((c) => c.trim()),
316
+ });
317
+ }
318
+ catch (err) {
319
+ const msg = err instanceof Error ? err.message : '';
320
+ if (msg.includes('UNIQUE constraint failed')) {
321
+ res.status(409).json({ error: 'A template with that name already exists' });
322
+ return;
323
+ }
324
+ console.error('[project-router] create template error:', err);
325
+ res.status(500).json({ error: 'Internal server error' });
326
+ return;
327
+ }
328
+ const created = templateToPublic((0, db_1.getTemplate)(ctx(req).db, id));
329
+ res.status(201).json({ template: created });
330
+ });
331
+ router.get('/:projectId/templates/:templateId', (req, res) => {
332
+ const row = (0, db_1.getTemplate)(ctx(req).db, req.params.templateId);
333
+ if (!row) {
334
+ res.status(404).json({ error: 'Template not found' });
335
+ return;
336
+ }
337
+ res.json({ template: templateToPublic(row) });
338
+ });
339
+ router.patch('/:projectId/templates/:templateId', (req, res) => {
340
+ const { db } = ctx(req);
341
+ const templateId = req.params.templateId;
342
+ const row = (0, db_1.getTemplate)(db, templateId);
343
+ if (!row) {
344
+ res.status(404).json({ error: 'Template not found' });
345
+ return;
346
+ }
347
+ const { name, description, commands } = req.body ?? {};
348
+ const patch = {};
349
+ if (name !== undefined) {
350
+ if (typeof name !== 'string' || !name.trim()) {
351
+ res.status(400).json({ error: 'name must be a non-empty string' });
352
+ return;
353
+ }
354
+ patch.name = name.trim();
355
+ }
356
+ if (description !== undefined) {
357
+ patch.description = description === null ? null : String(description).trim() || null;
358
+ }
359
+ if (commands !== undefined) {
360
+ if (!Array.isArray(commands) || commands.length === 0) {
361
+ res.status(400).json({ error: 'commands must be a non-empty array' });
362
+ return;
363
+ }
364
+ if (commands.some((c) => typeof c !== 'string' || !String(c).trim())) {
365
+ res.status(400).json({ error: 'each command must be a non-empty string' });
366
+ return;
367
+ }
368
+ patch.commands = commands.map((c) => c.trim());
369
+ }
370
+ try {
371
+ (0, db_1.updateTemplate)(db, templateId, patch);
372
+ }
373
+ catch (err) {
374
+ const msg = err instanceof Error ? err.message : '';
375
+ if (msg.includes('UNIQUE constraint failed')) {
376
+ res.status(409).json({ error: 'A template with that name already exists' });
377
+ return;
378
+ }
379
+ console.error('[project-router] update template error:', err);
380
+ res.status(500).json({ error: 'Internal server error' });
381
+ return;
382
+ }
383
+ const updated = templateToPublic((0, db_1.getTemplate)(db, templateId));
384
+ res.json({ ok: true, template: updated });
385
+ });
386
+ router.delete('/:projectId/templates/:templateId', (req, res) => {
387
+ const { db } = ctx(req);
388
+ const row = (0, db_1.getTemplate)(db, req.params.templateId);
389
+ if (!row) {
390
+ res.status(404).json({ error: 'Template not found' });
391
+ return;
392
+ }
393
+ (0, db_1.deleteTemplate)(db, req.params.templateId);
394
+ res.json({ ok: true });
395
+ });
396
+ router.post('/:projectId/templates/:templateId/run', (req, res) => {
397
+ const { db, queueManager } = ctx(req);
398
+ const row = (0, db_1.getTemplate)(db, req.params.templateId);
399
+ if (!row) {
400
+ res.status(404).json({ error: 'Template not found' });
401
+ return;
402
+ }
403
+ const commands = JSON.parse(row.commands);
404
+ const chain = req.body?.chain !== false; // default: chain jobs as pipeline
405
+ const jobIds = [];
406
+ try {
407
+ const pipelineId = chain && commands.length > 1 ? (0, ids_1.newId)() : undefined;
408
+ let prevJobId = null;
409
+ for (const command of commands) {
410
+ const job = queueManager.enqueue(command, 'normal', {
411
+ dependsOnJobId: chain ? (prevJobId ?? undefined) : undefined,
412
+ pipelineId,
413
+ });
414
+ jobIds.push(job.id);
415
+ prevJobId = job.id;
416
+ }
417
+ }
418
+ catch (err) {
419
+ if (err instanceof queue_manager_1.ClaudeNotFoundError) {
420
+ res.status(400).json({ error: err.message });
421
+ return;
422
+ }
423
+ console.error('[project-router] template run error:', err);
424
+ res.status(500).json({ error: 'Internal server error' });
425
+ return;
426
+ }
427
+ res.status(202).json({ ok: true, jobIds, templateId: row.id, templateName: row.name });
428
+ });
429
+ // ─── Integration contract ──────────────────────────────────────────────────
430
+ const DEFAULT_TICKET_CAPABILITIES = ['crud', 'labels', 'status', 'priorities', 'dependencies'];
431
+ const DEFAULT_TICKET_STORAGE_PATH = '.specrails/local-tickets.json';
432
+ // GET /:projectId/integration-contract — Return the project's integration contract with ticketProvider
433
+ router.get('/:projectId/integration-contract', (req, res) => {
434
+ const projectPath = ctx(req).project.path;
435
+ const contractFile = path_1.default.join(projectPath, '.claude', 'integration-contract.json');
436
+ let rawContract = {};
437
+ let source = 'default';
438
+ if (fs_1.default.existsSync(contractFile)) {
439
+ try {
440
+ rawContract = JSON.parse(fs_1.default.readFileSync(contractFile, 'utf-8'));
441
+ source = 'contract';
442
+ }
443
+ catch {
444
+ // malformed contract — fall back to defaults
445
+ }
446
+ }
447
+ const rawProvider = rawContract.ticketProvider;
448
+ const storagePath = rawProvider?.storagePath ?? DEFAULT_TICKET_STORAGE_PATH;
449
+ const ticketProvider = {
450
+ type: rawProvider?.type ?? 'local',
451
+ storagePath: path_1.default.resolve(projectPath, storagePath),
452
+ capabilities: rawProvider?.capabilities ?? DEFAULT_TICKET_CAPABILITIES,
453
+ };
454
+ res.json({ ticketProvider, source });
455
+ });
456
+ }